2024-01-22 11:33:49 +00:00
|
|
|
<script lang="ts">
|
2024-05-17 06:47:13 +00:00
|
|
|
import Spinner from '$lib/components/common/Spinner.svelte';
|
2024-01-22 11:33:49 +00:00
|
|
|
import { copyToClipboard } from '$lib/utils';
|
|
|
|
import hljs from 'highlight.js';
|
|
|
|
import 'highlight.js/styles/github-dark.min.css';
|
2024-05-17 05:21:08 +00:00
|
|
|
import { loadPyodide } from 'pyodide';
|
2024-05-17 03:49:28 +00:00
|
|
|
import { tick } from 'svelte';
|
|
|
|
|
|
|
|
export let id = '';
|
2024-01-22 11:33:49 +00:00
|
|
|
|
|
|
|
export let lang = '';
|
|
|
|
export let code = '';
|
|
|
|
|
2024-05-17 06:47:13 +00:00
|
|
|
let executing = false;
|
2024-05-17 05:21:08 +00:00
|
|
|
|
|
|
|
let stdout = null;
|
|
|
|
let stderr = null;
|
|
|
|
let result = null;
|
|
|
|
|
2024-01-22 11:33:49 +00:00
|
|
|
let copied = false;
|
|
|
|
|
|
|
|
const copyCode = async () => {
|
|
|
|
copied = true;
|
|
|
|
await copyToClipboard(code);
|
|
|
|
|
|
|
|
setTimeout(() => {
|
|
|
|
copied = false;
|
|
|
|
}, 1000);
|
|
|
|
};
|
|
|
|
|
2024-05-17 04:05:43 +00:00
|
|
|
const checkPythonCode = (str) => {
|
|
|
|
// Check if the string contains typical Python keywords, syntax, or functions
|
|
|
|
const pythonKeywords = [
|
|
|
|
'def',
|
|
|
|
'class',
|
|
|
|
'import',
|
|
|
|
'from',
|
|
|
|
'if',
|
|
|
|
'else',
|
|
|
|
'elif',
|
|
|
|
'for',
|
|
|
|
'while',
|
|
|
|
'try',
|
|
|
|
'except',
|
|
|
|
'finally',
|
|
|
|
'return',
|
|
|
|
'yield',
|
|
|
|
'lambda',
|
|
|
|
'assert',
|
|
|
|
'pass',
|
|
|
|
'break',
|
|
|
|
'continue',
|
|
|
|
'global',
|
|
|
|
'nonlocal',
|
|
|
|
'del',
|
|
|
|
'True',
|
|
|
|
'False',
|
|
|
|
'None',
|
|
|
|
'and',
|
|
|
|
'or',
|
|
|
|
'not',
|
|
|
|
'in',
|
|
|
|
'is',
|
|
|
|
'as',
|
|
|
|
'with'
|
|
|
|
];
|
|
|
|
|
|
|
|
for (let keyword of pythonKeywords) {
|
|
|
|
if (str.includes(keyword)) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check if the string contains typical Python syntax characters
|
|
|
|
const pythonSyntax = [
|
|
|
|
'def ',
|
|
|
|
'class ',
|
|
|
|
'import ',
|
|
|
|
'from ',
|
|
|
|
'if ',
|
|
|
|
'else:',
|
|
|
|
'elif ',
|
|
|
|
'for ',
|
|
|
|
'while ',
|
|
|
|
'try:',
|
|
|
|
'except:',
|
|
|
|
'finally:',
|
|
|
|
'return ',
|
|
|
|
'yield ',
|
|
|
|
'lambda ',
|
|
|
|
'assert ',
|
|
|
|
'pass',
|
|
|
|
'break',
|
|
|
|
'continue',
|
|
|
|
'global ',
|
|
|
|
'nonlocal ',
|
|
|
|
'del ',
|
|
|
|
'True',
|
|
|
|
'False',
|
|
|
|
'None',
|
|
|
|
' and ',
|
|
|
|
' or ',
|
|
|
|
' not ',
|
|
|
|
' in ',
|
|
|
|
' is ',
|
|
|
|
' as ',
|
|
|
|
' with ',
|
|
|
|
':',
|
|
|
|
'=',
|
|
|
|
'==',
|
|
|
|
'!=',
|
|
|
|
'>',
|
|
|
|
'<',
|
|
|
|
'>=',
|
|
|
|
'<=',
|
|
|
|
'+',
|
|
|
|
'-',
|
|
|
|
'*',
|
|
|
|
'/',
|
|
|
|
'%',
|
|
|
|
'**',
|
|
|
|
'//',
|
|
|
|
'(',
|
|
|
|
')',
|
|
|
|
'[',
|
|
|
|
']',
|
|
|
|
'{',
|
|
|
|
'}'
|
|
|
|
];
|
|
|
|
|
|
|
|
for (let syntax of pythonSyntax) {
|
|
|
|
if (str.includes(syntax)) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// If none of the above conditions met, it's probably not Python code
|
|
|
|
return false;
|
|
|
|
};
|
|
|
|
|
2024-05-17 05:21:08 +00:00
|
|
|
const executePython = async (code) => {
|
2024-05-17 08:54:37 +00:00
|
|
|
if (!code.includes('input') && !code.includes('matplotlib')) {
|
2024-05-17 08:39:07 +00:00
|
|
|
executePythonAsWorker(code);
|
|
|
|
} else {
|
|
|
|
result = null;
|
|
|
|
stdout = null;
|
|
|
|
stderr = null;
|
2024-05-17 05:22:10 +00:00
|
|
|
|
2024-05-17 08:39:07 +00:00
|
|
|
executing = true;
|
2024-05-17 03:49:28 +00:00
|
|
|
|
2024-05-17 09:15:39 +00:00
|
|
|
document.pyodideMplTarget = document.getElementById(`plt-canvas-${id}`);
|
2024-05-17 08:54:37 +00:00
|
|
|
|
2024-05-17 08:39:07 +00:00
|
|
|
let pyodide = await loadPyodide({
|
|
|
|
indexURL: '/pyodide/',
|
|
|
|
stdout: (text) => {
|
|
|
|
console.log('Python output:', text);
|
2024-05-17 05:21:08 +00:00
|
|
|
|
2024-05-17 08:39:07 +00:00
|
|
|
if (stdout) {
|
|
|
|
stdout += `${text}\n`;
|
|
|
|
} else {
|
|
|
|
stdout = `${text}\n`;
|
|
|
|
}
|
|
|
|
},
|
|
|
|
stderr: (text) => {
|
|
|
|
console.log('An error occured:', text);
|
|
|
|
if (stderr) {
|
|
|
|
stderr += `${text}\n`;
|
|
|
|
} else {
|
|
|
|
stderr = `${text}\n`;
|
|
|
|
}
|
2024-05-17 05:21:08 +00:00
|
|
|
}
|
2024-05-17 08:39:07 +00:00
|
|
|
});
|
2024-05-17 03:49:28 +00:00
|
|
|
|
2024-05-17 08:39:07 +00:00
|
|
|
try {
|
|
|
|
const res = await pyodide.loadPackage('micropip');
|
|
|
|
console.log(res);
|
2024-05-17 06:25:55 +00:00
|
|
|
|
2024-05-17 08:39:07 +00:00
|
|
|
const micropip = pyodide.pyimport('micropip');
|
2024-05-17 06:35:25 +00:00
|
|
|
|
2024-05-17 08:39:07 +00:00
|
|
|
await micropip.set_index_urls('https://pypi.org/pypi/{package_name}/json');
|
2024-05-17 06:25:55 +00:00
|
|
|
|
2024-05-17 08:39:07 +00:00
|
|
|
let packages = [
|
|
|
|
code.includes('requests') ? 'requests' : null,
|
|
|
|
code.includes('bs4') ? 'beautifulsoup4' : null,
|
|
|
|
code.includes('numpy') ? 'numpy' : null,
|
2024-05-17 08:54:37 +00:00
|
|
|
code.includes('pandas') ? 'pandas' : null,
|
|
|
|
code.includes('matplotlib') ? 'matplotlib' : null
|
2024-05-17 08:39:07 +00:00
|
|
|
].filter(Boolean);
|
2024-05-17 06:25:55 +00:00
|
|
|
|
2024-05-17 08:39:07 +00:00
|
|
|
console.log(packages);
|
|
|
|
await micropip.install(packages);
|
2024-05-17 06:25:55 +00:00
|
|
|
|
2024-05-17 08:39:07 +00:00
|
|
|
result = await pyodide.runPythonAsync(`from js import prompt
|
2024-05-17 07:09:53 +00:00
|
|
|
def input(p):
|
|
|
|
return prompt(p)
|
|
|
|
__builtins__.input = input`);
|
|
|
|
|
2024-05-17 08:39:07 +00:00
|
|
|
result = await pyodide.runPython(code);
|
|
|
|
|
|
|
|
if (!result) {
|
|
|
|
result = '[NO OUTPUT]';
|
|
|
|
}
|
2024-05-17 05:30:09 +00:00
|
|
|
|
2024-05-17 08:39:07 +00:00
|
|
|
console.log(result);
|
|
|
|
console.log(stdout);
|
|
|
|
console.log(stderr);
|
2024-05-17 09:19:31 +00:00
|
|
|
|
|
|
|
const pltCanvasElement = document.getElementById(`plt-canvas-${id}`);
|
|
|
|
|
|
|
|
if (pltCanvasElement?.innerHTML !== '') {
|
2024-05-17 09:20:21 +00:00
|
|
|
pltCanvasElement.classList.add('pt-4');
|
2024-05-17 09:19:31 +00:00
|
|
|
}
|
2024-05-17 08:39:07 +00:00
|
|
|
} catch (error) {
|
|
|
|
console.error('Error:', error);
|
|
|
|
stderr = error;
|
2024-05-17 06:53:50 +00:00
|
|
|
}
|
|
|
|
|
2024-05-17 08:39:07 +00:00
|
|
|
executing = false;
|
2024-05-17 05:30:09 +00:00
|
|
|
}
|
2024-05-17 08:39:07 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
const executePythonAsWorker = async (code) => {
|
|
|
|
result = null;
|
|
|
|
stdout = null;
|
|
|
|
stderr = null;
|
|
|
|
|
|
|
|
executing = true;
|
|
|
|
|
|
|
|
let packages = [
|
|
|
|
code.includes('requests') ? 'requests' : null,
|
|
|
|
code.includes('bs4') ? 'beautifulsoup4' : null,
|
|
|
|
code.includes('numpy') ? 'numpy' : null,
|
2024-05-17 08:54:37 +00:00
|
|
|
code.includes('pandas') ? 'pandas' : null,
|
|
|
|
code.includes('matplotlib') ? 'matplotlib' : null
|
2024-05-17 08:39:07 +00:00
|
|
|
].filter(Boolean);
|
|
|
|
|
|
|
|
const pyodideWorker = new Worker('/pyodide-worker.js');
|
|
|
|
|
|
|
|
pyodideWorker.postMessage({
|
|
|
|
id: id,
|
|
|
|
code: code,
|
|
|
|
packages: packages
|
|
|
|
});
|
|
|
|
|
|
|
|
setTimeout(() => {
|
|
|
|
if (executing) {
|
|
|
|
executing = false;
|
|
|
|
stderr = 'Execution Time Limit Exceeded';
|
|
|
|
pyodideWorker.terminate();
|
|
|
|
}
|
2024-05-17 08:39:46 +00:00
|
|
|
}, 60000);
|
2024-05-17 08:39:07 +00:00
|
|
|
|
|
|
|
pyodideWorker.onmessage = (event) => {
|
|
|
|
console.log('pyodideWorker.onmessage', event);
|
|
|
|
const { id, ...data } = event.data;
|
|
|
|
|
|
|
|
console.log(id, data);
|
|
|
|
|
|
|
|
data['stdout'] && (stdout = data['stdout']);
|
|
|
|
data['stderr'] && (stderr = data['stderr']);
|
|
|
|
data['result'] && (result = data['result']);
|
|
|
|
|
|
|
|
executing = false;
|
|
|
|
};
|
2024-05-17 06:47:13 +00:00
|
|
|
|
2024-05-17 08:39:07 +00:00
|
|
|
pyodideWorker.onerror = (event) => {
|
|
|
|
console.log('pyodideWorker.onerror', event);
|
|
|
|
executing = false;
|
|
|
|
};
|
2024-05-17 03:49:28 +00:00
|
|
|
};
|
|
|
|
|
2024-01-22 11:33:49 +00:00
|
|
|
$: highlightedCode = code ? hljs.highlightAuto(code, hljs.getLanguage(lang)?.aliases).value : '';
|
|
|
|
</script>
|
|
|
|
|
2024-01-22 12:14:07 +00:00
|
|
|
{#if code}
|
|
|
|
<div class="mb-4">
|
|
|
|
<div
|
|
|
|
class="flex justify-between bg-[#202123] text-white text-xs px-4 pt-1 pb-0.5 rounded-t-lg overflow-x-auto"
|
2024-01-22 11:33:49 +00:00
|
|
|
>
|
2024-01-22 12:14:07 +00:00
|
|
|
<div class="p-1">{@html lang}</div>
|
2024-05-17 03:49:28 +00:00
|
|
|
|
|
|
|
<div class="flex items-center">
|
2024-05-17 04:05:43 +00:00
|
|
|
{#if lang === 'python' || checkPythonCode(code)}
|
2024-05-17 06:47:13 +00:00
|
|
|
{#if executing}
|
|
|
|
<div class="copy-code-button bg-none border-none p-1 cursor-not-allowed">Running</div>
|
|
|
|
{:else}
|
|
|
|
<button
|
|
|
|
class="copy-code-button bg-none border-none p-1"
|
|
|
|
on:click={() => {
|
|
|
|
executePython(code);
|
|
|
|
}}>Run</button
|
|
|
|
>
|
|
|
|
{/if}
|
2024-05-17 03:49:28 +00:00
|
|
|
{/if}
|
|
|
|
<button class="copy-code-button bg-none border-none p-1" on:click={copyCode}
|
|
|
|
>{copied ? 'Copied' : 'Copy Code'}</button
|
|
|
|
>
|
|
|
|
</div>
|
2024-01-22 12:14:07 +00:00
|
|
|
</div>
|
2024-01-22 11:33:49 +00:00
|
|
|
|
2024-04-30 23:52:19 +00:00
|
|
|
<pre
|
|
|
|
class=" hljs p-4 px-5 overflow-x-auto"
|
2024-05-17 06:47:13 +00:00
|
|
|
style="border-top-left-radius: 0px; border-top-right-radius: 0px; {(executing ||
|
|
|
|
stdout ||
|
|
|
|
stderr ||
|
|
|
|
result) &&
|
2024-05-17 03:49:28 +00:00
|
|
|
'border-bottom-left-radius: 0px; border-bottom-right-radius: 0px;'}"><code
|
2024-01-22 12:14:07 +00:00
|
|
|
class="language-{lang} rounded-t-none whitespace-pre">{@html highlightedCode || code}</code
|
|
|
|
></pre>
|
2024-05-17 03:49:28 +00:00
|
|
|
|
2024-05-17 09:19:31 +00:00
|
|
|
<div id="plt-canvas-{id}" class="bg-[#202123] text-white" />
|
|
|
|
|
2024-05-17 06:47:13 +00:00
|
|
|
{#if executing}
|
2024-05-17 03:49:28 +00:00
|
|
|
<div class="bg-[#202123] text-white px-4 py-4 rounded-b-lg">
|
2024-05-17 03:55:27 +00:00
|
|
|
<div class=" text-gray-500 text-xs mb-1">STDOUT/STDERR</div>
|
2024-05-17 06:47:13 +00:00
|
|
|
<div class="text-sm">Running...</div>
|
|
|
|
</div>
|
2024-05-17 06:53:50 +00:00
|
|
|
{:else if stdout || stderr || result}
|
2024-05-17 06:47:13 +00:00
|
|
|
<div class="bg-[#202123] text-white px-4 py-4 rounded-b-lg">
|
|
|
|
<div class=" text-gray-500 text-xs mb-1">STDOUT/STDERR</div>
|
2024-05-17 06:53:50 +00:00
|
|
|
<div class="text-sm">{stdout || stderr || result}</div>
|
2024-05-17 03:49:28 +00:00
|
|
|
</div>
|
|
|
|
{/if}
|
2024-01-22 12:14:07 +00:00
|
|
|
</div>
|
|
|
|
{/if}
|