routes.py 35 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282
  1. import json
  2. from flask import render_template, url_for, flash, redirect, session, jsonify, Response, request, current_app, Blueprint, send_file
  3. from project.models import db, Game, Player, Object, Image, User, SingleplayerGame, get_game_images, reset_db, \
  4. ActionCard, GoalCard, CharacterCard
  5. from PIL import Image as PImage
  6. from flask_login import LoginManager, login_user, logout_user
  7. from project.forms import LoginForm
  8. from flask_limiter import Limiter
  9. from flask_limiter.util import get_remote_address
  10. from pathlib import Path
  11. from werkzeug.utils import secure_filename
  12. from base64 import b64encode
  13. from io import BytesIO
  14. from flask_login import login_required
  15. import os
  16. import glob
  17. limiter = Limiter(
  18. get_remote_address,
  19. default_limits=["20 per second"]
  20. )
  21. api = Blueprint('api', __name__)
  22. limiter.limit('20/second')(api)
  23. singleplayer = Blueprint('singleplayer', __name__)
  24. limiter.limit('20/second')(singleplayer)
  25. login_manager = LoginManager()
  26. def __is_game_closed(gameid):
  27. game = db.session.query(Game).filter(Game.id == gameid).first()
  28. if game is None:
  29. return True
  30. return game.closed
  31. # DONE check whether players are claimed before relevant functions
  32. # DONE restrict interaction with closed games (needed?)
  33. """
  34. ####################
  35. MULTIPLAYER
  36. ####################
  37. """
  38. @api.route("/create_game", methods=['POST'])
  39. @limiter.exempt
  40. def create_game():
  41. """ POST like this
  42. {
  43. players: <int> # numberof players
  44. location: <str>
  45. }
  46. returns game instance number(used to refer to game instance in other calls)
  47. """
  48. # extract values and check validity
  49. try:
  50. # content = request.get_json()
  51. # players = content["players"]
  52. # location = content["location"]
  53. players = request.form.get("players")
  54. location = request.form.get("location")
  55. except KeyError:
  56. return "Missing Data", 400
  57. if (players is None) or (location is None):
  58. return "Missing Data", 400
  59. players = int(players)
  60. # create Player objects
  61. player_list = []
  62. for i in range(0, players):
  63. p = Player(
  64. player_number=i + 1,
  65. )
  66. player_list.append(p)
  67. # create game object
  68. game = Game(
  69. number_of_players=players,
  70. location=location,
  71. players=player_list
  72. )
  73. # save game to db
  74. db.session.add(game)
  75. db.session.commit()
  76. # create a response json
  77. response = {
  78. 'data': game.id,
  79. 'message': f'Game {game.id} created successfully'
  80. }
  81. return response, 200, {'Content-Type': 'application/json; charset=utf-8'}
  82. @api.route("/is_player_available/<int:gameid>/<int:playerid>", methods=['GET'])
  83. @limiter.exempt
  84. def is_player_available(gameid=None, playerid=None):
  85. """
  86. :param gameid: game id
  87. :param playerid:game-specific player id
  88. :return: "True" or "False" as str
  89. """
  90. if gameid is None or playerid is None:
  91. return "Missing game id or player id", 400
  92. result = db.session.query(Player.is_taken).filter(Player.game_id == gameid,
  93. Player.player_number == playerid).first()
  94. if result is None:
  95. return "No such Game or Player", 400
  96. return str(not result[0]), 200
  97. @api.route("/select_player", methods=['POST'])
  98. @limiter.exempt
  99. def select_player():
  100. """ POST like this
  101. {
  102. game: <int> # game id
  103. player: <int> # player id in game
  104. }
  105. """
  106. # extract values and check validity
  107. try:
  108. # content = request.get_json()
  109. # playerid = content["player"]
  110. # gameid = content["game"]
  111. playerid = request.form.get("player")
  112. gameid = request.form.get("game")
  113. except KeyError:
  114. return "Missing Data", 400
  115. if (playerid is None) or (gameid is None):
  116. return "Missing Data", 400
  117. playerid = int(playerid)
  118. gameid = int(gameid)
  119. # check if game is closed. if yes or not found ignore request
  120. if __is_game_closed(gameid):
  121. return "Game is closed or does not exist", 400
  122. player = db.session.query(Player).filter(Player.game_id == gameid,
  123. Player.player_number == playerid).first()
  124. # no player found / player already taken
  125. if player is None:
  126. return "Game or Player does not exist", 400
  127. elif player.is_taken:
  128. return f"Player {playerid} is already taken", 400
  129. player.is_taken = True
  130. db.session.commit()
  131. return "Player claimed successfully", 200
  132. @api.route("/add_object", methods=['POST'])
  133. @limiter.exempt
  134. def add_object():
  135. """ POST like this
  136. {
  137. game: <int> # game id
  138. player: <int> # player id in game
  139. points: <int> # how many points?
  140. oid: <int> # id of object
  141. }
  142. """
  143. # extract values and check validity
  144. try:
  145. # content = request.get_json()
  146. # gameid = content["game"]
  147. # playerid = content["player"]
  148. # points = content["points"]
  149. # oid = content["oid"]
  150. gameid = request.form.get("game")
  151. playerid = request.form.get("player")
  152. points = request.form.get("points")
  153. oid = request.form.get("oid")
  154. longitude = request.form.get("longitude") # optional
  155. latitude = request.form.get("latitude") # optional
  156. name = request.form.get("name") # optional
  157. except KeyError:
  158. return "Missing Data", 400
  159. if (playerid is None) or (gameid is None) or (points is None) or (oid is None):
  160. return "Missing Data", 400
  161. gameid = int(gameid)
  162. playerid = int(playerid)
  163. points = int(points)
  164. oid = int(oid)
  165. # check if game is closed. if yes or not found ignore request
  166. if __is_game_closed(gameid):
  167. return "Game is closed or does not exist", 400
  168. # query player
  169. player = db.session.query(Player).filter(Player.game_id == gameid,
  170. Player.player_number == playerid).first()
  171. # no player found
  172. if player is None:
  173. return "Game or Player does not exist", 400
  174. # if player is not taken error
  175. if not player.is_taken:
  176. return "Player is not taken!", 400
  177. # it is not the players turn
  178. # if not player.is_player_turn:
  179. # return "Other players turn", 400
  180. # check whether object already in play
  181. already_in_play = db.session.query(Object).filter(Object.game_id == gameid,
  182. Object.object_type == oid).first() is not None
  183. if already_in_play:
  184. return f"Object {oid} is already in play", 400
  185. # create and add object to db
  186. obj = Object(
  187. game_id=gameid,
  188. player_number=playerid,
  189. object_type=oid,
  190. object_points=points,
  191. latitude=latitude,
  192. longitude=longitude,
  193. object_name=name,
  194. )
  195. db.session.add(obj)
  196. # add points to player
  197. player.points = player.points + points
  198. db.session.commit()
  199. return "Object played successfully", 200
  200. @api.route("/object_in_play/<int:gameid>/<int:objectid>", methods=['GET'])
  201. @limiter.exempt
  202. def object_in_play(gameid=None, objectid=None):
  203. # extract and validate
  204. if gameid is None or objectid is None:
  205. return "Missing game id or object id", 400
  206. # query object
  207. result = db.session.query(Object).filter(Object.game_id == gameid, Object.object_type == objectid).first()
  208. return str(result is not None), 200
  209. @api.route("/get_all_objects_in_game/<int:gameid>", methods=['GET'])
  210. @limiter.limit("1/2minute", override_defaults=True)
  211. def get_all_objects_in_game(gameid=None):
  212. objects = db.session.query(Object).filter(Object.game_id == gameid).all()
  213. obj_json = {}
  214. for [key, obj] in enumerate(objects):
  215. obj_json[key] = obj.to_json()
  216. response = json.dumps(obj_json)
  217. return response, 200, {'Content-Type': 'application/json; charset=utf-8'}
  218. @api.route("/delete_object", methods=['POST'])
  219. @limiter.exempt
  220. def delete_object():
  221. """ POST like this
  222. //TODO only take game and oid
  223. {
  224. game: <int> # game id
  225. oid: <int> # id/type of object
  226. }
  227. """
  228. # extract values and check validity
  229. try:
  230. # content = request.get_json()
  231. # gameid = content["game"]
  232. # playerid = content["player"]
  233. # points = content["points"]
  234. # oid = content["oid"]
  235. gameid = request.form.get("game")
  236. oid = request.form.get("oid")
  237. except KeyError:
  238. return "Missing Data", 400
  239. if (gameid is None) or (oid is None):
  240. return "Missing Data", 400
  241. gameid = int(gameid)
  242. oid = int(oid)
  243. # check if game is closed. if yes or not found ignore request
  244. if __is_game_closed(gameid):
  245. return "Game is closed or does not exist", 400
  246. obj = db.session.query(Object).filter(Object.game_id == gameid,
  247. Object.object_type == oid).first()
  248. if obj is None:
  249. return f"Game {gameid} has no Object with type {oid}", 400
  250. # query player
  251. player = db.session.query(Player).filter(Player.game_id == gameid,
  252. Player.player_number == obj.player_number).first()
  253. # no player found
  254. if player is None:
  255. return "Player does not exist", 400
  256. # if player is not taken error
  257. if not player.is_taken:
  258. return "Player is not taken!", 400
  259. player.points = player.points - obj.object_points
  260. db.session.delete(obj)
  261. db.session.commit()
  262. return "Object deleted successfully", 200
  263. """
  264. @api.route("/set_turn_to_next_player", methods=['POST'])
  265. def set_turn_to_next_player():
  266. try:
  267. # content = request.get_json()
  268. # gameid = content["game"]
  269. gameid = request.form.get("game")
  270. except KeyError:
  271. return "Missing Data", 400
  272. gameid = int(gameid)
  273. if type(gameid) is not int:
  274. return "Wrong Data", 400
  275. game = db.session.query(Game).filter(Game.id == gameid).first()
  276. if game is None:
  277. return f"Game {gameid} does not exist", 400
  278. # no longer current player's turn
  279. player_curr = db.session.query(Player).filter(Player.game_id == gameid,
  280. Player.is_player_turn).first()
  281. # who's turn is it next?
  282. number_of_player = game.number_of_players
  283. next_player = game.current_player + 1
  284. if next_player > number_of_player:
  285. next_player = 1
  286. # set turn to next players in Game and Player Objects
  287. game.current_player = next_player
  288. player_next = db.session.query(Player).filter(Player.game_id == gameid,
  289. Player.player_number == next_player).first()
  290. if not player_next.is_taken:
  291. return "Next player is not taken!", 400
  292. player_curr.is_player_turn = False
  293. player_next.is_player_turn = True
  294. db.session.commit()
  295. return f"Game {gameid}: Turn switched successfully", 200
  296. """
  297. """@api.route("/get_current_player_turn/<int:gameid>", methods=['GET'])
  298. def get_current_player_turn(gameid=None):
  299. if gameid is None:
  300. return "Missing game id", 400
  301. game = db.session.query(Game).filter(Game.id == gameid).first()
  302. if game is None:
  303. return f"Game {gameid} does not exist", 400
  304. current_player = game.current_player
  305. # create a response json
  306. response = {
  307. 'data': current_player,
  308. 'message': f"Game {game.id}: Player {current_player}'s turn"
  309. }
  310. return response, 200, {'Content-Type': 'application/json; charset=utf-8'}"""
  311. @api.route("/get_all_player_points/<int:gameid>", methods=['GET'])
  312. @limiter.exempt
  313. def get_all_player_points(gameid=None):
  314. """
  315. :param gameid: game id
  316. :return: {
  317. 1: 20 <playerid>: <points>
  318. 2: 15
  319. 3: 12
  320. }
  321. """
  322. if gameid is None:
  323. return "Missing game id", 400
  324. players = db.session.query(Player).filter(Player.game_id == gameid).all()
  325. if len(players) == 0:
  326. return f"Game {gameid} does not exist", 400
  327. players_points = {}
  328. for player in players:
  329. players_points[f'{player.player_number}'] = player.points
  330. return players_points, 200, {'Content-Type': 'application/json; charset=utf-8'}
  331. @api.route("/get_total_number_of_players/<int:gameid>", methods=['GET'])
  332. @limiter.exempt
  333. def get_total_number_of_players(gameid=None):
  334. if gameid is None:
  335. return "Missing game id", 400
  336. game = db.session.query(Game).filter(Game.id == gameid).first()
  337. if game is None:
  338. return f"Game {gameid} does not exist", 400
  339. # create a response json
  340. response = {
  341. 'data': game.number_of_players,
  342. 'message': f"Game {game.id} has {game.number_of_players} Players"
  343. }
  344. return response, 200, {'Content-Type': 'application/json; charset=utf-8'}
  345. @api.route("/get_cards_in_game/<int:gameid>", methods=['GET'])
  346. @limiter.exempt
  347. def get_cards_in_game(gameid=None):
  348. """
  349. :param gameid: game id
  350. :return: {
  351. 1: 20 <playerid>: <goal card>
  352. 2: 15
  353. 3: 12
  354. }
  355. """
  356. if gameid is None:
  357. return "Missing game id", 400
  358. players = db.session.query(Player).filter(Player.game_id == gameid).all()
  359. if len(players) == 0:
  360. return f"Game {gameid} does not exist", 400
  361. players_goal = {}
  362. for player in players:
  363. players_goal[f'{player.player_number}'] = player.goal_card_id
  364. return players_goal, 200, {'Content-Type': 'application/json; charset=utf-8'}
  365. @api.route("/add_card_to_player", methods=['POST'])
  366. @limiter.exempt
  367. def add_card_to_player():
  368. """ POST like this
  369. {
  370. game: <int> # game id
  371. player: <int> # player id in game
  372. cid: <int> # id of card
  373. }
  374. """
  375. try:
  376. # content = request.get_json()
  377. # gameid = content["game"]
  378. # playerid = content["player"]
  379. # cid = content["cid"]
  380. gameid = request.form.get("game")
  381. playerid = request.form.get("player")
  382. cid = request.form.get("cid")
  383. except KeyError:
  384. return "Missing Data", 400
  385. if (gameid is None) or (playerid is None) or (cid is None):
  386. return "Missing Data", 400
  387. gameid = int(gameid)
  388. playerid = int(playerid)
  389. cid = int(cid)
  390. # check if game is closed. if yes or not found ignore request
  391. if __is_game_closed(gameid):
  392. return "Game is closed or does not exist", 400
  393. player = db.session.query(Player).filter(Player.game_id == gameid,
  394. Player.player_number == playerid).first()
  395. # no player found
  396. if player is None:
  397. return "Game or Player does not exist", 400
  398. goal_card = db.session.query(GoalCard).filter(GoalCard.card_id == cid).first()
  399. # no such char card exists
  400. if goal_card is None:
  401. return f"There is no goal card with the id {cid}", 400
  402. # assign goal card
  403. goal_card.players.append(player)
  404. # player.goal_card_id = cid
  405. db.session.commit()
  406. return "Goal Card saved successfully", 200
  407. @api.route("/add_char_card_to_player", methods=['POST'])
  408. @limiter.exempt
  409. def add_char_card_to_player():
  410. """
  411. Adds the specified character card id to the specified player of a specified game
  412. POST like this
  413. {
  414. game: <int> # game id
  415. player: <int> # player id in game
  416. charid: <int> # id of character card
  417. }
  418. """
  419. try:
  420. # content = request.get_json()
  421. # gameid = content["game"]
  422. # playerid = content["player"]
  423. # cid = content["cid"]
  424. gameid = request.form.get("game")
  425. playerid = request.form.get("player")
  426. charid = request.form.get("charid")
  427. except KeyError:
  428. return "Missing Data", 400
  429. if (gameid is None) or (playerid is None) or (charid is None):
  430. return "Missing Data", 400
  431. gameid = int(gameid)
  432. playerid = int(playerid)
  433. charid = int(charid)
  434. # check if game is closed. if yes or not found ignore request
  435. if __is_game_closed(gameid):
  436. return "Game is closed or does not exist", 400
  437. player = db.session.query(Player).filter(Player.game_id == gameid,
  438. Player.player_number == playerid).first()
  439. # no player found
  440. if player is None:
  441. return "Game or Player does not exist", 400
  442. char_card = db.session.query(CharacterCard).filter(CharacterCard.card_id == charid).first()
  443. # no such char card exists
  444. if char_card is None:
  445. return f"There is no character card with the id {charid}", 400
  446. # assign char card
  447. char_card.players.append(player)
  448. # player.character_card_id = charid
  449. db.session.commit()
  450. return "Character Card saved successfully", 200
  451. @api.route("/add_action_card_to_player", methods=['POST'])
  452. @limiter.exempt
  453. def add_action_card_to_player():
  454. """
  455. Adds the specified action card to the specified player of a specified game
  456. POST like this
  457. {
  458. game: <int> # game id
  459. player: <int> # player id in game
  460. actionCardID: <int> # id of character card
  461. }
  462. """
  463. try:
  464. # content = request.get_json()
  465. # gameid = content["game"]
  466. # playerid = content["player"]
  467. # cid = content["cid"]
  468. gameid = request.form.get("game")
  469. playerid = request.form.get("player")
  470. actionid = request.form.get("actionCardID")
  471. except KeyError:
  472. return "Missing Data", 400
  473. if (gameid is None) or (playerid is None) or (actionid is None):
  474. return "Missing Data", 400
  475. gameid = int(gameid)
  476. playerid = int(playerid)
  477. actionid = int(actionid)
  478. # check if game is closed. if yes or not found ignore request
  479. if __is_game_closed(gameid):
  480. return "Game is closed or does not exist", 400
  481. player = db.session.query(Player).filter(Player.game_id == gameid,
  482. Player.player_number == playerid).first()
  483. # no player found
  484. if player is None:
  485. return "Game or Player does not exist", 400
  486. action_card = db.session.query(ActionCard).filter(ActionCard.card_id == actionid).first()
  487. # no such char card exists
  488. if action_card is None:
  489. return f"There is no action card with the id {actionid}", 400
  490. # assign char card
  491. player.drawn_action_cards.append(action_card)
  492. # player.character_card_id = charid
  493. db.session.commit()
  494. return "Action Card assigned successfully", 200
  495. @api.route("/get_char_cards_in_game/<int:gameid>", methods=['GET'])
  496. @limiter.exempt
  497. def get_char_cards_in_game(gameid=None):
  498. """
  499. returns a dictionary listing the character card id for every player of a game
  500. :param gameid: game id
  501. :return: {
  502. 1: 20 <playerid>: <character card>
  503. 2: 15
  504. 3: 12
  505. }
  506. """
  507. if gameid is None:
  508. return "Missing game id", 400
  509. players = db.session.query(Player).filter(Player.game_id == gameid).all()
  510. if len(players) == 0:
  511. return f"Game {gameid} does not exist", 400
  512. players_char = {}
  513. for player in players:
  514. players_char[f'{player.player_number}'] = player.character_card_id
  515. return players_char, 200, {'Content-Type': 'application/json; charset=utf-8'}
  516. @api.route("/add_action_card", methods=["POST"])
  517. @limiter.exempt()
  518. @login_required
  519. def add_action_card():
  520. """
  521. Add an action card to the game database
  522. POST like this
  523. {
  524. cardid: <int> # id of action card
  525. action: <str> # description of the action
  526. img: file # img file
  527. }
  528. """
  529. try:
  530. cardid = request.form.get('cardid')
  531. action = request.form.get('action')
  532. img = request.files['img']
  533. except KeyError:
  534. return "Missing data or wrong key name", 400
  535. if (cardid is None) or (action is None) or (img.content_type is None):
  536. return "Missing data or wrong key name", 400
  537. cardid = int(cardid)
  538. action_card = db.session.query(ActionCard).filter(ActionCard.card_id == cardid).first()
  539. if action_card is not None:
  540. return "An action card with this id already exists", 400
  541. filename = f"actioncard_{cardid}.png"
  542. img_path = os.path.join(f"{os.getenv('APP_FOLDER')}/project/images/", filename)
  543. img.save(img_path)
  544. ac = ActionCard(
  545. card_id=cardid,
  546. action=action,
  547. img_path=img_path
  548. )
  549. db.session.add(ac)
  550. db.session.commit()
  551. return "Action card added successfully", 200
  552. @api.route("/delete_action_card", methods=["POST"])
  553. @limiter.exempt()
  554. @login_required
  555. def delete_action_card():
  556. """
  557. Add an action card to the game database
  558. POST like this
  559. {
  560. cardid: <int> # id of action card
  561. action: <str> # description of the action
  562. img: file # img file
  563. }
  564. """
  565. try:
  566. cardid = request.form.get('cardid')
  567. except KeyError:
  568. return "Missing data or wrong key name", 400
  569. if (cardid is None):
  570. return "Missing data or wrong key name", 400
  571. cardid = int(cardid)
  572. action_card = db.session.query(ActionCard).filter(ActionCard.card_id == cardid).first()
  573. if action_card is None:
  574. return "An action card with this id does not exists", 400
  575. filename = f"actioncard_{cardid}.png"
  576. img_path = os.path.join(f"{os.getenv('APP_FOLDER')}/project/images/", filename)
  577. os.remove(img_path)
  578. db.session.delete(action_card)
  579. db.session.commit()
  580. return "Action card deleted successfully", 200
  581. @api.route("/add_goal_card", methods=["POST"])
  582. @limiter.exempt()
  583. @login_required
  584. def add_goal_card():
  585. """
  586. Add a goal card to the game database
  587. POST like this
  588. {
  589. cardid: <int> # id of goal card
  590. goal: <str> # description of the action
  591. img: file # img file
  592. }
  593. """
  594. try:
  595. cardid = request.form.get('cardid')
  596. goal = request.form.get('goal')
  597. img = request.files['img']
  598. except KeyError:
  599. return "Missing data or wrong key name", 400
  600. if (cardid is None) or (goal is None) or (img.content_type is None):
  601. return "Missing data or wrong key name", 400
  602. cardid = int(cardid)
  603. goal_card = db.session.query(GoalCard).filter(GoalCard.card_id == cardid).first()
  604. if goal_card is not None:
  605. return "A goal card with this id already exists", 400
  606. filename = f"goalcard_{cardid}.png"
  607. img_path = os.path.join(f"{os.getenv('APP_FOLDER')}/project/images/", filename)
  608. img.save(img_path)
  609. gc = GoalCard(
  610. card_id=cardid,
  611. goal=goal,
  612. img_path=img_path
  613. )
  614. db.session.add(gc)
  615. db.session.commit()
  616. return "Goal card added successfully", 200
  617. @api.route("/delete_goal_card", methods=["POST"])
  618. @limiter.exempt()
  619. @login_required
  620. def delete_goal_card():
  621. """
  622. Delete a Goal Card from the database
  623. POST like this
  624. {
  625. cardid: <int> # id of goal card
  626. }
  627. """
  628. try:
  629. cardid = request.form.get('cardid')
  630. except KeyError:
  631. return "Missing data or wrong key name", 400
  632. if (cardid is None):
  633. return "Missing data or wrong key name", 400
  634. cardid = int(cardid)
  635. goal_card = db.session.query(GoalCard).filter(GoalCard.card_id == cardid).first()
  636. if goal_card is None:
  637. return "A goal card with this id does not exists", 400
  638. filename = f"goalcard_{cardid}.png"
  639. img_path = os.path.join(f"{os.getenv('APP_FOLDER')}/project/images/", filename)
  640. os.remove(img_path)
  641. db.session.delete(goal_card)
  642. db.session.commit()
  643. return "Goal card deleted successfully", 200
  644. @api.route("/add_char_card", methods=["POST"])
  645. @limiter.exempt()
  646. @login_required
  647. def add_char_card():
  648. """
  649. Add a character card to the game database
  650. POST like this
  651. {
  652. cardid: <int> # id of character card
  653. img: file # img file
  654. }
  655. """
  656. try:
  657. cardid = request.form.get('cardid')
  658. name = request.form.get('name')
  659. age = request.form.get('age')
  660. role = request.form.get('role')
  661. interest = request.form.get('interest')
  662. quote = request.form.get('quote')
  663. img = request.files['img']
  664. except KeyError:
  665. return "Missing data or wrong key name", 400
  666. if (cardid is None) or (name is None) or (age is None) or (role is None) or (interest is None) or (
  667. quote is None) or (img.content_type is None):
  668. return "Missing data or wrong key name", 400
  669. cardid = int(cardid)
  670. char_card = db.session.query(CharacterCard).filter(CharacterCard.card_id == cardid).first()
  671. if char_card is not None:
  672. return "A character card with this id already exists", 400
  673. filename = f"charactercard_{cardid}.png"
  674. img_path = os.path.join(f"{os.getenv('APP_FOLDER')}/project/images/", filename)
  675. img.save(img_path)
  676. cc = CharacterCard(
  677. card_id=cardid,
  678. name=name,
  679. age=age,
  680. role=role,
  681. interest=interest,
  682. quote=quote,
  683. img_path=img_path
  684. )
  685. db.session.add(cc)
  686. db.session.commit()
  687. return "Character card added successfully", 200
  688. @api.route("/delete_char_card", methods=["POST"])
  689. @limiter.exempt()
  690. @login_required
  691. def delete_char_card():
  692. """
  693. delete a char card from the game database
  694. POST like this
  695. {
  696. cardid: <int> # id of action card
  697. }
  698. """
  699. try:
  700. cardid = request.form.get('cardid')
  701. except KeyError:
  702. return "Missing data or wrong key name", 400
  703. if (cardid is None):
  704. return "Missing data or wrong key name", 400
  705. cardid = int(cardid)
  706. char_card = db.session.query(CharacterCard).filter(CharacterCard.card_id == cardid).first()
  707. if char_card is None:
  708. return "A character card with this id does not exists", 400
  709. filename = f"charactercard_{cardid}.png"
  710. img_path = os.path.join(f"{os.getenv('APP_FOLDER')}/project/images/", filename)
  711. os.remove(img_path)
  712. db.session.delete(char_card)
  713. db.session.commit()
  714. return "Action card deleted successfully", 200
  715. @api.route('/get_card_image/<subpath>', methods=['GET'])
  716. @limiter.exempt()
  717. @login_required
  718. def get_card_image(subpath):
  719. # Here you would dynamically serve the file based on subpath
  720. # For demonstration purposes, let's assume all images are in /static/images/
  721. img_path = os.path.join(f"{os.getenv('APP_FOLDER')}/project/images", subpath)
  722. return send_file(img_path)
  723. @api.route("/add_image_to_game", methods=['POST'])
  724. @limiter.exempt
  725. def add_image_to_game():
  726. """
  727. send data as form-data, otherwise wont work
  728. 2 keys:
  729. 'game' which should contain game id
  730. 'img' the image file
  731. """
  732. try:
  733. gameid = request.form.get('game')
  734. img = request.files['img']
  735. except KeyError:
  736. return "Missing data or wrong key name", 400
  737. if gameid is None or img.content_type is None:
  738. return "Missing data or wrong key name", 400
  739. gameid = int(gameid)
  740. # does game exist?
  741. game = db.session.query(Game).filter(Game.id == gameid).first()
  742. if game is None:
  743. return f"Game {gameid} does not exist", 400
  744. if game.closed:
  745. return f"Game {gameid} is closed", 400
  746. # how many images are there already?
  747. image_count = db.session.query(Image).filter(Image.game_id == gameid).count()
  748. # if there are already 5 discard request
  749. if image_count >= 5:
  750. return f"Game {gameid} already has 5 images saved.", 400
  751. # else save image to path and create db entry
  752. # img = PImage.open(img)
  753. # this_folder = Path(__file__).parent.resolve()
  754. # img_path = str(this_folder / f"images/{gameid}_{image_count + 1}.png")
  755. ## img_path = f'/server/images/{gameid}_{image_count + 1}.png'
  756. # img.save(img_path)
  757. filename = f"{gameid}_{image_count + 1}.png"
  758. img_path = os.path.join(f"{os.getenv('APP_FOLDER')}/project/images", filename)
  759. img.save(img_path)
  760. image = Image(
  761. game_id=gameid,
  762. img_path=img_path
  763. )
  764. db.session.add(image)
  765. db.session.commit()
  766. return "Image saved successfully", 200
  767. @api.route("/get_next_game_id", methods=['GET'])
  768. @limiter.exempt
  769. def get_next_game_id():
  770. game = db.session.query(Game.id).order_by(Game.id.desc()).first()
  771. if game is None:
  772. return "1", 200
  773. next_id = game[0] + 1
  774. return f"{next_id}", 200
  775. @api.route("/close_game", methods=['POST'])
  776. @limiter.exempt
  777. def close_game():
  778. """ POST like this
  779. {
  780. game: <int> # game id to close
  781. }
  782. """
  783. try:
  784. # content = request.get_json()
  785. # gameid = content["game"]
  786. gameid = request.form.get("game")
  787. except KeyError:
  788. return "Missing data or wrong key name", 400
  789. if gameid is None:
  790. return "Missing Data", 400
  791. gameid = int(gameid)
  792. game = db.session.query(Game).filter(Game.id == gameid).first()
  793. if game is None:
  794. return "Game does not exist", 400
  795. if game.closed:
  796. return "Game already closed", 400
  797. game.closed = True
  798. db.session.commit()
  799. return f"Game {gameid} closed successfully", 200
  800. @api.route("/is_game_closed/<int:gameid>", methods=['GET'])
  801. @limiter.exempt
  802. def is_game_closed(gameid=None):
  803. if gameid is None:
  804. return "game id is missing", 400
  805. game = db.session.query(Game).filter(Game.id == gameid).first()
  806. if game is None:
  807. return "Game does not exist", 400
  808. return f"{game.closed}", 200
  809. @api.route("/get_object_owner/<int:gameid>/<int:oid>", methods=['GET'])
  810. @limiter.exempt
  811. def get_object_owner(gameid=None, oid=None):
  812. """
  813. :param gameid: id of game
  814. :param oid: object id
  815. :return: number of player who owns object
  816. """
  817. if gameid is None or oid is None:
  818. return "game id or object id is missing", 400
  819. owner = db.session.query(Object).filter(Object.game_id == gameid,
  820. Object.object_type == oid).first()
  821. if owner is None:
  822. return f"No player has played object {oid}", 400
  823. return f"{owner.player_number}", 200
  824. @api.route('/get_images/<int:id>', methods=["GET"])
  825. @limiter.exempt
  826. @login_required
  827. def get_images(id=None):
  828. if id is None:
  829. return "Attribute is None", 400
  830. imgs = get_game_images(id)
  831. images = {}
  832. for img in imgs:
  833. if f"{img.game_id}" not in images:
  834. images[f"{img.game_id}"] = []
  835. image = PImage.open(img.img_path)
  836. image_io = BytesIO()
  837. image.save(image_io, 'PNG')
  838. images[f"{img.game_id}"].append('data:image/png;base64,' + b64encode(image_io.getvalue()).decode('ascii'))
  839. return {'data': images}, 200, {'Content-Type': 'application/json; charset=utf-8'}
  840. @api.route('/reset_database', methods=["POST"])
  841. @limiter.exempt
  842. @login_required
  843. def reset_database():
  844. """
  845. Deletes every entry in database apart from User. Game ID sequence will not be resetted
  846. Also dleets all images from the file system
  847. :return:
  848. """
  849. try:
  850. content = request.get_json()
  851. password = content["password"]
  852. except KeyError:
  853. return "Missing data or wrong key name", 400
  854. user = db.session.query(User).first()
  855. if not user.verify_password(password):
  856. return "Wrong Password", 401
  857. db.session.query(Game).delete()
  858. db.session.query(SingleplayerGame).delete()
  859. db.session.query(Player).delete()
  860. db.session.query(Object).delete()
  861. db.session.query(CharacterCard).delete()
  862. db.session.query(GoalCard).delete()
  863. db.session.query(ActionCard).delete()
  864. files = glob.glob(f"{current_app.config['MEDIA_FOLDER']}/*")
  865. for f in files:
  866. print(f)
  867. os.remove(f)
  868. db.session.query(Image).delete()
  869. db.session.commit()
  870. # This resets the Primary Key sequence for Game table such that the game ids start with 1 again.
  871. # Alternatively one can also continue the ids however this is a problem when retrieving the last_value of the sequence
  872. # This is because for the 1st and 2nd interation of the sequence last_value returns 2, since nextval() does not increment on first call
  873. db.engine.execute("ALTER SEQUENCE game_id_seq RESTART WITH 1")
  874. return "DB is empty again", 200
  875. """
  876. ####################
  877. SINGLEPLAYER
  878. ####################
  879. """
  880. @singleplayer.route("/new_game", methods=["POST"])
  881. @limiter.exempt
  882. def new_game():
  883. game = SingleplayerGame()
  884. db.session.add(game)
  885. db.session.commit()
  886. return f"{game.id}", 200
  887. @singleplayer.route("/add_point_information", methods=["POST"])
  888. @limiter.exempt
  889. def add_point_information():
  890. """ POST like this
  891. {
  892. game: <int> # game id
  893. player: <int> # player id in game 1-5
  894. points: <int> # how many points?
  895. }
  896. """
  897. # extract values and check validity
  898. try:
  899. gameid = request.form.get("game")
  900. playerid = request.form.get("player")
  901. points = request.form.get("points")
  902. except KeyError:
  903. return "Missing Data", 400
  904. if (gameid is None) or (playerid is None) or (points is None):
  905. return "Missing Data", 400
  906. gameid = int(gameid)
  907. playerid = int(playerid)
  908. points = int(points)
  909. if playerid < 1 or playerid > 5:
  910. return f"Player {playerid} does not exist.", 400
  911. game = db.session.query(SingleplayerGame).filter(SingleplayerGame.id == gameid).first()
  912. if game is None:
  913. return f"Game {gameid} does not exist", 400
  914. setattr(game, f"player_{playerid}_points", getattr(game, f"player_{playerid}_points") + points)
  915. db.session.commit()
  916. return f"Successfully added {points} points to player {playerid}.", 200
  917. @singleplayer.route("/get_point_information/<int:gameid>", methods=['GET'])
  918. @limiter.exempt
  919. def get_point_information(gameid=None):
  920. game = db.session.query(SingleplayerGame).filter(SingleplayerGame.id == gameid).first()
  921. if game is None:
  922. return f"Game {gameid} does not exist", 400
  923. response_json = {
  924. 'player1': str(game.player_1_points),
  925. 'player2': str(game.player_2_points),
  926. 'player3': str(game.player_3_points),
  927. 'player4': str(game.player_4_points),
  928. 'player5': str(game.player_5_points)
  929. }
  930. return Response(json.dumps(response_json), status=200, mimetype='application/json')
  931. """
  932. ####################
  933. AUTHENTICATION
  934. ####################
  935. """
  936. auth = Blueprint('auth', __name__)
  937. limiter.limit('20/second')(auth)
  938. @login_manager.user_loader
  939. def get_user(user_id):
  940. return db.session.get(User, user_id)
  941. @auth.route("/login", methods=["GET", "POST"])
  942. @limiter.exempt
  943. def login():
  944. form = LoginForm()
  945. username = form.username.data
  946. password = form.password.data
  947. if form.validate_on_submit():
  948. user = User.query.filter_by(username=username).first()
  949. if user and user.verify_password(password):
  950. login_user(user)
  951. flash("logged in!")
  952. return redirect(url_for('index'))
  953. else:
  954. flash("login failed!")
  955. else:
  956. print(form.errors)
  957. return render_template('login.html', form=form)
  958. @auth.route('/logout')
  959. @limiter.exempt
  960. def logout():
  961. logout_user()
  962. return redirect(url_for('auth.login'))