Skip to content

Commit a13510b

Browse files
committed
Handle storm preconditions which throw exceptions for AuthDeny and NoSuchView with their own codes
Update the loginV1 handler to sent 403/404 codes on errors.
1 parent e807a1c commit a13510b

File tree

3 files changed

+32
-13
lines changed

3 files changed

+32
-13
lines changed
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,5 @@
11
---
2-
desc: Update Synapse HTTP APIs to set a non-200 HTTP status code when errors are
3-
returned. This does not apply to the ``api/v1/login`` endpoint.
2+
desc: Update Synapse HTTP APIs to set a non-200 HTTP status code when errors are returned.
43
prs: []
54
type: bug
65
...

synapse/lib/httpapi.py

+12-5
Original file line numberDiff line numberDiff line change
@@ -551,6 +551,13 @@ async def _reqValidOpts(self, opts: dict | None) -> dict | None:
551551

552552
return opts
553553

554+
def _handleStormErr(self, err: Exception):
555+
if isinstance(err, s_exc.AuthDeny):
556+
return self.sendRestExc(err, status_code=HTTPStatus.FORBIDDEN)
557+
if isinstance(err, s_exc.NoSuchView):
558+
return self.sendRestExc(err, status_code=HTTPStatus.NOT_FOUND)
559+
return self.sendRestExc(err, status_code=HTTPStatus.BAD_REQUEST)
560+
554561
class StormNodesV1(StormHandler):
555562

556563
async def post(self):
@@ -586,7 +593,7 @@ async def get(self):
586593
flushed = True
587594
except Exception as e:
588595
if not flushed:
589-
return self.sendRestExc(e, status_code=HTTPStatus.BAD_REQUEST)
596+
return self._handleStormErr(e)
590597

591598
class StormV1(StormHandler):
592599

@@ -621,7 +628,7 @@ async def get(self):
621628
flushed = True
622629
except Exception as e:
623630
if not flushed:
624-
return self.sendRestExc(e, status_code=HTTPStatus.BAD_REQUEST)
631+
return self._handleStormErr(e)
625632

626633
class StormCallV1(StormHandler):
627634

@@ -647,7 +654,7 @@ async def get(self):
647654
try:
648655
ret = await self.getCore().callStorm(query, opts=opts)
649656
except Exception as e:
650-
return self.sendRestExc(e, status_code=HTTPStatus.BAD_REQUEST)
657+
return self._handleStormErr(e)
651658
else:
652659
return self.sendRestRetn(ret)
653660

@@ -681,7 +688,7 @@ async def get(self):
681688
flushed = True
682689
except Exception as e:
683690
if not flushed:
684-
return self.sendRestExc(e, status_code=HTTPStatus.BAD_REQUEST)
691+
return self._handleStormErr(e)
685692

686693
class ReqValidStormV1(StormHandler):
687694

@@ -700,7 +707,7 @@ async def get(self):
700707
try:
701708
ret = await self.cell.reqValidStorm(query, opts)
702709
except Exception as e:
703-
return self.sendRestExc(e, status_code=HTTPStatus.BAD_REQUEST)
710+
return self._handleStormErr(e)
704711
else:
705712
return self.sendRestRetn(ret)
706713

synapse/tests/test_lib_httpapi.py

+19-6
Original file line numberDiff line numberDiff line change
@@ -430,20 +430,18 @@ async def test_http_auth(self):
430430
async with sess.post(f'https://localhost:{port}/api/v1/login') as resp:
431431
self.eq(resp.status, http.HTTPStatus.BAD_REQUEST)
432432
item = await resp.json()
433-
print(item)
434433
self.eq('SchemaViolation', item.get('code'))
435434
async with sess.post(f'https://localhost:{port}/api/v1/login', json=['newp',]) as resp:
436435
self.eq(resp.status, http.HTTPStatus.BAD_REQUEST)
437436
item = await resp.json()
438-
print(item)
439437
self.eq('SchemaViolation', item.get('code'))
440438

441439
async with self.getHttpSess() as sess:
442440

443441
info = {'user': 'hehe', 'passwd': 'newp'}
444442
with self.getAsyncLoggerStream('synapse.lib.httpapi', 'No such user.') as stream:
445443
async with sess.post(f'https://localhost:{port}/api/v1/login', json=info) as resp:
446-
self.eq(resp.status, http.HTTPStatus.OK)
444+
self.eq(resp.status, http.HTTPStatus.NOT_FOUND)
447445
item = await resp.json()
448446
self.eq('AuthDeny', item.get('code'))
449447
self.true(await stream.wait(timeout=6))
@@ -453,7 +451,7 @@ async def test_http_auth(self):
453451
await core.setUserLocked(visiiden, True)
454452
with self.getAsyncLoggerStream('synapse.lib.httpapi', 'User is locked.') as stream:
455453
async with sess.post(f'https://localhost:{port}/api/v1/login', json=info) as resp:
456-
self.eq(resp.status, http.HTTPStatus.OK)
454+
self.eq(resp.status, http.HTTPStatus.FORBIDDEN)
457455
item = await resp.json()
458456
self.eq('AuthDeny', item.get('code'))
459457
self.true(await stream.wait(timeout=6))
@@ -464,7 +462,7 @@ async def test_http_auth(self):
464462
info = {'user': 'visi', 'passwd': 'borked'}
465463
with self.getAsyncLoggerStream('synapse.lib.httpapi', 'Incorrect password.') as stream:
466464
async with sess.post(f'https://localhost:{port}/api/v1/login', json=info) as resp:
467-
self.eq(resp.status, http.HTTPStatus.OK)
465+
self.eq(resp.status, http.HTTPStatus.FORBIDDEN)
468466
item = await resp.json()
469467
self.eq('AuthDeny', item.get('code'))
470468
self.true(await stream.wait(timeout=6))
@@ -1602,6 +1600,21 @@ async def test_http_storm(self):
16021600
self.true(await task.waitfini(6))
16031601
self.len(0, core.boss.tasks)
16041602

1603+
fork = await core.callStorm('return($lib.view.get().fork().iden)')
1604+
lowuser = await core.auth.addUser('lowuser')
1605+
1606+
async with sess.get(f'https://localhost:{port}/api/v1/storm/nodes',
1607+
json={'query': '.created', 'opts': {'view': s_common.guid()}}) as resp:
1608+
self.eq(resp.status, http.HTTPStatus.NOT_FOUND)
1609+
1610+
async with sess.get(f'https://localhost:{port}/api/v1/storm',
1611+
json={'query': '.created', 'opts': {'view': s_common.guid()}}) as resp:
1612+
self.eq(resp.status, http.HTTPStatus.NOT_FOUND)
1613+
1614+
async with sess.get(f'https://localhost:{port}/api/v1/storm',
1615+
json={'query': '.created', 'opts': {'user': lowuser.iden, 'view': fork}}) as resp:
1616+
self.eq(resp.status, http.HTTPStatus.FORBIDDEN)
1617+
16051618
# check reqvalidstorm with various queries
16061619
tvs = (
16071620
('test:str=test', {}, 'ok'),
@@ -2193,7 +2206,7 @@ async def test_http_locked_admin(self):
21932206
self.eq('NotAuthenticated', item.get('code'))
21942207

21952208
resp = await sess.post(f'{root}/api/v1/login', json={'user': 'visi', 'passwd': 'secret123'})
2196-
self.eq(resp.status, http.HTTPStatus.OK)
2209+
self.eq(resp.status, http.HTTPStatus.FORBIDDEN)
21972210
retn = await resp.json()
21982211
self.eq(retn.get('status'), 'err')
21992212
self.eq(retn.get('code'), 'AuthDeny')

0 commit comments

Comments
 (0)