Add openapi description

This commit is contained in:
Peder Bergebakken Sundt 2018-02-25 02:40:58 +01:00
parent de4ce78bad
commit 92fcac7689
3 changed files with 55 additions and 23 deletions

View File

@ -18,6 +18,17 @@ def make_sanic_app(host="0.0.0.0", port=8080):
app.blueprint(openapi_blueprint) app.blueprint(openapi_blueprint)
app.blueprint(swagger_blueprint) app.blueprint(swagger_blueprint)
#openapi:
app.config.API_VERSION = '1.0'
app.config.API_TITLE = 'Grzegorz API'
app.config.API_DESCRIPTION \
= 'The Grzegorz Brzeczyszczykiewicz API, used to control a running MPV instance'
#app.config.API_TERMS_OF_SERVICE = ''# Supposed to be a link
app.config.API_PRODUCES_CONTENT_TYPES = ['application/json']
app.config.API_CONTACT_EMAIL = 'drift@pvv.ntnu.no'
#app.config.API_LICENSE_NAME = 'BSD 3-Clause License'
#app.config.API_LICENSE_URL = 'todo'
asyncio.set_event_loop(uvloop.new_event_loop()) asyncio.set_event_loop(uvloop.new_event_loop())
server_coro = app.create_server(host=host, port=port) server_coro = app.create_server(host=host, port=port)
loop = asyncio.get_event_loop() loop = asyncio.get_event_loop()

View File

@ -1,9 +1,10 @@
import asyncio import asyncio
from sanic import Blueprint, response from sanic import Blueprint, response
from sanic_openapi import doc
from functools import wraps from functools import wraps
from . import mpv from . import mpv
bp = Blueprint("grzegorz-api") bp = Blueprint("grzegorz-api", strict_slashes=True)
# this blueprint assumes a mpv.MPVControl instance is available # this blueprint assumes a mpv.MPVControl instance is available
# at request.app.config["mpv_control"] # at request.app.config["mpv_control"]
@ -12,15 +13,16 @@ def response_json(func):
@wraps(func) @wraps(func)
async def newfunc(*args, **kwargs): async def newfunc(*args, **kwargs):
try: try:
mpv_control = args[0].app.config["mpv_control"] request = args[0]
body = await func(*args, mpv_control, **kwargs) mpv_control = request.app.config["mpv_control"]
if "error" not in body: out = await func(*args, mpv_control, **kwargs)
body["error"] = False if "error" not in out:
if "request" in body: out["error"] = False
del body["request"] if "request" in out:
if "mpv_control" in body: del out["request"]
del body["mpv_control"] if "mpv_control" in out:
return response.json(body) del out["mpv_control"]
return response.json(out)
except Exception as e: except Exception as e:
return response.json({ return response.json({
"error": e.__class__.__name__, "error": e.__class__.__name__,
@ -37,89 +39,106 @@ def response_text(func):
class APIError(Exception): pass class APIError(Exception): pass
#routes: #routes:
@bp.get("/") @bp.get("")
@doc.exclude(True)
@response_text @response_text
async def root(request): async def root(request):
return "Hello World!" return "Hello friend, I hope you're having a lovely day"
@bp.post("/load") @bp.post("/load")
@doc.summary("Add item to playlist")
@doc.consumes({"path": doc.String("Link to the resource to enqueue")}, required=True)
@response_json @response_json
async def loadfile(request, mpv_control): async def loadfile(request, mpv_control):
if "path" not in request.form: if "path" not in request.args:
raise APIError("no form argument \"path\" provided") raise APIError("No query parameter \"path\" provided")
success = await mpv_control.loadfile(request.form["path"][0]) success = await mpv_control.loadfile(request.args["path"][0])
return locals() return locals()
@bp.get("/play") @bp.get("/play")
@doc.summary("Check whether the player is paused or playing")
@response_json @response_json
async def play_get(request, mpv_control): async def play_get(request, mpv_control):
value = await mpv_control.pause_get() == False value = await mpv_control.pause_get() == False
return locals() return locals()
@bp.post("/play") @bp.post("/play")
@doc.summary("Set whether the player is paused or playing")
@doc.consumes({"play": doc.Boolean("Whether to be playing or not")})
@response_json @response_json
async def play_set(request, mpv_control): async def play_set(request, mpv_control):
if "play" not in request.form: if "play" not in request.args:
raise APIError("No form argument \"play\" provided") raise APIError("No query parameter \"play\" provided")
success = await mpv_control \ success = await mpv_control \
.pause_set(request.form["play"][0] not in ["true", "1"]) .pause_set(request.args["play"][0] not in ["true", "1", True])
return locals() return locals()
@bp.get("/volume") @bp.get("/volume")
@doc.summary("Get the current player volume")
@response_json @response_json
async def volume_get(request, mpv_control): async def volume_get(request, mpv_control):
value = await mpv_control.volume_get() value = await mpv_control.volume_get()
return locals() return locals()
@bp.post("/volume") @bp.post("/volume")
@doc.summary("Set the player volume")
@doc.consumes({"volume": doc.Integer("A number between 0 and 100")})
@response_json @response_json
async def volume_set(request, mpv_control): async def volume_set(request, mpv_control):
if "volume" not in request.form: if "volume" not in request.args:
raise APIError("No form argument \"volume\" provided") raise APIError("No query parameter \"volume\" provided")
success = await mpv_control \ success = await mpv_control \
.volume_set(int(request.form["volume"][0])) .volume_set(int(request.args["volume"][0]))
return locals() return locals()
#@bp.get("/something") #@bp.get("/something")
@doc.summary("")
@response_json @response_json
async def noe(request, mpv_control): async def noe(request, mpv_control):
body = await mpv_control.time_pos_get() body = await mpv_control.time_pos_get()
return body return body
#@bp.get("/something") #@bp.get("/something")
@doc.summary("")
@response_json @response_json
async def noe(request, mpv_control): async def noe(request, mpv_control):
body = await mpv_control.time_remaining_get() body = await mpv_control.time_remaining_get()
return body return body
#@bp.get("/something") #@bp.get("/something")
@doc.summary("")
@response_json @response_json
async def noe(request, mpv_control): async def noe(request, mpv_control):
body = await mpv_control.seek_relative(seconds) body = await mpv_control.seek_relative(seconds)
return body return body
#@bp.get("/something") #@bp.get("/something")
@doc.summary("")
@response_json @response_json
async def noe(request, mpv_control): async def noe(request, mpv_control):
body = await mpv_control.seek_percent(percent) body = await mpv_control.seek_percent(percent)
return body return body
@bp.get("/playlist") @bp.get("/playlist")
@doc.summary("Get the current playlist")
@response_json @response_json
async def playlist_get(request, mpv_control): async def playlist_get(request, mpv_control):
value = await mpv_control.playlist_get() value = await mpv_control.playlist_get()
for i, v in enumerate(value): for i, v in enumerate(value):
if "current" in v and v["current"] == True: if "current" in v and v["current"] == True:
v["playing"] = await mpv_control.pause_get() == False v["playing"] = await mpv_control.pause_get() == False
del i, v
return locals() return locals()
@bp.post("/playlist/next") @bp.post("/playlist/next")
@doc.summary("Skip to the next item in the playlist")
@response_json @response_json
async def playlist_next(request, mpv_control): async def playlist_next(request, mpv_control):
success = await mpv_control.playlist_next() success = await mpv_control.playlist_next()
return locals() return locals()
@bp.post("/playlist/previous") @bp.post("/playlist/previous")
@doc.summary("Go back to the previous item in the playlist")
@response_json @response_json
async def playlist_previous(request, mpv_control): async def playlist_previous(request, mpv_control):
success = await mpv_control.playlist_prev() success = await mpv_control.playlist_prev()

View File

@ -104,11 +104,13 @@ class MPVControl:
return resp["error"] == "success" return resp["error"] == "success"
async def volume_get(self): async def volume_get(self):
resp = await self.send_request({"command":["get_property", "volume"]}) resp = await self.send_request({"command":["get_property", "volume"]})
if "error" in resp and resp["error"] != "success" or 1: if "error" in resp and resp["error"] != "success":
raise MPVError("Unable to set volume!") raise MPVError("Unable to get volume!")
return resp["data"] if "data" in resp else None return resp["data"] if "data" in resp else None
async def volume_set(self, volume): async def volume_set(self, volume):
resp = await self.send_request({"command":["set_property", "volume", volume]}) resp = await self.send_request({"command":["set_property", "volume", volume]})
if "error" in resp and resp["error"] != "success":
raise MPVError("Unable to set volume!")
return resp["error"] == "success" return resp["error"] == "success"
async def time_pos_get(self): async def time_pos_get(self):
return await self.send_request({"command":["get_property", "time-pos"]}) return await self.send_request({"command":["get_property", "time-pos"]})