| | <script> |
| | import { getContext, createEventDispatcher, onMount, tick } from 'svelte'; |
| |
|
| | const i18n = getContext('i18n'); |
| |
|
| | import CodeEditor from '$lib/components/common/CodeEditor.svelte'; |
| | import { goto } from '$app/navigation'; |
| | import ConfirmDialog from '$lib/components/common/ConfirmDialog.svelte'; |
| | import Badge from '$lib/components/common/Badge.svelte'; |
| | import ChevronLeft from '$lib/components/icons/ChevronLeft.svelte'; |
| | import Tooltip from '$lib/components/common/Tooltip.svelte'; |
| | import LockClosed from '$lib/components/icons/LockClosed.svelte'; |
| | import AccessControlModal from '../common/AccessControlModal.svelte'; |
| |
|
| | const dispatch = createEventDispatcher(); |
| |
|
| | let formElement = null; |
| | let loading = false; |
| |
|
| | let showConfirm = false; |
| | let showAccessControlModal = false; |
| |
|
| | export let edit = false; |
| | export let clone = false; |
| |
|
| | export let id = ''; |
| | export let name = ''; |
| | export let meta = { |
| | description: '' |
| | }; |
| | export let content = ''; |
| | export let accessControl = null; |
| |
|
| | let _content = ''; |
| |
|
| | $: if (content) { |
| | updateContent(); |
| | } |
| |
|
| | const updateContent = () => { |
| | _content = content; |
| | }; |
| |
|
| | $: if (name && !edit && !clone) { |
| | id = name.replace(/\s+/g, '_').toLowerCase(); |
| | } |
| |
|
| | let codeEditor; |
| | let boilerplate = `import os |
| | import requests |
| | from datetime import datetime |
| |
|
| |
|
| | class Tools: |
| | def __init__(self): |
| | pass |
| |
|
| | |
| | |
| | |
| |
|
| | def get_user_name_and_email_and_id(self, __user__: dict = {}) -> str: |
| | """ |
| | Get the user name, Email and ID from the user object. |
| | """ |
| |
|
| | |
| | |
| |
|
| | print(__user__) |
| | result = "" |
| |
|
| | if "name" in __user__: |
| | result += f"User: {__user__['name']}" |
| | if "id" in __user__: |
| | result += f" (ID: {__user__['id']})" |
| | if "email" in __user__: |
| | result += f" (Email: {__user__['email']})" |
| |
|
| | if result == "": |
| | result = "User: Unknown" |
| |
|
| | return result |
| |
|
| | def get_current_time(self) -> str: |
| | """ |
| | Get the current time in a more human-readable format. |
| | :return: The current time. |
| | """ |
| |
|
| | now = datetime.now() |
| | current_time = now.strftime("%I:%M:%S %p") |
| | current_date = now.strftime( |
| | "%A, %B %d, %Y" |
| | ) |
| |
|
| | return f"Current Date and Time = {current_date}, {current_time}" |
| |
|
| | def calculator(self, equation: str) -> str: |
| | """ |
| | Calculate the result of an equation. |
| | :param equation: The equation to calculate. |
| | """ |
| |
|
| | |
| | |
| | try: |
| | result = eval(equation) |
| | return f"{equation} = {result}" |
| | except Exception as e: |
| | print(e) |
| | return "Invalid equation" |
| |
|
| | def get_current_weather(self, city: str) -> str: |
| | """ |
| | Get the current weather for a given city. |
| | :param city: The name of the city to get the weather for. |
| | :return: The current weather information or an error message. |
| | """ |
| | api_key = os.getenv("OPENWEATHER_API_KEY") |
| | if not api_key: |
| | return ( |
| | "API key is not set in the environment variable 'OPENWEATHER_API_KEY'." |
| | ) |
| |
|
| | base_url = "http://api.openweathermap.org/data/2.5/weather" |
| | params = { |
| | "q": city, |
| | "appid": api_key, |
| | "units": "metric", |
| | } |
| |
|
| | try: |
| | response = requests.get(base_url, params=params) |
| | response.raise_for_status() |
| | data = response.json() |
| |
|
| | if data.get("cod") != 200: |
| | return f"Error fetching weather data: {data.get('message')}" |
| |
|
| | weather_description = data["weather"][0]["description"] |
| | temperature = data["main"]["temp"] |
| | humidity = data["main"]["humidity"] |
| | wind_speed = data["wind"]["speed"] |
| |
|
| | return f"Weather in {city}: {temperature}°C" |
| | except requests.RequestException as e: |
| | return f"Error fetching weather data: {str(e)}" |
| | `; |
| |
|
| | const saveHandler = async () => { |
| | loading = true; |
| | dispatch('save', { |
| | id, |
| | name, |
| | meta, |
| | content, |
| | access_control: accessControl |
| | }); |
| | }; |
| |
|
| | const submitHandler = async () => { |
| | if (codeEditor) { |
| | content = _content; |
| | await tick(); |
| |
|
| | const res = await codeEditor.formatPythonCodeHandler(); |
| | await tick(); |
| |
|
| | content = _content; |
| | await tick(); |
| |
|
| | if (res) { |
| | console.log('Code formatted successfully'); |
| |
|
| | saveHandler(); |
| | } |
| | } |
| | }; |
| | </script> |
| |
|
| | <AccessControlModal bind:show={showAccessControlModal} bind:accessControl /> |
| |
|
| | <div class=" flex flex-col justify-between w-full overflow-y-auto h-full"> |
| | <div class="mx-auto w-full md:px-0 h-full"> |
| | <form |
| | bind:this={formElement} |
| | class=" flex flex-col max-h-[100dvh] h-full" |
| | on:submit|preventDefault={() => { |
| | if (edit) { |
| | submitHandler(); |
| | } else { |
| | showConfirm = true; |
| | } |
| | }} |
| | > |
| | <div class="flex flex-col flex-1 overflow-auto h-0"> |
| | <div class="w-full mb-2 flex flex-col gap-0.5"> |
| | <div class="flex w-full items-center"> |
| | <div class=" flex-shrink-0 mr-2"> |
| | <Tooltip content={$i18n.t('Back')}> |
| | <button |
| | class="w-full text-left text-sm py-1.5 px-1 rounded-lg dark:text-gray-300 dark:hover:text-white hover:bg-black/5 dark:hover:bg-gray-850" |
| | on:click={() => { |
| | goto('/workspace/tools'); |
| | }} |
| | type="button" |
| | > |
| | <ChevronLeft strokeWidth="2.5" /> |
| | </button> |
| | </Tooltip> |
| | </div> |
| |
|
| | <div class="flex-1"> |
| | <Tooltip content={$i18n.t('e.g. My Tools')} placement="top-start"> |
| | <input |
| | class="w-full text-2xl font-semibold bg-transparent outline-none" |
| | type="text" |
| | placeholder={$i18n.t('Tool Name')} |
| | bind:value={name} |
| | required |
| | /> |
| | </Tooltip> |
| | </div> |
| |
|
| | <div class="self-center flex-shrink-0"> |
| | <button |
| | class="bg-gray-50 hover:bg-gray-100 text-black dark:bg-gray-850 dark:hover:bg-gray-800 dark:text-white transition px-2 py-1 rounded-full flex gap-1 items-center" |
| | type="button" |
| | on:click={() => { |
| | showAccessControlModal = true; |
| | }} |
| | > |
| | <LockClosed strokeWidth="2.5" className="size-3.5" /> |
| |
|
| | <div class="text-sm font-medium flex-shrink-0"> |
| | {$i18n.t('Access')} |
| | </div> |
| | </button> |
| | </div> |
| | </div> |
| |
|
| | <div class=" flex gap-2 px-1 items-center"> |
| | { |
| | <div class="text-sm text-gray-500 flex-shrink-0"> |
| | {id} |
| | </div> |
| | {:else} |
| | <Tooltip className="w-full" content={$i18n.t('e.g. my_tools')} placement="top-start"> |
| | <input |
| | class="w-full text-sm disabled:text-gray-500 bg-transparent outline-none" |
| | type="text" |
| | placeholder={$i18n.t('Tool ID')} |
| | bind:value={id} |
| | required |
| | disabled={edit} |
| | /> |
| | </Tooltip> |
| | {/if} |
| |
|
| | <Tooltip |
| | className="w-full self-center items-center flex" |
| | content={$i18n.t('e.g. Tools for performing various operations')} |
| | placement="top-start" |
| | > |
| | <input |
| | class="w-full text-sm bg-transparent outline-none" |
| | type="text" |
| | placeholder={$i18n.t('Tool Description')} |
| | bind:value={meta.description} |
| | required |
| | /> |
| | </Tooltip> |
| | </div> |
| | </div> |
| |
|
| | <div class="mb-2 flex-1 overflow-auto h-0 rounded-lg"> |
| | <CodeEditor |
| | bind:this={codeEditor} |
| | value={content} |
| | {boilerplate} |
| | lang="python" |
| | on:change={(e) => { |
| | _content = e.detail.value; |
| | }} |
| | on:save={() => { |
| | if (formElement) { |
| | formElement.requestSubmit(); |
| | } |
| | }} |
| | /> |
| | </div> |
| |
|
| | <div class="pb-3 flex justify-between"> |
| | <div class="flex-1 pr-3"> |
| | <div class="text-xs text-gray-500 line-clamp-2"> |
| | <span class=" font-semibold dark:text-gray-200">{$i18n.t('Warning:')}</span> |
| | {$i18n.t('Tools are a function calling system with arbitrary code execution')} <br />— |
| | <span class=" font-medium dark:text-gray-400" |
| | >{$i18n.t(`don't install random tools from sources you don't trust.`)}</span |
| | > |
| | </div> |
| | </div> |
| |
|
| | <button |
| | class="px-3.5 py-1.5 text-sm font-medium bg-black hover:bg-gray-900 text-white dark:bg-white dark:text-black dark:hover:bg-gray-100 transition rounded-full" |
| | type="submit" |
| | > |
| | {$i18n.t('Save')} |
| | </button> |
| | </div> |
| | </div> |
| | </form> |
| | </div> |
| | </div> |
| |
|
| | <ConfirmDialog |
| | bind:show={showConfirm} |
| | on:confirm={() => { |
| | submitHandler(); |
| | }} |
| | > |
| | <div class="text-sm text-gray-500"> |
| | <div class=" bg-yellow-500/20 text-yellow-700 dark:text-yellow-200 rounded-lg px-4 py-3"> |
| | <div>{$i18n.t('Please carefully review the following warnings:')}</div> |
| |
|
| | <ul class=" mt-1 list-disc pl-4 text-xs"> |
| | <li> |
| | {$i18n.t('Tools have a function calling system that allows arbitrary code execution.')} |
| | </li> |
| | <li>{$i18n.t('Do not install tools from sources you do not fully trust.')}</li> |
| | </ul> |
| | </div> |
| |
|
| | <div class="my-3"> |
| | {$i18n.t( |
| | 'I acknowledge that I have read and I understand the implications of my action. I am aware of the risks associated with executing arbitrary code and I have verified the trustworthiness of the source.' |
| | )} |
| | </div> |
| | </div> |
| | </ConfirmDialog> |
| |
|