Coverage for src/tests/test_fixtures.py: 98%

508 statements  

« prev     ^ index     » next       coverage.py v7.6.10, created at 2025-01-22 06:42 +0000

1import os 

2from concurrent import futures 

3from contextlib import contextmanager 

4from dataclasses import dataclass 

5from datetime import date 

6from pathlib import Path 

7from unittest.mock import patch 

8 

9import grpc 

10import pytest 

11from sqlalchemy.orm import close_all_sessions 

12from sqlalchemy.sql import or_, text 

13 

14from couchers.config import config 

15from couchers.constants import GUIDELINES_VERSION, TOS_VERSION 

16from couchers.crypto import random_hex 

17from couchers.db import _get_base_engine, session_scope 

18from couchers.descriptor_pool import get_descriptor_pool 

19from couchers.interceptors import AuthValidatorInterceptor, SessionInterceptor, _try_get_and_update_user_details 

20from couchers.jobs.worker import process_job 

21from couchers.models import ( 

22 Base, 

23 FriendRelationship, 

24 FriendStatus, 

25 HostingStatus, 

26 Language, 

27 LanguageAbility, 

28 LanguageFluency, 

29 MeetupStatus, 

30 Region, 

31 RegionLived, 

32 RegionVisited, 

33 Upload, 

34 User, 

35 UserBlock, 

36 UserSession, 

37) 

38from couchers.servicers.account import Account, Iris 

39from couchers.servicers.admin import Admin 

40from couchers.servicers.api import API 

41from couchers.servicers.auth import Auth, create_session 

42from couchers.servicers.blocking import Blocking 

43from couchers.servicers.bugs import Bugs 

44from couchers.servicers.communities import Communities 

45from couchers.servicers.conversations import Conversations 

46from couchers.servicers.discussions import Discussions 

47from couchers.servicers.donations import Donations, Stripe 

48from couchers.servicers.events import Events 

49from couchers.servicers.gis import GIS 

50from couchers.servicers.groups import Groups 

51from couchers.servicers.jail import Jail 

52from couchers.servicers.media import Media, get_media_auth_interceptor 

53from couchers.servicers.notifications import Notifications 

54from couchers.servicers.pages import Pages 

55from couchers.servicers.references import References 

56from couchers.servicers.reporting import Reporting 

57from couchers.servicers.requests import Requests 

58from couchers.servicers.resources import Resources 

59from couchers.servicers.search import Search 

60from couchers.servicers.threads import Threads 

61from couchers.sql import couchers_select as select 

62from couchers.utils import create_coordinate, now 

63from proto import ( 

64 account_pb2_grpc, 

65 admin_pb2_grpc, 

66 annotations_pb2, 

67 api_pb2_grpc, 

68 auth_pb2_grpc, 

69 blocking_pb2_grpc, 

70 bugs_pb2_grpc, 

71 communities_pb2_grpc, 

72 conversations_pb2_grpc, 

73 discussions_pb2_grpc, 

74 donations_pb2_grpc, 

75 events_pb2_grpc, 

76 gis_pb2_grpc, 

77 groups_pb2_grpc, 

78 iris_pb2_grpc, 

79 jail_pb2_grpc, 

80 media_pb2_grpc, 

81 notifications_pb2_grpc, 

82 pages_pb2_grpc, 

83 references_pb2_grpc, 

84 reporting_pb2_grpc, 

85 requests_pb2_grpc, 

86 resources_pb2_grpc, 

87 search_pb2_grpc, 

88 stripe_pb2_grpc, 

89 threads_pb2_grpc, 

90) 

91 

92 

93def drop_all(): 

94 """drop everything currently in the database""" 

95 with session_scope() as session: 

96 # postgis is required for all the Geographic Information System (GIS) stuff 

97 # pg_trgm is required for trigram based search 

98 # btree_gist is required for gist-based exclusion constraints 

99 session.execute( 

100 text( 

101 "DROP SCHEMA IF EXISTS public CASCADE;" 

102 "DROP SCHEMA IF EXISTS logging CASCADE;" 

103 "DROP EXTENSION IF EXISTS postgis CASCADE;" 

104 "CREATE SCHEMA public;" 

105 "CREATE SCHEMA logging;" 

106 "CREATE EXTENSION postgis;" 

107 "CREATE EXTENSION pg_trgm;" 

108 "CREATE EXTENSION btree_gist;" 

109 ) 

110 ) 

111 

112 # this resets the database connection pool, which caches some stuff postgres-side about objects and will otherwise 

113 # sometimes error out with "ERROR: no spatial operator found for 'st_contains': opfamily 203699 type 203585" 

114 # and similar errors 

115 _get_base_engine().dispose() 

116 

117 close_all_sessions() 

118 

119 

120def create_schema_from_models(): 

121 """ 

122 Create everything from the current models, not incrementally 

123 through migrations. 

124 """ 

125 

126 # create the slugify function 

127 functions = Path(__file__).parent / "slugify.sql" 

128 with open(functions) as f, session_scope() as session: 

129 session.execute(text(f.read())) 

130 

131 Base.metadata.create_all(_get_base_engine()) 

132 

133 

134def populate_testing_resources(session): 

135 """ 

136 Testing version of couchers.resources.copy_resources_to_database 

137 """ 

138 regions = [ 

139 ("AUS", "Australia"), 

140 ("CAN", "Canada"), 

141 ("CHE", "Switzerland"), 

142 ("CUB", "Cuba"), 

143 ("CXR", "Christmas Island"), 

144 ("CZE", "Czechia"), 

145 ("DEU", "Germany"), 

146 ("EGY", "Egypt"), 

147 ("ESP", "Spain"), 

148 ("EST", "Estonia"), 

149 ("FIN", "Finland"), 

150 ("FRA", "France"), 

151 ("GBR", "United Kingdom"), 

152 ("GEO", "Georgia"), 

153 ("GHA", "Ghana"), 

154 ("GRC", "Greece"), 

155 ("HKG", "Hong Kong"), 

156 ("IRL", "Ireland"), 

157 ("ISR", "Israel"), 

158 ("ITA", "Italy"), 

159 ("JPN", "Japan"), 

160 ("LAO", "Laos"), 

161 ("MEX", "Mexico"), 

162 ("MMR", "Myanmar"), 

163 ("NAM", "Namibia"), 

164 ("NLD", "Netherlands"), 

165 ("NZL", "New Zealand"), 

166 ("POL", "Poland"), 

167 ("PRK", "North Korea"), 

168 ("REU", "Réunion"), 

169 ("SGP", "Singapore"), 

170 ("SWE", "Sweden"), 

171 ("THA", "Thailand"), 

172 ("TUR", "Turkey"), 

173 ("TWN", "Taiwan"), 

174 ("USA", "United States"), 

175 ("VNM", "Vietnam"), 

176 ] 

177 

178 languages = [ 

179 ("arb", "Arabic (Standard)"), 

180 ("deu", "German"), 

181 ("eng", "English"), 

182 ("fin", "Finnish"), 

183 ("fra", "French"), 

184 ("heb", "Hebrew"), 

185 ("hun", "Hungarian"), 

186 ("jpn", "Japanese"), 

187 ("pol", "Polish"), 

188 ("swe", "Swedish"), 

189 ("cmn", "Chinese (Mandarin)"), 

190 ] 

191 

192 with open(Path(__file__).parent / ".." / ".." / "resources" / "timezone_areas.sql-fake", "r") as f: 

193 tz_sql = f.read() 

194 

195 for code, name in regions: 

196 session.add(Region(code=code, name=name)) 

197 

198 for code, name in languages: 

199 session.add(Language(code=code, name=name)) 

200 

201 session.execute(text(tz_sql)) 

202 

203 

204def recreate_database(): 

205 """ 

206 Connect to a running Postgres database, build it using metadata.create_all() 

207 """ 

208 

209 # running in non-UTC catches some timezone errors 

210 os.environ["TZ"] = "America/New_York" 

211 

212 # drop everything currently in the database 

213 drop_all() 

214 

215 # create everything from the current models, not incrementally through migrations 

216 create_schema_from_models() 

217 

218 with session_scope() as session: 

219 populate_testing_resources(session) 

220 

221 

222@pytest.fixture() 

223def db(): 

224 """ 

225 Pytest fixture to connect to a running Postgres database and build it using metadata.create_all() 

226 """ 

227 

228 recreate_database() 

229 

230 

231def generate_user(*, delete_user=False, complete_profile=True, **kwargs): 

232 """ 

233 Create a new user, return session token 

234 

235 The user is detached from any session, and you can access its static attributes, but you can't modify it 

236 

237 Use this most of the time 

238 """ 

239 auth = Auth() 

240 

241 with session_scope() as session: 

242 # default args 

243 username = "test_user_" + random_hex(16) 

244 user_opts = { 

245 "username": username, 

246 "email": f"{username}@dev.couchers.org", 

247 # password is just 'password' 

248 # this is hardcoded because the password is slow to hash (so would slow down tests otherwise) 

249 "hashed_password": b"$argon2id$v=19$m=65536,t=2,p=1$4cjGg1bRaZ10k+7XbIDmFg$tZG7JaLrkfyfO7cS233ocq7P8rf3znXR7SAfUt34kJg", 

250 "name": username.capitalize(), 

251 "hosting_status": HostingStatus.cant_host, 

252 "meetup_status": MeetupStatus.open_to_meetup, 

253 "city": "Testing city", 

254 "hometown": "Test hometown", 

255 "community_standing": 0.5, 

256 "birthdate": date(year=2000, month=1, day=1), 

257 "gender": "N/A", 

258 "pronouns": "", 

259 "occupation": "Tester", 

260 "education": "UST(esting)", 

261 "about_me": "I test things", 

262 "things_i_like": "Code", 

263 "about_place": "My place has a lot of testing paraphenelia", 

264 "additional_information": "I can be a bit testy", 

265 # you need to make sure to update this logic to make sure the user is jailed/not on request 

266 "accepted_tos": TOS_VERSION, 

267 "accepted_community_guidelines": GUIDELINES_VERSION, 

268 "geom": create_coordinate(40.7108, -73.9740), 

269 "geom_radius": 100, 

270 "onboarding_emails_sent": 1, 

271 "last_onboarding_email_sent": now(), 

272 "has_donated": True, 

273 } 

274 

275 for key, value in kwargs.items(): 

276 user_opts[key] = value 

277 

278 user = User(**user_opts) 

279 session.add(user) 

280 session.flush() 

281 

282 session.add(RegionVisited(user_id=user.id, region_code="CHE")) 

283 session.add(RegionVisited(user_id=user.id, region_code="REU")) 

284 session.add(RegionVisited(user_id=user.id, region_code="FIN")) 

285 

286 session.add(RegionLived(user_id=user.id, region_code="ESP")) 

287 session.add(RegionLived(user_id=user.id, region_code="FRA")) 

288 session.add(RegionLived(user_id=user.id, region_code="EST")) 

289 

290 session.add(LanguageAbility(user_id=user.id, language_code="fin", fluency=LanguageFluency.fluent)) 

291 session.add(LanguageAbility(user_id=user.id, language_code="fra", fluency=LanguageFluency.beginner)) 

292 

293 # this expires the user, so now it's "dirty" 

294 session.commit() 

295 

296 class _DummyContext: 

297 def invocation_metadata(self): 

298 return {} 

299 

300 token, _ = create_session(_DummyContext(), session, user, False, set_cookie=False) 

301 

302 # deleted user aborts session creation, hence this follows and necessitates a second commit 

303 if delete_user: 

304 user.is_deleted = True 

305 

306 user.recommendation_score = 1e10 - user.id 

307 

308 if complete_profile: 

309 key = random_hex(32) 

310 filename = random_hex(32) + ".jpg" 

311 session.add( 

312 Upload( 

313 key=key, 

314 filename=filename, 

315 creator_user_id=user.id, 

316 ) 

317 ) 

318 session.flush() 

319 user.avatar_key = key 

320 user.about_me = "I have a complete profile!\n" * 20 

321 

322 session.commit() 

323 

324 assert user.has_completed_profile == complete_profile 

325 

326 # refresh it, undoes the expiry 

327 session.refresh(user) 

328 

329 # this loads the user's timezone info which is lazy loaded, otherwise we'll get issues if we try to refer to it 

330 user.timezone # noqa: B018 

331 

332 # allows detaches the user from the session, allowing its use outside this session 

333 session.expunge(user) 

334 

335 return user, token 

336 

337 

338def get_user_id_and_token(session, username): 

339 user_id = session.execute(select(User).where(User.username == username)).scalar_one().id 

340 token = session.execute(select(UserSession).where(UserSession.user_id == user_id)).scalar_one().token 

341 return user_id, token 

342 

343 

344def make_friends(user1, user2): 

345 with session_scope() as session: 

346 friend_relationship = FriendRelationship( 

347 from_user_id=user1.id, 

348 to_user_id=user2.id, 

349 status=FriendStatus.accepted, 

350 ) 

351 session.add(friend_relationship) 

352 

353 

354def make_user_block(user1, user2): 

355 with session_scope() as session: 

356 user_block = UserBlock( 

357 blocking_user_id=user1.id, 

358 blocked_user_id=user2.id, 

359 ) 

360 session.add(user_block) 

361 session.commit() 

362 

363 

364def make_user_invisible(user_id): 

365 with session_scope() as session: 

366 session.execute(select(User).where(User.id == user_id)).scalar_one().is_banned = True 

367 

368 

369# This doubles as get_FriendRequest, since a friend request is just a pending friend relationship 

370def get_friend_relationship(user1, user2): 

371 with session_scope() as session: 

372 friend_relationship = session.execute( 

373 select(FriendRelationship).where( 

374 or_( 

375 (FriendRelationship.from_user_id == user1.id and FriendRelationship.to_user_id == user2.id), 

376 (FriendRelationship.from_user_id == user2.id and FriendRelationship.to_user_id == user1.id), 

377 ) 

378 ) 

379 ).scalar_one_or_none() 

380 

381 session.expunge(friend_relationship) 

382 return friend_relationship 

383 

384 

385class CookieMetadataPlugin(grpc.AuthMetadataPlugin): 

386 """ 

387 Injects the right `cookie: couchers-sesh=...` header into the metadata 

388 """ 

389 

390 def __init__(self, token): 

391 self.token = token 

392 

393 def __call__(self, context, callback): 

394 callback((("cookie", f"couchers-sesh={self.token}"),), None) 

395 

396 

397@contextmanager 

398def auth_api_session(grpc_channel_options=()): 

399 """ 

400 Create an Auth API for testing 

401 

402 This needs to use the real server since it plays around with headers 

403 """ 

404 with futures.ThreadPoolExecutor(1) as executor: 

405 server = grpc.server(executor, interceptors=[AuthValidatorInterceptor(), SessionInterceptor()]) 

406 port = server.add_secure_port("localhost:0", grpc.local_server_credentials()) 

407 auth_pb2_grpc.add_AuthServicer_to_server(Auth(), server) 

408 server.start() 

409 

410 try: 

411 with grpc.secure_channel( 

412 f"localhost:{port}", grpc.local_channel_credentials(), options=grpc_channel_options 

413 ) as channel: 

414 

415 class _MetadataKeeperInterceptor(grpc.UnaryUnaryClientInterceptor): 

416 def __init__(self): 

417 self.latest_headers = {} 

418 

419 def intercept_unary_unary(self, continuation, client_call_details, request): 

420 call = continuation(client_call_details, request) 

421 self.latest_headers = dict(call.initial_metadata()) 

422 self.latest_header_raw = call.initial_metadata() 

423 return call 

424 

425 metadata_interceptor = _MetadataKeeperInterceptor() 

426 channel = grpc.intercept_channel(channel, metadata_interceptor) 

427 yield auth_pb2_grpc.AuthStub(channel), metadata_interceptor 

428 finally: 

429 server.stop(None).wait() 

430 

431 

432@contextmanager 

433def api_session(token): 

434 """ 

435 Create an API for testing, uses the token for auth 

436 """ 

437 channel = fake_channel(token) 

438 api_pb2_grpc.add_APIServicer_to_server(API(), channel) 

439 yield api_pb2_grpc.APIStub(channel) 

440 

441 

442@contextmanager 

443def real_api_session(token): 

444 """ 

445 Create an API for testing, using TCP sockets, uses the token for auth 

446 """ 

447 with futures.ThreadPoolExecutor(1) as executor: 

448 server = grpc.server(executor, interceptors=[AuthValidatorInterceptor(), SessionInterceptor()]) 

449 port = server.add_secure_port("localhost:0", grpc.local_server_credentials()) 

450 api_pb2_grpc.add_APIServicer_to_server(API(), server) 

451 server.start() 

452 

453 call_creds = grpc.metadata_call_credentials(CookieMetadataPlugin(token)) 

454 comp_creds = grpc.composite_channel_credentials(grpc.local_channel_credentials(), call_creds) 

455 

456 try: 

457 with grpc.secure_channel(f"localhost:{port}", comp_creds) as channel: 

458 yield api_pb2_grpc.APIStub(channel) 

459 finally: 

460 server.stop(None).wait() 

461 

462 

463@contextmanager 

464def real_admin_session(token): 

465 """ 

466 Create a Admin service for testing, using TCP sockets, uses the token for auth 

467 """ 

468 with futures.ThreadPoolExecutor(1) as executor: 

469 server = grpc.server(executor, interceptors=[AuthValidatorInterceptor(), SessionInterceptor()]) 

470 port = server.add_secure_port("localhost:0", grpc.local_server_credentials()) 

471 admin_pb2_grpc.add_AdminServicer_to_server(Admin(), server) 

472 server.start() 

473 

474 call_creds = grpc.metadata_call_credentials(CookieMetadataPlugin(token)) 

475 comp_creds = grpc.composite_channel_credentials(grpc.local_channel_credentials(), call_creds) 

476 

477 try: 

478 with grpc.secure_channel(f"localhost:{port}", comp_creds) as channel: 

479 yield admin_pb2_grpc.AdminStub(channel) 

480 finally: 

481 server.stop(None).wait() 

482 

483 

484@contextmanager 

485def real_account_session(token): 

486 """ 

487 Create a Account service for testing, using TCP sockets, uses the token for auth 

488 """ 

489 with futures.ThreadPoolExecutor(1) as executor: 

490 server = grpc.server(executor, interceptors=[AuthValidatorInterceptor(), SessionInterceptor()]) 

491 port = server.add_secure_port("localhost:0", grpc.local_server_credentials()) 

492 account_pb2_grpc.add_AccountServicer_to_server(Account(), server) 

493 server.start() 

494 

495 call_creds = grpc.metadata_call_credentials(CookieMetadataPlugin(token)) 

496 comp_creds = grpc.composite_channel_credentials(grpc.local_channel_credentials(), call_creds) 

497 

498 try: 

499 with grpc.secure_channel(f"localhost:{port}", comp_creds) as channel: 

500 yield account_pb2_grpc.AccountStub(channel) 

501 finally: 

502 server.stop(None).wait() 

503 

504 

505@contextmanager 

506def real_jail_session(token): 

507 """ 

508 Create a Jail service for testing, using TCP sockets, uses the token for auth 

509 """ 

510 with futures.ThreadPoolExecutor(1) as executor: 

511 server = grpc.server(executor, interceptors=[AuthValidatorInterceptor(), SessionInterceptor()]) 

512 port = server.add_secure_port("localhost:0", grpc.local_server_credentials()) 

513 jail_pb2_grpc.add_JailServicer_to_server(Jail(), server) 

514 server.start() 

515 

516 call_creds = grpc.metadata_call_credentials(CookieMetadataPlugin(token)) 

517 comp_creds = grpc.composite_channel_credentials(grpc.local_channel_credentials(), call_creds) 

518 

519 try: 

520 with grpc.secure_channel(f"localhost:{port}", comp_creds) as channel: 

521 yield jail_pb2_grpc.JailStub(channel) 

522 finally: 

523 server.stop(None).wait() 

524 

525 

526@contextmanager 

527def gis_session(token): 

528 channel = fake_channel(token) 

529 gis_pb2_grpc.add_GISServicer_to_server(GIS(), channel) 

530 yield gis_pb2_grpc.GISStub(channel) 

531 

532 

533class FakeRpcError(grpc.RpcError): 

534 def __init__(self, code, details): 

535 self._code = code 

536 self._details = details 

537 

538 def code(self): 

539 return self._code 

540 

541 def details(self): 

542 return self._details 

543 

544 

545def _check_user_perms(method, user_id, is_jailed, is_superuser, token_expiry): 

546 # method is of the form "/org.couchers.api.core.API/GetUser" 

547 _, service_name, method_name = method.split("/") 

548 

549 service_options = get_descriptor_pool().FindServiceByName(service_name).GetOptions() 

550 auth_level = service_options.Extensions[annotations_pb2.auth_level] 

551 assert auth_level != annotations_pb2.AUTH_LEVEL_UNKNOWN 

552 assert auth_level in [ 

553 annotations_pb2.AUTH_LEVEL_OPEN, 

554 annotations_pb2.AUTH_LEVEL_JAILED, 

555 annotations_pb2.AUTH_LEVEL_SECURE, 

556 annotations_pb2.AUTH_LEVEL_ADMIN, 

557 ] 

558 

559 if not user_id: 

560 assert auth_level == annotations_pb2.AUTH_LEVEL_OPEN 

561 else: 

562 assert not (auth_level == annotations_pb2.AUTH_LEVEL_ADMIN and not is_superuser), ( 

563 "Non-superuser tried to call superuser API" 

564 ) 

565 assert not ( 

566 is_jailed and auth_level not in [annotations_pb2.AUTH_LEVEL_OPEN, annotations_pb2.AUTH_LEVEL_JAILED] 

567 ), "User is jailed but tried to call non-open/non-jailed API" 

568 

569 

570class FakeChannel: 

571 def __init__(self, user_id=None, is_jailed=None, is_superuser=None, token_expiry=None): 

572 self.handlers = {} 

573 self.user_id = user_id 

574 self._is_jailed = is_jailed 

575 self._is_superuser = is_superuser 

576 self._token_expiry = token_expiry 

577 

578 def abort(self, code, details): 

579 raise FakeRpcError(code, details) 

580 

581 def add_generic_rpc_handlers(self, generic_rpc_handlers): 

582 from grpc._server import _validate_generic_rpc_handlers 

583 

584 _validate_generic_rpc_handlers(generic_rpc_handlers) 

585 

586 self.handlers.update(generic_rpc_handlers[0]._method_handlers) 

587 

588 def unary_unary(self, uri, request_serializer, response_deserializer): 

589 handler = self.handlers[uri] 

590 

591 _check_user_perms(uri, self.user_id, self._is_jailed, self._is_superuser, self._token_expiry) 

592 

593 def fake_handler(request): 

594 # Do a full serialization cycle on the request and the 

595 # response to catch accidental use of unserializable data. 

596 request = handler.request_deserializer(request_serializer(request)) 

597 

598 with session_scope() as session: 

599 response = handler.unary_unary(request, self, session) 

600 

601 return response_deserializer(handler.response_serializer(response)) 

602 

603 return fake_handler 

604 

605 

606def fake_channel(token=None): 

607 if token: 

608 user_id, is_jailed, is_superuser, token_expiry = _try_get_and_update_user_details( 

609 token, is_api_key=False, ip_address="127.0.0.1", user_agent="Testing User-Agent" 

610 ) 

611 return FakeChannel(user_id=user_id, is_jailed=is_jailed, is_superuser=is_superuser, token_expiry=token_expiry) 

612 return FakeChannel() 

613 

614 

615@contextmanager 

616def conversations_session(token): 

617 """ 

618 Create a Conversations API for testing, uses the token for auth 

619 """ 

620 channel = fake_channel(token) 

621 conversations_pb2_grpc.add_ConversationsServicer_to_server(Conversations(), channel) 

622 yield conversations_pb2_grpc.ConversationsStub(channel) 

623 

624 

625@contextmanager 

626def requests_session(token): 

627 """ 

628 Create a Requests API for testing, uses the token for auth 

629 """ 

630 channel = fake_channel(token) 

631 requests_pb2_grpc.add_RequestsServicer_to_server(Requests(), channel) 

632 yield requests_pb2_grpc.RequestsStub(channel) 

633 

634 

635@contextmanager 

636def threads_session(token): 

637 channel = fake_channel(token) 

638 threads_pb2_grpc.add_ThreadsServicer_to_server(Threads(), channel) 

639 yield threads_pb2_grpc.ThreadsStub(channel) 

640 

641 

642@contextmanager 

643def discussions_session(token): 

644 channel = fake_channel(token) 

645 discussions_pb2_grpc.add_DiscussionsServicer_to_server(Discussions(), channel) 

646 yield discussions_pb2_grpc.DiscussionsStub(channel) 

647 

648 

649@contextmanager 

650def donations_session(token): 

651 channel = fake_channel(token) 

652 donations_pb2_grpc.add_DonationsServicer_to_server(Donations(), channel) 

653 yield donations_pb2_grpc.DonationsStub(channel) 

654 

655 

656@contextmanager 

657def real_stripe_session(): 

658 """ 

659 Create a Stripe service for testing, using TCP sockets 

660 """ 

661 with futures.ThreadPoolExecutor(1) as executor: 

662 server = grpc.server(executor, interceptors=[AuthValidatorInterceptor(), SessionInterceptor()]) 

663 port = server.add_secure_port("localhost:0", grpc.local_server_credentials()) 

664 stripe_pb2_grpc.add_StripeServicer_to_server(Stripe(), server) 

665 server.start() 

666 

667 creds = grpc.local_channel_credentials() 

668 

669 try: 

670 with grpc.secure_channel(f"localhost:{port}", creds) as channel: 

671 yield stripe_pb2_grpc.StripeStub(channel) 

672 finally: 

673 server.stop(None).wait() 

674 

675 

676@contextmanager 

677def real_iris_session(): 

678 with futures.ThreadPoolExecutor(1) as executor: 

679 server = grpc.server(executor, interceptors=[AuthValidatorInterceptor(), SessionInterceptor()]) 

680 port = server.add_secure_port("localhost:0", grpc.local_server_credentials()) 

681 iris_pb2_grpc.add_IrisServicer_to_server(Iris(), server) 

682 server.start() 

683 

684 creds = grpc.local_channel_credentials() 

685 

686 try: 

687 with grpc.secure_channel(f"localhost:{port}", creds) as channel: 

688 yield iris_pb2_grpc.IrisStub(channel) 

689 finally: 

690 server.stop(None).wait() 

691 

692 

693@contextmanager 

694def pages_session(token): 

695 channel = fake_channel(token) 

696 pages_pb2_grpc.add_PagesServicer_to_server(Pages(), channel) 

697 yield pages_pb2_grpc.PagesStub(channel) 

698 

699 

700@contextmanager 

701def communities_session(token): 

702 channel = fake_channel(token) 

703 communities_pb2_grpc.add_CommunitiesServicer_to_server(Communities(), channel) 

704 yield communities_pb2_grpc.CommunitiesStub(channel) 

705 

706 

707@contextmanager 

708def groups_session(token): 

709 channel = fake_channel(token) 

710 groups_pb2_grpc.add_GroupsServicer_to_server(Groups(), channel) 

711 yield groups_pb2_grpc.GroupsStub(channel) 

712 

713 

714@contextmanager 

715def blocking_session(token): 

716 channel = fake_channel(token) 

717 blocking_pb2_grpc.add_BlockingServicer_to_server(Blocking(), channel) 

718 yield blocking_pb2_grpc.BlockingStub(channel) 

719 

720 

721@contextmanager 

722def notifications_session(token): 

723 channel = fake_channel(token) 

724 notifications_pb2_grpc.add_NotificationsServicer_to_server(Notifications(), channel) 

725 yield notifications_pb2_grpc.NotificationsStub(channel) 

726 

727 

728@contextmanager 

729def account_session(token): 

730 """ 

731 Create a Account API for testing, uses the token for auth 

732 """ 

733 channel = fake_channel(token) 

734 account_pb2_grpc.add_AccountServicer_to_server(Account(), channel) 

735 yield account_pb2_grpc.AccountStub(channel) 

736 

737 

738@contextmanager 

739def search_session(token): 

740 """ 

741 Create a Search API for testing, uses the token for auth 

742 """ 

743 channel = fake_channel(token) 

744 search_pb2_grpc.add_SearchServicer_to_server(Search(), channel) 

745 yield search_pb2_grpc.SearchStub(channel) 

746 

747 

748@contextmanager 

749def references_session(token): 

750 """ 

751 Create a References API for testing, uses the token for auth 

752 """ 

753 channel = fake_channel(token) 

754 references_pb2_grpc.add_ReferencesServicer_to_server(References(), channel) 

755 yield references_pb2_grpc.ReferencesStub(channel) 

756 

757 

758@contextmanager 

759def reporting_session(token): 

760 channel = fake_channel(token) 

761 reporting_pb2_grpc.add_ReportingServicer_to_server(Reporting(), channel) 

762 yield reporting_pb2_grpc.ReportingStub(channel) 

763 

764 

765@contextmanager 

766def events_session(token): 

767 channel = fake_channel(token) 

768 events_pb2_grpc.add_EventsServicer_to_server(Events(), channel) 

769 yield events_pb2_grpc.EventsStub(channel) 

770 

771 

772@contextmanager 

773def bugs_session(token=None): 

774 channel = fake_channel(token) 

775 bugs_pb2_grpc.add_BugsServicer_to_server(Bugs(), channel) 

776 yield bugs_pb2_grpc.BugsStub(channel) 

777 

778 

779@contextmanager 

780def resources_session(): 

781 channel = fake_channel() 

782 resources_pb2_grpc.add_ResourcesServicer_to_server(Resources(), channel) 

783 yield resources_pb2_grpc.ResourcesStub(channel) 

784 

785 

786@contextmanager 

787def media_session(bearer_token): 

788 """ 

789 Create a fresh Media API for testing, uses the bearer token for media auth 

790 """ 

791 media_auth_interceptor = get_media_auth_interceptor(bearer_token) 

792 

793 with futures.ThreadPoolExecutor(1) as executor: 

794 server = grpc.server(executor, interceptors=[media_auth_interceptor, SessionInterceptor()]) 

795 port = server.add_secure_port("localhost:0", grpc.local_server_credentials()) 

796 servicer = Media() 

797 media_pb2_grpc.add_MediaServicer_to_server(servicer, server) 

798 server.start() 

799 

800 call_creds = grpc.access_token_call_credentials(bearer_token) 

801 comp_creds = grpc.composite_channel_credentials(grpc.local_channel_credentials(), call_creds) 

802 

803 try: 

804 with grpc.secure_channel(f"localhost:{port}", comp_creds) as channel: 

805 yield media_pb2_grpc.MediaStub(channel) 

806 finally: 

807 server.stop(None).wait() 

808 

809 

810@pytest.fixture(scope="class") 

811def testconfig(): 

812 prevconfig = config.copy() 

813 config.clear() 

814 config.update(prevconfig) 

815 

816 config["IN_TEST"] = True 

817 

818 config["DEV"] = True 

819 config["SECRET"] = bytes.fromhex("448697d3886aec65830a1ea1497cdf804981e0c260d2f812cf2787c4ed1a262b") 

820 config["VERSION"] = "testing_version" 

821 config["BASE_URL"] = "http://localhost:3000" 

822 config["BACKEND_BASE_URL"] = "http://localhost:8888" 

823 config["CONSOLE_BASE_URL"] = "http://localhost:8888" 

824 config["COOKIE_DOMAIN"] = "localhost" 

825 

826 config["ENABLE_SMS"] = False 

827 config["SMS_SENDER_ID"] = "invalid" 

828 

829 config["ENABLE_EMAIL"] = False 

830 config["NOTIFICATION_EMAIL_SENDER"] = "Couchers.org" 

831 config["NOTIFICATION_EMAIL_ADDRESS"] = "notify@couchers.org.invalid" 

832 config["NOTIFICATION_PREFIX"] = "[TEST] " 

833 config["REPORTS_EMAIL_RECIPIENT"] = "reports@couchers.org.invalid" 

834 config["CONTRIBUTOR_FORM_EMAIL_RECIPIENT"] = "forms@couchers.org.invalid" 

835 config["MODS_EMAIL_RECIPIENT"] = "mods@couchers.org.invalid" 

836 

837 config["ENABLE_DONATIONS"] = False 

838 config["STRIPE_API_KEY"] = "" 

839 config["STRIPE_WEBHOOK_SECRET"] = "" 

840 config["STRIPE_RECURRING_PRODUCT_ID"] = "" 

841 

842 config["ENABLE_STRONG_VERIFICATION"] = False 

843 config["IRIS_ID_PUBKEY"] = "" 

844 config["IRIS_ID_SECRET"] = "" 

845 # corresponds to private key e6c2fbf3756b387bc09a458a7b85935718ef3eb1c2777ef41d335c9f6c0ab272 

846 config["VERIFICATION_DATA_PUBLIC_KEY"] = bytes.fromhex( 

847 "dd740a2b2a35bf05041a28257ea439b30f76f056f3698000b71e6470cd82275f" 

848 ) 

849 

850 config["SMTP_HOST"] = "localhost" 

851 config["SMTP_PORT"] = 587 

852 config["SMTP_USERNAME"] = "username" 

853 config["SMTP_PASSWORD"] = "password" 

854 

855 config["ENABLE_MEDIA"] = True 

856 config["MEDIA_SERVER_SECRET_KEY"] = bytes.fromhex( 

857 "91e29bbacc74fa7e23c5d5f34cca5015cb896e338a620003de94a502a461f4bc" 

858 ) 

859 config["MEDIA_SERVER_BEARER_TOKEN"] = "c02d383897d3b82774ced09c9e17802164c37e7e105d8927553697bf4550e91e" 

860 config["MEDIA_SERVER_BASE_URL"] = "http://localhost:5001" 

861 config["MEDIA_SERVER_UPLOAD_BASE_URL"] = "http://localhost:5001" 

862 

863 config["BUG_TOOL_ENABLED"] = False 

864 config["BUG_TOOL_GITHUB_REPO"] = "org/repo" 

865 config["BUG_TOOL_GITHUB_USERNAME"] = "user" 

866 config["BUG_TOOL_GITHUB_TOKEN"] = "token" 

867 

868 config["LISTMONK_ENABLED"] = False 

869 config["LISTMONK_BASE_URL"] = "https://localhost" 

870 config["LISTMONK_API_USERNAME"] = "..." 

871 config["LISTMONK_API_KEY"] = "..." 

872 config["LISTMONK_LIST_ID"] = 3 

873 

874 config["PUSH_NOTIFICATIONS_ENABLED"] = True 

875 config["PUSH_NOTIFICATIONS_VAPID_PRIVATE_KEY"] = "uI1DCR4G1AdlmMlPfRLemMxrz9f3h4kvjfnI8K9WsVI" 

876 config["PUSH_NOTIFICATIONS_VAPID_SUBJECT"] = "mailto:testing@couchers.org.invalid" 

877 

878 yield None 

879 

880 config.clear() 

881 config.update(prevconfig) 

882 

883 

884@pytest.fixture 

885def fast_passwords(): 

886 # password hashing, by design, takes a lot of time, which slows down the tests. here we jump through some hoops to 

887 # make this fast by removing the hashing step 

888 

889 def fast_hash(password: bytes) -> bytes: 

890 return b"fake hash:" + password 

891 

892 def fast_verify(hashed: bytes, password: bytes) -> bool: 

893 return hashed == fast_hash(password) 

894 

895 with patch("couchers.crypto.nacl.pwhash.verify", fast_verify): 

896 with patch("couchers.crypto.nacl.pwhash.str", fast_hash): 

897 yield 

898 

899 

900def process_jobs(): 

901 while process_job(): 

902 pass 

903 

904 

905@contextmanager 

906def mock_notification_email(): 

907 with patch("couchers.email._queue_email") as mock: 

908 yield mock 

909 process_jobs() 

910 

911 

912@dataclass 

913class EmailData: 

914 sender_name: str 

915 sender_email: str 

916 recipient: str 

917 subject: str 

918 plain: str 

919 html: str 

920 source_data: str 

921 list_unsubscribe_header: str 

922 

923 

924def email_fields(mock, call_ix=0): 

925 _, kw = mock.call_args_list[call_ix] 

926 return EmailData( 

927 sender_name=kw.get("sender_name"), 

928 sender_email=kw.get("sender_email"), 

929 recipient=kw.get("recipient"), 

930 subject=kw.get("subject"), 

931 plain=kw.get("plain"), 

932 html=kw.get("html"), 

933 source_data=kw.get("source_data"), 

934 list_unsubscribe_header=kw.get("list_unsubscribe_header"), 

935 ) 

936 

937 

938@pytest.fixture 

939def push_collector(): 

940 """ 

941 See test_SendTestPushNotification for an example on how to use this fixture 

942 """ 

943 

944 class Push: 

945 """ 

946 This allows nice access to the push info via e.g. push.title instead of push["title"] 

947 """ 

948 

949 def __init__(self, kwargs): 

950 self.kwargs = kwargs 

951 

952 def __getattr__(self, attr): 

953 try: 

954 return self.kwargs[attr] 

955 except KeyError: 

956 raise AttributeError(f"'{self.__class__.__name__}' object has no attribute '{attr}'") from None 

957 

958 def __repr__(self): 

959 kwargs_disp = ", ".join(f"'{key}'='{val}'" for key, val in self.kwargs.items()) 

960 return f"Push({kwargs_disp})" 

961 

962 class PushCollector: 

963 def __init__(self): 

964 # pairs of (user_id, push) 

965 self.pushes = [] 

966 

967 def by_user(self, user_id): 

968 return [kwargs for uid, kwargs in self.pushes if uid == user_id] 

969 

970 def push_to_user(self, session, user_id, **kwargs): 

971 self.pushes.append((user_id, Push(kwargs=kwargs))) 

972 

973 def assert_user_has_count(self, user_id, count): 

974 assert len(self.by_user(user_id)) == count 

975 

976 def assert_user_push_matches_fields(self, user_id, ix=0, **kwargs): 

977 push = self.by_user(user_id)[ix] 

978 for kwarg in kwargs: 

979 assert kwarg in push.kwargs, f"Push notification {user_id=}, {ix=} missing field '{kwarg}'" 

980 assert push.kwargs[kwarg] == kwargs[kwarg], ( 

981 f"Push notification {user_id=}, {ix=} mismatch in field '{kwarg}', expected '{kwargs[kwarg]}' but got '{push.kwargs[kwarg]}'" 

982 ) 

983 

984 def assert_user_has_single_matching(self, user_id, **kwargs): 

985 self.assert_user_has_count(user_id, 1) 

986 self.assert_user_push_matches_fields(user_id, ix=0, **kwargs) 

987 

988 collector = PushCollector() 

989 

990 with patch("couchers.notifications.push._push_to_user", collector.push_to_user): 

991 yield collector