kopia lustrzana https://github.com/cirospaciari/socketify.py
an example of using high-order functions as middlewares sync and async
rodzic
5f5d43f288
commit
87abd89f7d
|
@ -0,0 +1,37 @@
|
||||||
|
from socketify import App
|
||||||
|
|
||||||
|
|
||||||
|
def middleware(*functions):
|
||||||
|
def middleware_route(res, req):
|
||||||
|
data = None
|
||||||
|
#cicle to all middlewares
|
||||||
|
for function in functions:
|
||||||
|
#call middlewares
|
||||||
|
data = function(res, req, data)
|
||||||
|
#stops if returns Falsy
|
||||||
|
if not data:
|
||||||
|
break
|
||||||
|
|
||||||
|
return middleware_route
|
||||||
|
|
||||||
|
def get_user(authorization_header):
|
||||||
|
if authorization_header:
|
||||||
|
return { 'greeting': 'Hello, World' }
|
||||||
|
return None
|
||||||
|
|
||||||
|
def auth(res, req, data=None):
|
||||||
|
user = get_user(req.get_header('authorization'))
|
||||||
|
if not user:
|
||||||
|
res.write_status(403).end("not authorized")
|
||||||
|
return False
|
||||||
|
|
||||||
|
#returns extra data
|
||||||
|
return user
|
||||||
|
|
||||||
|
def home(res, req, user=None):
|
||||||
|
res.end(user.get('greeting', None))
|
||||||
|
|
||||||
|
app = App()
|
||||||
|
app.get("/", middleware(auth, home))
|
||||||
|
app.listen(3000, lambda config: print("Listening on port http://localhost:%d now\n" % config.port))
|
||||||
|
app.run()
|
|
@ -0,0 +1,46 @@
|
||||||
|
from socketify import App
|
||||||
|
|
||||||
|
|
||||||
|
async def get_user(authorization):
|
||||||
|
if authorization:
|
||||||
|
#do actually something async here
|
||||||
|
return { 'greeting': 'Hello, World' }
|
||||||
|
return None
|
||||||
|
|
||||||
|
def auth(home, queries=[]):
|
||||||
|
#in async query string, arguments and headers are only valid until the first await
|
||||||
|
async def auth_middleware(res, req):
|
||||||
|
#get_headers will preserve headers (and cookies) after await
|
||||||
|
headers = req.get_headers()
|
||||||
|
#get_parameters will preserve all params after await
|
||||||
|
params = req.get_parameters()
|
||||||
|
|
||||||
|
#preserve desired query string data
|
||||||
|
query_data = {}
|
||||||
|
for query in queries:
|
||||||
|
value = req.get_query(query)
|
||||||
|
if value:
|
||||||
|
query_data[query] = value
|
||||||
|
|
||||||
|
user = await get_user(headers.get('authorization', None))
|
||||||
|
if user:
|
||||||
|
return home(res, req, user, query_data)
|
||||||
|
|
||||||
|
return res.write_status(403).cork_end("not authorized")
|
||||||
|
|
||||||
|
return auth_middleware
|
||||||
|
|
||||||
|
|
||||||
|
def home(res, req, user=None, query={}):
|
||||||
|
theme = query.get("theme_color", "light")
|
||||||
|
greeting = user.get('greeting', None)
|
||||||
|
user_id = req.get_parameter(0)
|
||||||
|
res.cork_end(f"{greeting} <br/> theme: {theme} <br/> id: {user_id}")
|
||||||
|
|
||||||
|
app = App()
|
||||||
|
app.get("/user/:id", auth(home, ['theme_color']))
|
||||||
|
app.listen(3000, lambda config: print("Listening on port http://localhost:%d now\n" % config.port))
|
||||||
|
app.run()
|
||||||
|
|
||||||
|
|
||||||
|
#curl --location --request GET 'http://localhost:3000/user/10?theme_color=dark' --header 'Authorization: Bearer 23456789'
|
|
@ -867,13 +867,20 @@ class AppRequest:
|
||||||
self.jar_parsed = False
|
self.jar_parsed = False
|
||||||
self._for_each_header_handler = None
|
self._for_each_header_handler = None
|
||||||
self._ptr = ffi.new_handle(self)
|
self._ptr = ffi.new_handle(self)
|
||||||
|
self._headers = None
|
||||||
|
self._params = None
|
||||||
|
|
||||||
|
|
||||||
def get_cookie(self, name):
|
def get_cookie(self, name):
|
||||||
if self.read_jar == None:
|
if self.read_jar == None:
|
||||||
if self.jar_parsed:
|
if self.jar_parsed:
|
||||||
return None
|
return None
|
||||||
|
|
||||||
|
if self._headers:
|
||||||
|
raw_cookies = self._headers.get("cookie", None)
|
||||||
|
else:
|
||||||
raw_cookies = self.get_header("cookie")
|
raw_cookies = self.get_header("cookie")
|
||||||
|
|
||||||
if raw_cookies:
|
if raw_cookies:
|
||||||
self.jar_parsed = True
|
self.jar_parsed = True
|
||||||
self.read_jar = cookies.SimpleCookie(raw_cookies)
|
self.read_jar = cookies.SimpleCookie(raw_cookies)
|
||||||
|
@ -925,11 +932,14 @@ class AppRequest:
|
||||||
lib.uws_req_for_each_header(self.req, uws_req_for_each_header_handler, self._ptr)
|
lib.uws_req_for_each_header(self.req, uws_req_for_each_header_handler, self._ptr)
|
||||||
|
|
||||||
def get_headers(self):
|
def get_headers(self):
|
||||||
headers = {}
|
if not self._headers is None:
|
||||||
|
return self._headers
|
||||||
|
|
||||||
|
self._headers = {}
|
||||||
def copy_headers(key, value):
|
def copy_headers(key, value):
|
||||||
headers[key] = value
|
self._headers[key] = value
|
||||||
self.for_each_header(copy_headers)
|
self.for_each_header(copy_headers)
|
||||||
return headers
|
return self._headers
|
||||||
|
|
||||||
def get_header(self, lower_case_header):
|
def get_header(self, lower_case_header):
|
||||||
if isinstance(lower_case_header, str):
|
if isinstance(lower_case_header, str):
|
||||||
|
@ -966,7 +976,28 @@ class AppRequest:
|
||||||
return ffi.unpack(buffer_address, length).decode("utf-8")
|
return ffi.unpack(buffer_address, length).decode("utf-8")
|
||||||
except Exception: #invalid utf-8
|
except Exception: #invalid utf-8
|
||||||
return None
|
return None
|
||||||
|
|
||||||
|
def get_parameters(self):
|
||||||
|
if self._params:
|
||||||
|
return self._params
|
||||||
|
self._params = []
|
||||||
|
i = 0
|
||||||
|
while True:
|
||||||
|
value = self.get_parameter(i)
|
||||||
|
if value:
|
||||||
|
self._params.append(value)
|
||||||
|
else:
|
||||||
|
break
|
||||||
|
i = i + 1
|
||||||
|
return self._params
|
||||||
|
|
||||||
def get_parameter(self, index):
|
def get_parameter(self, index):
|
||||||
|
if self._params:
|
||||||
|
try:
|
||||||
|
return self._params[index]
|
||||||
|
except:
|
||||||
|
return None
|
||||||
|
|
||||||
buffer = ffi.new("char**")
|
buffer = ffi.new("char**")
|
||||||
length = lib.uws_req_get_parameter(self.req, ffi.cast("unsigned short", index), buffer)
|
length = lib.uws_req_get_parameter(self.req, ffi.cast("unsigned short", index), buffer)
|
||||||
buffer_address = ffi.addressof(buffer, 0)[0]
|
buffer_address = ffi.addressof(buffer, 0)[0]
|
||||||
|
@ -1350,9 +1381,9 @@ class AppResponse:
|
||||||
return 0
|
return 0
|
||||||
|
|
||||||
def has_responded(self):
|
def has_responded(self):
|
||||||
if not self.aborted:
|
if self.aborted:
|
||||||
return False
|
return False
|
||||||
return bool(lib.uws_res_has_responded(self.SSL, self.res, data, len(data)))
|
return bool(lib.uws_res_has_responded(self.SSL, self.res))
|
||||||
|
|
||||||
def on_aborted(self, handler):
|
def on_aborted(self, handler):
|
||||||
if hasattr(handler, '__call__'):
|
if hasattr(handler, '__call__'):
|
||||||
|
|
Ładowanie…
Reference in New Issue