rest_api.py 2.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105
  1. import logging
  2. from base64 import b64encode
  3. from copy import deepcopy
  4. from json import dumps, loads
  5. from os import environ, urandom
  6. from flask import Flask, jsonify, render_template, request
  7. from scriptshifter.exceptions import ApiError
  8. from scriptshifter.tables import list_tables, load_table
  9. from scriptshifter.trans import transliterate
  10. logger = logging.getLogger(__name__)
  11. logging.basicConfig(level=environ.get("TXL_LOGLEVEL", logging.INFO))
  12. def create_app():
  13. flask_env = environ.get("TXL_APP_MODE", "production")
  14. app = Flask(__name__)
  15. app.config.update({
  16. "ENV": flask_env,
  17. "SECRET_KEY": environ.get("TXL_FLASK_SECRET", b64encode(urandom(64))),
  18. "JSON_AS_ASCII": False,
  19. "JSONIFY_PRETTYPRINT_REGULAR": True,
  20. })
  21. return app
  22. app = create_app()
  23. @app.errorhandler(ApiError)
  24. def handle_exception(e: ApiError):
  25. return ({
  26. "warnings": [
  27. "ScriptShifter HTTP request failed with status code "
  28. f"{e.status_code}: {e.msg}"
  29. ],
  30. "output": "",
  31. }, e.status_code)
  32. @app.route("/", methods=["GET"])
  33. def index():
  34. return render_template("index.html", languages=list_tables())
  35. @app.route("/health", methods=["GET"])
  36. def health_check():
  37. return "I'm alive!\n"
  38. @app.route("/languages", methods=["GET"])
  39. def list_languages():
  40. return jsonify(list_tables())
  41. @app.route("/table/<lang>")
  42. def dump_table(lang):
  43. """
  44. Dump parsed transliteration table for a language.
  45. """
  46. tbl = deepcopy(load_table(lang))
  47. for sec_name in ("roman_to_script", "script_to_roman"):
  48. if sec_name in tbl:
  49. for hname, fn_defs in tbl[sec_name].get("hooks", {}).items():
  50. tbl[sec_name]["hooks"][hname] = [
  51. (fn.__name__, kw) for (fn, kw) in fn_defs]
  52. return jsonify(tbl)
  53. @app.route("/options/<lang>", methods=["GET"])
  54. def get_options(lang):
  55. """
  56. Get extra options for a table.
  57. """
  58. tbl = load_table(lang)
  59. return jsonify(tbl.get("options", []))
  60. @app.route("/trans", methods=["POST"])
  61. def transliterate_req():
  62. lang = request.form["lang"]
  63. in_txt = request.form["text"]
  64. capitalize = request.form.get("capitalize", False)
  65. t_dir = request.form.get("t_dir", "s2r")
  66. if t_dir not in ("s2r", "r2s"):
  67. return f"Invalid direction: {t_dir}", 400
  68. if not len(in_txt):
  69. return ("No input text provided! ", 400)
  70. options = loads(request.form.get("options", "{}"))
  71. logger.debug(f"Extra options: {options}")
  72. try:
  73. out, warnings = transliterate(in_txt, lang, t_dir, capitalize, options)
  74. except (NotImplementedError, ValueError) as e:
  75. return (str(e), 400)
  76. return {"output": out, "warnings": warnings}