File size: 7,641 Bytes
fc0b4cb
 
 
 
9e8ab62
 
fc0b4cb
57a2c17
 
 
7928d62
9e8ab62
 
 
 
 
 
 
 
 
 
7928d62
9e8ab62
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
7928d62
57a2c17
43d514c
 
 
 
fc0b4cb
57a2c17
 
 
 
43d514c
57a2c17
 
 
fc0b4cb
 
 
 
 
 
57a2c17
 
 
 
fc0b4cb
 
 
 
57a2c17
 
 
 
 
 
 
 
fc0b4cb
 
57a2c17
 
 
 
 
 
fc0b4cb
 
 
 
 
 
 
 
 
7928d62
9e8ab62
 
 
 
 
 
57a2c17
9e8ab62
 
 
 
 
 
fc0b4cb
9e8ab62
 
 
 
 
 
 
 
 
fc0b4cb
9e8ab62
 
fc0b4cb
9e8ab62
57a2c17
 
 
9e8ab62
 
 
 
 
57a2c17
 
 
 
 
 
fc0b4cb
57a2c17
 
 
 
fc0b4cb
57a2c17
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
9e8ab62
 
57a2c17
9e8ab62
 
 
 
57a2c17
9e8ab62
fc0b4cb
9e8ab62
 
 
57a2c17
9e8ab62
7928d62
fc0b4cb
 
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
import os
import importlib.util
import sys
import tempfile
import re
import ast
import gradio as gr
import openai
import requests
import json

# Extract code blocks from text
def extract_code_blocks(text):
    pattern = r'```(?:python)?\s*([\s\S]*?)```'
    matches = re.findall(pattern, text)
    
    if not matches and text.strip():
        if re.search(r'import\s+\w+|def\s+\w+\(|class\s+\w+:|if\s+__name__\s*==\s*[\'"]__main__[\'"]:', text):
            return [text.strip()]
    
    return [match.strip() for match in matches]

# Validate that the code is safe to run
def validate_gradio_code(code):
    try:
        tree = ast.parse(code)
        
        # Check imports
        allowed_modules = ['gradio', 'numpy', 'pandas', 'matplotlib', 'PIL', 'os', 'io', 'base64', 
                          'time', 'datetime', 'json', 'random', 'math', 'sys', 're', 'pathlib',
                          'collections', 'typing', 'warnings']
        
        for node in ast.walk(tree):
            if isinstance(node, ast.Import):
                for name in node.names:
                    if name.name not in allowed_modules:
                        return False, f"Unauthorized import: {name.name}"
            
            elif isinstance(node, ast.ImportFrom):
                if node.module not in allowed_modules and node.module is not None:
                    return False, f"Unauthorized import from: {node.module}"
        
        # Check for harmful operations
        code_str = code.lower()
        harmful_ops = ['subprocess', 'system(', 'popen(', 'execve(', 'chmod(', 'rmdir(', 
                      'remove(', 'unlink(', 'rmtree(', 'socket', 'eval(', 'exec(']
        
        for op in harmful_ops:
            if op in code_str:
                return False, f"Potentially harmful operation detected: {op}"
        
        return True, None
        
    except SyntaxError as e:
        return False, f"Syntax error in the code: {str(e)}"
    except Exception as e:
        return False, f"Error validating code: {str(e)}"

# Generate app using direct API call
def generate_gradio_app(api_key, prompt):
    if not api_key or len(api_key) < 20:
        return None, "Please provide a valid OpenAI API key"
    
    try:
        headers = {
            "Content-Type": "application/json",
            "Authorization": f"Bearer {api_key}"
        }
        
        data = {
            "model": "gpt-4o",
            "messages": [
                {"role": "system", "content": """You are an expert Gradio developer. 
                Create a standalone Gradio application based on the user's prompt.
                Your response should ONLY include Python code without any explanation.
                The code must:
                1. Import all necessary libraries
                2. Define a complete, functional Gradio interface
                3. DO NOT include a launch command or if __name__ == "__main__" block
                4. The interface should be assigned to a variable named 'demo'
                5. Handle errors gracefully
                6. Be completely self-contained in a single script
                """
                },
                {"role": "user", "content": prompt}
            ],
            "temperature": 0.2,
            "max_tokens": 4000
        }
        
        response = requests.post(
            "https://api.openai.com/v1/chat/completions",
            headers=headers,
            json=data
        )
        
        if response.status_code != 200:
            return None, f"API Error: {response.status_code} - {response.text}"
        
        result = response.json()
        generated_code = result["choices"][0]["message"]["content"]
        
        code_blocks = extract_code_blocks(generated_code)
        
        if code_blocks:
            return code_blocks[0], None
        else:
            return generated_code, None
            
    except Exception as e:
        return None, str(e)

# Create the main UI
with gr.Blocks(title="AI Gradio App Generator") as demo:
    gr.Markdown("# 🤖 AI Gradio App Generator")
    gr.Markdown("Describe the app you want, and I'll generate it for you using OpenAI's API.")
    
    with gr.Row():
        with gr.Column():
            api_key = gr.Textbox(
                label="OpenAI API Key", 
                placeholder="sk-...",
                type="password",
                info="Your key is used for this session only and not stored"
            )
            
            prompt = gr.Textbox(
                label="App Description", 
                placeholder="Describe the Gradio app you want to create...",
                lines=5
            )
            
            with gr.Row():
                submit_btn = gr.Button("Generate App", variant="primary")
                clear_btn = gr.Button("Clear", variant="secondary")
            
            with gr.Accordion("Generated Code", open=False):
                code_output = gr.Code(language="python", label="Generated Code")
            
            status_output = gr.Markdown("")
    
    app_container = gr.HTML(visible=False)
    generated_app_container = gr.Group(visible=False)
    
    def on_submit(api_key_input, prompt_text):
        # Generate code
        code, error = generate_gradio_app(api_key_input, prompt_text)
        if error:
            return None, f"⚠️ Error generating code: {error}", gr.update(visible=False), gr.update(visible=False)
        
        # Validate code
        is_valid, error_msg = validate_gradio_code(code)
        if not is_valid:
            return code, f"⚠️ Error validating code: {error_msg}", gr.update(visible=False), gr.update(visible=False)
        
        # Save code to temp file
        with tempfile.NamedTemporaryFile(suffix='.py', delete=False) as f:
            f.write(code.encode('utf-8'))
            temp_file = f.name
        
        try:
            # Import module
            module_name = os.path.basename(temp_file).replace('.py', '')
            spec = importlib.util.spec_from_file_location(module_name, temp_file)
            module = importlib.util.module_from_spec(spec)
            sys.modules[module_name] = module
            spec.loader.exec_module(module)
            
            # Get the Gradio interface
            if hasattr(module, 'demo'):
                generated_app = module.demo
                
                # Clear the container and add the new app
                return (
                    code, 
                    "✅ App generated successfully!", 
                    gr.update(visible=False),
                    gr.update(visible=True, value=generated_app)
                )
            else:
                return code, "⚠️ No 'demo' variable found in the generated code", gr.update(visible=False), gr.update(visible=False)
                
        except Exception as e:
            return code, f"⚠️ Error executing the generated code: {str(e)}", gr.update(visible=False), gr.update(visible=False)
        finally:
            try:
                os.unlink(temp_file)
            except:
                pass
    
    def on_clear():
        return "", "", gr.update(visible=False), gr.update(visible=False)
    
    submit_btn.click(
        on_submit,
        inputs=[api_key, prompt],
        outputs=[code_output, status_output, app_container, generated_app_container]
    )
    
    clear_btn.click(
        on_clear,
        inputs=[],
        outputs=[prompt, status_output, app_container, generated_app_container]
    )

if __name__ == "__main__":
    demo.launch(server_name="0.0.0.0", server_port=7860)