mirror of
				https://github.com/ItzCrazyKns/Perplexica.git
				synced 2025-10-25 00:18:14 +00:00 
			
		
		
		
	Compare commits
	
		
			9 Commits
		
	
	
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | 4f50462f1d | ||
|  | 231bc22a36 | ||
|  | cb1d85e458 | ||
|  | ce78b4ff62 | ||
|  | 88ae67065b | ||
|  | f35d12f94c | ||
|  | 3d17975d83 | ||
|  | 950717e0cf | ||
|  | 4f39b5746a | 
										
											Binary file not shown.
										
									
								
							| Before Width: | Height: | Size: 2.1 MiB After Width: | Height: | Size: 2.1 MiB | 
							
								
								
									
										52
									
								
								Dockerfile
									
									
									
									
									
								
							
							
						
						
									
										52
									
								
								Dockerfile
									
									
									
									
									
								
							| @@ -17,22 +17,11 @@ RUN yarn build | |||||||
|  |  | ||||||
| FROM node:24.5.0-slim | FROM node:24.5.0-slim | ||||||
|  |  | ||||||
| RUN apt-get update && \ | RUN apt-get update && apt-get install -y \ | ||||||
|     apt-get install -y \ |     python3-dev python3-babel python3-venv python-is-python3 \ | ||||||
|     python3 \ |     uwsgi uwsgi-plugin-python3 \ | ||||||
|     python3-pip \ |     git build-essential libxslt-dev zlib1g-dev libffi-dev libssl-dev \ | ||||||
|     python3-venv \ |     curl sudo \ | ||||||
|     python3-dev \ |  | ||||||
|     sqlite3 \ |  | ||||||
|     git \ |  | ||||||
|     build-essential \ |  | ||||||
|     libxslt-dev \ |  | ||||||
|     zlib1g-dev \ |  | ||||||
|     libffi-dev \ |  | ||||||
|     libssl-dev \ |  | ||||||
|     uwsgi \ |  | ||||||
|     uwsgi-plugin-python3 \ |  | ||||||
|     curl \ |  | ||||||
|     && rm -rf /var/lib/apt/lists/* |     && rm -rf /var/lib/apt/lists/* | ||||||
|  |  | ||||||
| WORKDIR /home/perplexica | WORKDIR /home/perplexica | ||||||
| @@ -45,28 +34,39 @@ COPY drizzle ./drizzle | |||||||
|  |  | ||||||
| RUN mkdir /home/perplexica/uploads | RUN mkdir /home/perplexica/uploads | ||||||
|  |  | ||||||
| RUN useradd --system --home-dir /usr/local/searxng --shell /bin/sh searxng | RUN useradd --shell /bin/bash --system \ | ||||||
|  |     --home-dir "/usr/local/searxng" \ | ||||||
| WORKDIR /usr/local/searxng |     --comment 'Privacy-respecting metasearch engine' \ | ||||||
| RUN git clone https://github.com/searxng/searxng.git . && \ |     searxng | ||||||
|     python3 -m venv venv && \ |  | ||||||
|     . venv/bin/activate && \ |  | ||||||
|     pip install --upgrade pip setuptools wheel pyyaml && \ |  | ||||||
|     pip install -r requirements.txt && \ |  | ||||||
|     pip install uwsgi |  | ||||||
|  |  | ||||||
|  | RUN mkdir "/usr/local/searxng" | ||||||
| RUN mkdir -p /etc/searxng | RUN mkdir -p /etc/searxng | ||||||
|  | RUN chown -R "searxng:searxng" "/usr/local/searxng" | ||||||
|  |  | ||||||
| COPY searxng/settings.yml /etc/searxng/settings.yml | COPY searxng/settings.yml /etc/searxng/settings.yml | ||||||
| COPY searxng/limiter.toml /etc/searxng/limiter.toml | COPY searxng/limiter.toml /etc/searxng/limiter.toml | ||||||
| COPY searxng/uwsgi.ini /etc/searxng/uwsgi.ini | COPY searxng/uwsgi.ini /etc/searxng/uwsgi.ini | ||||||
|  | RUN chown -R searxng:searxng /etc/searxng | ||||||
|  |  | ||||||
| RUN chown -R searxng:searxng /usr/local/searxng /etc/searxng | USER searxng | ||||||
|  |  | ||||||
|  | RUN git clone "https://github.com/searxng/searxng" \ | ||||||
|  |                    "/usr/local/searxng/searxng-src" | ||||||
|  |  | ||||||
|  | RUN python3 -m venv "/usr/local/searxng/searx-pyenv" | ||||||
|  | RUN "/usr/local/searxng/searx-pyenv/bin/pip" install --upgrade pip setuptools wheel pyyaml msgspec | ||||||
|  | RUN cd "/usr/local/searxng/searxng-src" && \ | ||||||
|  |     "/usr/local/searxng/searx-pyenv/bin/pip" install --use-pep517 --no-build-isolation -e . | ||||||
|  |  | ||||||
|  | USER root | ||||||
|  |  | ||||||
| WORKDIR /home/perplexica | WORKDIR /home/perplexica | ||||||
| COPY entrypoint.sh ./entrypoint.sh | COPY entrypoint.sh ./entrypoint.sh | ||||||
| RUN chmod +x ./entrypoint.sh | RUN chmod +x ./entrypoint.sh | ||||||
| RUN sed -i 's/\r$//' ./entrypoint.sh || true | RUN sed -i 's/\r$//' ./entrypoint.sh || true | ||||||
|  |  | ||||||
|  | RUN echo "searxng ALL=(ALL) NOPASSWD: ALL" >> /etc/sudoers | ||||||
|  |  | ||||||
| EXPOSE 3000 8080 | EXPOSE 3000 8080 | ||||||
|  |  | ||||||
| ENV SEARXNG_API_URL=http://localhost:8080 | ENV SEARXNG_API_URL=http://localhost:8080 | ||||||
|   | |||||||
| @@ -79,19 +79,19 @@ There are mainly 2 ways of installing Perplexica - With Docker, Without Docker. | |||||||
| Perplexica can be easily run using Docker. Simply run the following command: | Perplexica can be easily run using Docker. Simply run the following command: | ||||||
|  |  | ||||||
| ```bash | ```bash | ||||||
| docker run -p 3000:3000 --name perplexica itzcrazykns1337/perplexica:latest | docker run -d -p 3000:3000 -v perplexica-data:/home/perplexica/data -v perplexica-uploads:/home/perplexica/uploads --name perplexica itzcrazykns1337/perplexica:latest | ||||||
| ``` | ``` | ||||||
|  |  | ||||||
| This will pull and start the Perplexica container with the bundled SearxNG search engine. Once running, open your browser and navigate to http://localhost:3000. You can then configure your settings (API keys, models, etc.) directly in the setup screen. | This will pull and start the Perplexica container with the bundled SearxNG search engine. Once running, open your browser and navigate to http://localhost:3000. You can then configure your settings (API keys, models, etc.) directly in the setup screen. | ||||||
|  |  | ||||||
| **Note**: The image includes both Perplexica and SearxNG, so no additional setup is required. | **Note**: The image includes both Perplexica and SearxNG, so no additional setup is required. The `-v` flags create persistent volumes for your data and uploaded files. | ||||||
|  |  | ||||||
| #### Using Perplexica with Your Own SearxNG Instance | #### Using Perplexica with Your Own SearxNG Instance | ||||||
|  |  | ||||||
| If you already have SearxNG running, you can use the slim version of Perplexica: | If you already have SearxNG running, you can use the slim version of Perplexica: | ||||||
|  |  | ||||||
| ```bash | ```bash | ||||||
| docker run -p 3000:3000 -e SEARXNG_API_URL=http://your-searxng-url:8080 --name perplexica itzcrazykns1337/perplexica:slim-latest | docker run -d -p 3000:3000 -e SEARXNG_API_URL=http://your-searxng-url:8080 -v perplexica-data:/home/perplexica/data -v perplexica-uploads:/home/perplexica/uploads --name perplexica itzcrazykns1337/perplexica:slim-latest | ||||||
| ``` | ``` | ||||||
|  |  | ||||||
| **Important**: Make sure your SearxNG instance has: | **Important**: Make sure your SearxNG instance has: | ||||||
| @@ -118,7 +118,7 @@ If you prefer to build from source or need more control: | |||||||
|  |  | ||||||
|    ```bash |    ```bash | ||||||
|    docker build -t perplexica . |    docker build -t perplexica . | ||||||
|    docker run -p 3000:3000 --name perplexica perplexica |    docker run -d -p 3000:3000 -v perplexica-data:/home/perplexica/data -v perplexica-uploads:/home/perplexica/uploads --name perplexica perplexica | ||||||
|    ``` |    ``` | ||||||
|  |  | ||||||
| 5. Access Perplexica at http://localhost:3000 and configure your settings in the setup screen. | 5. Access Perplexica at http://localhost:3000 and configure your settings in the setup screen. | ||||||
|   | |||||||
| @@ -1,37 +0,0 @@ | |||||||
| FROM node:24.5.0-slim AS builder |  | ||||||
|  |  | ||||||
| RUN apt-get update && apt-get install -y python3 python3-pip sqlite3 && rm -rf /var/lib/apt/lists/* |  | ||||||
|  |  | ||||||
| WORKDIR /home/perplexica |  | ||||||
|  |  | ||||||
| COPY package.json yarn.lock ./ |  | ||||||
| RUN yarn install --frozen-lockfile --network-timeout 600000 |  | ||||||
|  |  | ||||||
| COPY tsconfig.json next.config.mjs next-env.d.ts postcss.config.js drizzle.config.ts tailwind.config.ts ./ |  | ||||||
| COPY src ./src |  | ||||||
| COPY public ./public |  | ||||||
| COPY drizzle ./drizzle |  | ||||||
|  |  | ||||||
| RUN mkdir -p /home/perplexica/data |  | ||||||
| RUN yarn build |  | ||||||
|  |  | ||||||
| FROM node:24.5.0-slim |  | ||||||
|  |  | ||||||
| RUN apt-get update && apt-get install -y python3 python3-pip sqlite3 && rm -rf /var/lib/apt/lists/* |  | ||||||
|  |  | ||||||
| WORKDIR /home/perplexica |  | ||||||
|  |  | ||||||
| COPY --from=builder /home/perplexica/public ./public |  | ||||||
| COPY --from=builder /home/perplexica/.next/static ./public/_next/static |  | ||||||
|  |  | ||||||
| COPY --from=builder /home/perplexica/.next/standalone ./ |  | ||||||
| COPY --from=builder /home/perplexica/data ./data |  | ||||||
| COPY drizzle ./drizzle |  | ||||||
|  |  | ||||||
| RUN mkdir /home/perplexica/uploads |  | ||||||
|  |  | ||||||
| COPY entrypoint.sh ./entrypoint.sh |  | ||||||
| RUN chmod +x ./entrypoint.sh |  | ||||||
| RUN sed -i 's/\r$//' ./entrypoint.sh || true |  | ||||||
|  |  | ||||||
| CMD ["/home/perplexica/entrypoint.sh"] |  | ||||||
							
								
								
									
										15
									
								
								docker-compose.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										15
									
								
								docker-compose.yaml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,15 @@ | |||||||
|  | services: | ||||||
|  |   perplexica: | ||||||
|  |     image: itzcrazykns1337/perplexica:latest | ||||||
|  |     ports: | ||||||
|  |       - "3000:3000" | ||||||
|  |     volumes: | ||||||
|  |       - data:/home/perplexica/data | ||||||
|  |       - uploads:/home/perplexica/uploads | ||||||
|  |     restart: unless-stopped | ||||||
|  |  | ||||||
|  | volumes: | ||||||
|  |   data: | ||||||
|  |     name: 'perplexica-data' | ||||||
|  |   uploads: | ||||||
|  |     name: 'perplexica-uploads' | ||||||
| @@ -10,7 +10,7 @@ Simply pull the latest image and restart your container: | |||||||
| docker pull itzcrazykns1337/perplexica:latest | docker pull itzcrazykns1337/perplexica:latest | ||||||
| docker stop perplexica | docker stop perplexica | ||||||
| docker rm perplexica | docker rm perplexica | ||||||
| docker run -p 3000:3000 --name perplexica itzcrazykns1337/perplexica:latest | docker run -d -p 3000:3000 -v perplexica-data:/home/perplexica/data -v perplexica-uploads:/home/perplexica/uploads --name perplexica itzcrazykns1337/perplexica:latest | ||||||
| ``` | ``` | ||||||
|  |  | ||||||
| For slim version: | For slim version: | ||||||
| @@ -19,7 +19,7 @@ For slim version: | |||||||
| docker pull itzcrazykns1337/perplexica:slim-latest | docker pull itzcrazykns1337/perplexica:slim-latest | ||||||
| docker stop perplexica | docker stop perplexica | ||||||
| docker rm perplexica | docker rm perplexica | ||||||
| docker run -p 3000:3000 -e SEARXNG_API_URL=http://your-searxng-url:8080 --name perplexica itzcrazykns1337/perplexica:slim-latest | docker run -d -p 3000:3000 -e SEARXNG_API_URL=http://your-searxng-url:8080 -v perplexica-data:/home/perplexica/data -v perplexica-uploads:/home/perplexica/uploads --name perplexica itzcrazykns1337/perplexica:slim-latest | ||||||
| ``` | ``` | ||||||
|  |  | ||||||
| Once updated, go to http://localhost:3000 and verify the latest changes. Your settings are preserved automatically. | Once updated, go to http://localhost:3000 and verify the latest changes. Your settings are preserved automatically. | ||||||
|   | |||||||
| @@ -1,24 +1,32 @@ | |||||||
| #!/bin/sh | #!/bin/sh | ||||||
| set -e | set -e | ||||||
|  |  | ||||||
| cd /usr/local/searxng |  | ||||||
| export SEARXNG_SETTINGS_PATH=/etc/searxng/settings.yml |  | ||||||
|  |  | ||||||
| # Start SearXNG in background with all output redirected to /dev/null |  | ||||||
| /usr/local/searxng/venv/bin/uwsgi \ |  | ||||||
|   --http-socket 0.0.0.0:8080 \ |  | ||||||
|   --ini /etc/searxng/uwsgi.ini \ |  | ||||||
|   --virtualenv /usr/local/searxng/venv \ |  | ||||||
|   --disable-logging > /dev/null 2>&1 & |  | ||||||
|  |  | ||||||
| echo "Starting SearXNG..." | echo "Starting SearXNG..." | ||||||
|  |  | ||||||
|  | sudo -H -u searxng bash -c "cd /usr/local/searxng/searxng-src && export SEARXNG_SETTINGS_PATH='/etc/searxng/settings.yml' && export FLASK_APP=searx/webapp.py && /usr/local/searxng/searx-pyenv/bin/python -m flask run --host=0.0.0.0 --port=8080" & | ||||||
|  | SEARXNG_PID=$! | ||||||
|  |  | ||||||
|  | echo "Waiting for SearXNG to be ready..." | ||||||
| sleep 5 | sleep 5 | ||||||
|  |  | ||||||
|  | COUNTER=0 | ||||||
|  | MAX_TRIES=30 | ||||||
| until curl -s http://localhost:8080 > /dev/null 2>&1; do | until curl -s http://localhost:8080 > /dev/null 2>&1; do | ||||||
|  |   COUNTER=$((COUNTER+1)) | ||||||
|  |   if [ $COUNTER -ge $MAX_TRIES ]; then | ||||||
|  |     echo "Warning: SearXNG health check timeout, but continuing..." | ||||||
|  |     break | ||||||
|  |   fi | ||||||
|   sleep 1 |   sleep 1 | ||||||
| done | done | ||||||
| echo "SearXNG started successfully" |  | ||||||
|  | if curl -s http://localhost:8080 > /dev/null 2>&1; then | ||||||
|  |   echo "SearXNG started successfully (PID: $SEARXNG_PID)" | ||||||
|  | else | ||||||
|  |   echo "SearXNG may not be fully ready, but continuing (PID: $SEARXNG_PID)" | ||||||
|  | fi | ||||||
|  |  | ||||||
| cd /home/perplexica | cd /home/perplexica | ||||||
| echo "Starting Perplexica..." | echo "Starting Perplexica..." | ||||||
|  |  | ||||||
| exec node server.js | exec node server.js | ||||||
| @@ -1,6 +1,6 @@ | |||||||
| { | { | ||||||
|   "name": "perplexica-frontend", |   "name": "perplexica-frontend", | ||||||
|   "version": "1.11.0", |   "version": "1.11.1", | ||||||
|   "license": "MIT", |   "license": "MIT", | ||||||
|   "author": "ItzCrazyKns", |   "author": "ItzCrazyKns", | ||||||
|   "scripts": { |   "scripts": { | ||||||
|   | |||||||
| @@ -112,7 +112,6 @@ const ModelProvider = ({ | |||||||
|           > |           > | ||||||
|             <div className="border-t border-light-200 dark:border-dark-200" /> |             <div className="border-t border-light-200 dark:border-dark-200" /> | ||||||
|             <div className="flex flex-col gap-y-4 px-5 py-4"> |             <div className="flex flex-col gap-y-4 px-5 py-4"> | ||||||
|               {modelProvider.chatModels.length > 0 && ( |  | ||||||
|               <div className="flex flex-col gap-y-2"> |               <div className="flex flex-col gap-y-2"> | ||||||
|                 <div className="flex flex-row w-full justify-between items-center"> |                 <div className="flex flex-row w-full justify-between items-center"> | ||||||
|                   <p className="text-[11px] lg:text-xs text-black/70 dark:text-white/70"> |                   <p className="text-[11px] lg:text-xs text-black/70 dark:text-white/70"> | ||||||
| @@ -157,8 +156,6 @@ const ModelProvider = ({ | |||||||
|                   )} |                   )} | ||||||
|                 </div> |                 </div> | ||||||
|               </div> |               </div> | ||||||
|               )} |  | ||||||
|               {modelProvider.embeddingModels.length > 0 && ( |  | ||||||
|               <div className="flex flex-col gap-y-2"> |               <div className="flex flex-col gap-y-2"> | ||||||
|                 <div className="flex flex-row w-full justify-between items-center"> |                 <div className="flex flex-row w-full justify-between items-center"> | ||||||
|                   <p className="text-[11px] lg:text-xs text-black/70 dark:text-white/70"> |                   <p className="text-[11px] lg:text-xs text-black/70 dark:text-white/70"> | ||||||
| @@ -205,7 +202,6 @@ const ModelProvider = ({ | |||||||
|                   )} |                   )} | ||||||
|                 </div> |                 </div> | ||||||
|               </div> |               </div> | ||||||
|               )} |  | ||||||
|             </div> |             </div> | ||||||
|           </motion.div> |           </motion.div> | ||||||
|         )} |         )} | ||||||
|   | |||||||
| @@ -11,14 +11,16 @@ const ModelSelect = ({ | |||||||
|   type: 'chat' | 'embedding'; |   type: 'chat' | 'embedding'; | ||||||
| }) => { | }) => { | ||||||
|   const [selectedModel, setSelectedModel] = useState<string>( |   const [selectedModel, setSelectedModel] = useState<string>( | ||||||
|     `${providers[0]?.id}/${providers[0].embeddingModels[0]?.key}`, |     type === 'chat' | ||||||
|  |       ? `${localStorage.getItem('chatModelProviderId')}/${localStorage.getItem('chatModelKey')}` | ||||||
|  |       : `${localStorage.getItem('embeddingModelProviderId')}/${localStorage.getItem('embeddingModelKey')}`, | ||||||
|   ); |   ); | ||||||
|   const [loading, setLoading] = useState(false); |   const [loading, setLoading] = useState(false); | ||||||
|  |  | ||||||
|   const handleSave = async (newValue: string) => { |   const handleSave = async (newValue: string) => { | ||||||
|     setLoading(true); |     setLoading(true); | ||||||
|     setSelectedModel(newValue); |     setSelectedModel(newValue); | ||||||
|     console.log(newValue); |  | ||||||
|     try { |     try { | ||||||
|       if (type === 'chat') { |       if (type === 'chat') { | ||||||
|         localStorage.setItem('chatModelProviderId', newValue.split('/')[0]); |         localStorage.setItem('chatModelProviderId', newValue.split('/')[0]); | ||||||
|   | |||||||
| @@ -63,8 +63,7 @@ const SetupConfig = ({ | |||||||
|     } |     } | ||||||
|   }; |   }; | ||||||
|  |  | ||||||
|   const hasProviders = |   const hasProviders = providers.length > 0; | ||||||
|     providers.filter((p) => p.chatModels.length > 0).length > 0; |  | ||||||
|  |  | ||||||
|   return ( |   return ( | ||||||
|     <div className="w-[95vw] md:w-[80vw] lg:w-[65vw] mx-auto px-2 sm:px-4 md:px-6 flex flex-col space-y-6"> |     <div className="w-[95vw] md:w-[80vw] lg:w-[65vw] mx-auto px-2 sm:px-4 md:px-6 flex flex-col space-y-6"> | ||||||
|   | |||||||
| @@ -50,6 +50,25 @@ class ConfigManager { | |||||||
|           'e.g., "Respond in a friendly and concise tone" or "Use British English and format answers as bullet points."', |           'e.g., "Respond in a friendly and concise tone" or "Use British English and format answers as bullet points."', | ||||||
|         scope: 'client', |         scope: 'client', | ||||||
|       }, |       }, | ||||||
|  |       { | ||||||
|  |         name: 'Measurement Unit', | ||||||
|  |         key: 'measureUnit', | ||||||
|  |         type: 'select', | ||||||
|  |         options: [ | ||||||
|  |           { | ||||||
|  |             name: 'Imperial', | ||||||
|  |             value: 'Imperial', | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             name: 'Metric', | ||||||
|  |             value: 'Metric', | ||||||
|  |           }, | ||||||
|  |         ], | ||||||
|  |         required: false, | ||||||
|  |         description: 'Choose between Metric  and Imperial measurement unit.', | ||||||
|  |         default: 'Metric', | ||||||
|  |         scope: 'client', | ||||||
|  |       }, | ||||||
|     ], |     ], | ||||||
|     modelProviders: [], |     modelProviders: [], | ||||||
|     search: [ |     search: [ | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user