rest_api.py 4.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168
  1. import logging
  2. from base64 import b64encode
  3. from copy import deepcopy
  4. from email.message import EmailMessage
  5. from json import dumps, loads
  6. from os import environ, urandom
  7. from smtplib import SMTP
  8. from flask import Flask, jsonify, render_template, request
  9. from werkzeug.exceptions import BadRequest
  10. from scriptshifter import EMAIL_FROM, EMAIL_TO, SMTP_HOST, SMTP_PORT
  11. from scriptshifter.exceptions import ApiError
  12. from scriptshifter.tables import list_tables, load_table
  13. from scriptshifter.trans import transliterate
  14. logger = logging.getLogger(__name__)
  15. logging.basicConfig(level=environ.get("TXL_LOGLEVEL", logging.INFO))
  16. def create_app():
  17. flask_env = environ.get("TXL_APP_MODE", "production")
  18. app = Flask(__name__)
  19. app.config.update({
  20. "ENV": flask_env,
  21. "SECRET_KEY": environ.get("TXL_FLASK_SECRET", b64encode(urandom(64))),
  22. "JSON_AS_ASCII": False,
  23. "JSONIFY_PRETTYPRINT_REGULAR": True,
  24. })
  25. return app
  26. app = create_app()
  27. @app.errorhandler(ApiError)
  28. def handle_exception(e: ApiError):
  29. return ({
  30. "warnings": [
  31. "ScriptShifter HTTP request failed with status code "
  32. f"{e.status_code}: {e.msg}"
  33. ],
  34. "output": "",
  35. }, e.status_code)
  36. @app.errorhandler(BadRequest)
  37. def handle_400(e):
  38. if logging.DEBUG >= logging.root.level:
  39. body = {
  40. "debug": {
  41. "form_data": request.form,
  42. }
  43. }
  44. else:
  45. body = ""
  46. return body, 400
  47. @app.route("/", methods=["GET"])
  48. def index():
  49. return render_template(
  50. "index.html",
  51. languages=list_tables(),
  52. feedback_form=SMTP_HOST is not None)
  53. @app.route("/health", methods=["GET"])
  54. def health_check():
  55. return "I'm alive!\n"
  56. @app.route("/languages", methods=["GET"])
  57. def list_languages():
  58. return jsonify(list_tables())
  59. @app.route("/table/<lang>")
  60. def dump_table(lang):
  61. """
  62. Dump parsed transliteration table for a language.
  63. """
  64. tbl = deepcopy(load_table(lang))
  65. for sec_name in ("roman_to_script", "script_to_roman"):
  66. if sec_name in tbl:
  67. for hname, fn_defs in tbl[sec_name].get("hooks", {}).items():
  68. tbl[sec_name]["hooks"][hname] = [
  69. (fn.__name__, kw) for (fn, kw) in fn_defs]
  70. return jsonify(tbl)
  71. @app.route("/options/<lang>", methods=["GET"])
  72. def get_options(lang):
  73. """
  74. Get extra options for a table.
  75. """
  76. tbl = load_table(lang)
  77. return jsonify(tbl.get("options", []))
  78. @app.route("/trans", methods=["POST"])
  79. def transliterate_req():
  80. lang = request.form["lang"]
  81. in_txt = request.form["text"]
  82. capitalize = request.form.get("capitalize", False)
  83. t_dir = request.form.get("t_dir", "s2r")
  84. if t_dir not in ("s2r", "r2s"):
  85. return f"Invalid direction: {t_dir}", 400
  86. if not len(in_txt):
  87. return ("No input text provided! ", 400)
  88. options = loads(request.form.get("options", "{}"))
  89. logger.debug(f"Extra options: {options}")
  90. try:
  91. out, warnings = transliterate(in_txt, lang, t_dir, capitalize, options)
  92. except (NotImplementedError, ValueError) as e:
  93. return (str(e), 400)
  94. return {"output": out, "warnings": warnings}
  95. @app.route("/feedback", methods=["POST"])
  96. def feedback():
  97. """
  98. Allows users to provide feedback to improve a specific result.
  99. """
  100. lang = request.form["lang"]
  101. src = request.form["src"]
  102. t_dir = request.form.get("t_dir", "s2r")
  103. result = request.form["result"]
  104. expected = request.form["expected"]
  105. options = request.form.get("options", {})
  106. notes = request.form.get("notes")
  107. contact = request.form.get("contact")
  108. msg = EmailMessage()
  109. msg["subject"] = "Scriptshifter feedback report"
  110. msg["from"] = EMAIL_FROM
  111. msg["to"] = EMAIL_TO
  112. if contact:
  113. msg["cc"] = contact
  114. msg.set_content(f"""
  115. *Scriptshifter feedback report from {contact or 'anonymous'}*\n\n
  116. *Language:* {lang}\n
  117. *Direction:* {
  118. 'Roman to Script' if t_dir == 'r2s'
  119. else 'Script to Roman'}\n
  120. *Source:* {src}\n
  121. *Result:* {result}\n
  122. *Expected result:* {expected}\n
  123. *Applied options:* {dumps(options)}\n
  124. *Notes:*\n
  125. {notes}""")
  126. # TODO This uses a test SMTP server:
  127. # python -m smtpd -n -c DebuggingServer localhost:1025
  128. smtp = SMTP(SMTP_HOST, SMTP_PORT)
  129. smtp.send_message(msg)
  130. smtp.quit()
  131. return {"message": "Feedback message sent."}