123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381 |
- <script>
- import { getContext, createEventDispatcher, onMount } from 'svelte';
- import { goto } from '$app/navigation';
- const dispatch = createEventDispatcher();
- const i18n = getContext('i18n');
- import CodeEditor from '$lib/components/common/CodeEditor.svelte';
- import ConfirmDialog from '$lib/components/common/ConfirmDialog.svelte';
- let formElement = null;
- let loading = false;
- let showConfirm = false;
- export let edit = false;
- export let clone = false;
- export let id = '';
- export let name = '';
- export let meta = {
- description: ''
- };
- export let content = '';
- $: if (name && !edit && !clone) {
- id = name.replace(/\s+/g, '_').toLowerCase();
- }
- let codeEditor;
- let boilerplate = `from pydantic import BaseModel
- from typing import Optional
- class Filter:
- class Valves(BaseModel):
- max_turns: int = 4
- pass
- def __init__(self):
- # Indicates custom file handling logic. This flag helps disengage default routines in favor of custom
- # implementations, informing the WebUI to defer file-related operations to designated methods within this class.
- # Alternatively, you can remove the files directly from the body in from the inlet hook
- self.file_handler = True
- # Initialize 'valves' with specific configurations. Using 'Valves' instance helps encapsulate settings,
- # which ensures settings are managed cohesively and not confused with operational flags like 'file_handler'.
- self.valves = self.Valves(**{"max_turns": 2})
- pass
- def inlet(self, body: dict, user: Optional[dict] = None) -> dict:
- # Modify the request body or validate it before processing by the chat completion API.
- # This function is the pre-processor for the API where various checks on the input can be performed.
- # It can also modify the request before sending it to the API.
- print(f"inlet:{__name__}")
- print(f"inlet:body:{body}")
- print(f"inlet:user:{user}")
- if user.get("role", "admin") in ["user", "admin"]:
- messages = body.get("messages", [])
- if len(messages) > self.valves.max_turns:
- raise Exception(
- f"Conversation turn limit exceeded. Max turns: {self.valves.max_turns}"
- )
- return body
- def outlet(self, body: dict, user: Optional[dict] = None) -> dict:
- # Modify or analyze the response body after processing by the API.
- # This function is the post-processor for the API, which can be used to modify the response
- # or perform additional checks and analytics.
- print(f"outlet:{__name__}")
- print(f"outlet:body:{body}")
- print(f"outlet:user:{user}")
- messages = [
- {
- **message,
- "content": f"{message['content']} - @@Modified from Filter Outlet",
- }
- for message in body.get("messages", [])
- ]
- return {"messages": messages}
- `;
- const _boilerplate = `from pydantic import BaseModel
- from typing import Optional, Union, Generator, Iterator
- from utils.misc import get_last_user_message
- import os
- import requests
- # Filter Class: This class is designed to serve as a pre-processor and post-processor
- # for request and response modifications. It checks and transforms requests and responses
- # to ensure they meet specific criteria before further processing or returning to the user.
- class Filter:
- class Valves(BaseModel):
- max_turns: int = 4
- pass
- def __init__(self):
- # Indicates custom file handling logic. This flag helps disengage default routines in favor of custom
- # implementations, informing the WebUI to defer file-related operations to designated methods within this class.
- # Alternatively, you can remove the files directly from the body in from the inlet hook
- self.file_handler = True
- # Initialize 'valves' with specific configurations. Using 'Valves' instance helps encapsulate settings,
- # which ensures settings are managed cohesively and not confused with operational flags like 'file_handler'.
- self.valves = self.Valves(**{"max_turns": 2})
- pass
- def inlet(self, body: dict, user: Optional[dict] = None) -> dict:
- # Modify the request body or validate it before processing by the chat completion API.
- # This function is the pre-processor for the API where various checks on the input can be performed.
- # It can also modify the request before sending it to the API.
- print(f"inlet:{__name__}")
- print(f"inlet:body:{body}")
- print(f"inlet:user:{user}")
- if user.get("role", "admin") in ["user", "admin"]:
- messages = body.get("messages", [])
- if len(messages) > self.valves.max_turns:
- raise Exception(
- f"Conversation turn limit exceeded. Max turns: {self.valves.max_turns}"
- )
- return body
- def outlet(self, body: dict, user: Optional[dict] = None) -> dict:
- # Modify or analyze the response body after processing by the API.
- # This function is the post-processor for the API, which can be used to modify the response
- # or perform additional checks and analytics.
- print(f"outlet:{__name__}")
- print(f"outlet:body:{body}")
- print(f"outlet:user:{user}")
- messages = [
- {
- **message,
- "content": f"{message['content']} - @@Modified from Filter Outlet",
- }
- for message in body.get("messages", [])
- ]
- return {"messages": messages}
- # Pipe Class: This class functions as a customizable pipeline.
- # It can be adapted to work with any external or internal models,
- # making it versatile for various use cases outside of just OpenAI models.
- class Pipe:
- class Valves(BaseModel):
- OPENAI_API_BASE_URL: str = "https://api.openai.com/v1"
- OPENAI_API_KEY: str = "your-key"
- pass
- def __init__(self):
- self.type = "manifold"
- self.valves = self.Valves()
- self.pipes = self.get_openai_models()
- pass
- def get_openai_models(self):
- if self.valves.OPENAI_API_KEY:
- try:
- headers = {}
- headers["Authorization"] = f"Bearer {self.valves.OPENAI_API_KEY}"
- headers["Content-Type"] = "application/json"
- r = requests.get(
- f"{self.valves.OPENAI_API_BASE_URL}/models", headers=headers
- )
- models = r.json()
- return [
- {
- "id": model["id"],
- "name": model["name"] if "name" in model else model["id"],
- }
- for model in models["data"]
- if "gpt" in model["id"]
- ]
- except Exception as e:
- print(f"Error: {e}")
- return [
- {
- "id": "error",
- "name": "Could not fetch models from OpenAI, please update the API Key in the valves.",
- },
- ]
- else:
- return []
- def pipe(self, body: dict) -> Union[str, Generator, Iterator]:
- # This is where you can add your custom pipelines like RAG.
- print(f"pipe:{__name__}")
- if "user" in body:
- print(body["user"])
- del body["user"]
- headers = {}
- headers["Authorization"] = f"Bearer {self.valves.OPENAI_API_KEY}"
- headers["Content-Type"] = "application/json"
- model_id = body["model"][body["model"].find(".") + 1 :]
- payload = {**body, "model": model_id}
- print(payload)
- try:
- r = requests.post(
- url=f"{self.valves.OPENAI_API_BASE_URL}/chat/completions",
- json=payload,
- headers=headers,
- stream=True,
- )
- r.raise_for_status()
- if body["stream"]:
- return r.iter_lines()
- else:
- return r.json()
- except Exception as e:
- return f"Error: {e}"
- `;
- const saveHandler = async () => {
- loading = true;
- dispatch('save', {
- id,
- name,
- meta,
- content
- });
- };
- const submitHandler = async () => {
- if (codeEditor) {
- const res = await codeEditor.formatPythonCodeHandler();
- if (res) {
- console.log('Code formatted successfully');
- saveHandler();
- }
- }
- };
- </script>
- <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="mb-2.5">
- <button
- class="flex space-x-1"
- on:click={() => {
- goto('/workspace/functions');
- }}
- type="button"
- >
- <div class=" self-center">
- <svg
- xmlns="http://www.w3.org/2000/svg"
- viewBox="0 0 20 20"
- fill="currentColor"
- class="w-4 h-4"
- >
- <path
- fill-rule="evenodd"
- d="M17 10a.75.75 0 01-.75.75H5.612l4.158 3.96a.75.75 0 11-1.04 1.08l-5.5-5.25a.75.75 0 010-1.08l5.5-5.25a.75.75 0 111.04 1.08L5.612 9.25H16.25A.75.75 0 0117 10z"
- clip-rule="evenodd"
- />
- </svg>
- </div>
- <div class=" self-center font-medium text-sm">{$i18n.t('Back')}</div>
- </button>
- </div>
- <div class="flex flex-col flex-1 overflow-auto h-0 rounded-lg">
- <div class="w-full mb-2 flex flex-col gap-1.5">
- <div class="flex gap-2 w-full">
- <input
- class="w-full px-3 py-2 text-sm font-medium bg-gray-50 dark:bg-gray-850 dark:text-gray-200 rounded-lg outline-none"
- type="text"
- placeholder="Function Name (e.g. My Filter)"
- bind:value={name}
- required
- />
- <input
- class="w-full px-3 py-2 text-sm font-medium disabled:text-gray-300 dark:disabled:text-gray-700 bg-gray-50 dark:bg-gray-850 dark:text-gray-200 rounded-lg outline-none"
- type="text"
- placeholder="Function ID (e.g. my_filter)"
- bind:value={id}
- required
- disabled={edit}
- />
- </div>
- <input
- class="w-full px-3 py-2 text-sm font-medium bg-gray-50 dark:bg-gray-850 dark:text-gray-200 rounded-lg outline-none"
- type="text"
- placeholder="Function Description (e.g. A filter to remove profanity from text)"
- bind:value={meta.description}
- required
- />
- </div>
- <div class="mb-2 flex-1 overflow-auto h-0 rounded-lg">
- <CodeEditor
- bind:value={content}
- bind:this={codeEditor}
- {boilerplate}
- 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">Warning:</span> Functions allow
- arbitrary code execution <br />—
- <span class=" font-medium dark:text-gray-400"
- >don't install random functions from sources you don't trust.</span
- >
- </div>
- </div>
- <button
- class="px-3 py-1.5 text-sm font-medium bg-emerald-600 hover:bg-emerald-700 text-gray-50 transition rounded-lg"
- 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>Please carefully review the following warnings:</div>
- <ul class=" mt-1 list-disc pl-4 text-xs">
- <li>Functions allow arbitrary code execution.</li>
- <li>Do not install functions from sources you do not fully trust.</li>
- </ul>
- </div>
- <div class="my-3">
- 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>
|