Compare commits
	
		
			3 Commits
		
	
	
		
			main
			...
			b56619a2e6
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| b56619a2e6 | |||
| 90e0b4ff7f | |||
| 47059dabdc | 
							
								
								
									
										40
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										40
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @ -174,3 +174,43 @@ cython_debug/ | |||||||
| pyvenv.cfg | pyvenv.cfg | ||||||
| .venv | .venv | ||||||
| pip-selfcheck.json | pip-selfcheck.json | ||||||
|  |  | ||||||
|  |  | ||||||
|  | # Logs | ||||||
|  | logs | ||||||
|  | *.log | ||||||
|  | npm-debug.log* | ||||||
|  |  | ||||||
|  | # Runtime data | ||||||
|  | pids | ||||||
|  | *.pid | ||||||
|  | *.seed | ||||||
|  |  | ||||||
|  | # Directory for instrumented libs generated by jscoverage/JSCover | ||||||
|  | lib-cov | ||||||
|  |  | ||||||
|  | # Coverage directory used by tools like istanbul | ||||||
|  | coverage | ||||||
|  |  | ||||||
|  | # nyc test coverage | ||||||
|  | .nyc_output | ||||||
|  |  | ||||||
|  | # Grunt intermediate storage (http://gruntjs.com/creating-plugins#storing-task-files) | ||||||
|  | .grunt | ||||||
|  |  | ||||||
|  | # node-waf configuration | ||||||
|  | .lock-wscript | ||||||
|  |  | ||||||
|  | # Compiled binary addons (http://nodejs.org/api/addons.html) | ||||||
|  | build/Release | ||||||
|  |  | ||||||
|  | # Dependency directories | ||||||
|  | node_modules | ||||||
|  | jspm_packages | ||||||
|  |  | ||||||
|  | # Optional npm cache directory | ||||||
|  | .npm | ||||||
|  |  | ||||||
|  | # Optional REPL history | ||||||
|  | .node_repl_history | ||||||
|  | .next | ||||||
							
								
								
									
										3
									
								
								dewey/.eslintrc.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								dewey/.eslintrc.json
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,3 @@ | |||||||
|  | { | ||||||
|  |   "extends": ["next/core-web-vitals", "next/typescript"] | ||||||
|  | } | ||||||
							
								
								
									
										36
									
								
								dewey/.gitignore
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										36
									
								
								dewey/.gitignore
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @ -0,0 +1,36 @@ | |||||||
|  | # See https://help.github.com/articles/ignoring-files/ for more about ignoring files. | ||||||
|  |  | ||||||
|  | # dependencies | ||||||
|  | /node_modules | ||||||
|  | /.pnp | ||||||
|  | .pnp.js | ||||||
|  | .yarn/install-state.gz | ||||||
|  |  | ||||||
|  | # testing | ||||||
|  | /coverage | ||||||
|  |  | ||||||
|  | # next.js | ||||||
|  | /.next/ | ||||||
|  | /out/ | ||||||
|  |  | ||||||
|  | # production | ||||||
|  | /build | ||||||
|  |  | ||||||
|  | # misc | ||||||
|  | .DS_Store | ||||||
|  | *.pem | ||||||
|  |  | ||||||
|  | # debug | ||||||
|  | npm-debug.log* | ||||||
|  | yarn-debug.log* | ||||||
|  | yarn-error.log* | ||||||
|  |  | ||||||
|  | # local env files | ||||||
|  | .env*.local | ||||||
|  |  | ||||||
|  | # vercel | ||||||
|  | .vercel | ||||||
|  |  | ||||||
|  | # typescript | ||||||
|  | *.tsbuildinfo | ||||||
|  | next-env.d.ts | ||||||
							
								
								
									
										36
									
								
								dewey/README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										36
									
								
								dewey/README.md
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,36 @@ | |||||||
|  | This is a [Next.js](https://nextjs.org) project bootstrapped with [`create-next-app`](https://nextjs.org/docs/app/api-reference/cli/create-next-app). | ||||||
|  |  | ||||||
|  | ## Getting Started | ||||||
|  |  | ||||||
|  | First, run the development server: | ||||||
|  |  | ||||||
|  | ```bash | ||||||
|  | npm run dev | ||||||
|  | # or | ||||||
|  | yarn dev | ||||||
|  | # or | ||||||
|  | pnpm dev | ||||||
|  | # or | ||||||
|  | bun dev | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | Open [http://localhost:3000](http://localhost:3000) with your browser to see the result. | ||||||
|  |  | ||||||
|  | You can start editing the page by modifying `app/page.tsx`. The page auto-updates as you edit the file. | ||||||
|  |  | ||||||
|  | This project uses [`next/font`](https://nextjs.org/docs/app/building-your-application/optimizing/fonts) to automatically optimize and load [Geist](https://vercel.com/font), a new font family for Vercel. | ||||||
|  |  | ||||||
|  | ## Learn More | ||||||
|  |  | ||||||
|  | To learn more about Next.js, take a look at the following resources: | ||||||
|  |  | ||||||
|  | - [Next.js Documentation](https://nextjs.org/docs) - learn about Next.js features and API. | ||||||
|  | - [Learn Next.js](https://nextjs.org/learn) - an interactive Next.js tutorial. | ||||||
|  |  | ||||||
|  | You can check out [the Next.js GitHub repository](https://github.com/vercel/next.js) - your feedback and contributions are welcome! | ||||||
|  |  | ||||||
|  | ## Deploy on Vercel | ||||||
|  |  | ||||||
|  | The easiest way to deploy your Next.js app is to use the [Vercel Platform](https://vercel.com/new?utm_medium=default-template&filter=next.js&utm_source=create-next-app&utm_campaign=create-next-app-readme) from the creators of Next.js. | ||||||
|  |  | ||||||
|  | Check out our [Next.js deployment documentation](https://nextjs.org/docs/app/building-your-application/deploying) for more details. | ||||||
							
								
								
									
										
											BIN
										
									
								
								dewey/app/favicon.ico
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								dewey/app/favicon.ico
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 25 KiB | 
							
								
								
									
										
											BIN
										
									
								
								dewey/app/fonts/GeistMonoVF.woff
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								dewey/app/fonts/GeistMonoVF.woff
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										
											BIN
										
									
								
								dewey/app/fonts/GeistVF.woff
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								dewey/app/fonts/GeistVF.woff
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										27
									
								
								dewey/app/globals.css
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										27
									
								
								dewey/app/globals.css
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,27 @@ | |||||||
|  | @tailwind base; | ||||||
|  | @tailwind components; | ||||||
|  | @tailwind utilities; | ||||||
|  |  | ||||||
|  | :root { | ||||||
|  |   --background: #ffffff; | ||||||
|  |   --foreground: #171717; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | @media (prefers-color-scheme: dark) { | ||||||
|  |   :root { | ||||||
|  |     --background: #0a0a0a; | ||||||
|  |     --foreground: #ededed; | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | body { | ||||||
|  |   color: var(--foreground); | ||||||
|  |   background: var(--background); | ||||||
|  |   font-family: Arial, Helvetica, sans-serif; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | @layer utilities { | ||||||
|  |   .text-balance { | ||||||
|  |     text-wrap: balance; | ||||||
|  |   } | ||||||
|  | } | ||||||
							
								
								
									
										35
									
								
								dewey/app/layout.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										35
									
								
								dewey/app/layout.tsx
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,35 @@ | |||||||
|  | import type { Metadata } from "next"; | ||||||
|  | import localFont from "next/font/local"; | ||||||
|  | import "./globals.css"; | ||||||
|  |  | ||||||
|  | const geistSans = localFont({ | ||||||
|  |   src: "./fonts/GeistVF.woff", | ||||||
|  |   variable: "--font-geist-sans", | ||||||
|  |   weight: "100 900", | ||||||
|  | }); | ||||||
|  | const geistMono = localFont({ | ||||||
|  |   src: "./fonts/GeistMonoVF.woff", | ||||||
|  |   variable: "--font-geist-mono", | ||||||
|  |   weight: "100 900", | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | export const metadata: Metadata = { | ||||||
|  |   title: "Create Next App", | ||||||
|  |   description: "Generated by create next app", | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | export default function RootLayout({ | ||||||
|  |   children, | ||||||
|  | }: Readonly<{ | ||||||
|  |   children: React.ReactNode; | ||||||
|  | }>) { | ||||||
|  |   return ( | ||||||
|  |     <html lang="en"> | ||||||
|  |       <body | ||||||
|  |         className={`${geistSans.variable} ${geistMono.variable} antialiased`} | ||||||
|  |       > | ||||||
|  |         {children} | ||||||
|  |       </body> | ||||||
|  |     </html> | ||||||
|  |   ); | ||||||
|  | } | ||||||
							
								
								
									
										101
									
								
								dewey/app/page.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										101
									
								
								dewey/app/page.tsx
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,101 @@ | |||||||
|  | import Image from "next/image"; | ||||||
|  |  | ||||||
|  | export default function Home() { | ||||||
|  |   return ( | ||||||
|  |     <div className="grid grid-rows-[20px_1fr_20px] items-center justify-items-center min-h-screen p-8 pb-20 gap-16 sm:p-20 font-[family-name:var(--font-geist-sans)]"> | ||||||
|  |       <main className="flex flex-col gap-8 row-start-2 items-center sm:items-start"> | ||||||
|  |         <Image | ||||||
|  |           className="dark:invert" | ||||||
|  |           src="https://nextjs.org/icons/next.svg" | ||||||
|  |           alt="Next.js logo" | ||||||
|  |           width={180} | ||||||
|  |           height={38} | ||||||
|  |           priority | ||||||
|  |         /> | ||||||
|  |         <ol className="list-inside list-decimal text-sm text-center sm:text-left font-[family-name:var(--font-geist-mono)]"> | ||||||
|  |           <li className="mb-2"> | ||||||
|  |             Get started by editing{" "} | ||||||
|  |             <code className="bg-black/[.05] dark:bg-white/[.06] px-1 py-0.5 rounded font-semibold"> | ||||||
|  |               app/page.tsx | ||||||
|  |             </code> | ||||||
|  |             . | ||||||
|  |           </li> | ||||||
|  |           <li>Save and see your changes instantly.</li> | ||||||
|  |         </ol> | ||||||
|  |  | ||||||
|  |         <div className="flex gap-4 items-center flex-col sm:flex-row"> | ||||||
|  |           <a | ||||||
|  |             className="rounded-full border border-solid border-transparent transition-colors flex items-center justify-center bg-foreground text-background gap-2 hover:bg-[#383838] dark:hover:bg-[#ccc] text-sm sm:text-base h-10 sm:h-12 px-4 sm:px-5" | ||||||
|  |             href="https://vercel.com/new?utm_source=create-next-app&utm_medium=appdir-template-tw&utm_campaign=create-next-app" | ||||||
|  |             target="_blank" | ||||||
|  |             rel="noopener noreferrer" | ||||||
|  |           > | ||||||
|  |             <Image | ||||||
|  |               className="dark:invert" | ||||||
|  |               src="https://nextjs.org/icons/vercel.svg" | ||||||
|  |               alt="Vercel logomark" | ||||||
|  |               width={20} | ||||||
|  |               height={20} | ||||||
|  |             /> | ||||||
|  |             Deploy now | ||||||
|  |           </a> | ||||||
|  |           <a | ||||||
|  |             className="rounded-full border border-solid border-black/[.08] dark:border-white/[.145] transition-colors flex items-center justify-center hover:bg-[#f2f2f2] dark:hover:bg-[#1a1a1a] hover:border-transparent text-sm sm:text-base h-10 sm:h-12 px-4 sm:px-5 sm:min-w-44" | ||||||
|  |             href="https://nextjs.org/docs?utm_source=create-next-app&utm_medium=appdir-template-tw&utm_campaign=create-next-app" | ||||||
|  |             target="_blank" | ||||||
|  |             rel="noopener noreferrer" | ||||||
|  |           > | ||||||
|  |             Read our docs | ||||||
|  |           </a> | ||||||
|  |         </div> | ||||||
|  |       </main> | ||||||
|  |       <footer className="row-start-3 flex gap-6 flex-wrap items-center justify-center"> | ||||||
|  |         <a | ||||||
|  |           className="flex items-center gap-2 hover:underline hover:underline-offset-4" | ||||||
|  |           href="https://nextjs.org/learn?utm_source=create-next-app&utm_medium=appdir-template-tw&utm_campaign=create-next-app" | ||||||
|  |           target="_blank" | ||||||
|  |           rel="noopener noreferrer" | ||||||
|  |         > | ||||||
|  |           <Image | ||||||
|  |             aria-hidden | ||||||
|  |             src="https://nextjs.org/icons/file.svg" | ||||||
|  |             alt="File icon" | ||||||
|  |             width={16} | ||||||
|  |             height={16} | ||||||
|  |           /> | ||||||
|  |           Learn | ||||||
|  |         </a> | ||||||
|  |         <a | ||||||
|  |           className="flex items-center gap-2 hover:underline hover:underline-offset-4" | ||||||
|  |           href="https://vercel.com/templates?framework=next.js&utm_source=create-next-app&utm_medium=appdir-template-tw&utm_campaign=create-next-app" | ||||||
|  |           target="_blank" | ||||||
|  |           rel="noopener noreferrer" | ||||||
|  |         > | ||||||
|  |           <Image | ||||||
|  |             aria-hidden | ||||||
|  |             src="https://nextjs.org/icons/window.svg" | ||||||
|  |             alt="Window icon" | ||||||
|  |             width={16} | ||||||
|  |             height={16} | ||||||
|  |           /> | ||||||
|  |           Examples | ||||||
|  |         </a> | ||||||
|  |         <a | ||||||
|  |           className="flex items-center gap-2 hover:underline hover:underline-offset-4" | ||||||
|  |           href="https://nextjs.org?utm_source=create-next-app&utm_medium=appdir-template-tw&utm_campaign=create-next-app" | ||||||
|  |           target="_blank" | ||||||
|  |           rel="noopener noreferrer" | ||||||
|  |         > | ||||||
|  |           <Image | ||||||
|  |             aria-hidden | ||||||
|  |             src="https://nextjs.org/icons/globe.svg" | ||||||
|  |             alt="Globe icon" | ||||||
|  |             width={16} | ||||||
|  |             height={16} | ||||||
|  |           /> | ||||||
|  |           Go to nextjs.org → | ||||||
|  |         </a> | ||||||
|  |       </footer> | ||||||
|  |     </div> | ||||||
|  |   ); | ||||||
|  | } | ||||||
							
								
								
									
										4
									
								
								dewey/next.config.mjs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										4
									
								
								dewey/next.config.mjs
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,4 @@ | |||||||
|  | /** @type {import('next').NextConfig} */ | ||||||
|  | const nextConfig = {}; | ||||||
|  |  | ||||||
|  | export default nextConfig; | ||||||
							
								
								
									
										5308
									
								
								dewey/package-lock.json
									
									
									
										generated
									
									
									
										Normal file
									
								
							
							
						
						
									
										5308
									
								
								dewey/package-lock.json
									
									
									
										generated
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										26
									
								
								dewey/package.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										26
									
								
								dewey/package.json
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,26 @@ | |||||||
|  | { | ||||||
|  |   "name": "dewey", | ||||||
|  |   "version": "0.1.0", | ||||||
|  |   "private": true, | ||||||
|  |   "scripts": { | ||||||
|  |     "dev": "next dev", | ||||||
|  |     "build": "next build", | ||||||
|  |     "start": "next start", | ||||||
|  |     "lint": "next lint" | ||||||
|  |   }, | ||||||
|  |   "dependencies": { | ||||||
|  |     "react": "^18", | ||||||
|  |     "react-dom": "^18", | ||||||
|  |     "next": "14.2.13" | ||||||
|  |   }, | ||||||
|  |   "devDependencies": { | ||||||
|  |     "typescript": "^5", | ||||||
|  |     "@types/node": "^20", | ||||||
|  |     "@types/react": "^18", | ||||||
|  |     "@types/react-dom": "^18", | ||||||
|  |     "postcss": "^8", | ||||||
|  |     "tailwindcss": "^3.4.1", | ||||||
|  |     "eslint": "^8", | ||||||
|  |     "eslint-config-next": "14.2.13" | ||||||
|  |   } | ||||||
|  | } | ||||||
							
								
								
									
										8
									
								
								dewey/postcss.config.mjs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										8
									
								
								dewey/postcss.config.mjs
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,8 @@ | |||||||
|  | /** @type {import('postcss-load-config').Config} */ | ||||||
|  | const config = { | ||||||
|  |   plugins: { | ||||||
|  |     tailwindcss: {}, | ||||||
|  |   }, | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | export default config; | ||||||
							
								
								
									
										19
									
								
								dewey/tailwind.config.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										19
									
								
								dewey/tailwind.config.ts
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,19 @@ | |||||||
|  | import type { Config } from "tailwindcss"; | ||||||
|  |  | ||||||
|  | const config: Config = { | ||||||
|  |   content: [ | ||||||
|  |     "./pages/**/*.{js,ts,jsx,tsx,mdx}", | ||||||
|  |     "./components/**/*.{js,ts,jsx,tsx,mdx}", | ||||||
|  |     "./app/**/*.{js,ts,jsx,tsx,mdx}", | ||||||
|  |   ], | ||||||
|  |   theme: { | ||||||
|  |     extend: { | ||||||
|  |       colors: { | ||||||
|  |         background: "var(--background)", | ||||||
|  |         foreground: "var(--foreground)", | ||||||
|  |       }, | ||||||
|  |     }, | ||||||
|  |   }, | ||||||
|  |   plugins: [], | ||||||
|  | }; | ||||||
|  | export default config; | ||||||
							
								
								
									
										26
									
								
								dewey/tsconfig.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										26
									
								
								dewey/tsconfig.json
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,26 @@ | |||||||
|  | { | ||||||
|  |   "compilerOptions": { | ||||||
|  |     "lib": ["dom", "dom.iterable", "esnext"], | ||||||
|  |     "allowJs": true, | ||||||
|  |     "skipLibCheck": true, | ||||||
|  |     "strict": true, | ||||||
|  |     "noEmit": true, | ||||||
|  |     "esModuleInterop": true, | ||||||
|  |     "module": "esnext", | ||||||
|  |     "moduleResolution": "bundler", | ||||||
|  |     "resolveJsonModule": true, | ||||||
|  |     "isolatedModules": true, | ||||||
|  |     "jsx": "preserve", | ||||||
|  |     "incremental": true, | ||||||
|  |     "plugins": [ | ||||||
|  |       { | ||||||
|  |         "name": "next" | ||||||
|  |       } | ||||||
|  |     ], | ||||||
|  |     "paths": { | ||||||
|  |       "@/*": ["./*"] | ||||||
|  |     } | ||||||
|  |   }, | ||||||
|  |   "include": ["next-env.d.ts", "**/*.ts", "**/*.tsx", ".next/types/**/*.ts"], | ||||||
|  |   "exclude": ["node_modules"] | ||||||
|  | } | ||||||
							
								
								
									
										391
									
								
								index.html
									
									
									
									
									
								
							
							
						
						
									
										391
									
								
								index.html
									
									
									
									
									
								
							| @ -9,6 +9,8 @@ | |||||||
|     <script src="https://cdn.jsdelivr.net/npm/moment@2.29.4/moment.min.js"></script> |     <script src="https://cdn.jsdelivr.net/npm/moment@2.29.4/moment.min.js"></script> | ||||||
|     <script src="https://cdn.jsdelivr.net/npm/chart.js"></script> |     <script src="https://cdn.jsdelivr.net/npm/chart.js"></script> | ||||||
|     <script src="https://cdn.jsdelivr.net/npm/chartjs-adapter-moment@1.0.1/dist/chartjs-adapter-moment.min.js"></script> |     <script src="https://cdn.jsdelivr.net/npm/chartjs-adapter-moment@1.0.1/dist/chartjs-adapter-moment.min.js"></script> | ||||||
|  |     <script src="https://cdnjs.cloudflare.com/ajax/libs/highlight.js/11.7.0/highlight.min.js"></script> | ||||||
|  |     <link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/highlight.js/11.7.0/styles/default.min.css"> | ||||||
|     <link href="https://fonts.googleapis.com/css2?family=Noto+Sans+Mono:wght@400;700&display=swap" rel="stylesheet"> |     <link href="https://fonts.googleapis.com/css2?family=Noto+Sans+Mono:wght@400;700&display=swap" rel="stylesheet"> | ||||||
|     <style> |     <style> | ||||||
|         body { |         body { | ||||||
| @ -86,13 +88,17 @@ | |||||||
|             color: #888; |             color: #888; | ||||||
|         } |         } | ||||||
|         .thought-summary { |         .thought-summary { | ||||||
|             cursor: pointer; |  | ||||||
|             color: #fff; |  | ||||||
|             margin-bottom: 5px; |  | ||||||
|             font-weight: bold; |             font-weight: bold; | ||||||
|             display: flex; |             margin-bottom: 5px; | ||||||
|             align-items: center; |             padding: 5px; | ||||||
|  |             border-radius: 3px; | ||||||
|         } |         } | ||||||
|  |         .thought-summary.plan { background-color: #2c3e50; } | ||||||
|  |         .thought-summary.decision { background-color: #34495e; } | ||||||
|  |         .thought-summary.tool_call { background-color: #16a085; } | ||||||
|  |         .thought-summary.tool_result { background-color: #27ae60; } | ||||||
|  |         .thought-summary.think_more { background-color: #2980b9; } | ||||||
|  |         .thought-summary.answer { background-color: #8e44ad; } | ||||||
|         .thought-details { |         .thought-details { | ||||||
|             display: none; |             display: none; | ||||||
|             margin-left: 20px; |             margin-left: 20px; | ||||||
| @ -238,11 +244,111 @@ | |||||||
|                 transform: translateX(0); |                 transform: translateX(0); | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|  |  | ||||||
|  |         .conversation-history-container { | ||||||
|  |             margin-top: 20px; | ||||||
|  |             background-color: #222; | ||||||
|  |             border-radius: 5px; | ||||||
|  |             padding: 10px; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         #conversation-history { | ||||||
|  |             color: #fff; | ||||||
|  |             font-family: 'Noto Sans Mono', monospace; | ||||||
|  |             font-size: 12px; | ||||||
|  |         } | ||||||
|  |         .history-card { | ||||||
|  |             background-color: #2c3e50; | ||||||
|  |             border-radius: 5px; | ||||||
|  |             padding: 10px; | ||||||
|  |             margin-bottom: 10px; | ||||||
|  |         } | ||||||
|  |         .history-role { | ||||||
|  |             font-weight: bold; | ||||||
|  |             margin-bottom: 5px; | ||||||
|  |         } | ||||||
|  |         .history-content { | ||||||
|  |             white-space: pre-wrap; | ||||||
|  |             word-break: break-word; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         .error-message { | ||||||
|  |             background-color: #ff6b6b; | ||||||
|  |             color: #fff; | ||||||
|  |             padding: 10px; | ||||||
|  |             border-radius: 5px; | ||||||
|  |             margin-bottom: 10px; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         .retrying { | ||||||
|  |             background-color: #feca57; | ||||||
|  |             color: #333; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         #clear-history-button { | ||||||
|  |             background-color: #e74c3c; | ||||||
|  |             color: white; | ||||||
|  |             border: none; | ||||||
|  |             padding: 10px; | ||||||
|  |             margin-bottom: 10px; | ||||||
|  |             cursor: pointer; | ||||||
|  |             font-family: 'Noto Sans Mono', monospace; | ||||||
|  |             font-size: 14px; | ||||||
|  |             border-radius: 5px; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         #clear-history-button:hover { | ||||||
|  |             background-color: #c0392b; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         #chat-tabs { | ||||||
|  |             display: flex; | ||||||
|  |             background-color: #222; | ||||||
|  |             padding: 10px 10px 0 10px; | ||||||
|  |         } | ||||||
|  |          | ||||||
|  |         .chat-tab { | ||||||
|  |             background-color: #444; | ||||||
|  |             color: #fff; | ||||||
|  |             border: none; | ||||||
|  |             padding: 10px 20px; | ||||||
|  |             margin-right: 5px; | ||||||
|  |             cursor: pointer; | ||||||
|  |             border-top-left-radius: 5px; | ||||||
|  |             border-top-right-radius: 5px; | ||||||
|  |         } | ||||||
|  |          | ||||||
|  |         .chat-tab.active { | ||||||
|  |             background-color: #666; | ||||||
|  |         } | ||||||
|  |          | ||||||
|  |         #new-chat-button { | ||||||
|  |             background-color: #27ae60; | ||||||
|  |             color: #fff; | ||||||
|  |             border: none; | ||||||
|  |             padding: 10px 20px; | ||||||
|  |             cursor: pointer; | ||||||
|  |             border-top-left-radius: 5px; | ||||||
|  |             border-top-right-radius: 5px; | ||||||
|  |         } | ||||||
|  |          | ||||||
|  |         .close-tab { | ||||||
|  |             margin-left: 10px; | ||||||
|  |             color: #ff6b6b; | ||||||
|  |             cursor: pointer; | ||||||
|  |         } | ||||||
|  |          | ||||||
|  |         .thinking-section { | ||||||
|  |             margin-bottom: 20px; | ||||||
|  |             border-left: 2px solid #444; | ||||||
|  |             padding-left: 10px; | ||||||
|  |         } | ||||||
|     </style> |     </style> | ||||||
| </head> | </head> | ||||||
| <body> | <body> | ||||||
|     <div id="main-container"> |     <div id="main-container"> | ||||||
|         <div id="chat-area"> |         <div id="chat-area"> | ||||||
|  |             <div id="chat-tabs"></div> | ||||||
|             <div id="chat-container"></div> |             <div id="chat-container"></div> | ||||||
|             <div id="input-container" class="pdp-panel"> |             <div id="input-container" class="pdp-panel"> | ||||||
|                 <div class="pdp-label">INPUT:</div> |                 <div class="pdp-label">INPUT:</div> | ||||||
| @ -272,6 +378,12 @@ | |||||||
|                 <div class="graph-title">GPU Memory</div> |                 <div class="graph-title">GPU Memory</div> | ||||||
|                 <canvas id="gpuMemoryChart"></canvas> |                 <canvas id="gpuMemoryChart"></canvas> | ||||||
|             </div> |             </div> | ||||||
|  |              | ||||||
|  |             <!-- Add this new section for conversation history --> | ||||||
|  |             <div class="conversation-history-container"> | ||||||
|  |                 <div class="graph-title">Conversation History</div> | ||||||
|  |                 <div id="conversation-history"></div> | ||||||
|  |             </div> | ||||||
|         </div> |         </div> | ||||||
|     </div> |     </div> | ||||||
|  |  | ||||||
| @ -280,10 +392,80 @@ | |||||||
|         const chatContainer = document.getElementById('chat-container'); |         const chatContainer = document.getElementById('chat-container'); | ||||||
|         const userInput = document.getElementById('user-input'); |         const userInput = document.getElementById('user-input'); | ||||||
|         const sendButton = document.getElementById('send-button'); |         const sendButton = document.getElementById('send-button'); | ||||||
|  |         const chatTabs = document.getElementById('chat-tabs'); | ||||||
|  |  | ||||||
|         let thinkingElement = null; |         let currentChatId = null; | ||||||
|         let thinkingDetails = null; |         let chats = {}; | ||||||
|         let thinkingStartTime = null; |  | ||||||
|  |         function createNewChat() { | ||||||
|  |             const chatId = Date.now().toString(); | ||||||
|  |             chats[chatId] = { | ||||||
|  |                 messages: [], | ||||||
|  |                 thinkingSections: [] | ||||||
|  |             }; | ||||||
|  |             addChatTab(chatId); | ||||||
|  |             switchToChat(chatId); | ||||||
|  |             saveChats(); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         function addChatTab(chatId) { | ||||||
|  |             const tab = document.createElement('button'); | ||||||
|  |             tab.classList.add('chat-tab'); | ||||||
|  |             tab.textContent = `Chat ${Object.keys(chats).length}`; | ||||||
|  |             tab.onclick = () => switchToChat(chatId); | ||||||
|  |  | ||||||
|  |             const closeButton = document.createElement('span'); | ||||||
|  |             closeButton.classList.add('close-tab'); | ||||||
|  |             closeButton.textContent = '×'; | ||||||
|  |             closeButton.onclick = (e) => { | ||||||
|  |                 e.stopPropagation(); | ||||||
|  |                 closeChat(chatId); | ||||||
|  |             }; | ||||||
|  |  | ||||||
|  |             tab.appendChild(closeButton); | ||||||
|  |             chatTabs.insertBefore(tab, chatTabs.lastElementChild); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         function switchToChat(chatId) { | ||||||
|  |             currentChatId = chatId; | ||||||
|  |             document.querySelectorAll('.chat-tab').forEach(tab => tab.classList.remove('active')); | ||||||
|  |             document.querySelector(`.chat-tab:nth-child(${Object.keys(chats).indexOf(chatId) + 1})`).classList.add('active'); | ||||||
|  |             renderChat(chatId); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         function closeChat(chatId) { | ||||||
|  |             delete chats[chatId]; | ||||||
|  |             saveChats(); | ||||||
|  |             const tabToRemove = Array.from(chatTabs.children).find(tab => tab.textContent.includes(`Chat ${Object.keys(chats).indexOf(chatId) + 1}`)); | ||||||
|  |             if (tabToRemove) { | ||||||
|  |                 chatTabs.removeChild(tabToRemove); | ||||||
|  |             } | ||||||
|  |             if (currentChatId === chatId) { | ||||||
|  |                 const remainingChatIds = Object.keys(chats); | ||||||
|  |                 if (remainingChatIds.length > 0) { | ||||||
|  |                     switchToChat(remainingChatIds[0]); | ||||||
|  |                 } else { | ||||||
|  |                     createNewChat(); | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         function renderChat(chatId) { | ||||||
|  |             chatContainer.innerHTML = ''; | ||||||
|  |             const chat = chats[chatId]; | ||||||
|  |             chat.messages.forEach(message => addMessage(message.content, message.isUser)); | ||||||
|  |             chat.thinkingSections.forEach(section => { | ||||||
|  |                 const thinkingSection = createThinkingSection(); | ||||||
|  |                 section.thoughts.forEach(thought => addThought(thought.type, thought.content, thought.details, thinkingSection)); | ||||||
|  |             }); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         function createThinkingSection() { | ||||||
|  |             const section = document.createElement('div'); | ||||||
|  |             section.classList.add('thinking-section'); | ||||||
|  |             chatContainer.appendChild(section); | ||||||
|  |             return section; | ||||||
|  |         } | ||||||
|  |  | ||||||
|         function addMessage(message, isUser) { |         function addMessage(message, isUser) { | ||||||
|             const messageElement = document.createElement('div'); |             const messageElement = document.createElement('div'); | ||||||
| @ -292,65 +474,40 @@ | |||||||
|             messageElement.innerHTML = isUser ? message : marked.parse(message); |             messageElement.innerHTML = isUser ? message : marked.parse(message); | ||||||
|             chatContainer.appendChild(messageElement); |             chatContainer.appendChild(messageElement); | ||||||
|             chatContainer.scrollTop = chatContainer.scrollHeight; |             chatContainer.scrollTop = chatContainer.scrollHeight; | ||||||
|  |  | ||||||
|  |             if (currentChatId) { | ||||||
|  |                 chats[currentChatId].messages.push({ content: message, isUser: isUser }); | ||||||
|  |                 saveChats(); | ||||||
|  |             } | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         function startThinking() { |         function addThought(type, content, details = '', thinkingSection) { | ||||||
|             thinkingElement = document.createElement('div'); |  | ||||||
|             thinkingElement.classList.add('thought-summary', 'collapsible'); |  | ||||||
|              |  | ||||||
|             const led = document.createElement('div'); |  | ||||||
|             led.classList.add('led', 'blinking'); |  | ||||||
|              |  | ||||||
|             const textNode = document.createTextNode('Thinking...'); |  | ||||||
|              |  | ||||||
|             thinkingElement.appendChild(led); |  | ||||||
|             thinkingElement.appendChild(textNode); |  | ||||||
|             thinkingElement.onclick = toggleThinkingDetails; |  | ||||||
|              |  | ||||||
|             thinkingDetails = document.createElement('div'); |  | ||||||
|             thinkingDetails.classList.add('thought-details'); |  | ||||||
|              |  | ||||||
|             chatContainer.appendChild(thinkingElement); |  | ||||||
|             chatContainer.appendChild(thinkingDetails); |  | ||||||
|              |  | ||||||
|             thinkingStartTime = Date.now(); |  | ||||||
|             chatContainer.scrollTop = chatContainer.scrollHeight; |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         function addThought(step, content) { |  | ||||||
|             if (thinkingDetails) { |  | ||||||
|             const stepElement = document.createElement('div'); |             const stepElement = document.createElement('div'); | ||||||
|                 stepElement.classList.add('thought-summary', 'collapsible'); |             stepElement.classList.add('thought-summary', 'collapsible', type); | ||||||
|                 stepElement.textContent = step; |             stepElement.textContent = type.charAt(0).toUpperCase() + type.slice(1).replace('_', ' ') + ':'; | ||||||
|             stepElement.onclick = toggleStepDetails; |             stepElement.onclick = toggleStepDetails; | ||||||
|  |  | ||||||
|             const stepDetails = document.createElement('div'); |             const stepDetails = document.createElement('div'); | ||||||
|             stepDetails.classList.add('thought-details'); |             stepDetails.classList.add('thought-details'); | ||||||
|                 stepDetails.innerHTML = content; |  | ||||||
|              |              | ||||||
|                 thinkingDetails.appendChild(stepElement); |             if (type === 'error') { | ||||||
|                 thinkingDetails.appendChild(stepDetails); |                 stepElement.classList.add('error-message'); | ||||||
|  |                 if (content.includes('retrying')) { | ||||||
|  |                     stepElement.classList.add('retrying'); | ||||||
|  |                 } | ||||||
|  |                 stepDetails.innerHTML = marked.parse(content + '\n\nDetails:\n```\n' + details + '\n```'); | ||||||
|  |             } else { | ||||||
|  |                 stepDetails.innerHTML = marked.parse(content); | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             thinkingSection.appendChild(stepElement); | ||||||
|  |             thinkingSection.appendChild(stepDetails); | ||||||
|             chatContainer.scrollTop = chatContainer.scrollHeight; |             chatContainer.scrollTop = chatContainer.scrollHeight; | ||||||
|             } |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         function endThinking(thinkingTime) { |             if (currentChatId) { | ||||||
|             if (thinkingElement) { |                 const currentThinkingSection = chats[currentChatId].thinkingSections[chats[currentChatId].thinkingSections.length - 1]; | ||||||
|                 const textNode = thinkingElement.childNodes[1]; |                 currentThinkingSection.thoughts.push({ type, content, details }); | ||||||
|                 textNode.nodeValue = `Thinking... (${thinkingTime}s)`; |                 saveChats(); | ||||||
|                 const led = thinkingElement.querySelector('.led'); |  | ||||||
|                 led.classList.remove('blinking'); |  | ||||||
|                 led.style.backgroundColor = '#0f0'; |  | ||||||
|                 led.style.boxShadow = '0 0 10px #0f0'; |  | ||||||
|                 thinkingStartTime = null; |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         function toggleThinkingDetails() { |  | ||||||
|             this.classList.toggle('open'); |  | ||||||
|             const details = this.nextElementSibling; |  | ||||||
|             if (details) { |  | ||||||
|                 details.style.display = details.style.display === 'none' ? 'block' : 'none'; |  | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|  |  | ||||||
| @ -362,34 +519,71 @@ | |||||||
|             } |             } | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         socket.on('thinking', (data) => { |         function saveChats() { | ||||||
|             if (!thinkingElement) startThinking(); |             localStorage.setItem('chats', JSON.stringify(chats)); | ||||||
|             addThought(data.step, 'Started'); |         } | ||||||
|         }); |  | ||||||
|  |  | ||||||
|         socket.on('thought', (data) => { |         function loadChats() { | ||||||
|             addThought('Result', data.content); |             const storedChats = localStorage.getItem('chats'); | ||||||
|         }); |             if (storedChats) { | ||||||
|  |                 chats = JSON.parse(storedChats); | ||||||
|         socket.on('chat_response', (data) => { |                 Object.keys(chats).forEach(chatId => addChatTab(chatId)); | ||||||
|             endThinking(data.thinking_time); |                 if (Object.keys(chats).length > 0) { | ||||||
|             addMessage(data.response, false); |                     switchToChat(Object.keys(chats)[0]); | ||||||
|         }); |                 } else { | ||||||
|  |                     createNewChat(); | ||||||
|         socket.on('error', (data) => { |                 } | ||||||
|             endThinking(data.thinking_time); |             } else { | ||||||
|             addMessage(`Error: ${data.message}`, false); |                 createNewChat(); | ||||||
|         }); |             } | ||||||
|  |         } | ||||||
|  |  | ||||||
|         function sendMessage() { |         function sendMessage() { | ||||||
|             const message = userInput.value.trim(); |             const message = userInput.value.trim(); | ||||||
|             if (message) { |             if (message && currentChatId) { | ||||||
|                 addMessage(message, true); |                 addMessage(message, true); | ||||||
|                 socket.emit('chat_request', { message: message }); |                 chats[currentChatId].thinkingSections.push({ thoughts: [] }); | ||||||
|  |                 socket.emit('chat_request', {  | ||||||
|  |                     message: message, | ||||||
|  |                     conversation_history: chats[currentChatId].messages.filter(m => !m.isUser).map(m => ({ role: 'assistant', content: m.content })) | ||||||
|  |                         .concat(chats[currentChatId].messages.filter(m => m.isUser).map(m => ({ role: 'user', content: m.content }))) | ||||||
|  |                 }); | ||||||
|                 userInput.value = ''; |                 userInput.value = ''; | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|  |  | ||||||
|  |         socket.on('thinking', (data) => { | ||||||
|  |             if (currentChatId) { | ||||||
|  |                 const newThinkingSection = createThinkingSection(); | ||||||
|  |                 chats[currentChatId].thinkingSections.push({ thoughts: [] }); | ||||||
|  |                 addThought(data.step, 'Started', '', newThinkingSection); | ||||||
|  |             } | ||||||
|  |         }); | ||||||
|  |  | ||||||
|  |         socket.on('thought', (data) => { | ||||||
|  |             if (currentChatId) { | ||||||
|  |                 const currentThinkingSection = chatContainer.querySelector('.thinking-section:last-child'); | ||||||
|  |                 addThought(data.type, data.content, data.details, currentThinkingSection); | ||||||
|  |             } | ||||||
|  |         }); | ||||||
|  |  | ||||||
|  |         socket.on('chat_response', (data) => { | ||||||
|  |             if (currentChatId) { | ||||||
|  |                 addMessage(data.response, false); | ||||||
|  |             } | ||||||
|  |         }); | ||||||
|  |  | ||||||
|  |         socket.on('error', (data) => { | ||||||
|  |             if (currentChatId) { | ||||||
|  |                 const currentThinkingSection = chatContainer.querySelector('.thinking-section:last-child'); | ||||||
|  |                 if (data.type === 'retrying') { | ||||||
|  |                     addThought('error', data.content, '', currentThinkingSection); | ||||||
|  |                 } else { | ||||||
|  |                     addThought('error', data.message, '', currentThinkingSection); | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         }); | ||||||
|  |  | ||||||
|         sendButton.addEventListener('click', sendMessage); |         sendButton.addEventListener('click', sendMessage); | ||||||
|         userInput.addEventListener('keypress', function(e) { |         userInput.addEventListener('keypress', function(e) { | ||||||
|             if (e.key === 'Enter' && !e.shiftKey) { |             if (e.key === 'Enter' && !e.shiftKey) { | ||||||
| @ -398,6 +592,16 @@ | |||||||
|             } |             } | ||||||
|         }); |         }); | ||||||
|  |  | ||||||
|  |         // Add new chat button | ||||||
|  |         const newChatButton = document.createElement('button'); | ||||||
|  |         newChatButton.id = 'new-chat-button'; | ||||||
|  |         newChatButton.textContent = '+ New Chat'; | ||||||
|  |         newChatButton.onclick = createNewChat; | ||||||
|  |         chatTabs.appendChild(newChatButton); | ||||||
|  |  | ||||||
|  |         // Load chats when the page loads | ||||||
|  |         loadChats(); | ||||||
|  |  | ||||||
|         const chartOptions = { |         const chartOptions = { | ||||||
|             type: 'line', |             type: 'line', | ||||||
|             options: { |             options: { | ||||||
| @ -570,6 +774,41 @@ | |||||||
|  |  | ||||||
|         window.addEventListener('resize', checkWindowSize); |         window.addEventListener('resize', checkWindowSize); | ||||||
|         checkWindowSize(); // Initial check |         checkWindowSize(); // Initial check | ||||||
|  |  | ||||||
|  |         // Add this new function to update the conversation history | ||||||
|  |         function updateConversationHistory(history) { | ||||||
|  |             const conversationHistoryElement = document.getElementById('conversation-history'); | ||||||
|  |             conversationHistoryElement.innerHTML = ''; | ||||||
|  |              | ||||||
|  |             history.forEach(item => { | ||||||
|  |                 const card = document.createElement('div'); | ||||||
|  |                 card.classList.add('history-card'); | ||||||
|  |                  | ||||||
|  |                 const role = document.createElement('div'); | ||||||
|  |                 role.classList.add('history-role'); | ||||||
|  |                 role.textContent = item.role.charAt(0).toUpperCase() + item.role.slice(1); | ||||||
|  |                  | ||||||
|  |                 const content = document.createElement('pre'); | ||||||
|  |                 content.classList.add('history-content'); | ||||||
|  |                 content.innerHTML = hljs.highlightAuto(item.content).value; | ||||||
|  |                  | ||||||
|  |                 card.appendChild(role); | ||||||
|  |                 card.appendChild(content); | ||||||
|  |                 conversationHistoryElement.appendChild(card); | ||||||
|  |             }); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         // Add this new socket listener | ||||||
|  |         socket.on('conversation_history', (data) => { | ||||||
|  |             updateConversationHistory(data.history); | ||||||
|  |         }); | ||||||
|  |  | ||||||
|  |         // Add event listener for the clear history button | ||||||
|  |         clearHistoryButton.addEventListener('click', () => { | ||||||
|  |             if (confirm('Are you sure you want to clear the conversation history?')) { | ||||||
|  |                 clearConversationHistory(); | ||||||
|  |             } | ||||||
|  |         }); | ||||||
|     </script> |     </script> | ||||||
| </body> | </body> | ||||||
| </html> | </html> | ||||||
							
								
								
									
										267
									
								
								main.py
									
									
									
									
									
								
							
							
						
						
									
										267
									
								
								main.py
									
									
									
									
									
								
							| @ -1,4 +1,4 @@ | |||||||
| from flask import Flask, send_from_directory | from flask import Flask, send_from_directory, request | ||||||
| from flask_socketio import SocketIO, emit | from flask_socketio import SocketIO, emit | ||||||
| from flask_openapi3 import OpenAPI, Info | from flask_openapi3 import OpenAPI, Info | ||||||
| from pydantic import BaseModel | from pydantic import BaseModel | ||||||
| @ -10,15 +10,20 @@ import psutil | |||||||
| import GPUtil | import GPUtil | ||||||
| import threading | import threading | ||||||
| import os | import os | ||||||
|  | from tools import DefaultToolManager | ||||||
|  | import ollama | ||||||
|  | import re | ||||||
|  | import json | ||||||
|  | from datetime import datetime | ||||||
|  | import pprint | ||||||
| logger = structlog.get_logger() | logger = structlog.get_logger() | ||||||
|  |  | ||||||
| openapi = OpenAPI(__name__, info=Info(title="LLM Chat Server", version="1.0.0")) | openapi = OpenAPI(__name__, info=Info(title="LLM Chat Server", version="1.0.0")) | ||||||
| app = openapi | app = openapi | ||||||
| socketio = SocketIO(app, cors_allowed_origins="*") | socketio = SocketIO(app, cors_allowed_origins="*") | ||||||
|  |  | ||||||
|  | tool_manager = DefaultToolManager() | ||||||
|  |  | ||||||
| @app.route('/') | @app.route('/') | ||||||
| def index(): | def index(): | ||||||
|     logger.info("Serving index.html") |     logger.info("Serving index.html") | ||||||
| @ -33,51 +38,13 @@ class ChatResponse(BaseModel): | |||||||
| @socketio.on('chat_request') | @socketio.on('chat_request') | ||||||
| def handle_chat_request(data): | def handle_chat_request(data): | ||||||
|     user_input = data['message'] |     user_input = data['message'] | ||||||
|     logger.info("Received chat request", user_input=user_input) |     conversation_history = data.get('conversation_history', []) | ||||||
|  |     conversation_history = [{"role": "system", "content": ANSWER_QUESTION_PROMPT}] + conversation_history | ||||||
|  |     logger.info("Received chat request", user_input=user_input, conversation_history=conversation_history) | ||||||
|      |      | ||||||
|     start_time = time.time() |     start_time = time.time() | ||||||
|     full_context = "" |  | ||||||
|     try: |     try: | ||||||
|         # Step 1: Generate a plan using the initial LLM |         final_response = answer_question_tools(user_input, conversation_history) | ||||||
|         emit('thinking', {'step': 'Generating plan'}) |  | ||||||
|         plan, plan_generation = generate_plan(user_input) |  | ||||||
|         full_context += f"Plan Thinking:\n{plan_generation}" |  | ||||||
|         full_context += f"Plan:\n{plan}" |  | ||||||
|         emit('thought', {'content': f"Plan Thinking:\n{plan_generation}"}) |  | ||||||
|         emit('thought', {'content': f"Plan:\n{plan}"}) |  | ||||||
|  |  | ||||||
|         if plan[0].strip().lower() == "direct_answer": |  | ||||||
|             final_response = plan[1] |  | ||||||
|             thinking_time = round(time.time() - start_time, 2) |  | ||||||
|             emit('chat_response', { |  | ||||||
|                 'response': final_response, |  | ||||||
|                 'thinking_time': thinking_time |  | ||||||
|             }) |  | ||||||
|             return |  | ||||||
|          |  | ||||||
|         # Step 2: Execute each step of the plan |  | ||||||
|         step_results = [] |  | ||||||
|         for i, step in enumerate(plan): |  | ||||||
|             emit('thinking', {'step': f'Executing step {i+1}'}) |  | ||||||
|             while True: |  | ||||||
|                 best_model, model_selection = select_best_model(step, step_results, full_context) |  | ||||||
|                 if best_model in model_manager.model_capabilities: |  | ||||||
|                     break |  | ||||||
|                 logger.warning(f"Selected model {best_model} is not in the list of available models. Retrying...") |  | ||||||
|             emit('thought', {'content': f"Selected model for step {i+1}:\n{model_selection}"}) |  | ||||||
|             # summary, summary_generation = summarize_context(f"Plan: {plan}\n\nSteps: {step_results}") |  | ||||||
|             # emit('thought', {'content': f"Context summary:\n{summary_generation}"}) |  | ||||||
|             step_result, step_execution = execute_step(step, best_model, step_results, full_context) |  | ||||||
|             emit('thought', {'content': f"Step {i+1} result:\n{step_execution}"}) |  | ||||||
|             emit('thought', {'content': f"Result {i+1}:\n{step_result}"}) |  | ||||||
|             step_results.append(step_result) |  | ||||||
|             full_context += f"Step {i+1} result:\n{step_execution}" |  | ||||||
|          |  | ||||||
|         # Step 3: Generate final response |  | ||||||
|         emit('thinking', {'step': 'Generating final response'}) |  | ||||||
|         final_response, final_generation = generate_final_response(user_input, plan, step_results) |  | ||||||
|         emit('thought', {'content': f"Final response generation:\n{final_generation}"}) |  | ||||||
|          |  | ||||||
|         end_time = time.time() |         end_time = time.time() | ||||||
|         thinking_time = round(end_time - start_time, 2) |         thinking_time = round(end_time - start_time, 2) | ||||||
|          |          | ||||||
| @ -94,146 +61,98 @@ def handle_chat_request(data): | |||||||
|             'thinking_time': thinking_time |             'thinking_time': thinking_time | ||||||
|         }) |         }) | ||||||
|  |  | ||||||
| PLAN_GENERATE_PROMPT = """ | def answer_question_tools(user_input: str, conversation_history: List[dict], max_retries: int = 100): | ||||||
| You are building a "chain of thought" workflow for a series of LLMs to complete a task provided by a user. |     global tool_manager | ||||||
| Your first task is to "think" through the problem provided by the user. Probe what it would take to complete the task, see if there are hidden nuances, what constrains might be relevant, how to be efficient. |  | ||||||
| This thinking should set question the premise of the task, and sets the scene for a plan of attack to be created. |  | ||||||
| Verbalize your thoughts out loud, allow the user to see your thought process. This thought process will also be used as context for processing the generated plan. |  | ||||||
| This thought process should mimic the process of a human, and not be a simple list of steps, but should be a narrative of thought that a human would have. |  | ||||||
| Each step in the formulated plan is a step that a seperate LLM will complete. The LLM that will complete the step will be selected based on the scope of the step and the capabilities of the available models. |  | ||||||
| There are models that are good at coding and math, and there are models that are good at reasoning and planning. Some models that are generalists, multilingual, or conversational. And even some that are vision models. |  | ||||||
| Use this context of the possible models to shape each step such that a LLM can complete the step given the step and some context. |  | ||||||
| Steps should follow a logical "chain of thought" in order to best complete the overall task. |  | ||||||
| Steps should be self contained and be designed such that the results of one step can be passed on to the next step. |  | ||||||
| Steps should be phrased in such a way that it acts as a prompt or instruction to the LLM that will complete the step. |  | ||||||
| Each step will return a result, and a thought process. The thought process is extremely important, it is the "chain of thought" that the LLM went through to complete the step. This thought process is critical for the next step in the plan. |  | ||||||
| Consider how results from one step can be combined with results from another step and consider how the chain of thought from one step can inform the next step when designing each step. |  | ||||||
| Try and minimize the number of steps required to complete the task since running a lot of steps is expensive.  |  | ||||||
| Your output should be your thought process, followed by a single line titled "STEPS", followed by each step to take, one step per line. |  | ||||||
| Do not add any sort of markdown formatting, code formatting, or any other formatting. |  | ||||||
| Do not add any preamble, postamble, or other text, only the thought process and the steps. |  | ||||||
|      |      | ||||||
| Consider the following example: |     # If conversation_history is empty, initialize it with the system prompt | ||||||
|  |     if not conversation_history: | ||||||
|  |         conversation_history = [ | ||||||
|  |             {"role": "system", "content": ANSWER_QUESTION_PROMPT}, | ||||||
|  |         ] | ||||||
|      |      | ||||||
| Prompt: Write a program to reverse a string, then output ASCII block art of that reversed string. Do this in python. |     logger.info("Starting chat", user_input=user_input, conversation_history=conversation_history) | ||||||
|  |     # Add the new user input to the conversation history | ||||||
|  |     conversation_history.append({"role": "user", "content": user_input}) | ||||||
|      |      | ||||||
| So there are two parts to this task. First, we need to reverse the input string. Then we need to print the ASCII block art for each character in the reversed string. |     emit('thinking', {'step': 'Starting'}) | ||||||
| We should be able to reverse the string using either a simple loop, or a python slice. Slicing is simpler, so we should use that. |     emit('conversation_history', {'history': conversation_history}) | ||||||
| For the ASCII block art, the challenge is in creating a mapping between each character and its block art representation. There are a few ways to go about this: |  | ||||||
|  - Find a library that converts text to block art |  | ||||||
|  - Create our own mapping from characters to block art |  | ||||||
|  - Create a procedurally generated mapping from characters to block art |  | ||||||
| Procedural generation could be done with an algorithm, but coming up with a good algorithm could be challenging. |  | ||||||
| Generating a dictionary could be a good approach, but there are 26 letters in the alphabet, and 10 digits, so we would need 36 different outputs for the block art. |  | ||||||
| We should search for a library that already does this, import it, and call it on the result of the string reversal. We would also need to tell the user to install the library. |  | ||||||
|  |  | ||||||
| We're now ready to create our plan. |     for iteration in range(max_retries): | ||||||
|  |         response = ollama.chat(model=PRIMARY_MODEL, messages=conversation_history, tools=tool_manager.get_tools_for_ollama_dict(), stream=False) | ||||||
|  |         assistant_message = response['message'] | ||||||
|          |          | ||||||
| STEPS |         conversation_history.append(assistant_message) | ||||||
| 1. Write a function that takes a string and reverses it. |         emit('conversation_history', {'history': conversation_history}) | ||||||
| 2. Write a function that takes a string and returns the ASCII block art for each character in the string, this must be done using a library. |         pprint.pp(assistant_message) | ||||||
| 3. Combine the two functions into a single program. |  | ||||||
|  |  | ||||||
| --- |         if 'tool_calls' in assistant_message: | ||||||
|  |             emit('thought', {'type': 'decision', 'content': "Tool Call\n\n" + assistant_message['content']}) | ||||||
|  |             for tool_call in assistant_message['tool_calls']: | ||||||
|  |                 tool_name = tool_call['function']['name'] | ||||||
|  |                 tool_args = tool_call['function']['arguments'] | ||||||
|  |                 emit('thought', {'type': 'tool_call', 'content': f"Tool: {tool_name}\nArguments: {tool_args}"}) | ||||||
|  |                 tool_response = tool_manager.get_tool(tool_name).execute(tool_args) | ||||||
|  |                 conversation_history.append({ | ||||||
|  |                     "role": "tool", | ||||||
|  |                     "content": tool_response | ||||||
|  |                 }) | ||||||
|  |                 emit('conversation_history', {'history': conversation_history}) | ||||||
|  |                 emit('thought', {'type': 'tool_result', 'content': tool_response}) | ||||||
|  |  | ||||||
| Now you try. |             reflection_prompt = "Reflect on the tool results. If there were any errors, propose multiple alternative approaches to solve the problem. If successful, consider if the result fully answers the user's query or if additional steps are needed." | ||||||
| """ |             conversation_history.append({ | ||||||
| _REMINADER_PT =""" |                 "role": "assistant", | ||||||
| Each task you create should be should be self contained and be designed such that the results of one step can be passed on to the next step.  |                 "content": reflection_prompt | ||||||
| Try and minimize the number of steps required to complete the task.  |             }) | ||||||
| Output only a numbered list of steps, each step should be a seperate line. |             emit('conversation_history', {'history': conversation_history}) | ||||||
| Do not output any preamble or other text, only the list of steps. |         else: | ||||||
| If you think a task can be completed by a single step, then you can output a single step.  |             if "<answer>" in assistant_message['content'].lower(): | ||||||
| If you can directly answer the question, you must begin your response with a single line containing the text "DIRECT_ANSWER" and then provide the answer to the question on the next line. |                 answer_content = re.search(r'<answer>(.*?)</answer>', assistant_message['content'], re.DOTALL) | ||||||
|  |                 if answer_content: | ||||||
|  |                     final_answer = answer_content.group(1).strip() | ||||||
|  |                     emit('thought', {'type': 'answer', 'content': final_answer}) | ||||||
|  |                     return final_answer | ||||||
|  |             else: | ||||||
|  |                 emit('thought', {'type': 'decision', 'content': "Think/Plan/Decision/Action\n\n" + assistant_message['content']}) | ||||||
|  |                 reflection_prompt = "Your last response didn't provide a final answer. Please reflect on your current understanding of the problem and consider if you need to use any tools or if you can now provide a final answer. If you're ready to give a final answer, put your response in tags <answer></answer>" | ||||||
|  |                 conversation_history.append({"role": "assistant", "content": reflection_prompt}) | ||||||
|  |                 emit('conversation_history', {'history': conversation_history}) | ||||||
|  |  | ||||||
| Here are some samples: |     return f"Max iterations reached. Last response: {assistant_message['content']}" | ||||||
|  |  | ||||||
| Input: Write a program to reverse a string, then output the ASCII art of that reversed string. Do this in python. | ANSWER_QUESTION_PROMPT = f""" | ||||||
| Steps: | The current date is {datetime.now().strftime("%A, %B %d, %Y")}, your knowledge cutoff was December 2023. | ||||||
| 1. Define a template for a program that prints the ASCII art of the reversed string. | You are Dewey, an AI assistant with access to external tools and the ability to think through complex problems. Your role is to assist users by leveraging tools when necessary, thinking deeply about problems, and providing accurate and helpful information, all with a cheerful, but witty personality. Here are the tools available to you: | ||||||
| 2. Fill in the logic to reverse the string. |  | ||||||
| 3. Fill in the logic to print the ASCII art of the reversed string. |  | ||||||
| 4. Output the final program. |  | ||||||
|  |  | ||||||
| Input: What are the oceans of the world? | {tool_manager.get_tools_and_descriptions_for_prompt()} | ||||||
| Steps: |  | ||||||
| 1. Use the encyclopedia tool to get the page on the oceans of the world, parse, and output the results. |  | ||||||
|  |  | ||||||
| Input: What is the perfect gas law? | When addressing a query, follow these steps: | ||||||
| Steps: |  | ||||||
| DIRECT_ANSWER | 1. Analyze: Thoroughly analyze the query and consider multiple approaches to solving it. | ||||||
| The perfect gas law is the equation of state of a hypothetical ideal gas. The formula is $$PV = nRT$$ where P is pressure, V is volume, n is the number of moles, R is the ideal gas constant, and T is temperature. |  | ||||||
|  | 2. Plan: Develop a plan of action, considering whether you need to use any tools or if you can answer directly. | ||||||
|  |  | ||||||
|  | 3. Execute: If you need to use a tool, call it as you would a function. If not, proceed with your reasoning. | ||||||
|  |  | ||||||
|  | 4. Reflect: After each step or tool use, reflect on the results: | ||||||
|  |    - If successful, consider if the result fully answers the user's query or if additional steps are needed. | ||||||
|  |    - If there were errors or the result is unsatisfactory, don't give up! Use Tree of Thoughts reasoning: | ||||||
|  |      a) Generate multiple alternative approaches or modifications to your previous approach. | ||||||
|  |      b) Briefly evaluate the potential of each alternative. | ||||||
|  |      c) Choose the most promising alternative and execute it. | ||||||
|  |      d) Repeat this process if needed, building upon your growing understanding of the problem. | ||||||
|  |      e) You cannot return a final answer after an error using a tool, you must try again. | ||||||
|  |  | ||||||
|  | 5. Iterate: Continue this process of execution and reflection, exploring different branches of thought as needed. | ||||||
|  |  | ||||||
|  | 6. Conclude: When you believe you have a comprehensive answer to the user's query, provide your final answer. | ||||||
|  |  | ||||||
|  | Always explain your thought process, including your reasoning for each decision and how you arrived at your conclusions. If you're providing a final answer, put your response in tags <answer></answer>. | ||||||
|  |  | ||||||
|  | Remember, complex problems often require multiple steps and iterations. Don't hesitate to break down the problem, use tools multiple times, or explore different approaches to arrive at the best solution. | ||||||
| """ | """ | ||||||
|  |  | ||||||
| def generate_plan(user_input: str) -> tuple[List[str], str]: | PRIMARY_MODEL = "llama3.1:8b" | ||||||
|     logger.debug("Generating plan", prompt=user_input, system=PLAN_GENERATE_PROMPT) |  | ||||||
|     response = model_manager.generate_text("qwen2.5:7b", user_input, max_length=1024, system=PLAN_GENERATE_PROMPT) |  | ||||||
|     plan = response.split("STEPS")[1].strip() |  | ||||||
|     response_no_steps = response.split("STEPS")[0].strip() |  | ||||||
|     return [step.strip() for step in plan.split("\n") if step.strip()], response_no_steps |  | ||||||
|  |  | ||||||
|  |  | ||||||
| SELECT_BEST_MODEL_PROMPT = f""" |  | ||||||
| You are a large language model whos job it is to evaluate a step that is part of a larger plan, and determine what LLM would be best suited to complete the step based on the capabilities of the LLM. |  | ||||||
|  |  | ||||||
| The LLMs and their capabilities are as follows: |  | ||||||
| {"\n".join([f"{k}: {','.join(v)}" for k,v in model_manager.model_capabilities.items()])} |  | ||||||
|  |  | ||||||
| You will be provided with the current step of execution, the results of the previous steps in order, and the current chain of thought so far. |  | ||||||
| If the chain of thought is too long, a summary of the current chain of thought will be provided. |  | ||||||
| Your job is to use all this information to determine which of the provided LLMs would be best suited to complete the provided step given the capabilities of the LLM. |  | ||||||
| Your response should be the full name of the LLM that should complete the step. |  | ||||||
| Reply with only one of the following values: \n{'\n'.join(list(model_manager.model_capabilities.keys()))} |  | ||||||
| """ |  | ||||||
|  |  | ||||||
| def select_best_model(step: str, results: List[str], context: str) -> tuple[str, str]: |  | ||||||
|     prompt = f"Current Step: {step}\n\nResults So Far: {results}\n\nCurrent Chain of Thought: {context}" |  | ||||||
|     logger.debug("Selecting best model", prompt=prompt, system=SELECT_BEST_MODEL_PROMPT) |  | ||||||
|     response = model_manager.generate_text("llama3.2:3b", prompt, max_length=50, system=SELECT_BEST_MODEL_PROMPT) |  | ||||||
|     model_name = response.strip().lower() |  | ||||||
|     return model_name, response |  | ||||||
|  |  | ||||||
|  |  | ||||||
| def summarize_context(context: str) -> tuple[str, str]: |  | ||||||
|     prompt = f"Summarize the following context: {context}" |  | ||||||
|     logger.debug("Summarizing context", prompt=prompt) |  | ||||||
|     response = model_manager.generate_text("llama3.2:3b", prompt, max_length=300) |  | ||||||
|     return response, response |  | ||||||
|  |  | ||||||
| EXECUTE_STEP_PROMPT = """ |  | ||||||
| You are a large language model that has been selected to complete a step within a larger task. |  | ||||||
| You have been selected to complete this step due to your specific capabilities. |  | ||||||
| You will be provided with the job to do in this current step, the results of the previous steps in order, and the current chain of thought so far. |  | ||||||
| If the chain of thought is too long, a summary of the current chain of thought will be provided. |  | ||||||
| Your job is to use all this information to complete the step. |  | ||||||
| Your response should be in two parts. The first part should be your thought process in completing the step, how you went about solving the step, assumptions made, relation to previous steps, and challenges faced.  |  | ||||||
| You must then output a line with the word "RESPONSE". |  | ||||||
| The second part should be the result of completing your step. |  | ||||||
| The second part should contain nothing except the result of completing your step. |  | ||||||
| Only complete your part of the step. Do not extrapolate beyond the bounds of the step. Do not trample on the results of previous steps. Build on the results of previous steps, and use them to inform your work. |  | ||||||
| Do not include any preamble or other text, only the result of completing your step. |  | ||||||
| Do not use any markdown formatting, code formatting, or any other formatting. |  | ||||||
| """ |  | ||||||
|  |  | ||||||
| def execute_step(step: str, model: str, results: List[str], context: str) -> tuple[str, str]: |  | ||||||
|     prompt = f"Current Step: {step}\n\nResults So Far: {results}\n\nCurrent Chain of Thought: {context}" |  | ||||||
|     logger.debug("Executing step", step=step, model=model, prompt=prompt) |  | ||||||
|     response = model_manager.generate_text(model, prompt, max_length=1024, system=EXECUTE_STEP_PROMPT) |  | ||||||
|     response_step = response.split("RESPONSE")[1].strip() |  | ||||||
|     response_thinking = response.split("RESPONSE")[0].strip() |  | ||||||
|     return response_step, response_thinking |  | ||||||
|  |  | ||||||
| def generate_final_response(user_input: str, plan: List[str], step_results: List[str]) -> tuple[str, str]: |  | ||||||
|     prompt = f"Question: {user_input}\n\nPlan:\n" |  | ||||||
|     for i, step in enumerate(plan): |  | ||||||
|         prompt += f"{i+1}. {step}\n" |  | ||||||
|     prompt += "\nResults:\n" |  | ||||||
|     for i, result in enumerate(step_results): |  | ||||||
|         prompt += f"Step {i+1} result: {result}\n" |  | ||||||
|     prompt += "\nBased on the above information, provide a comprehensive answer to the original question." |  | ||||||
|     logger.debug("Generating final response", prompt=prompt) |  | ||||||
|     response = model_manager.generate_text("qwen2.5:7b", prompt, max_length=500) |  | ||||||
|     return response, response |  | ||||||
|  |  | ||||||
| UPDATE_INTERVAL = 0.1  # 100ms, configurable | UPDATE_INTERVAL = 0.1  # 100ms, configurable | ||||||
|  |  | ||||||
| @ -284,4 +203,4 @@ def send_system_resources(): | |||||||
| if __name__ == "__main__": | if __name__ == "__main__": | ||||||
|     logger.info("Starting LLM Chat Server") |     logger.info("Starting LLM Chat Server") | ||||||
|     threading.Thread(target=send_system_resources, daemon=True).start() |     threading.Thread(target=send_system_resources, daemon=True).start() | ||||||
|     socketio.run(app, debug=True, host="0.0.0.0", port=5000) |     socketio.run(app, debug=True, host="0.0.0.0", port=5001) | ||||||
							
								
								
									
										11
									
								
								models.py
									
									
									
									
									
								
							
							
						
						
									
										11
									
								
								models.py
									
									
									
									
									
								
							| @ -6,9 +6,9 @@ logger = structlog.get_logger() | |||||||
| class ModelManager: | class ModelManager: | ||||||
|     def __init__(self): |     def __init__(self): | ||||||
|         self.model_capabilities = { |         self.model_capabilities = { | ||||||
|             "qwen2.5:7b": ["general_knowledge", "structured_output", "multilingual", "instruction_following", "structured_data"], |             "ajindal/llama3.1-storm:8b": ["general_knowledge", "reasoning", "tool_calling", "conversation", "multilingual", "instruction_following"], | ||||||
|             "llama3.1:8b": ["general_knowledge", "reasoning", "tool_calling", "conversation", "multilingual", "instruction_following"], |             "llama3.1:8b": ["general_knowledge", "reasoning", "tool_calling", "conversation", "multilingual", "instruction_following"], | ||||||
|             "qwen2.5-coder:7b": ["code_generation", "code_analysis", "instruction_following", "math_reasoning"], |             "qwen2.5:7b": ["general_knowledge", "reasoning", "tool_calling", "conversation", "multilingual", "instruction_following"], | ||||||
|             "llama3.2:3b": ["summarization", "instruction_following", "tool_calling", "multilingual"], |             "llama3.2:3b": ["summarization", "instruction_following", "tool_calling", "multilingual"], | ||||||
|             "llava:7b": ["visual_reasoning", "visual_conversation", "visual_tool_calling", "vision", "ocr", "multimodal"], |             "llava:7b": ["visual_reasoning", "visual_conversation", "visual_tool_calling", "vision", "ocr", "multimodal"], | ||||||
|         } |         } | ||||||
| @ -25,8 +25,7 @@ class ModelManager: | |||||||
|         logger.info("Selected best model", required_capability=required_capability, selected_model=selected_model) |         logger.info("Selected best model", required_capability=required_capability, selected_model=selected_model) | ||||||
|         return selected_model |         return selected_model | ||||||
|  |  | ||||||
|     def generate_text(self, model_name, prompt, max_length=100, system="You are a helpful assistant."): |     def generate_text(self, model_name, prompt, max_length=100, system="You are a helpful assistant.", tools=[]): | ||||||
|         logger.debug("Generating text", model=model_name, prompt=prompt, max_length=max_length) |  | ||||||
|         # Check if model exists |         # Check if model exists | ||||||
|         try: |         try: | ||||||
|             ollama.pull(model_name) |             ollama.pull(model_name) | ||||||
| @ -38,7 +37,9 @@ class ModelManager: | |||||||
|             else: |             else: | ||||||
|                 logger.exception("Error pulling model", model=model_name, error=str(e)) |                 logger.exception("Error pulling model", model=model_name, error=str(e)) | ||||||
|                 raise e |                 raise e | ||||||
|         response = ollama.generate(model=model_name, prompt=prompt, system=system) |              | ||||||
|  |  | ||||||
|  |         response = ollama.generate(model=model_name, prompt=prompt, system=system, tools=tools, max_tokens=max_length) | ||||||
|         logger.debug("Text generated", model=model_name, response=response['response']) |         logger.debug("Text generated", model=model_name, response=response['response']) | ||||||
|         return response['response'] |         return response['response'] | ||||||
|  |  | ||||||
|  | |||||||
							
								
								
									
										116
									
								
								tools.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										116
									
								
								tools.py
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,116 @@ | |||||||
|  | import duckduckgo_search | ||||||
|  | import requests | ||||||
|  | from readability.readability import Document | ||||||
|  | from markdownify import markdownify as md | ||||||
|  | import sys | ||||||
|  | import time | ||||||
|  | import io | ||||||
|  | import subprocess | ||||||
|  | class Tool: | ||||||
|  |     def __init__(self, name: str, description: str, arguments: dict, returns: str): | ||||||
|  |         self.name = name | ||||||
|  |         self.description = description | ||||||
|  |         self.arguments = arguments | ||||||
|  |         self.returns = returns | ||||||
|  |  | ||||||
|  |     def execute(self, arguments: dict) -> str: | ||||||
|  |         pass | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class ToolManager: | ||||||
|  |     def __init__(self): | ||||||
|  |         self.tools = [] | ||||||
|  |  | ||||||
|  |     def add_tool(self, tool: Tool): | ||||||
|  |         self.tools.append(tool) | ||||||
|  |  | ||||||
|  |     def get_tool(self, name: str) -> Tool: | ||||||
|  |         for tool in self.tools: | ||||||
|  |             if tool.name == name: | ||||||
|  |                 return tool | ||||||
|  |         return None | ||||||
|  |      | ||||||
|  |     def get_tools_and_descriptions_for_prompt(self): | ||||||
|  |         return "\n".join([f"{tool.name}: {tool.description}" for tool in self.tools]) | ||||||
|  |      | ||||||
|  |     def get_tools_for_ollama_dict(self): | ||||||
|  |         return [{'type': 'function', 'function': {'name': tool.name, 'description': tool.description, 'parameters': tool.arguments}} for tool in self.tools] | ||||||
|  |      | ||||||
|  |  | ||||||
|  | class DefaultToolManager(ToolManager): | ||||||
|  |     def __init__(self): | ||||||
|  |         super().__init__() | ||||||
|  |         self.add_tool(SearchTool()) | ||||||
|  |         self.add_tool(GetReadablePageContentsTool()) | ||||||
|  |         self.add_tool(CalculatorTool()) | ||||||
|  |         self.add_tool(PythonCodeTool()) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class SearchTool(Tool): | ||||||
|  |     def __init__(self): | ||||||
|  |         super().__init__("search_web", "Search the internet for information", {'type': 'object', 'properties': {'query': {'type': 'string', 'description': 'The search query'}}}, "results:list[string]") | ||||||
|  |  | ||||||
|  |     def execute(self, arg: dict) -> str: | ||||||
|  |         res = duckduckgo_search.DDGS().text(arg['query'], max_results=5) | ||||||
|  |         return '\n\n'.join([f"{r['title']}\n{r['body']}\n{r['href']}" for r in res]) | ||||||
|  |      | ||||||
|  |  | ||||||
|  | def get_readable_page_contents(url: str) -> str: | ||||||
|  |     try: | ||||||
|  |         response = requests.get(url) | ||||||
|  |         response.raise_for_status() | ||||||
|  |         doc = Document(response.content) | ||||||
|  |         content = doc.summary() | ||||||
|  |         return md(content) | ||||||
|  |     except Exception as e: | ||||||
|  |         return f"Error fetching readable content: {str(e)}" | ||||||
|  |      | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class GetReadablePageContentsTool(Tool): | ||||||
|  |     def __init__(self): | ||||||
|  |         super().__init__("get_readable_page_contents", "Get the contents of a web page in a readable format", {'type': 'object', 'properties': {'url': {'type': 'string', 'description': 'The url of the web page'}}}, "contents:string") | ||||||
|  |  | ||||||
|  |     def execute(self, arg: dict) -> str: | ||||||
|  |         return get_readable_page_contents(arg['url']) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class CalculatorTool(Tool): | ||||||
|  |     def __init__(self): | ||||||
|  |         super().__init__("calculator", "Perform a calculation", {'type': 'object', 'properties': {'expression': {'type': 'string', 'description': 'The mathematical expression to evaluate, should be a python mathematical expression'}}}, "result:string") | ||||||
|  |  | ||||||
|  |     def execute(self, arg: dict) -> str: | ||||||
|  |         try: | ||||||
|  |             return str(exec(arg["expression"])) | ||||||
|  |         except Exception as e: | ||||||
|  |             return f"Error executing code: {str(e)}" | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class PythonCodeTool(Tool): | ||||||
|  |     def __init__(self): | ||||||
|  |         super().__init__("python_code", "Execute python code", {'type': 'object', 'properties': {'code': {'type': 'string', 'description': 'The python code to execute, should be a single line of valid python'}}}, "result:string") | ||||||
|  |  | ||||||
|  |     def execute(self, arg: dict) -> str: | ||||||
|  |         try: | ||||||
|  |             start_time = time.time() | ||||||
|  |             process = subprocess.Popen(['python', '-c', arg['code']],  | ||||||
|  |                                        stdout=subprocess.PIPE,  | ||||||
|  |                                        stderr=subprocess.PIPE,  | ||||||
|  |                                        text=True) | ||||||
|  |             stdout, stderr = process.communicate(timeout=10)  # 10 second timeout | ||||||
|  |             end_time = time.time() | ||||||
|  |             execution_time = end_time - start_time | ||||||
|  |              | ||||||
|  |             result = { | ||||||
|  |                 'stdout': stdout, | ||||||
|  |                 'stderr': stderr, | ||||||
|  |                 'return_value': process.returncode, | ||||||
|  |                 'execution_time': execution_time | ||||||
|  |             } | ||||||
|  |         except subprocess.TimeoutExpired: | ||||||
|  |             process.kill() | ||||||
|  |             return "Error: Code execution timed out after 10 seconds" | ||||||
|  |         except Exception as e: | ||||||
|  |             return f"Error executing code: {str(e)}" | ||||||
|  |          | ||||||
|  |         return '\n'.join([f"{k}: {v}" for k, v in result.items()]) | ||||||
		Reference in New Issue
	
	Block a user