open-webui/src/lib/components/chat/Messages/CodeBlock.svelte

277 lines
6.6 KiB
Svelte
Raw Normal View History

2024-01-22 11:33:49 +00:00
<script lang="ts">
import hljs from 'highlight.js';
import { loadPyodide } from 'pyodide';
2024-08-08 13:07:24 +00:00
import { getContext, getAllContexts } from 'svelte';
2024-08-08 12:24:47 +00:00
import { copyToClipboard } from '$lib/utils';
import 'highlight.js/styles/github-dark.min.css';
2024-05-19 12:19:48 +00:00
import PyodideWorker from '$lib/workers/pyodide.worker?worker';
2024-05-17 03:49:28 +00:00
2024-08-08 12:24:47 +00:00
const i18n = getContext('i18n');
2024-05-17 03:49:28 +00:00
export let id = '';
2024-01-22 11:33:49 +00:00
export let lang = '';
export let code = '';
2024-05-25 09:47:09 +00:00
let highlightedCode = null;
let executing = false;
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 syntax characters
const pythonSyntax = [
'def ',
'else:',
'elif ',
'try:',
'except:',
'finally:',
'yield ',
'lambda ',
'assert ',
'nonlocal ',
'del ',
'True',
'False',
'None',
' and ',
' or ',
' not ',
' in ',
' is ',
2024-05-19 17:07:43 +00:00
' with '
2024-05-17 04:05:43 +00:00
];
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;
};
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 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`;
}
},
packages: ['micropip']
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 micropip = pyodide.pyimport('micropip');
2024-05-17 06:35:25 +00:00
2024-05-21 02:20:49 +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,
2024-05-18 21:28:16 +00:00
code.includes('matplotlib') ? 'matplotlib' : null,
2024-05-18 21:29:39 +00:00
code.includes('sklearn') ? 'scikit-learn' : null,
2024-05-18 21:28:16 +00:00
code.includes('scipy') ? 'scipy' : null,
code.includes('re') ? 'regex' : null,
code.includes('seaborn') ? 'seaborn' : 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,
2024-05-18 21:29:39 +00:00
code.includes('sklearn') ? 'scikit-learn' : null,
2024-05-18 21:28:16 +00:00
code.includes('scipy') ? 'scipy' : null,
code.includes('re') ? 'regex' : null,
code.includes('seaborn') ? 'seaborn' : null
2024-05-17 08:39:07 +00:00
].filter(Boolean);
2024-05-18 21:29:39 +00:00
console.log(packages);
const pyodideWorker = new PyodideWorker();
2024-05-17 08:39:07 +00:00
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 08:39:07 +00:00
pyodideWorker.onerror = (event) => {
console.log('pyodideWorker.onerror', event);
executing = false;
};
2024-05-17 03:49:28 +00:00
};
2024-06-20 19:27:34 +00:00
let debounceTimeout;
2024-05-25 09:47:09 +00:00
$: if (code) {
2024-06-20 19:27:34 +00:00
// Function to perform the code highlighting
const highlightCode = () => {
highlightedCode = hljs.highlightAuto(code, hljs.getLanguage(lang)?.aliases).value || code;
};
// Clear the previous timeout if it exists
clearTimeout(debounceTimeout);
// Set a new timeout to debounce the code highlighting
debounceTimeout = setTimeout(highlightCode, 10);
2024-05-25 09:47:09 +00:00
}
2024-01-22 11:33:49 +00:00
</script>
2024-08-05 17:43:51 +00:00
<div class="my-2" dir="ltr">
2024-05-25 09:47:09 +00:00
<div
class="flex justify-between bg-[#202123] text-white text-xs px-4 pt-1 pb-0.5 rounded-t-lg overflow-x-auto"
>
<div class="p-1">{@html lang}</div>
<div class="flex items-center">
{#if lang.toLowerCase() === 'python' || lang.toLowerCase() === 'py' || (lang === '' && checkPythonCode(code))}
2024-05-25 09:47:09 +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);
2024-08-08 13:07:24 +00:00
}}>{$i18n.t('Run')}</button
2024-05-25 09:47:09 +00:00
>
2024-05-17 03:49:28 +00:00
{/if}
2024-05-25 09:47:09 +00:00
{/if}
<button class="copy-code-button bg-none border-none p-1" on:click={copyCode}
2024-08-08 13:07:24 +00:00
>{copied ? $i18n.t('Copied') : $i18n.t('Copy Code')}</button
2024-05-25 09:47:09 +00:00
>
2024-01-22 12:14:07 +00:00
</div>
</div>
2024-05-25 09:47:09 +00:00
<pre
class=" hljs p-4 px-5 overflow-x-auto"
style="border-top-left-radius: 0px; border-top-right-radius: 0px; {(executing ||
stdout ||
stderr ||
result) &&
'border-bottom-left-radius: 0px; border-bottom-right-radius: 0px;'}"><code
2024-07-09 07:41:56 +00:00
class="language-{lang} rounded-t-none whitespace-pre"
>{#if highlightedCode}{@html highlightedCode}{:else}{code}{/if}</code
2024-05-25 09:47:09 +00:00
></pre>
<div
id="plt-canvas-{id}"
class="bg-[#202123] text-white max-w-full overflow-x-auto scrollbar-hidden"
/>
{#if executing}
<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>
<div class="text-sm">Running...</div>
</div>
{:else if stdout || stderr || result}
<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>
<div class="text-sm">{stdout || stderr || result}</div>
</div>
{/if}
</div>