|
1 | | -import inspect |
2 | | -import json |
3 | 1 | import time |
4 | | -from urllib import request as urllib_request |
5 | | -from urllib.error import URLError |
6 | 2 |
|
7 | 3 | from flask import Flask, Response, request |
8 | 4 |
|
9 | | -from mesop.env.env import AI_SERVICE_BASE_URL |
10 | 5 | from mesop.runtime import runtime |
11 | | -from mesop.server.server_utils import ( |
12 | | - check_editor_access, |
13 | | - make_sse_response, |
14 | | - sse_request, |
15 | | -) |
16 | 6 |
|
17 | 7 |
|
18 | 8 | def configure_debug_routes(flask_app: Flask): |
19 | | - @flask_app.route("/__editor__/commit", methods=["POST"]) |
20 | | - def page_commit() -> Response: |
21 | | - check_editor_access() |
22 | | - |
23 | | - try: |
24 | | - data = request.get_json() |
25 | | - except json.JSONDecodeError: |
26 | | - return Response("Invalid JSON format", status=400) |
27 | | - code = data.get("code") |
28 | | - path = data.get("path") |
29 | | - page_config = runtime().get_page_config(path=path) |
30 | | - assert page_config |
31 | | - module = inspect.getmodule(page_config.page_fn) |
32 | | - assert module |
33 | | - module_file = module.__file__ |
34 | | - assert module_file |
35 | | - module_file_path = module.__file__ |
36 | | - assert module_file_path |
37 | | - with open(module_file_path, "w") as file: |
38 | | - file.write(code) |
39 | | - |
40 | | - response_data = {"message": "Page commit successful"} |
41 | | - return Response( |
42 | | - json.dumps(response_data), status=200, mimetype="application/json" |
43 | | - ) |
44 | | - |
45 | | - @flask_app.route("/__editor__/save-interaction", methods=["POST"]) |
46 | | - def save_interaction() -> Response | dict[str, str]: |
47 | | - check_editor_access() |
48 | | - |
49 | | - data = request.get_json() |
50 | | - if not data: |
51 | | - return Response("Invalid JSON data", status=400) |
52 | | - |
53 | | - try: |
54 | | - req = urllib_request.Request( |
55 | | - AI_SERVICE_BASE_URL + "/save-interaction", |
56 | | - data=json.dumps(data).encode("utf-8"), |
57 | | - headers={"Content-Type": "application/json"}, |
58 | | - ) |
59 | | - with urllib_request.urlopen(req) as response: |
60 | | - if response.status == 200: |
61 | | - folder = json.loads(response.read().decode("utf-8"))["folder"] |
62 | | - return {"folder": folder} |
63 | | - else: |
64 | | - print(f"Error from AI service: {response.read().decode('utf-8')}") |
65 | | - return Response( |
66 | | - f"Error from AI service: {response.read().decode('utf-8')}", |
67 | | - status=500, |
68 | | - ) |
69 | | - except URLError as e: |
70 | | - return Response(f"Error making request to AI service: {e!s}", status=500) |
71 | | - |
72 | | - @flask_app.route("/__editor__/generate", methods=["POST"]) |
73 | | - def page_generate(): |
74 | | - check_editor_access() |
75 | | - |
76 | | - try: |
77 | | - data = request.get_json() |
78 | | - except json.JSONDecodeError: |
79 | | - return Response("Invalid JSON format", status=400) |
80 | | - if not data: |
81 | | - return Response("Invalid JSON data", status=400) |
82 | | - |
83 | | - prompt = data.get("prompt") |
84 | | - if not prompt: |
85 | | - return Response("Missing 'prompt' in JSON data", status=400) |
86 | | - |
87 | | - path = data.get("path") |
88 | | - page_config = runtime().get_page_config(path=path) |
89 | | - |
90 | | - line_number = data.get("lineNumber") |
91 | | - assert page_config |
92 | | - module = inspect.getmodule(page_config.page_fn) |
93 | | - if module is None: |
94 | | - return Response("Could not retrieve module source code.", status=500) |
95 | | - module_file = module.__file__ |
96 | | - assert module_file |
97 | | - with open(module_file) as file: |
98 | | - source_code = file.read() |
99 | | - print(f"Source code of module {module.__name__}:") |
100 | | - |
101 | | - def generate(): |
102 | | - try: |
103 | | - for event in sse_request( |
104 | | - AI_SERVICE_BASE_URL + "/adjust-mesop-app", |
105 | | - {"prompt": prompt, "code": source_code, "lineNumber": line_number}, |
106 | | - ): |
107 | | - if event.get("type") == "end": |
108 | | - sse_data = { |
109 | | - "type": "end", |
110 | | - "prompt": prompt, |
111 | | - "path": path, |
112 | | - "beforeCode": source_code, |
113 | | - "afterCode": event["code"], |
114 | | - "diff": event["diff"], |
115 | | - "message": "Prompt processed successfully", |
116 | | - } |
117 | | - yield f"data: {json.dumps(sse_data)}\n\n" |
118 | | - break |
119 | | - elif event.get("type") == "progress": |
120 | | - sse_data = {"data": event["data"], "type": "progress"} |
121 | | - yield f"data: {json.dumps(sse_data)}\n\n" |
122 | | - elif event.get("type") == "error": |
123 | | - sse_data = {"error": event["error"], "type": "error"} |
124 | | - yield f"data: {json.dumps(sse_data)}\n\n" |
125 | | - break |
126 | | - else: |
127 | | - raise Exception(f"Unknown event type: {event}") |
128 | | - except Exception as e: |
129 | | - sse_data = { |
130 | | - "error": "Could not connect to AI service: " + str(e), |
131 | | - "type": "error", |
132 | | - } |
133 | | - yield f"data: {json.dumps(sse_data)}\n\n" |
134 | | - |
135 | | - return make_sse_response(generate()) |
136 | | - |
137 | 9 | @flask_app.route("/__hot-reload__") |
138 | 10 | def hot_reload() -> Response: |
139 | 11 | counter = int(request.args["counter"]) |
|
0 commit comments