Compare commits
	
		
			2 Commits
		
	
	
		
			v0.3.14-rc
			...
			mattw/pyth
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|   | 05162c56aa | ||
|   | edd1a2b6e8 | 
| @@ -1,11 +1,9 @@ | |||||||
| .vscode | .vscode | ||||||
| ollama | ollama | ||||||
| app | app | ||||||
| macapp |  | ||||||
| dist | dist | ||||||
| llm/llama.cpp | scripts | ||||||
|  | llm/llama.cpp/ggml | ||||||
|  | llm/llama.cpp/gguf | ||||||
| .env | .env | ||||||
| .cache | .cache | ||||||
| test_data |  | ||||||
| llm/build |  | ||||||
| llama/build |  | ||||||
|   | |||||||
							
								
								
									
										12
									
								
								.gitattributes
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -1,12 +0,0 @@ | |||||||
| llm/ext_server/* linguist-vendored |  | ||||||
| llama/**/*.cpp linguist-vendored |  | ||||||
| llama/**/*.hpp linguist-vendored |  | ||||||
| llama/**/*.h linguist-vendored |  | ||||||
| llama/**/*.c linguist-vendored |  | ||||||
| llama/**/*.cu linguist-vendored |  | ||||||
| llama/**/*.cuh linguist-vendored |  | ||||||
| llama/**/*.m linguist-vendored |  | ||||||
| llama/**/*.metal linguist-vendored |  | ||||||
|  |  | ||||||
| * text=auto |  | ||||||
| *.go text eol=lf |  | ||||||
							
								
								
									
										60
									
								
								.github/ISSUE_TEMPLATE/10_bug_report.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -1,60 +0,0 @@ | |||||||
| name: Bug report |  | ||||||
| labels: [bug] |  | ||||||
| description: Something isn't working right. |  | ||||||
| body: |  | ||||||
|   - type: textarea |  | ||||||
|     id: description |  | ||||||
|     attributes: |  | ||||||
|       label: What is the issue? |  | ||||||
|       description: What happened? What did you expect to happen? |  | ||||||
|     validations: |  | ||||||
|       required: true |  | ||||||
|   - type: dropdown |  | ||||||
|     id: os |  | ||||||
|     attributes: |  | ||||||
|       label: OS |  | ||||||
|       description: Which operating system are you using? |  | ||||||
|       multiple: true |  | ||||||
|       options: |  | ||||||
|         - Linux |  | ||||||
|         - macOS |  | ||||||
|         - Windows |  | ||||||
|         - Docker |  | ||||||
|         - WSL2 |  | ||||||
|     validations: |  | ||||||
|       required: false |  | ||||||
|   - type: dropdown |  | ||||||
|     id: gpu |  | ||||||
|     attributes: |  | ||||||
|       label: GPU |  | ||||||
|       description: Which GPU are you using? |  | ||||||
|       multiple: true |  | ||||||
|       options: |  | ||||||
|         - Nvidia |  | ||||||
|         - AMD |  | ||||||
|         - Intel |  | ||||||
|         - Apple |  | ||||||
|         - Other |  | ||||||
|     validations: |  | ||||||
|       required: false |  | ||||||
|   - type: dropdown |  | ||||||
|     id: cpu |  | ||||||
|     attributes: |  | ||||||
|       label: CPU |  | ||||||
|       description: Which CPU are you using? |  | ||||||
|       multiple: true |  | ||||||
|       options: |  | ||||||
|         - Intel |  | ||||||
|         - AMD |  | ||||||
|         - Apple |  | ||||||
|         - Other |  | ||||||
|     validations: |  | ||||||
|       required: false |  | ||||||
|   - type: input |  | ||||||
|     id: version |  | ||||||
|     attributes: |  | ||||||
|       label: Ollama version |  | ||||||
|       description: What version of Ollama are you using? (`ollama --version`) |  | ||||||
|       placeholder: e.g., 0.1.32 |  | ||||||
|     validations: |  | ||||||
|       required: false |  | ||||||
							
								
								
									
										6
									
								
								.github/ISSUE_TEMPLATE/20_feature_request.md
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -1,6 +0,0 @@ | |||||||
| --- |  | ||||||
| name: Feature request |  | ||||||
| about: Request a new feature |  | ||||||
| labels: feature request |  | ||||||
| --- |  | ||||||
|  |  | ||||||
							
								
								
									
										5
									
								
								.github/ISSUE_TEMPLATE/30_model_request.md
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -1,5 +0,0 @@ | |||||||
| --- |  | ||||||
| name: Model request |  | ||||||
| about: Request support for a new model to be added to Ollama |  | ||||||
| labels: model request |  | ||||||
| --- |  | ||||||
							
								
								
									
										8
									
								
								.github/ISSUE_TEMPLATE/config.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -1,8 +0,0 @@ | |||||||
| blank_issues_enabled: true |  | ||||||
| contact_links: |  | ||||||
|   - name: Help |  | ||||||
|     url: https://discord.com/invite/ollama |  | ||||||
|     about: Please join our Discord server for help using Ollama |  | ||||||
|   - name: Troubleshooting |  | ||||||
|     url: https://github.com/ollama/ollama/blob/main/docs/faq.md#faq |  | ||||||
|     about: See the FAQ for common issues and solutions |  | ||||||
							
								
								
									
										24
									
								
								.github/workflows/latest.yaml
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -1,24 +0,0 @@ | |||||||
| name: latest |  | ||||||
|  |  | ||||||
| on: |  | ||||||
|   release: |  | ||||||
|     types: [released] |  | ||||||
|  |  | ||||||
| jobs: |  | ||||||
|   update-latest: |  | ||||||
|     environment: release |  | ||||||
|     runs-on: linux |  | ||||||
|     steps: |  | ||||||
|       - uses: actions/checkout@v4 |  | ||||||
|       - name: Login to Docker Hub |  | ||||||
|         uses: docker/login-action@v3 |  | ||||||
|         with: |  | ||||||
|           username: ${{ vars.DOCKER_USER }} |  | ||||||
|           password: ${{ secrets.DOCKER_ACCESS_TOKEN }} |  | ||||||
|       - name: Tag images as latest |  | ||||||
|         env: |  | ||||||
|           PUSH: "1" |  | ||||||
|         shell: bash |  | ||||||
|         run: | |  | ||||||
|           export "VERSION=${GITHUB_REF_NAME#v}" |  | ||||||
|           ./scripts/tag_latest.sh |  | ||||||
							
								
								
									
										775
									
								
								.github/workflows/release.yaml
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -1,775 +0,0 @@ | |||||||
| name: release |  | ||||||
|  |  | ||||||
| on: |  | ||||||
|   push: |  | ||||||
|     tags: |  | ||||||
|       - 'v*' |  | ||||||
|  |  | ||||||
| jobs: |  | ||||||
|   # Full build of the Mac assets |  | ||||||
|   build-darwin: |  | ||||||
|     runs-on: macos-12 |  | ||||||
|     environment: release |  | ||||||
|     steps: |  | ||||||
|       - uses: actions/checkout@v4 |  | ||||||
|       - name: Set Version |  | ||||||
|         shell: bash |  | ||||||
|         run: | |  | ||||||
|           echo "VERSION=${GITHUB_REF_NAME#v}" >> $GITHUB_ENV |  | ||||||
|           echo "RELEASE_VERSION=$(echo ${GITHUB_REF_NAME} | cut -f1 -d-)" >> $GITHUB_ENV |  | ||||||
|       - name: key |  | ||||||
|         env: |  | ||||||
|           MACOS_SIGNING_KEY: ${{ secrets.MACOS_SIGNING_KEY }} |  | ||||||
|           MACOS_SIGNING_KEY_PASSWORD: ${{ secrets.MACOS_SIGNING_KEY_PASSWORD }} |  | ||||||
|         run: | |  | ||||||
|           echo $MACOS_SIGNING_KEY | base64 --decode > certificate.p12 |  | ||||||
|           security create-keychain -p password build.keychain |  | ||||||
|           security default-keychain -s build.keychain |  | ||||||
|           security unlock-keychain -p password build.keychain |  | ||||||
|           security import certificate.p12 -k build.keychain -P $MACOS_SIGNING_KEY_PASSWORD -T /usr/bin/codesign |  | ||||||
|           security set-key-partition-list -S apple-tool:,apple:,codesign: -s -k password build.keychain |  | ||||||
|           security set-keychain-settings -lut 3600 build.keychain |  | ||||||
|       - uses: actions/setup-go@v5 |  | ||||||
|         with: |  | ||||||
|           go-version-file: go.mod |  | ||||||
|           cache: true |  | ||||||
|       - name: Build Darwin |  | ||||||
|         env: |  | ||||||
|           APPLE_IDENTITY: ${{ secrets.APPLE_IDENTITY }} |  | ||||||
|           APPLE_PASSWORD: ${{ secrets.APPLE_PASSWORD }} |  | ||||||
|           APPLE_TEAM_ID: ${{ vars.APPLE_TEAM_ID }} |  | ||||||
|           APPLE_ID: ${{ vars.APPLE_ID }} |  | ||||||
|           SDKROOT: /Applications/Xcode_13.4.1.app/Contents/Developer/Platforms/MacOSX.platform/Developer/SDKs/MacOSX.sdk |  | ||||||
|           DEVELOPER_DIR: /Applications/Xcode_13.4.1.app/Contents/Developer |  | ||||||
|         run: | |  | ||||||
|           ./scripts/build_darwin.sh |  | ||||||
|  |  | ||||||
|       - uses: actions/upload-artifact@v4 |  | ||||||
|         with: |  | ||||||
|           name: dist-darwin |  | ||||||
|           path: | |  | ||||||
|             dist/*arwin* |  | ||||||
|             !dist/*-cov |  | ||||||
|  |  | ||||||
|   # Windows builds take a long time to both install the dependencies and build, so parallelize |  | ||||||
|   # CPU generation step |  | ||||||
|   generate-windows-cpu: |  | ||||||
|     environment: release |  | ||||||
|     runs-on: windows |  | ||||||
|     env: |  | ||||||
|       KEY_CONTAINER: ${{ vars.KEY_CONTAINER }} |  | ||||||
|     steps: |  | ||||||
|       - uses: actions/checkout@v4 |  | ||||||
|       - name: Set Version |  | ||||||
|         shell: bash |  | ||||||
|         run: echo "VERSION=${GITHUB_REF_NAME#v}" >> $GITHUB_ENV |  | ||||||
|       - uses: 'google-github-actions/auth@v2' |  | ||||||
|         with: |  | ||||||
|           project_id: 'ollama' |  | ||||||
|           credentials_json: '${{ secrets.GOOGLE_SIGNING_CREDENTIALS }}' |  | ||||||
|       - run: echo "${{ vars.OLLAMA_CERT }}" > ollama_inc.crt |  | ||||||
|       - name: install Windows SDK 8.1 to get signtool |  | ||||||
|         run: | |  | ||||||
|           $ErrorActionPreference = "Stop" |  | ||||||
|           write-host "downloading SDK" |  | ||||||
|           Invoke-WebRequest -Uri "https://go.microsoft.com/fwlink/p/?LinkId=323507" -OutFile "${env:RUNNER_TEMP}\sdksetup.exe" |  | ||||||
|           Start-Process "${env:RUNNER_TEMP}\sdksetup.exe" -ArgumentList @("/q") -NoNewWindow -Wait |  | ||||||
|           write-host "Win SDK 8.1 installed" |  | ||||||
|           gci -path 'C:\Program Files (x86)\Windows Kits\' -r -fi 'signtool.exe' |  | ||||||
|       - name: install signing plugin |  | ||||||
|         run: | |  | ||||||
|           $ErrorActionPreference = "Stop" |  | ||||||
|           write-host "downloading plugin" |  | ||||||
|           Invoke-WebRequest -Uri "https://github.com/GoogleCloudPlatform/kms-integrations/releases/download/cng-v1.0/kmscng-1.0-windows-amd64.zip" -OutFile "${env:RUNNER_TEMP}\plugin.zip" |  | ||||||
|           Expand-Archive -Path "${env:RUNNER_TEMP}\plugin.zip" -DestinationPath ${env:RUNNER_TEMP}\plugin\ |  | ||||||
|           write-host "Installing plugin" |  | ||||||
|           & "${env:RUNNER_TEMP}\plugin\*\kmscng.msi" /quiet |  | ||||||
|           write-host "plugin installed" |  | ||||||
|       - uses: actions/setup-go@v5 |  | ||||||
|         with: |  | ||||||
|           go-version-file: go.mod |  | ||||||
|           cache: true |  | ||||||
|       - run: go get ./... |  | ||||||
|       - run: | |  | ||||||
|           $gopath=(get-command go).source | split-path -parent |  | ||||||
|           & "C:\Program Files (x86)\Microsoft Visual Studio\2019\Enterprise\Common7\Tools\Launch-VsDevShell.ps1" |  | ||||||
|           cd $env:GITHUB_WORKSPACE |  | ||||||
|           $env:CMAKE_SYSTEM_VERSION="10.0.22621.0" |  | ||||||
|           $env:PATH="$gopath;$env:PATH" |  | ||||||
|           go generate -x ./... |  | ||||||
|         name: go generate |  | ||||||
|       - uses: actions/upload-artifact@v4 |  | ||||||
|         with: |  | ||||||
|           name: generate-windows-cpu |  | ||||||
|           path: | |  | ||||||
|             build/**/* |  | ||||||
|             build/**/*.a |  | ||||||
|             llm/build/**/*.a |  | ||||||
|             dist/windows-amd64/** |  | ||||||
|  |  | ||||||
|   # ROCm generation step |  | ||||||
|   generate-windows-rocm: |  | ||||||
|     environment: release |  | ||||||
|     runs-on: windows |  | ||||||
|     env: |  | ||||||
|       KEY_CONTAINER: ${{ vars.KEY_CONTAINER }} |  | ||||||
|     steps: |  | ||||||
|       - uses: actions/checkout@v4 |  | ||||||
|       - name: Set Version |  | ||||||
|         shell: bash |  | ||||||
|         run: echo "VERSION=${GITHUB_REF_NAME#v}" >> $GITHUB_ENV |  | ||||||
|       - uses: 'google-github-actions/auth@v2' |  | ||||||
|         with: |  | ||||||
|           project_id: 'ollama' |  | ||||||
|           credentials_json: '${{ secrets.GOOGLE_SIGNING_CREDENTIALS }}' |  | ||||||
|       - run: echo "${{ vars.OLLAMA_CERT }}" > ollama_inc.crt |  | ||||||
|       - name: install Windows SDK 8.1 to get signtool |  | ||||||
|         run: | |  | ||||||
|           $ErrorActionPreference = "Stop" |  | ||||||
|           write-host "downloading SDK" |  | ||||||
|           Invoke-WebRequest -Uri "https://go.microsoft.com/fwlink/p/?LinkId=323507" -OutFile "${env:RUNNER_TEMP}\sdksetup.exe" |  | ||||||
|           Start-Process "${env:RUNNER_TEMP}\sdksetup.exe" -ArgumentList @("/q") -NoNewWindow -Wait |  | ||||||
|           write-host "Win SDK 8.1 installed" |  | ||||||
|           gci -path 'C:\Program Files (x86)\Windows Kits\' -r -fi 'signtool.exe' |  | ||||||
|       - name: install signing plugin |  | ||||||
|         run: | |  | ||||||
|           $ErrorActionPreference = "Stop" |  | ||||||
|           write-host "downloading plugin" |  | ||||||
|           Invoke-WebRequest -Uri "https://github.com/GoogleCloudPlatform/kms-integrations/releases/download/cng-v1.0/kmscng-1.0-windows-amd64.zip" -OutFile "${env:RUNNER_TEMP}\plugin.zip" |  | ||||||
|           Expand-Archive -Path "${env:RUNNER_TEMP}\plugin.zip" -DestinationPath ${env:RUNNER_TEMP}\plugin\ |  | ||||||
|           write-host "Installing plugin" |  | ||||||
|           & "${env:RUNNER_TEMP}\plugin\*\kmscng.msi" /quiet |  | ||||||
|           write-host "plugin installed" |  | ||||||
|       - uses: actions/setup-go@v5 |  | ||||||
|         with: |  | ||||||
|           go-version-file: go.mod |  | ||||||
|           cache: true |  | ||||||
|       - name: 'Install ROCm' |  | ||||||
|         run: | |  | ||||||
|           $ErrorActionPreference = "Stop" |  | ||||||
|           write-host "downloading AMD HIP Installer" |  | ||||||
|           Invoke-WebRequest -Uri "https://download.amd.com/developer/eula/rocm-hub/AMD-Software-PRO-Edition-24.Q3-WinSvr2022-For-HIP.exe" -OutFile "${env:RUNNER_TEMP}\rocm-install.exe" |  | ||||||
|           write-host "Installing AMD HIP" |  | ||||||
|           Start-Process "${env:RUNNER_TEMP}\rocm-install.exe" -ArgumentList '-install' -NoNewWindow -Wait |  | ||||||
|           write-host "Completed AMD HIP" |  | ||||||
|       - name: 'Verify ROCm' |  | ||||||
|         run: | |  | ||||||
|           & 'C:\Program Files\AMD\ROCm\*\bin\clang.exe' --version |  | ||||||
|       - run: go get ./... |  | ||||||
|       - run: | |  | ||||||
|           $gopath=(get-command go).source | split-path -parent |  | ||||||
|           & "C:\Program Files (x86)\Microsoft Visual Studio\2019\Enterprise\Common7\Tools\Launch-VsDevShell.ps1" |  | ||||||
|           cd $env:GITHUB_WORKSPACE |  | ||||||
|           $env:CMAKE_SYSTEM_VERSION="10.0.22621.0" |  | ||||||
|           $env:PATH="$gopath;$env:PATH" |  | ||||||
|           $env:OLLAMA_SKIP_CPU_GENERATE="1" |  | ||||||
|           $env:HIP_PATH=$(Resolve-Path 'C:\Program Files\AMD\ROCm\*\bin\clang.exe' | split-path | split-path) |  | ||||||
|           go generate -x ./... |  | ||||||
|         name: go generate |  | ||||||
|       - name: 'gather rocm dependencies' |  | ||||||
|         run: | |  | ||||||
|           $HIP_PATH=$(Resolve-Path 'C:\Program Files\AMD\ROCm\*\bin\clang.exe' | split-path | split-path) |  | ||||||
|           md "dist\deps\bin\rocblas\library" |  | ||||||
|           cp "${HIP_PATH}\bin\hipblas.dll" "dist\deps\bin\" |  | ||||||
|           cp "${HIP_PATH}\bin\rocblas.dll" "dist\deps\bin\" |  | ||||||
|           cp "${HIP_PATH}\bin\rocblas\library\*" "dist\deps\bin\rocblas\library\" |  | ||||||
|       - uses: actions/upload-artifact@v4 |  | ||||||
|         with: |  | ||||||
|           name: generate-windows-rocm |  | ||||||
|           path: | |  | ||||||
|             build/**/* |  | ||||||
|             dist/windows-amd64/** |  | ||||||
|       - uses: actions/upload-artifact@v4 |  | ||||||
|         with: |  | ||||||
|           name: windows-rocm-deps |  | ||||||
|           path: dist/deps/* |  | ||||||
|  |  | ||||||
|   # CUDA generation step |  | ||||||
|   generate-windows-cuda: |  | ||||||
|     environment: release |  | ||||||
|     runs-on: windows |  | ||||||
|     strategy: |  | ||||||
|       matrix: |  | ||||||
|         cuda: |  | ||||||
|           - version: "11" |  | ||||||
|             url: 'https://developer.download.nvidia.com/compute/cuda/11.3.1/local_installers/cuda_11.3.1_465.89_win10.exe' |  | ||||||
|           - version: "12" |  | ||||||
|             url: 'https://developer.download.nvidia.com/compute/cuda/12.4.0/local_installers/cuda_12.4.0_551.61_windows.exe' |  | ||||||
|     env: |  | ||||||
|       KEY_CONTAINER: ${{ vars.KEY_CONTAINER }} |  | ||||||
|     steps: |  | ||||||
|       - uses: actions/checkout@v4 |  | ||||||
|       - name: Set Version |  | ||||||
|         shell: bash |  | ||||||
|         run: echo "VERSION=${GITHUB_REF_NAME#v}" >> $GITHUB_ENV |  | ||||||
|       - uses: 'google-github-actions/auth@v2' |  | ||||||
|         with: |  | ||||||
|           project_id: 'ollama' |  | ||||||
|           credentials_json: '${{ secrets.GOOGLE_SIGNING_CREDENTIALS }}' |  | ||||||
|       - run: echo "${{ vars.OLLAMA_CERT }}" > ollama_inc.crt |  | ||||||
|       - name: install Windows SDK 8.1 to get signtool |  | ||||||
|         run: | |  | ||||||
|           $ErrorActionPreference = "Stop" |  | ||||||
|           write-host "downloading SDK" |  | ||||||
|           Invoke-WebRequest -Uri "https://go.microsoft.com/fwlink/p/?LinkId=323507" -OutFile "${env:RUNNER_TEMP}\sdksetup.exe" |  | ||||||
|           Start-Process "${env:RUNNER_TEMP}\sdksetup.exe" -ArgumentList @("/q") -NoNewWindow -Wait |  | ||||||
|           write-host "Win SDK 8.1 installed" |  | ||||||
|           gci -path 'C:\Program Files (x86)\Windows Kits\' -r -fi 'signtool.exe' |  | ||||||
|       - name: install signing plugin |  | ||||||
|         run: | |  | ||||||
|           $ErrorActionPreference = "Stop" |  | ||||||
|           write-host "downloading plugin" |  | ||||||
|           Invoke-WebRequest -Uri "https://github.com/GoogleCloudPlatform/kms-integrations/releases/download/cng-v1.0/kmscng-1.0-windows-amd64.zip" -OutFile "${env:RUNNER_TEMP}\plugin.zip" |  | ||||||
|           Expand-Archive -Path "${env:RUNNER_TEMP}\plugin.zip" -DestinationPath ${env:RUNNER_TEMP}\plugin\ |  | ||||||
|           write-host "Installing plugin" |  | ||||||
|           & "${env:RUNNER_TEMP}\plugin\*\kmscng.msi" /quiet |  | ||||||
|           write-host "plugin installed" |  | ||||||
|       - uses: actions/setup-go@v5 |  | ||||||
|         with: |  | ||||||
|           go-version-file: go.mod |  | ||||||
|           cache: true |  | ||||||
|       - name: 'Install CUDA ${{ matrix.cuda.version }}' |  | ||||||
|         run: | |  | ||||||
|           $ErrorActionPreference = "Stop" |  | ||||||
|           write-host "downloading CUDA Installer" |  | ||||||
|           Invoke-WebRequest -Uri "${{ matrix.cuda.url }}" -OutFile "${env:RUNNER_TEMP}\cuda-install.exe" |  | ||||||
|           write-host "Installing CUDA" |  | ||||||
|           Start-Process "${env:RUNNER_TEMP}\cuda-install.exe" -ArgumentList '-s' -NoNewWindow -Wait |  | ||||||
|           write-host "Completed CUDA" |  | ||||||
|           $cudaPath=((resolve-path "c:\Program Files\NVIDIA*\CUDA\v*\bin\nvcc.exe")[0].path | split-path | split-path) |  | ||||||
|           $cudaVer=($cudaPath | split-path -leaf ) -replace 'v(\d+).(\d+)', '$1_$2'  |  | ||||||
|           echo "$cudaPath\bin" >> $env:GITHUB_PATH |  | ||||||
|           echo "CUDA_PATH=$cudaPath" >> $env:GITHUB_ENV |  | ||||||
|           echo "CUDA_PATH_V${cudaVer}=$cudaPath" >> $env:GITHUB_ENV |  | ||||||
|           echo "CUDA_PATH_VX_Y=CUDA_PATH_V${cudaVer}" >> $env:GITHUB_ENV |  | ||||||
|       - name: 'Verify CUDA' |  | ||||||
|         run: nvcc -V |  | ||||||
|       - run: go get ./... |  | ||||||
|       - name: go generate |  | ||||||
|         run: | |  | ||||||
|           $gopath=(get-command go).source | split-path -parent |  | ||||||
|           $cudabin=(get-command nvcc).source | split-path |  | ||||||
|           & "C:\Program Files (x86)\Microsoft Visual Studio\2019\Enterprise\Common7\Tools\Launch-VsDevShell.ps1" |  | ||||||
|           cd $env:GITHUB_WORKSPACE |  | ||||||
|           $env:CMAKE_SYSTEM_VERSION="10.0.22621.0" |  | ||||||
|           $env:PATH="$gopath;$cudabin;$env:PATH" |  | ||||||
|           $env:OLLAMA_SKIP_CPU_GENERATE="1" |  | ||||||
|           go generate -x ./... |  | ||||||
|       - name: 'gather cuda dependencies' |  | ||||||
|         run: | |  | ||||||
|           $NVIDIA_DIR=(resolve-path 'C:\Program Files\NVIDIA GPU Computing Toolkit\CUDA\*\bin\')[0] |  | ||||||
|           md "dist\deps" |  | ||||||
|           cp "${NVIDIA_DIR}\cudart64_*.dll" "dist\deps\" |  | ||||||
|           cp "${NVIDIA_DIR}\cublas64_*.dll" "dist\deps\" |  | ||||||
|           cp "${NVIDIA_DIR}\cublasLt64_*.dll" "dist\deps\" |  | ||||||
|       - uses: actions/upload-artifact@v4 |  | ||||||
|         with: |  | ||||||
|           name: generate-windows-cuda-${{ matrix.cuda.version }} |  | ||||||
|           path: | |  | ||||||
|             build/**/* |  | ||||||
|             dist/windows-amd64/** |  | ||||||
|       - uses: actions/upload-artifact@v4 |  | ||||||
|         with: |  | ||||||
|           name: windows-cuda-deps-${{ matrix.cuda.version }} |  | ||||||
|           path: dist/deps/* |  | ||||||
|  |  | ||||||
|  |  | ||||||
|   # windows arm64 generate, go build, and zip file (no installer) |  | ||||||
|   # Output of this build is aggregated into the final x86 build |  | ||||||
|   # for a unified windows installer |  | ||||||
|   windows-arm64: |  | ||||||
|     runs-on: windows-arm64 |  | ||||||
|     environment: release |  | ||||||
|     env: |  | ||||||
|       KEY_CONTAINER: ${{ vars.KEY_CONTAINER }} |  | ||||||
|     steps: |  | ||||||
|       # The current Windows arm64 beta image has effectively zero dev tools installed... |  | ||||||
|       - name: Install git and gzip |  | ||||||
|         run: | |  | ||||||
|           Set-ExecutionPolicy Bypass -Scope Process -Force |  | ||||||
|           [System.Net.ServicePointManager]::SecurityProtocol = [System.Net.ServicePointManager]::SecurityProtocol -bor 3072 |  | ||||||
|           iex ((New-Object System.Net.WebClient).DownloadString('https://community.chocolatey.org/install.ps1')) |  | ||||||
|           choco install -y --no-progress git gzip |  | ||||||
|           echo "C:\Program Files\Git\cmd" | Out-File -FilePath $env:GITHUB_PATH -Encoding utf8 -Append |  | ||||||
|           echo "C:\ProgramData\chocolatey\bin" | Out-File -FilePath $env:GITHUB_PATH -Encoding utf8 -Append |  | ||||||
|       - name: Install Visual Studio 2022 |  | ||||||
|         run: | |  | ||||||
|           $components = @( |  | ||||||
|             "Microsoft.VisualStudio.Component.CoreEditor", |  | ||||||
|             "Microsoft.VisualStudio.Workload.CoreEditor", |  | ||||||
|             "Microsoft.VisualStudio.Component.Roslyn.Compiler", |  | ||||||
|             "Microsoft.Component.MSBuild", |  | ||||||
|             "Microsoft.VisualStudio.Component.TextTemplating", |  | ||||||
|             "Microsoft.VisualStudio.Component.Debugger.JustInTime", |  | ||||||
|             "Microsoft.VisualStudio.Component.VC.CoreIde", |  | ||||||
|             "Microsoft.VisualStudio.Component.VC.Tools.x86.x64", |  | ||||||
|             "Microsoft.VisualStudio.Component.Windows11SDK.22621", |  | ||||||
|             "Microsoft.VisualStudio.Component.VC.Tools.ARM64EC", |  | ||||||
|             "Microsoft.VisualStudio.Component.VC.Tools.ARM64", |  | ||||||
|             "Microsoft.VisualStudio.Component.VC.ATL", |  | ||||||
|             "Microsoft.VisualStudio.Component.VC.ATL.ARM64", |  | ||||||
|             "Microsoft.VisualStudio.Component.Graphics", |  | ||||||
|             "Microsoft.VisualStudio.Component.VC.Redist.14.Latest", |  | ||||||
|             "Microsoft.VisualStudio.ComponentGroup.NativeDesktop.Core", |  | ||||||
|             "Microsoft.VisualStudio.Component.Windows11Sdk.WindowsPerformanceToolkit", |  | ||||||
|             "Microsoft.VisualStudio.Component.CppBuildInsights", |  | ||||||
|             "Microsoft.VisualStudio.Component.VC.DiagnosticTools", |  | ||||||
|             "Microsoft.VisualStudio.ComponentGroup.WebToolsExtensions.CMake", |  | ||||||
|             "Microsoft.VisualStudio.Component.VC.CMake.Project", |  | ||||||
|             "Microsoft.VisualStudio.Component.VC.ASAN", |  | ||||||
|             "Microsoft.VisualStudio.Component.Vcpkg", |  | ||||||
|             "Microsoft.VisualStudio.Workload.NativeDesktop" |  | ||||||
|           ) |  | ||||||
|           $config = @{ |  | ||||||
|                 "version" = "1.0" |  | ||||||
|                 "components"  = $components |  | ||||||
|                 "extensions"  = @() |  | ||||||
|             } |  | ||||||
|           $configPath = "${env:RUNNER_TEMP}\vsconfig" |  | ||||||
|           $config | ConvertTo-Json | Out-File -FilePath $configPath |  | ||||||
|           $bootstrapperFilePath = "${env:RUNNER_TEMP}\vs_community.exe" |  | ||||||
|           write-host "Downloading Visual Studio 2022" |  | ||||||
|           Invoke-WebRequest -Uri "https://aka.ms/vs/17/release/vs_community.exe" -outfile $bootstrapperFilePath |  | ||||||
|           $bootstrapperArgumentList = ('/c', $bootstrapperFilePath, '--config', $configPath, '--quiet', '--wait' ) |  | ||||||
|           write-host "Installing Visual Studio 2022" |  | ||||||
|           $process = Start-Process -FilePath cmd.exe -ArgumentList $bootstrapperArgumentList -Wait -PassThru |  | ||||||
|           $exitCode = $process.ExitCode |  | ||||||
|           write-host $exitCode |  | ||||||
|       # pacman in mingw/msys2 is ~broken on windows arm right now - hangs consistently during attempts to install |  | ||||||
|       # so we'll use this alternative GCC binary |  | ||||||
|       - name: Install llvm-mingw GCC |  | ||||||
|         run: | |  | ||||||
|           $gcc_url="https://github.com/mstorsjo/llvm-mingw/releases/download/20240619/llvm-mingw-20240619-ucrt-aarch64.zip" |  | ||||||
|           write-host "Downloading llvm-mingw" |  | ||||||
|           Invoke-WebRequest -Uri "${gcc_url}" -OutFile "${env:RUNNER_TEMP}\gcc.zip" |  | ||||||
|           write-host "Unpacking llvm-mingw" |  | ||||||
|           expand-archive -path "${env:RUNNER_TEMP}\gcc.zip" -destinationpath "c:\" |  | ||||||
|           mv c:\llvm-mingw-* c:\llvm-mingw |  | ||||||
|           echo "c:\llvm-mingw\bin" | Out-File -FilePath $env:GITHUB_PATH -Encoding utf8 -Append |  | ||||||
|       - name: Verify GCC |  | ||||||
|         run: | |  | ||||||
|           echo $env:PATH |  | ||||||
|           gcc --version |  | ||||||
|       - uses: actions/checkout@v4 |  | ||||||
|       - name: Set Version |  | ||||||
|         run: | |  | ||||||
|           $ver=${env:GITHUB_REF_NAME}.trim("v") |  | ||||||
|           echo VERSION=$ver | Out-File -FilePath ${env:GITHUB_ENV} -Encoding utf8 -Append |  | ||||||
|       - uses: 'google-github-actions/auth@v2' |  | ||||||
|         with: |  | ||||||
|           project_id: 'ollama' |  | ||||||
|           credentials_json: '${{ secrets.GOOGLE_SIGNING_CREDENTIALS }}' |  | ||||||
|       - run: echo "${{ vars.OLLAMA_CERT }}" | Out-File -FilePath ollama_inc.crt -Encoding utf8 |  | ||||||
|       - name: install Windows SDK 8.1 to get signtool |  | ||||||
|         run: | |  | ||||||
|           $ErrorActionPreference = "Stop" |  | ||||||
|           write-host "downloading SDK" |  | ||||||
|           Invoke-WebRequest -Uri "https://go.microsoft.com/fwlink/p/?LinkId=323507" -OutFile "${env:RUNNER_TEMP}\sdksetup.exe" |  | ||||||
|           Start-Process "${env:RUNNER_TEMP}\sdksetup.exe" -ArgumentList @("/q") -NoNewWindow -Wait |  | ||||||
|           write-host "Win SDK 8.1 installed" |  | ||||||
|           gci -path 'C:\Program Files (x86)\Windows Kits\' -r -fi 'signtool.exe' |  | ||||||
|       - name: install signing plugin |  | ||||||
|         run: | |  | ||||||
|           $ErrorActionPreference = "Stop" |  | ||||||
|           write-host "downloading plugin" |  | ||||||
|           Invoke-WebRequest -Uri "https://github.com/GoogleCloudPlatform/kms-integrations/releases/download/cng-v1.0/kmscng-1.0-windows-amd64.zip" -OutFile "${env:RUNNER_TEMP}\plugin.zip" |  | ||||||
|           Expand-Archive -Path "${env:RUNNER_TEMP}\plugin.zip" -DestinationPath ${env:RUNNER_TEMP}\plugin\ |  | ||||||
|           write-host "Installing plugin" |  | ||||||
|           & "${env:RUNNER_TEMP}\plugin\*\kmscng.msi" /quiet |  | ||||||
|           write-host "plugin installed" |  | ||||||
|       - uses: actions/setup-go@v5 |  | ||||||
|         with: |  | ||||||
|           go-version-file: go.mod |  | ||||||
|           cache: true |  | ||||||
|       - run: go get ./... |  | ||||||
|       - run: | |  | ||||||
|           $gopath=(get-command go).source | split-path -parent |  | ||||||
|           $gccpath=(get-command gcc).source | split-path -parent |  | ||||||
|           & "C:\Program Files\Microsoft Visual Studio\2022\Community\Common7\Tools\Launch-VsDevShell.ps1" |  | ||||||
|           cd $env:GITHUB_WORKSPACE |  | ||||||
|           $env:CMAKE_SYSTEM_VERSION="10.0.22621.0" |  | ||||||
|           $env:PATH="$gopath;$gccpath;$env:PATH;C:\Program Files\Microsoft Visual Studio\2022\Community\Common7\IDE\CommonExtensions\Microsoft\CMake\CMake\bin" |  | ||||||
|           echo $env:PATH |  | ||||||
|           $env:ARCH="arm64" |  | ||||||
|           .\scripts\build_windows.ps1 buildOllama buildApp gatherDependencies distZip |  | ||||||
|         name: 'Windows Build' |  | ||||||
|       - uses: actions/upload-artifact@v4 |  | ||||||
|         with: |  | ||||||
|           name: windows-arm64 |  | ||||||
|           path: | |  | ||||||
|             dist/windows-arm64/** |  | ||||||
|             dist/windows-arm64-app.exe |  | ||||||
|             dist/ollama-windows-arm64.zip |  | ||||||
|  |  | ||||||
|   # Import the prior generation steps plus the full arm64 build, and build the final windows assets |  | ||||||
|   build-windows: |  | ||||||
|     environment: release |  | ||||||
|     runs-on: windows |  | ||||||
|     needs: |  | ||||||
|       - generate-windows-cuda |  | ||||||
|       - generate-windows-rocm |  | ||||||
|       - generate-windows-cpu |  | ||||||
|       - windows-arm64 |  | ||||||
|     env: |  | ||||||
|       KEY_CONTAINER: ${{ vars.KEY_CONTAINER }} |  | ||||||
|     steps: |  | ||||||
|       - uses: actions/checkout@v4 |  | ||||||
|         with: |  | ||||||
|           submodules: recursive |  | ||||||
|       - name: Set Version |  | ||||||
|         shell: bash |  | ||||||
|         run: echo "VERSION=${GITHUB_REF_NAME#v}" >> $GITHUB_ENV |  | ||||||
|       - uses: 'google-github-actions/auth@v2' |  | ||||||
|         with: |  | ||||||
|           project_id: 'ollama' |  | ||||||
|           credentials_json: '${{ secrets.GOOGLE_SIGNING_CREDENTIALS }}' |  | ||||||
|       - run: echo "${{ vars.OLLAMA_CERT }}" > ollama_inc.crt |  | ||||||
|       - name: install Windows SDK 8.1 to get signtool |  | ||||||
|         run: | |  | ||||||
|           $ErrorActionPreference = "Stop" |  | ||||||
|           write-host "downloading SDK" |  | ||||||
|           Invoke-WebRequest -Uri "https://go.microsoft.com/fwlink/p/?LinkId=323507" -OutFile "${env:RUNNER_TEMP}\sdksetup.exe" |  | ||||||
|           Start-Process "${env:RUNNER_TEMP}\sdksetup.exe" -ArgumentList @("/q") -NoNewWindow -Wait |  | ||||||
|           write-host "Win SDK 8.1 installed" |  | ||||||
|           gci -path 'C:\Program Files (x86)\Windows Kits\' -r -fi 'signtool.exe' |  | ||||||
|       - name: install signing plugin |  | ||||||
|         run: | |  | ||||||
|           $ErrorActionPreference = "Stop" |  | ||||||
|           write-host "downloading plugin" |  | ||||||
|           Invoke-WebRequest -Uri "https://github.com/GoogleCloudPlatform/kms-integrations/releases/download/cng-v1.0/kmscng-1.0-windows-amd64.zip" -OutFile "${env:RUNNER_TEMP}\plugin.zip" |  | ||||||
|           Expand-Archive -Path "${env:RUNNER_TEMP}\plugin.zip" -DestinationPath ${env:RUNNER_TEMP}\plugin\ |  | ||||||
|           write-host "Installing plugin" |  | ||||||
|           & "${env:RUNNER_TEMP}\plugin\*\kmscng.msi" /quiet |  | ||||||
|           write-host "plugin installed" |  | ||||||
|       - uses: actions/setup-go@v5 |  | ||||||
|         with: |  | ||||||
|           go-version-file: go.mod |  | ||||||
|           cache: true |  | ||||||
|       - run: go get |  | ||||||
|       - uses: actions/download-artifact@v4 |  | ||||||
|         with: |  | ||||||
|           name: generate-windows-cpu |  | ||||||
|       - uses: actions/download-artifact@v4 |  | ||||||
|         with: |  | ||||||
|           name: generate-windows-cuda-11 |  | ||||||
|       - uses: actions/download-artifact@v4 |  | ||||||
|         with: |  | ||||||
|           name: generate-windows-cuda-12 |  | ||||||
|       - uses: actions/download-artifact@v4 |  | ||||||
|         with: |  | ||||||
|           name: windows-cuda-deps-11 |  | ||||||
|       - uses: actions/download-artifact@v4 |  | ||||||
|         with: |  | ||||||
|           name: windows-cuda-deps-12 |  | ||||||
|       - uses: actions/download-artifact@v4 |  | ||||||
|         with: |  | ||||||
|           name: windows-rocm-deps |  | ||||||
|       - uses: actions/download-artifact@v4 |  | ||||||
|         with: |  | ||||||
|           name: generate-windows-rocm |  | ||||||
|       - uses: actions/download-artifact@v4 |  | ||||||
|         with: |  | ||||||
|           name: windows-arm64 |  | ||||||
|           path: dist |  | ||||||
|       - run: dir build |  | ||||||
|       - run: | |  | ||||||
|           $gopath=(get-command go).source | split-path -parent |  | ||||||
|           & "C:\Program Files (x86)\Microsoft Visual Studio\2019\Enterprise\Common7\Tools\Launch-VsDevShell.ps1" |  | ||||||
|           cd $env:GITHUB_WORKSPACE |  | ||||||
|           $env:CMAKE_SYSTEM_VERSION="10.0.22621.0" |  | ||||||
|           $env:PATH="$gopath;$env:PATH" |  | ||||||
|           $env:OLLAMA_SKIP_GENERATE="1" |  | ||||||
|           & .\scripts\build_windows.ps1 |  | ||||||
|       - uses: actions/upload-artifact@v4 |  | ||||||
|         with: |  | ||||||
|           name: dist-windows |  | ||||||
|           path: | |  | ||||||
|             dist/OllamaSetup.exe |  | ||||||
|             dist/ollama-windows-*.zip |  | ||||||
|  |  | ||||||
|   # Linux x86 assets built using the container based build |  | ||||||
|   build-linux-amd64: |  | ||||||
|     environment: release |  | ||||||
|     runs-on: linux |  | ||||||
|     env: |  | ||||||
|       PLATFORM: linux/amd64 |  | ||||||
|     steps: |  | ||||||
|       - uses: actions/checkout@v4 |  | ||||||
|         with: |  | ||||||
|           submodules: recursive |  | ||||||
|       - name: Set Version |  | ||||||
|         shell: bash |  | ||||||
|         run: echo "VERSION=${GITHUB_REF_NAME#v}" >> $GITHUB_ENV |  | ||||||
|       - run: | |  | ||||||
|           ./scripts/build_linux.sh |  | ||||||
|       - uses: actions/upload-artifact@v4 |  | ||||||
|         with: |  | ||||||
|           name: dist-linux-amd64 |  | ||||||
|           path: | |  | ||||||
|             dist/*linux* |  | ||||||
|             !dist/*-cov |  | ||||||
|  |  | ||||||
|   # Linux ARM assets built using the container based build |  | ||||||
|   # (at present, docker isn't pre-installed on arm ubunutu images) |  | ||||||
|   build-linux-arm64: |  | ||||||
|     environment: release |  | ||||||
|     runs-on: linux-arm64 |  | ||||||
|     env: |  | ||||||
|       PLATFORM: linux/arm64 |  | ||||||
|     steps: |  | ||||||
|       - uses: actions/checkout@v4 |  | ||||||
|         with: |  | ||||||
|           submodules: recursive |  | ||||||
|       - name: Set Version |  | ||||||
|         shell: bash |  | ||||||
|         run: echo "VERSION=${GITHUB_REF_NAME#v}" >> $GITHUB_ENV |  | ||||||
|       - name: 'Install Docker' |  | ||||||
|         run: | |  | ||||||
|           # Add Docker's official GPG key: |  | ||||||
|           env |  | ||||||
|           uname -a |  | ||||||
|           sudo apt-get update |  | ||||||
|           sudo apt-get install -y ca-certificates curl |  | ||||||
|           sudo install -m 0755 -d /etc/apt/keyrings |  | ||||||
|           sudo curl -fsSL https://download.docker.com/linux/ubuntu/gpg -o /etc/apt/keyrings/docker.asc |  | ||||||
|           sudo chmod a+r /etc/apt/keyrings/docker.asc |  | ||||||
|  |  | ||||||
|           # Add the repository to Apt sources: |  | ||||||
|           echo \ |  | ||||||
|             "deb [arch=$(dpkg --print-architecture) signed-by=/etc/apt/keyrings/docker.asc] https://download.docker.com/linux/ubuntu \ |  | ||||||
|             $(. /etc/os-release && echo "$VERSION_CODENAME") stable" | \ |  | ||||||
|             sudo tee /etc/apt/sources.list.d/docker.list > /dev/null |  | ||||||
|           sudo apt-get update |  | ||||||
|           sudo apt-get install -y docker-ce docker-ce-cli containerd.io |  | ||||||
|           sudo usermod -aG docker $USER |  | ||||||
|           sudo apt-get install acl |  | ||||||
|           sudo setfacl --modify user:$USER:rw /var/run/docker.sock |  | ||||||
|       - run: | |  | ||||||
|           ./scripts/build_linux.sh |  | ||||||
|       - uses: actions/upload-artifact@v4 |  | ||||||
|         with: |  | ||||||
|           name: dist-linux-arm64 |  | ||||||
|           path: | |  | ||||||
|             dist/*linux* |  | ||||||
|             !dist/*-cov |  | ||||||
|  |  | ||||||
|   # Container image build |  | ||||||
|   build-container-image: |  | ||||||
|     environment: release |  | ||||||
|     strategy: |  | ||||||
|       matrix: |  | ||||||
|         runner: |  | ||||||
|           - linux |  | ||||||
|           - linux-arm64 |  | ||||||
|     runs-on: ${{ matrix.runner }} |  | ||||||
|     env: |  | ||||||
|       FINAL_IMAGE_REPO: ollama/ollama |  | ||||||
|     steps: |  | ||||||
|       - uses: actions/checkout@v4 |  | ||||||
|         with: |  | ||||||
|           submodules: recursive |  | ||||||
|       - name: 'Install Docker' |  | ||||||
|         if: ${{ startsWith(matrix.runner, 'linux-arm64') }} |  | ||||||
|         run: | |  | ||||||
|           sudo apt-get update |  | ||||||
|           sudo apt-get install -y ca-certificates curl |  | ||||||
|           sudo install -m 0755 -d /etc/apt/keyrings |  | ||||||
|           sudo curl -fsSL https://download.docker.com/linux/ubuntu/gpg -o /etc/apt/keyrings/docker.asc |  | ||||||
|           sudo chmod a+r /etc/apt/keyrings/docker.asc |  | ||||||
|           echo "deb [arch=$(dpkg --print-architecture) signed-by=/etc/apt/keyrings/docker.asc] https://download.docker.com/linux/ubuntu \ |  | ||||||
|             $(. /etc/os-release && echo "$VERSION_CODENAME") stable" | \ |  | ||||||
|             sudo tee /etc/apt/sources.list.d/docker.list > /dev/null |  | ||||||
|           sudo apt-get update |  | ||||||
|           sudo apt-get install -y docker-ce docker-ce-cli containerd.io |  | ||||||
|           sudo usermod -aG docker $USER |  | ||||||
|           sudo apt-get install acl |  | ||||||
|           sudo setfacl --modify user:$USER:rw /var/run/docker.sock |  | ||||||
|       - name: Docker meta |  | ||||||
|         id: meta |  | ||||||
|         uses: docker/metadata-action@v5 |  | ||||||
|         with: |  | ||||||
|           images: ${{ env.FINAL_IMAGE_REPO }} |  | ||||||
|           flavor: | |  | ||||||
|             latest=false |  | ||||||
|           tags: | |  | ||||||
|             type=ref,enable=true,priority=600,prefix=0.0.0-pr,suffix=,event=pr |  | ||||||
|             type=semver,pattern={{version}} |  | ||||||
|       - name: Set Version |  | ||||||
|         shell: bash |  | ||||||
|         run: | |  | ||||||
|           machine=$(uname -m) |  | ||||||
|           case ${machine} in |  | ||||||
|             x86_64) echo ARCH=amd64; echo PLATFORM_PAIR=linux-amd64 ;; |  | ||||||
|             aarch64) echo ARCH=arm64; echo PLATFORM_PAIR=linux-arm64 ;; |  | ||||||
|           esac >>$GITHUB_ENV |  | ||||||
|           echo GOFLAGS="'-ldflags=-w -s \"-X=github.com/ollama/ollama/version.Version=${{ env.DOCKER_METADATA_OUTPUT_VERSION }}\" \"-X=github.com/ollama/ollama/server.mode=release\"'" >>$GITHUB_ENV |  | ||||||
|       - name: Set up Docker Buildx |  | ||||||
|         uses: docker/setup-buildx-action@v3 |  | ||||||
|       - name: Login to Docker Hub |  | ||||||
|         uses: docker/login-action@v3 |  | ||||||
|         with: |  | ||||||
|           username: ${{ vars.DOCKER_USER }} |  | ||||||
|           password: ${{ secrets.DOCKER_ACCESS_TOKEN }} |  | ||||||
|       - name: Build and push by digest |  | ||||||
|         id: build |  | ||||||
|         uses: docker/build-push-action@v6 |  | ||||||
|         with: |  | ||||||
|           context: "." |  | ||||||
|           platforms: linux/${{ env.ARCH }} |  | ||||||
|           build-args: | |  | ||||||
|             GOFLAGS |  | ||||||
|           outputs: type=image,name=${{ env.FINAL_IMAGE_REPO }},push-by-digest=true,name-canonical=true,push=true |  | ||||||
|       - name: Export digest |  | ||||||
|         run: | |  | ||||||
|           mkdir -p /tmp/digests |  | ||||||
|           digest="${{ steps.build.outputs.digest }}" |  | ||||||
|           touch "/tmp/digests/${digest#sha256:}" |  | ||||||
|       - name: Upload digest |  | ||||||
|         uses: actions/upload-artifact@v4 |  | ||||||
|         with: |  | ||||||
|           name: digests-${{ env.PLATFORM_PAIR }} |  | ||||||
|           path: /tmp/digests/* |  | ||||||
|           if-no-files-found: error |  | ||||||
|           retention-days: 1 |  | ||||||
|   merge: |  | ||||||
|     environment: release |  | ||||||
|     runs-on: linux |  | ||||||
|     needs: |  | ||||||
|       - build-container-image |  | ||||||
|     env: |  | ||||||
|       FINAL_IMAGE_REPO: ollama/ollama |  | ||||||
|     steps: |  | ||||||
|       - uses: actions/checkout@v4 |  | ||||||
|         with: |  | ||||||
|           submodules: recursive |  | ||||||
|       - name: Download digests |  | ||||||
|         uses: actions/download-artifact@v4 |  | ||||||
|         with: |  | ||||||
|           path: /tmp/digests |  | ||||||
|           pattern: digests-* |  | ||||||
|           merge-multiple: true |  | ||||||
|       - name: Set up Docker Buildx |  | ||||||
|         uses: docker/setup-buildx-action@v3 |  | ||||||
|       - name: Docker meta |  | ||||||
|         id: meta |  | ||||||
|         uses: docker/metadata-action@v5 |  | ||||||
|         with: |  | ||||||
|           images: ${{ env.FINAL_IMAGE_REPO }} |  | ||||||
|           flavor: | |  | ||||||
|             latest=false |  | ||||||
|           tags: | |  | ||||||
|             type=ref,enable=true,priority=600,prefix=0.0.0-pr,suffix=,event=pr |  | ||||||
|             type=semver,pattern={{version}} |  | ||||||
|       - name: Set Version |  | ||||||
|         shell: bash |  | ||||||
|         run: | |  | ||||||
|           machine=$(uname -m) |  | ||||||
|           case ${machine} in |  | ||||||
|             x86_64) echo ARCH=amd64; echo PLATFORM_PAIR=linux-amd64 ;; |  | ||||||
|             aarch64) echo ARCH=arm64; echo PLATFORM_PAIR=linux-arm64 ;; |  | ||||||
|           esac >>$GITHUB_ENV |  | ||||||
|           echo GOFLAGS="'-ldflags=-w -s \"-X=github.com/ollama/ollama/version.Version=${{ env.DOCKER_METADATA_OUTPUT_VERSION }}\" \"-X=github.com/ollama/ollama/server.mode=release\"'" >>$GITHUB_ENV |  | ||||||
|       - name: Login to Docker Hub |  | ||||||
|         uses: docker/login-action@v3 |  | ||||||
|         with: |  | ||||||
|           username: ${{ vars.DOCKER_USER }} |  | ||||||
|           password: ${{ secrets.DOCKER_ACCESS_TOKEN }} |  | ||||||
|       - name: Create manifest list and push |  | ||||||
|         working-directory: /tmp/digests |  | ||||||
|         run: | |  | ||||||
|           docker buildx imagetools create $(jq -cr '.tags | map("-t " + .) | join(" ")' <<< "$DOCKER_METADATA_OUTPUT_JSON") \ |  | ||||||
|             $(printf '${{ env.FINAL_IMAGE_REPO }}@sha256:%s ' *) |  | ||||||
|       - name: Inspect image |  | ||||||
|         run: | |  | ||||||
|           docker buildx imagetools inspect ${{ env.FINAL_IMAGE_REPO }}:${{ steps.meta.outputs.version }}           |  | ||||||
|   build-container-image-rocm: |  | ||||||
|     environment: release |  | ||||||
|     runs-on: linux |  | ||||||
|     env: |  | ||||||
|       FINAL_IMAGE_REPO: ollama/ollama |  | ||||||
|       ARCH: amd64 |  | ||||||
|       PLATFORM_PAIR: linux-amd64 |  | ||||||
|     steps: |  | ||||||
|       - uses: actions/checkout@v4 |  | ||||||
|         with: |  | ||||||
|           submodules: recursive |  | ||||||
|       - name: Docker meta |  | ||||||
|         id: meta |  | ||||||
|         uses: docker/metadata-action@v5 |  | ||||||
|         with: |  | ||||||
|           images: ${{ env.FINAL_IMAGE_REPO }} |  | ||||||
|           flavor: | |  | ||||||
|             latest=false |  | ||||||
|           tags: | |  | ||||||
|             type=ref,enable=true,priority=600,prefix=0.0.0-pr,suffix=,event=pr |  | ||||||
|             type=semver,pattern={{version}} |  | ||||||
|       - name: Set Version |  | ||||||
|         shell: bash |  | ||||||
|         run: | |  | ||||||
|           echo GOFLAGS="'-ldflags=-w -s \"-X=github.com/ollama/ollama/version.Version=${{ env.DOCKER_METADATA_OUTPUT_VERSION }}\" \"-X=github.com/ollama/ollama/server.mode=release\"'" >>$GITHUB_ENV |  | ||||||
|       - name: Set up Docker Buildx |  | ||||||
|         uses: docker/setup-buildx-action@v3 |  | ||||||
|       - name: Login to Docker Hub |  | ||||||
|         uses: docker/login-action@v3 |  | ||||||
|         with: |  | ||||||
|           username: ${{ vars.DOCKER_USER }} |  | ||||||
|           password: ${{ secrets.DOCKER_ACCESS_TOKEN }} |  | ||||||
|       - name: Build and push by digest |  | ||||||
|         id: build |  | ||||||
|         uses: docker/build-push-action@v6 |  | ||||||
|         with: |  | ||||||
|           context: "." |  | ||||||
|           target: runtime-rocm |  | ||||||
|           build-args: | |  | ||||||
|             GOFLAGS |  | ||||||
|           tags: ${{ env.FINAL_IMAGE_REPO }}:${{ env.DOCKER_METADATA_OUTPUT_VERSION}}-rocm |  | ||||||
|           push: true |  | ||||||
|  |  | ||||||
|   # Aggregate all the assets and ship a release |  | ||||||
|   release: |  | ||||||
|     needs: |  | ||||||
|       - build-darwin |  | ||||||
|       - build-windows |  | ||||||
|       - build-linux-amd64 |  | ||||||
|       - build-linux-arm64 |  | ||||||
|     runs-on: linux |  | ||||||
|     environment: release |  | ||||||
|     permissions: |  | ||||||
|       contents: write |  | ||||||
|     env: |  | ||||||
|       GH_TOKEN: ${{ github.token }} |  | ||||||
|     steps: |  | ||||||
|       - uses: actions/checkout@v4 |  | ||||||
|       - name: Set Version |  | ||||||
|         shell: bash |  | ||||||
|         run: | |  | ||||||
|           echo "VERSION=${GITHUB_REF_NAME#v}" >> $GITHUB_ENV |  | ||||||
|           echo "RELEASE_VERSION=$(echo ${GITHUB_REF_NAME} | cut -f1 -d-)" >> $GITHUB_ENV |  | ||||||
|       - name: Retrieve built artifact |  | ||||||
|         uses: actions/download-artifact@v4 |  | ||||||
|         with: |  | ||||||
|           path: dist |  | ||||||
|           pattern: dist-* |  | ||||||
|           merge-multiple: true |  | ||||||
|       - run: | |  | ||||||
|           ls -lh dist/ |  | ||||||
|           (cd dist; find . -type f | xargs sha256sum > ../sha256sum.txt) |  | ||||||
|           mv sha256sum.txt dist/ |  | ||||||
|           cat dist/sha256sum.txt |  | ||||||
|       - name: Create or update Release |  | ||||||
|         run: | |  | ||||||
|           echo "Looking for existing release for ${{ env.RELEASE_VERSION }}" |  | ||||||
|           OLD_TAG=$(gh release ls --json name,tagName | jq -r ".[] | select(.name == \"${{ env.RELEASE_VERSION }}\") | .tagName") |  | ||||||
|           if [ -n "$OLD_TAG" ]; then |  | ||||||
|             echo "Updating release ${{ env.RELEASE_VERSION }} to point to new tag ${GITHUB_REF_NAME}" |  | ||||||
|             gh release edit ${OLD_TAG} --tag ${GITHUB_REF_NAME} |  | ||||||
|           else |  | ||||||
|             echo "Creating new release ${{ env.RELEASE_VERSION }} pointing to tag ${GITHUB_REF_NAME}" |  | ||||||
|             gh release create ${GITHUB_REF_NAME} \ |  | ||||||
|               --title ${{ env.RELEASE_VERSION }} \ |  | ||||||
|               --draft \ |  | ||||||
|               --generate-notes \ |  | ||||||
|               --prerelease |  | ||||||
|           fi |  | ||||||
|           echo "Uploading artifacts for tag ${GITHUB_REF_NAME}" |  | ||||||
|           gh release upload ${GITHUB_REF_NAME} dist/* --clobber |  | ||||||
							
								
								
									
										336
									
								
								.github/workflows/test.yaml
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -1,336 +0,0 @@ | |||||||
| name: test |  | ||||||
|  |  | ||||||
| concurrency: |  | ||||||
|   # For PRs, later CI runs preempt previous ones. e.g. a force push on a PR |  | ||||||
|   # cancels running CI jobs and starts all new ones. |  | ||||||
|   # |  | ||||||
|   # For non-PR pushes, concurrency.group needs to be unique for every distinct |  | ||||||
|   # CI run we want to have happen. Use run_id, which in practice means all |  | ||||||
|   # non-PR CI runs will be allowed to run without preempting each other. |  | ||||||
|   group: ${{ github.workflow }}-$${{ github.pull_request.number || github.run_id }} |  | ||||||
|   cancel-in-progress: true |  | ||||||
|  |  | ||||||
| on: |  | ||||||
|   pull_request: |  | ||||||
|     paths: |  | ||||||
|       - '**/*' |  | ||||||
|       - '!docs/**' |  | ||||||
|       - '!README.md' |  | ||||||
|  |  | ||||||
| jobs: |  | ||||||
|   changes: |  | ||||||
|     runs-on: ubuntu-latest |  | ||||||
|     outputs: |  | ||||||
|       GENERATE: ${{ steps.changes.outputs.GENERATE }} |  | ||||||
|       GENERATE_CUDA: ${{ steps.changes.outputs.GENERATE_CUDA }} |  | ||||||
|       GENERATE_ROCM: ${{ steps.changes.outputs.GENERATE_ROCM }} |  | ||||||
|       RUNNERS: ${{ steps.changes.outputs.RUNNERS }} |  | ||||||
|     steps: |  | ||||||
|       - uses: actions/checkout@v4 |  | ||||||
|         with: |  | ||||||
|           fetch-depth: 0 |  | ||||||
|       - id: changes |  | ||||||
|         run: | |  | ||||||
|           changed() { |  | ||||||
|             git diff-tree -r --no-commit-id --name-only \ |  | ||||||
|               $(git merge-base ${{ github.event.pull_request.base.sha }} ${{ github.event.pull_request.head.sha }}) \ |  | ||||||
|               ${{ github.event.pull_request.head.sha }} \ |  | ||||||
|               | xargs python3 -c "import sys; from pathlib import Path; print(any(Path(x).match(glob) for x in sys.argv[1:] for glob in '$*'.split(' ')))" |  | ||||||
|           } |  | ||||||
|  |  | ||||||
|           { |  | ||||||
|             echo GENERATE=$(changed 'llm/llama.cpp' 'llm/patches/**' 'llm/ext_server/**' 'llm/generate/**') |  | ||||||
|             echo GENERATE_CUDA=$(changed 'llm/llama.cpp' 'llm/patches/**' 'llm/ext_server/**' 'llm/generate/**') |  | ||||||
|             echo GENERATE_ROCM=$(changed 'llm/llama.cpp' 'llm/patches/**' 'llm/ext_server/**' 'llm/generate/**') |  | ||||||
|             echo RUNNERS=$(changed 'llama/**') |  | ||||||
|           } >>$GITHUB_OUTPUT |  | ||||||
|  |  | ||||||
|   generate: |  | ||||||
|     needs: [changes] |  | ||||||
|     if: ${{ needs.changes.outputs.GENERATE == 'True' }} |  | ||||||
|     strategy: |  | ||||||
|       matrix: |  | ||||||
|         os: [ubuntu-latest, macos-latest, windows-2019] |  | ||||||
|         arch: [amd64, arm64] |  | ||||||
|         exclude: |  | ||||||
|           - os: ubuntu-latest |  | ||||||
|             arch: arm64 |  | ||||||
|           - os: windows-2019 |  | ||||||
|             arch: arm64 |  | ||||||
|     runs-on: ${{ matrix.os }} |  | ||||||
|     env: |  | ||||||
|       GOARCH: ${{ matrix.arch }} |  | ||||||
|       CGO_ENABLED: '1' |  | ||||||
|     steps: |  | ||||||
|       - uses: actions/checkout@v4 |  | ||||||
|       - uses: actions/setup-go@v5 |  | ||||||
|         with: |  | ||||||
|           go-version-file: go.mod |  | ||||||
|           cache: true |  | ||||||
|       - run: go get ./... |  | ||||||
|       - run: | |  | ||||||
|           $gopath=(get-command go).source | split-path -parent |  | ||||||
|           $gccpath=(get-command gcc).source | split-path -parent |  | ||||||
|           & "C:\Program Files (x86)\Microsoft Visual Studio\2019\Enterprise\Common7\Tools\Launch-VsDevShell.ps1" |  | ||||||
|           cd $env:GITHUB_WORKSPACE |  | ||||||
|           $env:CMAKE_SYSTEM_VERSION="10.0.22621.0" |  | ||||||
|           $env:PATH="$gopath;$gccpath;$env:PATH" |  | ||||||
|           echo $env:PATH |  | ||||||
|           go generate -x ./... |  | ||||||
|         if: ${{ startsWith(matrix.os, 'windows-') }} |  | ||||||
|         name: 'Windows Go Generate' |  | ||||||
|       - run: go generate -x ./... |  | ||||||
|         if: ${{ ! startsWith(matrix.os, 'windows-') }} |  | ||||||
|         name: 'Unix Go Generate' |  | ||||||
|       - run: go build . |  | ||||||
|   generate-cuda: |  | ||||||
|     needs: [changes] |  | ||||||
|     if: ${{ needs.changes.outputs.GENERATE_CUDA == 'True' }} |  | ||||||
|     strategy: |  | ||||||
|       matrix: |  | ||||||
|         cuda-version: |  | ||||||
|           - '11.8.0' |  | ||||||
|     runs-on: linux |  | ||||||
|     container: nvidia/cuda:${{ matrix.cuda-version }}-devel-ubuntu20.04 |  | ||||||
|     steps: |  | ||||||
|       - run: | |  | ||||||
|           apt-get update && apt-get install -y git build-essential curl |  | ||||||
|           curl -fsSL https://github.com/Kitware/CMake/releases/download/v3.28.1/cmake-3.28.1-linux-x86_64.tar.gz \ |  | ||||||
|             | tar -zx -C /usr --strip-components 1 |  | ||||||
|         env: |  | ||||||
|           DEBIAN_FRONTEND: noninteractive |  | ||||||
|       - uses: actions/checkout@v4 |  | ||||||
|       - uses: actions/setup-go@v4 |  | ||||||
|         with: |  | ||||||
|           go-version-file: go.mod |  | ||||||
|           cache: true |  | ||||||
|       - run: go get ./... |  | ||||||
|       - run: | |  | ||||||
|           git config --global --add safe.directory /__w/ollama/ollama |  | ||||||
|           go generate -x ./... |  | ||||||
|         env: |  | ||||||
|           OLLAMA_SKIP_CPU_GENERATE: '1' |  | ||||||
|   generate-rocm: |  | ||||||
|     needs: [changes] |  | ||||||
|     if: ${{ needs.changes.outputs.GENERATE_ROCM == 'True' }} |  | ||||||
|     strategy: |  | ||||||
|       matrix: |  | ||||||
|         rocm-version: |  | ||||||
|           - '6.1.2' |  | ||||||
|     runs-on: linux |  | ||||||
|     container: rocm/dev-ubuntu-20.04:${{ matrix.rocm-version }} |  | ||||||
|     steps: |  | ||||||
|       - run: | |  | ||||||
|           apt-get update && apt-get install -y git build-essential curl rocm-libs |  | ||||||
|           curl -fsSL https://github.com/Kitware/CMake/releases/download/v3.28.1/cmake-3.28.1-linux-x86_64.tar.gz \ |  | ||||||
|             | tar -zx -C /usr --strip-components 1 |  | ||||||
|         env: |  | ||||||
|           DEBIAN_FRONTEND: noninteractive |  | ||||||
|       - uses: actions/checkout@v4 |  | ||||||
|       - uses: actions/setup-go@v4 |  | ||||||
|         with: |  | ||||||
|           go-version-file: go.mod |  | ||||||
|           cache: true |  | ||||||
|       - run: go get ./... |  | ||||||
|       - run: | |  | ||||||
|           git config --global --add safe.directory /__w/ollama/ollama |  | ||||||
|           go generate -x ./... |  | ||||||
|         env: |  | ||||||
|           OLLAMA_SKIP_CPU_GENERATE: '1' |  | ||||||
|  |  | ||||||
|   # ROCm generation step |  | ||||||
|   generate-windows-rocm: |  | ||||||
|     needs: [changes] |  | ||||||
|     if: ${{ needs.changes.outputs.GENERATE_ROCM == 'True' }} |  | ||||||
|     runs-on: windows |  | ||||||
|     steps: |  | ||||||
|       - uses: actions/checkout@v4 |  | ||||||
|       - uses: actions/setup-go@v5 |  | ||||||
|         with: |  | ||||||
|           go-version-file: go.mod |  | ||||||
|           cache: true |  | ||||||
|       - name: 'Install ROCm' |  | ||||||
|         run: | |  | ||||||
|           $ErrorActionPreference = "Stop" |  | ||||||
|           write-host "downloading AMD HIP Installer" |  | ||||||
|           Invoke-WebRequest -Uri "https://download.amd.com/developer/eula/rocm-hub/AMD-Software-PRO-Edition-24.Q3-WinSvr2022-For-HIP.exe" -OutFile "${env:RUNNER_TEMP}\rocm-install.exe" |  | ||||||
|           write-host "Installing AMD HIP" |  | ||||||
|           Start-Process "${env:RUNNER_TEMP}\rocm-install.exe" -ArgumentList '-install' -NoNewWindow -Wait |  | ||||||
|           write-host "Completed AMD HIP" |  | ||||||
|       - name: 'Verify ROCm' |  | ||||||
|         run: | |  | ||||||
|           & 'C:\Program Files\AMD\ROCm\*\bin\clang.exe' --version |  | ||||||
|       - run: go get ./... |  | ||||||
|       - run: | |  | ||||||
|           $gopath=(get-command go).source | split-path -parent |  | ||||||
|           & "C:\Program Files (x86)\Microsoft Visual Studio\2019\Enterprise\Common7\Tools\Launch-VsDevShell.ps1" |  | ||||||
|           cd $env:GITHUB_WORKSPACE |  | ||||||
|           $env:CMAKE_SYSTEM_VERSION="10.0.22621.0" |  | ||||||
|           $env:PATH="$gopath;$env:PATH" |  | ||||||
|           $env:OLLAMA_SKIP_CPU_GENERATE="1" |  | ||||||
|           $env:HIP_PATH=$(Resolve-Path 'C:\Program Files\AMD\ROCm\*\bin\clang.exe' | split-path | split-path) |  | ||||||
|           go generate -x ./... |  | ||||||
|         name: go generate |  | ||||||
|         env: |  | ||||||
|           OLLAMA_SKIP_CPU_GENERATE: '1' |  | ||||||
|  |  | ||||||
|   # CUDA generation step |  | ||||||
|   generate-windows-cuda: |  | ||||||
|     needs: [changes] |  | ||||||
|     if: ${{ needs.changes.outputs.GENERATE_CUDA == 'True' }} |  | ||||||
|     runs-on: windows |  | ||||||
|     steps: |  | ||||||
|       - uses: actions/checkout@v4 |  | ||||||
|       - uses: actions/setup-go@v5 |  | ||||||
|         with: |  | ||||||
|           go-version-file: go.mod |  | ||||||
|           cache: true |  | ||||||
|       - name: 'Install CUDA' |  | ||||||
|         run: | |  | ||||||
|           $ErrorActionPreference = "Stop" |  | ||||||
|           write-host "downloading CUDA Installer" |  | ||||||
|           Invoke-WebRequest -Uri "https://developer.download.nvidia.com/compute/cuda/11.3.1/local_installers/cuda_11.3.1_465.89_win10.exe" -OutFile "${env:RUNNER_TEMP}\cuda-install.exe" |  | ||||||
|           write-host "Installing CUDA" |  | ||||||
|           Start-Process "${env:RUNNER_TEMP}\cuda-install.exe" -ArgumentList '-s' -NoNewWindow -Wait |  | ||||||
|           write-host "Completed CUDA" |  | ||||||
|           $cudaPath=((resolve-path "c:\Program Files\NVIDIA*\CUDA\v*\bin\nvcc.exe")[0].path | split-path | split-path) |  | ||||||
|           $cudaVer=($cudaPath | split-path -leaf ) -replace 'v(\d+).(\d+)', '$1_$2'  |  | ||||||
|           echo "$cudaPath\bin" >> $env:GITHUB_PATH |  | ||||||
|           echo "CUDA_PATH=$cudaPath" >> $env:GITHUB_ENV |  | ||||||
|           echo "CUDA_PATH_V${cudaVer}=$cudaPath" >> $env:GITHUB_ENV |  | ||||||
|           echo "CUDA_PATH_VX_Y=CUDA_PATH_V${cudaVer}" >> $env:GITHUB_ENV |  | ||||||
|       - name: 'Verify CUDA' |  | ||||||
|         run: nvcc -V |  | ||||||
|       - run: go get ./... |  | ||||||
|       - name: go generate |  | ||||||
|         run: | |  | ||||||
|           $gopath=(get-command go).source | split-path -parent |  | ||||||
|           $cudabin=(get-command nvcc).source | split-path |  | ||||||
|           & "C:\Program Files (x86)\Microsoft Visual Studio\2019\Enterprise\Common7\Tools\Launch-VsDevShell.ps1" |  | ||||||
|           cd $env:GITHUB_WORKSPACE |  | ||||||
|           $env:CMAKE_SYSTEM_VERSION="10.0.22621.0" |  | ||||||
|           $env:PATH="$gopath;$cudabin;$env:PATH" |  | ||||||
|           $env:OLLAMA_SKIP_CPU_GENERATE="1" |  | ||||||
|           go generate -x ./... |  | ||||||
|         env: |  | ||||||
|           OLLAMA_SKIP_CPU_GENERATE: '1' |  | ||||||
|  |  | ||||||
|   runners: |  | ||||||
|     needs: [changes] |  | ||||||
|     if: ${{ needs.changes.outputs.RUNNERS == 'True' }} |  | ||||||
|     strategy: |  | ||||||
|       matrix: |  | ||||||
|         os: [ubuntu-latest, macos-latest, windows-2019] |  | ||||||
|         arch: [amd64, arm64] |  | ||||||
|         exclude: |  | ||||||
|           - os: ubuntu-latest |  | ||||||
|             arch: arm64 |  | ||||||
|           - os: windows-2019 |  | ||||||
|             arch: arm64 |  | ||||||
|     runs-on: ${{ matrix.os }} |  | ||||||
|     env: |  | ||||||
|       GOARCH: ${{ matrix.arch }} |  | ||||||
|       ARCH: ${{ matrix.arch }} |  | ||||||
|       CGO_ENABLED: '1' |  | ||||||
|     steps: |  | ||||||
|       - uses: actions/checkout@v4 |  | ||||||
|       - uses: actions/setup-go@v5 |  | ||||||
|         with: |  | ||||||
|           go-version-file: go.mod |  | ||||||
|           cache: true |  | ||||||
|       - run: go get ./... |  | ||||||
|       - name: 'Build Windows Go Runners' |  | ||||||
|         if: ${{ startsWith(matrix.os, 'windows-') }} |  | ||||||
|         run: | |  | ||||||
|           $gopath=(get-command go).source | split-path -parent |  | ||||||
|           $gccpath=(get-command gcc).source | split-path -parent |  | ||||||
|           & "C:\Program Files (x86)\Microsoft Visual Studio\2019\Enterprise\Common7\Tools\Launch-VsDevShell.ps1" |  | ||||||
|           cd $env:GITHUB_WORKSPACE |  | ||||||
|           $env:CMAKE_SYSTEM_VERSION="10.0.22621.0" |  | ||||||
|           $env:PATH="$gopath;$gccpath;$env:PATH" |  | ||||||
|           echo $env:PATH |  | ||||||
|           make -C llama -j 4       |  | ||||||
|       - name: 'Build Unix Go Runners' |  | ||||||
|         if: ${{ ! startsWith(matrix.os, 'windows-') }} |  | ||||||
|         run: make -C llama -j 4 |  | ||||||
|       - run: go build . |  | ||||||
|  |  | ||||||
|   lint: |  | ||||||
|     strategy: |  | ||||||
|       matrix: |  | ||||||
|         os: [ubuntu-latest, macos-latest, windows-2019] |  | ||||||
|         arch: [amd64, arm64] |  | ||||||
|         exclude: |  | ||||||
|           - os: ubuntu-latest |  | ||||||
|             arch: arm64 |  | ||||||
|           - os: windows-2019 |  | ||||||
|             arch: arm64 |  | ||||||
|           - os: macos-latest |  | ||||||
|             arch: amd64 |  | ||||||
|     runs-on: ${{ matrix.os }} |  | ||||||
|     env: |  | ||||||
|       GOARCH: ${{ matrix.arch }} |  | ||||||
|       CGO_ENABLED: '1' |  | ||||||
|     steps: |  | ||||||
|       - uses: actions/checkout@v4 |  | ||||||
|         with: |  | ||||||
|           submodules: recursive |  | ||||||
|       - uses: actions/setup-go@v5 |  | ||||||
|         with: |  | ||||||
|           go-version-file: go.mod |  | ||||||
|           cache: false |  | ||||||
|       - run: | |  | ||||||
|           case ${{ matrix.arch }} in |  | ||||||
|             amd64) echo ARCH=x86_64 ;; |  | ||||||
|             arm64) echo ARCH=arm64 ;; |  | ||||||
|           esac >>$GITHUB_ENV |  | ||||||
|         shell: bash |  | ||||||
|       - uses: golangci/golangci-lint-action@v6 |  | ||||||
|         with: |  | ||||||
|           args: --timeout 8m0s -v |  | ||||||
|   test: |  | ||||||
|     strategy: |  | ||||||
|       matrix: |  | ||||||
|         os: [ubuntu-latest, macos-latest, windows-2019] |  | ||||||
|         arch: [amd64] |  | ||||||
|         exclude: |  | ||||||
|           - os: ubuntu-latest |  | ||||||
|             arch: arm64 |  | ||||||
|           - os: windows-2019 |  | ||||||
|             arch: arm64 |  | ||||||
|     runs-on: ${{ matrix.os }} |  | ||||||
|     env: |  | ||||||
|       GOARCH: ${{ matrix.arch }} |  | ||||||
|       CGO_ENABLED: '1' |  | ||||||
|       OLLAMA_CPU_TARGET: 'static' |  | ||||||
|       OLLAMA_SKIP_CPU_GENERATE: '1' |  | ||||||
|       OLLAMA_SKIP_METAL_GENERATE: '1' |  | ||||||
|     steps: |  | ||||||
|       - uses: actions/checkout@v4 |  | ||||||
|         with: |  | ||||||
|           submodules: recursive |  | ||||||
|       - uses: actions/setup-go@v5 |  | ||||||
|         with: |  | ||||||
|           go-version-file: go.mod |  | ||||||
|           cache: true |  | ||||||
|       - run: | |  | ||||||
|           case ${{ matrix.arch }} in |  | ||||||
|             amd64) echo ARCH=amd64 ;; |  | ||||||
|             arm64) echo ARCH=arm64 ;; |  | ||||||
|           esac >>$GITHUB_ENV |  | ||||||
|         shell: bash |  | ||||||
|       - run: go generate ./... |  | ||||||
|       - run: go build |  | ||||||
|       - run: go test -v ./... |  | ||||||
|  |  | ||||||
|   patches: |  | ||||||
|     needs: [changes] |  | ||||||
|     if: ${{ needs.changes.outputs.RUNNERS == 'True' }} |  | ||||||
|     runs-on: ubuntu-latest |  | ||||||
|     steps: |  | ||||||
|       - uses: actions/checkout@v4 |  | ||||||
|         with: |  | ||||||
|           submodules: recursive |  | ||||||
|       - name: Verify patches carry all the changes |  | ||||||
|         run: | |  | ||||||
|           cd llama && ./sync.sh && git diff --compact-summary --exit-code . |  | ||||||
							
								
								
									
										10
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -5,13 +5,7 @@ | |||||||
| .swp | .swp | ||||||
| dist | dist | ||||||
| ollama | ollama | ||||||
|  | ggml-metal.metal | ||||||
| .cache | .cache | ||||||
| *.exe | *.exe | ||||||
| .idea | .idea | ||||||
| test_data |  | ||||||
| *.crt |  | ||||||
| llm/build |  | ||||||
| build/*/*/* |  | ||||||
| !build/**/placeholder |  | ||||||
| llama/build |  | ||||||
| __debug_bin* |  | ||||||
							
								
								
									
										14
									
								
								.gitmodules
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -1,4 +1,10 @@ | |||||||
| [submodule "llama.cpp"] | [submodule "llm/llama.cpp/ggml"] | ||||||
| 	path = llm/llama.cpp |     path = llm/llama.cpp/ggml | ||||||
| 	url = https://github.com/ggerganov/llama.cpp.git |     url = https://github.com/ggerganov/llama.cpp.git | ||||||
| 	shallow = true |     ignore = dirty | ||||||
|  |     shallow = true | ||||||
|  | [submodule "llm/llama.cpp/gguf"] | ||||||
|  |     path = llm/llama.cpp/gguf | ||||||
|  |     url = https://github.com/ggerganov/llama.cpp.git | ||||||
|  |     ignore = dirty | ||||||
|  |     shallow = true | ||||||
|   | |||||||
| @@ -1,47 +0,0 @@ | |||||||
| run: |  | ||||||
|   timeout: 5m |  | ||||||
| linters: |  | ||||||
|   enable: |  | ||||||
|     - asasalint |  | ||||||
|     - bidichk |  | ||||||
|     - bodyclose |  | ||||||
|     - containedctx |  | ||||||
|     - contextcheck |  | ||||||
|     - errcheck |  | ||||||
|     - exportloopref |  | ||||||
|     - gci |  | ||||||
|     - gocheckcompilerdirectives |  | ||||||
|     - gofmt |  | ||||||
|     - gofumpt |  | ||||||
|     - gosimple |  | ||||||
|     - govet |  | ||||||
|     - ineffassign |  | ||||||
|     - intrange |  | ||||||
|     - makezero |  | ||||||
|     - misspell |  | ||||||
|     - nilerr |  | ||||||
|     - nolintlint |  | ||||||
|     - nosprintfhostport |  | ||||||
|     - staticcheck |  | ||||||
|     - tenv |  | ||||||
|     - unconvert |  | ||||||
|     - unused |  | ||||||
|     - usestdlibvars |  | ||||||
|     - wastedassign |  | ||||||
|     - whitespace |  | ||||||
| linters-settings: |  | ||||||
|   gci: |  | ||||||
|     sections: [standard, default, localmodule] |  | ||||||
|   staticcheck: |  | ||||||
|     checks: |  | ||||||
|       - all |  | ||||||
|       - -SA1019 # omit Deprecated check |  | ||||||
| severity: |  | ||||||
|   default-severity: error |  | ||||||
|   rules: |  | ||||||
|     - linters: |  | ||||||
|         - gofmt |  | ||||||
|         - goimports |  | ||||||
|         - intrange |  | ||||||
|         - usestdlibvars |  | ||||||
|       severity: info |  | ||||||
| @@ -1,37 +0,0 @@ | |||||||
| # Contributing to Ollama |  | ||||||
|  |  | ||||||
| Thank you for your interest in contributing to Ollama! Here are a few guidelines to help get you started. |  | ||||||
|  |  | ||||||
| ## Set up |  | ||||||
|  |  | ||||||
| See the [development documentation](./docs/development.md) for instructions on how to build and run Ollama locally. |  | ||||||
|  |  | ||||||
| ## Pull requests |  | ||||||
|  |  | ||||||
| ### Ideal issues |  | ||||||
|  |  | ||||||
| * [Bugs](https://github.com/ollama/ollama/issues?q=is%3Aissue+is%3Aopen+label%3Abug): issues where Ollama stops working or where it results in an unexpected error. |  | ||||||
| * [Performance](https://github.com/ollama/ollama/issues?q=is%3Aissue+is%3Aopen+label%3Aperformance): issues to make Ollama faster at model inference, downloading or uploading. |  | ||||||
| * [Security](https://github.com/ollama/ollama/blob/main/SECURITY.md): issues that could lead to a security vulnerability. As mentioned in [SECURITY.md](https://github.com/ollama/ollama/blob/main/SECURITY.md), please do not disclose security vulnerabilities publicly. |  | ||||||
|  |  | ||||||
| ### Issues that are harder to review |  | ||||||
|  |  | ||||||
| * New features: new features (e.g. API fields, environment variables) add surface area to Ollama and make it harder to maintain in the long run as they cannot be removed without potentially breaking users in the future. |  | ||||||
| * Refactoring: large code improvements are important, but can be harder or take longer to review and merge. |  | ||||||
| * Documentation: small updates to fill in or correct missing documentation is helpful, however large documentation additions can be hard to maintain over time. |  | ||||||
|  |  | ||||||
| ### Issues that may not be accepted |  | ||||||
|  |  | ||||||
| * Changes that break backwards compatibility in Ollama's API (including the OpenAI-compatible API) |  | ||||||
| * Changes that add significant friction to the user experience |  | ||||||
| * Changes that create a large future maintenance burden for maintainers and contributors |  | ||||||
|  |  | ||||||
| ### Best practices |  | ||||||
|  |  | ||||||
| * Commit messages: please leave both a title and a description in your commit messages. The title should be a short summary of the changes, with a leading word that explains the section of the code being changed (e.g. `api: fix parsing of prompt field`) . In the description, leave a short 2-3 sentences that explain more about the change and its impact. |  | ||||||
| * Tests: please add test coverage to changes where possible. |  | ||||||
| * Minimize dependencies: avoid adding new dependencies unless absolutely necessary. |  | ||||||
|  |  | ||||||
| ## Need help? |  | ||||||
|  |  | ||||||
| If you need help with anything, feel free to reach out to us on our [Discord server](https://discord.gg/ollama). |  | ||||||
							
								
								
									
										257
									
								
								Dockerfile
									
									
									
									
									
								
							
							
						
						| @@ -1,250 +1,23 @@ | |||||||
| ARG GOLANG_VERSION=1.22.5 | FROM nvidia/cuda:11.8.0-devel-ubuntu22.04 | ||||||
| ARG CMAKE_VERSION=3.22.1 |  | ||||||
| ARG CUDA_VERSION_11=11.3.1 |  | ||||||
| ARG CUDA_V11_ARCHITECTURES="50;52;53;60;61;62;70;72;75;80;86" |  | ||||||
| ARG CUDA_VERSION_12=12.4.0 |  | ||||||
| ARG CUDA_V12_ARCHITECTURES="60;61;62;70;72;75;80;86;87;89;90;90a" |  | ||||||
| ARG ROCM_VERSION=6.1.2 |  | ||||||
|  |  | ||||||
| # Copy the minimal context we need to run the generate scripts | ARG TARGETARCH | ||||||
| FROM scratch AS llm-code | ARG GOFLAGS="'-ldflags=-w -s'" | ||||||
| COPY .git .git |  | ||||||
| COPY .gitmodules .gitmodules |  | ||||||
| COPY llm llm |  | ||||||
|  |  | ||||||
| FROM --platform=linux/amd64 nvidia/cuda:$CUDA_VERSION_11-devel-centos7 AS cuda-11-build-amd64 | WORKDIR /go/src/github.com/jmorganca/ollama | ||||||
| ARG CMAKE_VERSION | RUN apt-get update && apt-get install -y git build-essential cmake | ||||||
| COPY ./scripts/rh_linux_deps.sh / | ADD https://dl.google.com/go/go1.21.3.linux-$TARGETARCH.tar.gz /tmp/go1.21.3.tar.gz | ||||||
| RUN CMAKE_VERSION=${CMAKE_VERSION} sh /rh_linux_deps.sh | RUN mkdir -p /usr/local && tar xz -C /usr/local </tmp/go1.21.3.tar.gz | ||||||
| ENV PATH=/opt/rh/devtoolset-10/root/usr/bin:$PATH |  | ||||||
| COPY --from=llm-code / /go/src/github.com/ollama/ollama/ |  | ||||||
| WORKDIR /go/src/github.com/ollama/ollama/llm/generate |  | ||||||
| ARG CGO_CFLAGS |  | ||||||
| ARG CUDA_V11_ARCHITECTURES |  | ||||||
| ENV GOARCH=amd64 |  | ||||||
| RUN --mount=type=cache,target=/root/.ccache \ |  | ||||||
|     OLLAMA_SKIP_STATIC_GENERATE=1 \ |  | ||||||
|     OLLAMA_SKIP_CPU_GENERATE=1 \ |  | ||||||
|     CMAKE_CUDA_ARCHITECTURES="${CUDA_V11_ARCHITECTURES}" \ |  | ||||||
|     CUDA_VARIANT="_v11" \ |  | ||||||
|     bash gen_linux.sh |  | ||||||
|  |  | ||||||
| FROM --platform=linux/amd64 nvidia/cuda:$CUDA_VERSION_12-devel-centos7 AS cuda-12-build-amd64 |  | ||||||
| ARG CMAKE_VERSION |  | ||||||
| COPY ./scripts/rh_linux_deps.sh / |  | ||||||
| RUN CMAKE_VERSION=${CMAKE_VERSION} sh /rh_linux_deps.sh |  | ||||||
| ENV PATH=/opt/rh/devtoolset-10/root/usr/bin:$PATH |  | ||||||
| COPY --from=llm-code / /go/src/github.com/ollama/ollama/ |  | ||||||
| WORKDIR /go/src/github.com/ollama/ollama/llm/generate |  | ||||||
| ARG CGO_CFLAGS |  | ||||||
| ARG CUDA_V12_ARCHITECTURES |  | ||||||
| ENV GOARCH=amd64 |  | ||||||
| RUN --mount=type=cache,target=/root/.ccache \ |  | ||||||
|     OLLAMA_SKIP_STATIC_GENERATE=1 \ |  | ||||||
|     OLLAMA_SKIP_CPU_GENERATE=1 \ |  | ||||||
|     CMAKE_CUDA_ARCHITECTURES="${CUDA_V12_ARCHITECTURES}" \ |  | ||||||
|     CUDA_VARIANT="_v12" \ |  | ||||||
|     OLLAMA_CUSTOM_CUDA_DEFS="-DGGML_CUDA_USE_GRAPHS=on" \ |  | ||||||
|     bash gen_linux.sh |  | ||||||
|  |  | ||||||
| FROM --platform=linux/arm64 nvidia/cuda:$CUDA_VERSION_11-devel-rockylinux8 AS cuda-11-build-runner-arm64 |  | ||||||
| ARG CMAKE_VERSION |  | ||||||
| COPY ./scripts/rh_linux_deps.sh / |  | ||||||
| RUN CMAKE_VERSION=${CMAKE_VERSION} sh /rh_linux_deps.sh |  | ||||||
| ENV PATH=/opt/rh/gcc-toolset-10/root/usr/bin:$PATH |  | ||||||
| COPY --from=llm-code / /go/src/github.com/ollama/ollama/ |  | ||||||
| WORKDIR /go/src/github.com/ollama/ollama/llm/generate |  | ||||||
| ARG CGO_CFLAGS |  | ||||||
| ARG CUDA_V11_ARCHITECTURES |  | ||||||
| ENV GOARCH=arm64 |  | ||||||
| RUN OLLAMA_SKIP_STATIC_GENERATE=1 \ |  | ||||||
|     OLLAMA_SKIP_CPU_GENERATE=1 \ |  | ||||||
|     CMAKE_CUDA_ARCHITECTURES="${CUDA_V11_ARCHITECTURES}" \ |  | ||||||
|     CUDA_VARIANT="_v11" \ |  | ||||||
|     bash gen_linux.sh |  | ||||||
|  |  | ||||||
| FROM --platform=linux/arm64 nvidia/cuda:$CUDA_VERSION_12-devel-rockylinux8 AS cuda-12-build-runner-arm64 |  | ||||||
| ARG CMAKE_VERSION |  | ||||||
| COPY ./scripts/rh_linux_deps.sh / |  | ||||||
| RUN CMAKE_VERSION=${CMAKE_VERSION} sh /rh_linux_deps.sh |  | ||||||
| ENV PATH=/opt/rh/gcc-toolset-10/root/usr/bin:$PATH |  | ||||||
| COPY --from=llm-code / /go/src/github.com/ollama/ollama/ |  | ||||||
| WORKDIR /go/src/github.com/ollama/ollama/llm/generate |  | ||||||
| ARG CGO_CFLAGS |  | ||||||
| ARG CUDA_V12_ARCHITECTURES |  | ||||||
| ENV GOARCH=arm64 |  | ||||||
| RUN --mount=type=cache,target=/root/.ccache \ |  | ||||||
|     OLLAMA_SKIP_STATIC_GENERATE=1 \ |  | ||||||
|     OLLAMA_SKIP_CPU_GENERATE=1 \ |  | ||||||
|     CMAKE_CUDA_ARCHITECTURES="${CUDA_V12_ARCHITECTURES}" \ |  | ||||||
|     CUDA_VARIANT="_v12" \ |  | ||||||
|     OLLAMA_CUSTOM_CUDA_DEFS="-DGGML_CUDA_USE_GRAPHS=on" \ |  | ||||||
|     bash gen_linux.sh |  | ||||||
|  |  | ||||||
|  |  | ||||||
| FROM --platform=linux/amd64 rocm/dev-centos-7:${ROCM_VERSION}-complete AS rocm-build-amd64 |  | ||||||
| ARG CMAKE_VERSION |  | ||||||
| COPY ./scripts/rh_linux_deps.sh / |  | ||||||
| RUN CMAKE_VERSION=${CMAKE_VERSION} sh /rh_linux_deps.sh |  | ||||||
| ENV PATH=/opt/rh/devtoolset-10/root/usr/bin:$PATH |  | ||||||
| ENV LIBRARY_PATH=/opt/amdgpu/lib64 |  | ||||||
| COPY --from=llm-code / /go/src/github.com/ollama/ollama/ |  | ||||||
| WORKDIR /go/src/github.com/ollama/ollama/llm/generate |  | ||||||
| ARG CGO_CFLAGS |  | ||||||
| ARG AMDGPU_TARGETS |  | ||||||
| ENV GOARCH=amd64 |  | ||||||
| RUN --mount=type=cache,target=/root/.ccache \ |  | ||||||
|     OLLAMA_SKIP_STATIC_GENERATE=1 OLLAMA_SKIP_CPU_GENERATE=1 bash gen_linux.sh |  | ||||||
| RUN mkdir -p ../../dist/linux-amd64-rocm/lib/ollama && \ |  | ||||||
|     (cd /opt/rocm/lib && tar cf - rocblas/library) | (cd ../../dist/linux-amd64-rocm/lib/ollama && tar xf - ) |  | ||||||
|  |  | ||||||
| FROM --platform=linux/amd64 centos:7 AS cpu-builder-amd64 |  | ||||||
| ARG CMAKE_VERSION |  | ||||||
| ARG GOLANG_VERSION |  | ||||||
| COPY ./scripts/rh_linux_deps.sh / |  | ||||||
| RUN CMAKE_VERSION=${CMAKE_VERSION} GOLANG_VERSION=${GOLANG_VERSION} sh /rh_linux_deps.sh |  | ||||||
| ENV PATH=/opt/rh/devtoolset-10/root/usr/bin:$PATH |  | ||||||
| COPY --from=llm-code / /go/src/github.com/ollama/ollama/ |  | ||||||
| ARG OLLAMA_CUSTOM_CPU_DEFS |  | ||||||
| ARG CGO_CFLAGS |  | ||||||
| ENV GOARCH=amd64 |  | ||||||
| WORKDIR /go/src/github.com/ollama/ollama/llm/generate |  | ||||||
|  |  | ||||||
| FROM --platform=linux/amd64 cpu-builder-amd64 AS cpu-build-amd64 |  | ||||||
| RUN --mount=type=cache,target=/root/.ccache \ |  | ||||||
|     OLLAMA_SKIP_STATIC_GENERATE=1 OLLAMA_CPU_TARGET="cpu" bash gen_linux.sh |  | ||||||
| FROM --platform=linux/amd64 cpu-builder-amd64 AS cpu_avx-build-amd64 |  | ||||||
| RUN --mount=type=cache,target=/root/.ccache \ |  | ||||||
|     OLLAMA_SKIP_STATIC_GENERATE=1 OLLAMA_CPU_TARGET="cpu_avx" bash gen_linux.sh |  | ||||||
| FROM --platform=linux/amd64 cpu-builder-amd64 AS cpu_avx2-build-amd64 |  | ||||||
| RUN --mount=type=cache,target=/root/.ccache \ |  | ||||||
|     OLLAMA_SKIP_STATIC_GENERATE=1 OLLAMA_CPU_TARGET="cpu_avx2" bash gen_linux.sh |  | ||||||
|  |  | ||||||
| FROM --platform=linux/arm64 rockylinux:8 AS cpu-builder-arm64 |  | ||||||
| ARG CMAKE_VERSION |  | ||||||
| ARG GOLANG_VERSION |  | ||||||
| COPY ./scripts/rh_linux_deps.sh / |  | ||||||
| RUN CMAKE_VERSION=${CMAKE_VERSION} GOLANG_VERSION=${GOLANG_VERSION} sh /rh_linux_deps.sh |  | ||||||
| ENV PATH=/opt/rh/gcc-toolset-10/root/usr/bin:$PATH |  | ||||||
| COPY --from=llm-code / /go/src/github.com/ollama/ollama/ |  | ||||||
| ARG OLLAMA_CUSTOM_CPU_DEFS |  | ||||||
| ARG CGO_CFLAGS |  | ||||||
| ENV GOARCH=arm64 |  | ||||||
| WORKDIR /go/src/github.com/ollama/ollama/llm/generate |  | ||||||
|  |  | ||||||
| FROM --platform=linux/arm64 cpu-builder-arm64 AS cpu-build-arm64 |  | ||||||
| RUN --mount=type=cache,target=/root/.ccache \ |  | ||||||
|     OLLAMA_SKIP_STATIC_GENERATE=1 OLLAMA_CPU_TARGET="cpu" bash gen_linux.sh |  | ||||||
|  |  | ||||||
|  |  | ||||||
| # Intermediate stages used for ./scripts/build_linux.sh |  | ||||||
| FROM --platform=linux/amd64 cpu-build-amd64 AS build-amd64 |  | ||||||
| ENV CGO_ENABLED=1 |  | ||||||
| WORKDIR /go/src/github.com/ollama/ollama |  | ||||||
| COPY . . | COPY . . | ||||||
| COPY --from=cpu_avx-build-amd64 /go/src/github.com/ollama/ollama/build/ build/ | ENV GOARCH=$TARGETARCH | ||||||
| COPY --from=cpu_avx2-build-amd64 /go/src/github.com/ollama/ollama/build/ build/ | ENV GOFLAGS=$GOFLAGS | ||||||
| COPY --from=cuda-11-build-amd64 /go/src/github.com/ollama/ollama/dist/ dist/ | RUN /usr/local/go/bin/go generate ./... \ | ||||||
| COPY --from=cuda-11-build-amd64 /go/src/github.com/ollama/ollama/build/ build/ |     && /usr/local/go/bin/go build . | ||||||
| COPY --from=cuda-12-build-amd64 /go/src/github.com/ollama/ollama/dist/ dist/ |  | ||||||
| COPY --from=cuda-12-build-amd64 /go/src/github.com/ollama/ollama/build/ build/ |  | ||||||
| COPY --from=rocm-build-amd64 /go/src/github.com/ollama/ollama/dist/ dist/ |  | ||||||
| COPY --from=rocm-build-amd64 /go/src/github.com/ollama/ollama/build/ build/ |  | ||||||
| ARG GOFLAGS |  | ||||||
| ARG CGO_CFLAGS |  | ||||||
| RUN --mount=type=cache,target=/root/.ccache \ |  | ||||||
|     go build -trimpath -o dist/linux-amd64/bin/ollama . |  | ||||||
| RUN cd dist/linux-$GOARCH && \ |  | ||||||
|     tar --exclude runners -cf - . | pigz --best > ../ollama-linux-$GOARCH.tgz |  | ||||||
| RUN cd dist/linux-$GOARCH-rocm && \ |  | ||||||
|     tar -cf - . | pigz --best > ../ollama-linux-$GOARCH-rocm.tgz |  | ||||||
|  |  | ||||||
| FROM --platform=linux/arm64 cpu-build-arm64 AS build-arm64 | FROM ubuntu:22.04 | ||||||
| ENV CGO_ENABLED=1 | RUN apt-get update && apt-get install -y ca-certificates | ||||||
| ARG GOLANG_VERSION | COPY --from=0 /go/src/github.com/jmorganca/ollama/ollama /bin/ollama | ||||||
| WORKDIR /go/src/github.com/ollama/ollama |  | ||||||
| COPY . . |  | ||||||
| COPY --from=cuda-11-build-runner-arm64 /go/src/github.com/ollama/ollama/dist/ dist/ |  | ||||||
| COPY --from=cuda-11-build-runner-arm64 /go/src/github.com/ollama/ollama/build/ build/ |  | ||||||
| COPY --from=cuda-12-build-runner-arm64 /go/src/github.com/ollama/ollama/dist/ dist/ |  | ||||||
| COPY --from=cuda-12-build-runner-arm64 /go/src/github.com/ollama/ollama/build/ build/ |  | ||||||
| ARG GOFLAGS |  | ||||||
| ARG CGO_CFLAGS |  | ||||||
| RUN --mount=type=cache,target=/root/.ccache \ |  | ||||||
|     go build -trimpath -o dist/linux-arm64/bin/ollama . |  | ||||||
| RUN cd dist/linux-$GOARCH && \ |  | ||||||
|     tar --exclude runners -cf - . | pigz --best > ../ollama-linux-$GOARCH.tgz |  | ||||||
|  |  | ||||||
| FROM --platform=linux/amd64 scratch AS dist-amd64 |  | ||||||
| COPY --from=build-amd64 /go/src/github.com/ollama/ollama/dist/ollama-linux-*.tgz / |  | ||||||
| FROM --platform=linux/arm64 scratch AS dist-arm64 |  | ||||||
| COPY --from=build-arm64 /go/src/github.com/ollama/ollama/dist/ollama-linux-*.tgz / |  | ||||||
| FROM dist-$TARGETARCH as dist |  | ||||||
|  |  | ||||||
|  |  | ||||||
| # Optimized container images do not cary nested payloads |  | ||||||
| FROM --platform=linux/amd64 cpu-builder-amd64 AS container-build-amd64 |  | ||||||
| WORKDIR /go/src/github.com/ollama/ollama |  | ||||||
| COPY . . |  | ||||||
| ARG GOFLAGS |  | ||||||
| ARG CGO_CFLAGS |  | ||||||
| RUN --mount=type=cache,target=/root/.ccache \ |  | ||||||
|     go build -trimpath -o dist/linux-amd64/bin/ollama . |  | ||||||
|  |  | ||||||
| FROM --platform=linux/arm64 cpu-builder-arm64 AS container-build-arm64 |  | ||||||
| WORKDIR /go/src/github.com/ollama/ollama |  | ||||||
| COPY . . |  | ||||||
| ARG GOFLAGS |  | ||||||
| ARG CGO_CFLAGS |  | ||||||
| RUN --mount=type=cache,target=/root/.ccache \ |  | ||||||
|     go build -trimpath -o dist/linux-arm64/bin/ollama . |  | ||||||
|  |  | ||||||
| FROM --platform=linux/amd64 ubuntu:22.04 AS runtime-amd64 |  | ||||||
| RUN apt-get update && \ |  | ||||||
|     apt-get install -y ca-certificates && \ |  | ||||||
|     apt-get clean && rm -rf /var/lib/apt/lists/* |  | ||||||
| COPY --from=container-build-amd64 /go/src/github.com/ollama/ollama/dist/linux-amd64/bin/ /bin/ |  | ||||||
| COPY --from=cpu-build-amd64 /go/src/github.com/ollama/ollama/dist/linux-amd64/lib/ /lib/ |  | ||||||
| COPY --from=cpu_avx-build-amd64 /go/src/github.com/ollama/ollama/dist/linux-amd64/lib/ /lib/ |  | ||||||
| COPY --from=cpu_avx2-build-amd64 /go/src/github.com/ollama/ollama/dist/linux-amd64/lib/ /lib/ |  | ||||||
| COPY --from=cuda-11-build-amd64 /go/src/github.com/ollama/ollama/dist/linux-amd64/lib/ /lib/ |  | ||||||
| COPY --from=cuda-12-build-amd64 /go/src/github.com/ollama/ollama/dist/linux-amd64/lib/ /lib/ |  | ||||||
|  |  | ||||||
| FROM --platform=linux/arm64 ubuntu:22.04 AS runtime-arm64 |  | ||||||
| RUN apt-get update && \ |  | ||||||
|     apt-get install -y ca-certificates && \ |  | ||||||
|     apt-get clean && rm -rf /var/lib/apt/lists/* |  | ||||||
| COPY --from=container-build-arm64 /go/src/github.com/ollama/ollama/dist/linux-arm64/bin/ /bin/ |  | ||||||
| COPY --from=cpu-build-arm64 /go/src/github.com/ollama/ollama/dist/linux-arm64/lib/ /lib/ |  | ||||||
| COPY --from=cuda-11-build-runner-arm64 /go/src/github.com/ollama/ollama/dist/linux-arm64/lib/ /lib/ |  | ||||||
| COPY --from=cuda-12-build-runner-arm64 /go/src/github.com/ollama/ollama/dist/linux-arm64/lib/ /lib/ |  | ||||||
|  |  | ||||||
| # ROCm libraries larger so we keep it distinct from the CPU/CUDA image |  | ||||||
| FROM --platform=linux/amd64 ubuntu:22.04 AS runtime-rocm |  | ||||||
| # Frontload the rocm libraries which are large, and rarely change to increase chance of a common layer |  | ||||||
| # across releases |  | ||||||
| COPY --from=rocm-build-amd64 /go/src/github.com/ollama/ollama/dist/linux-amd64-rocm/lib/ /lib/ |  | ||||||
| RUN apt-get update && \ |  | ||||||
|     apt-get install -y ca-certificates && \ |  | ||||||
|     apt-get clean && rm -rf /var/lib/apt/lists/* |  | ||||||
| COPY --from=container-build-amd64 /go/src/github.com/ollama/ollama/dist/linux-amd64/bin/ /bin/ |  | ||||||
| COPY --from=cpu-build-amd64 /go/src/github.com/ollama/ollama/dist/linux-amd64/lib/ /lib/ |  | ||||||
| COPY --from=cpu_avx-build-amd64 /go/src/github.com/ollama/ollama/dist/linux-amd64/lib/ /lib/ |  | ||||||
| COPY --from=cpu_avx2-build-amd64 /go/src/github.com/ollama/ollama/dist/linux-amd64/lib/ /lib/ |  | ||||||
| COPY --from=rocm-build-amd64 /go/src/github.com/ollama/ollama/dist/linux-amd64/lib/ /lib/ |  | ||||||
| EXPOSE 11434 | EXPOSE 11434 | ||||||
| ENV OLLAMA_HOST=0.0.0.0 | ENV OLLAMA_HOST 0.0.0.0 | ||||||
|  |  | ||||||
| ENTRYPOINT ["/bin/ollama"] |  | ||||||
| CMD ["serve"] |  | ||||||
|  |  | ||||||
| FROM runtime-$TARGETARCH |  | ||||||
| EXPOSE 11434 |  | ||||||
| ENV OLLAMA_HOST=0.0.0.0 |  | ||||||
| ENV PATH=/usr/local/sbin:/usr/local/bin:/usr/sbin:/usr/bin:/sbin:/bin |  | ||||||
| ENV LD_LIBRARY_PATH=/usr/local/nvidia/lib:/usr/local/nvidia/lib64 |  | ||||||
| ENV NVIDIA_DRIVER_CAPABILITIES=compute,utility |  | ||||||
| ENV NVIDIA_VISIBLE_DEVICES=all |  | ||||||
|  |  | ||||||
| ENTRYPOINT ["/bin/ollama"] | ENTRYPOINT ["/bin/ollama"] | ||||||
| CMD ["serve"] | CMD ["serve"] | ||||||
|   | |||||||
							
								
								
									
										31
									
								
								Dockerfile.build
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,31 @@ | |||||||
|  | # centos7 amd64 dependencies | ||||||
|  | FROM --platform=linux/amd64 nvidia/cuda:11.3.1-devel-centos7 AS base-amd64 | ||||||
|  | RUN yum install -y https://repo.ius.io/ius-release-el7.rpm centos-release-scl && \ | ||||||
|  |     yum update -y && \ | ||||||
|  |     yum install -y devtoolset-10-gcc devtoolset-10-gcc-c++ git236 wget | ||||||
|  | RUN wget "https://github.com/Kitware/CMake/releases/download/v3.27.6/cmake-3.27.6-linux-x86_64.sh" -O cmake-installer.sh && chmod +x cmake-installer.sh && ./cmake-installer.sh --skip-license --prefix=/usr/local | ||||||
|  | ENV PATH /opt/rh/devtoolset-10/root/usr/bin:$PATH | ||||||
|  |  | ||||||
|  | # centos8 arm64 dependencies | ||||||
|  | FROM --platform=linux/arm64 nvidia/cuda-arm64:11.3.1-devel-centos8 AS base-arm64 | ||||||
|  | RUN sed -i -e 's/mirrorlist/#mirrorlist/g' -e 's|#baseurl=http://mirror.centos.org|baseurl=http://vault.centos.org|g' /etc/yum.repos.d/CentOS-* | ||||||
|  | RUN yum install -y git cmake | ||||||
|  |  | ||||||
|  | FROM base-${TARGETARCH} | ||||||
|  | ARG TARGETARCH | ||||||
|  | ARG GOFLAGS="'-ldflags -w -s'" | ||||||
|  |  | ||||||
|  | # install go | ||||||
|  | ADD https://dl.google.com/go/go1.21.3.linux-$TARGETARCH.tar.gz /tmp/go1.21.3.tar.gz | ||||||
|  | RUN mkdir -p /usr/local && tar xz -C /usr/local </tmp/go1.21.3.tar.gz | ||||||
|  |  | ||||||
|  | # build the final binary | ||||||
|  | WORKDIR /go/src/github.com/jmorganca/ollama | ||||||
|  | COPY . . | ||||||
|  |  | ||||||
|  | ENV GOOS=linux | ||||||
|  | ENV GOARCH=$TARGETARCH | ||||||
|  | ENV GOFLAGS=$GOFLAGS | ||||||
|  |  | ||||||
|  | RUN /usr/local/go/bin/go generate ./... && \ | ||||||
|  |     /usr/local/go/bin/go build . | ||||||
							
								
								
									
										292
									
								
								README.md
									
									
									
									
									
								
							
							
						
						| @@ -1,77 +1,66 @@ | |||||||
| <div align="center"> | <div align="center"> | ||||||
|  <img alt="ollama" height="200px" src="https://github.com/ollama/ollama/assets/3325447/0d0b44e2-8f4a-4e99-9b52-a5c1c741c8f7"> |   <picture> | ||||||
|  |     <source media="(prefers-color-scheme: dark)" height="200px" srcset="https://github.com/jmorganca/ollama/assets/3325447/56ea1849-1284-4645-8970-956de6e51c3c"> | ||||||
|  |     <img alt="logo" height="200px" src="https://github.com/jmorganca/ollama/assets/3325447/0d0b44e2-8f4a-4e99-9b52-a5c1c741c8f7"> | ||||||
|  |   </picture> | ||||||
| </div> | </div> | ||||||
|  |  | ||||||
| # Ollama | # Ollama | ||||||
|  |  | ||||||
| [](https://discord.gg/ollama) | [](https://discord.gg/ollama) | ||||||
|  |  | ||||||
| Get up and running with large language models. | Get up and running with large language models locally. | ||||||
|  |  | ||||||
| ### macOS | ### macOS | ||||||
|  |  | ||||||
| [Download](https://ollama.com/download/Ollama-darwin.zip) | [Download](https://ollama.ai/download/Ollama-darwin.zip) | ||||||
|  |  | ||||||
| ### Windows preview | ### Windows | ||||||
|  |  | ||||||
| [Download](https://ollama.com/download/OllamaSetup.exe) | Coming soon! | ||||||
|  |  | ||||||
| ### Linux | ### Linux & WSL2 | ||||||
|  |  | ||||||
| ``` | ``` | ||||||
| curl -fsSL https://ollama.com/install.sh | sh | curl https://ollama.ai/install.sh | sh | ||||||
| ``` | ``` | ||||||
|  |  | ||||||
| [Manual install instructions](https://github.com/ollama/ollama/blob/main/docs/linux.md) | [Manual install instructions](https://github.com/jmorganca/ollama/blob/main/docs/linux.md) | ||||||
|  |  | ||||||
| ### Docker | ### Docker | ||||||
|  |  | ||||||
| The official [Ollama Docker image](https://hub.docker.com/r/ollama/ollama) `ollama/ollama` is available on Docker Hub. | The official [Ollama Docker image](https://hub.docker.com/r/ollama/ollama) `ollama/ollama` is available on Docker Hub. | ||||||
|  |  | ||||||
| ### Libraries |  | ||||||
|  |  | ||||||
| - [ollama-python](https://github.com/ollama/ollama-python) |  | ||||||
| - [ollama-js](https://github.com/ollama/ollama-js) |  | ||||||
|  |  | ||||||
| ## Quickstart | ## Quickstart | ||||||
|  |  | ||||||
| To run and chat with [Llama 3.2](https://ollama.com/library/llama3.2): | To run and chat with [Llama 2](https://ollama.ai/library/llama2): | ||||||
|  |  | ||||||
| ``` | ``` | ||||||
| ollama run llama3.2 | ollama run llama2 | ||||||
| ``` | ``` | ||||||
|  |  | ||||||
| ## Model library | ## Model library | ||||||
|  |  | ||||||
| Ollama supports a list of models available on [ollama.com/library](https://ollama.com/library 'ollama model library') | Ollama supports a list of open-source models available on [ollama.ai/library](https://ollama.ai/library 'ollama model library') | ||||||
|  |  | ||||||
| Here are some example models that can be downloaded: | Here are some example open-source models that can be downloaded: | ||||||
|  |  | ||||||
| | Model              | Parameters | Size  | Download                       | | | Model              | Parameters | Size  | Download                       | | ||||||
| | ------------------ | ---------- | ----- | ------------------------------ | | | ------------------ | ---------- | ----- | ------------------------------ | | ||||||
| | Llama 3.2          | 3B         | 2.0GB | `ollama run llama3.2`          | |  | ||||||
| | Llama 3.2          | 1B         | 1.3GB | `ollama run llama3.2:1b`       | |  | ||||||
| | Llama 3.1          | 8B         | 4.7GB | `ollama run llama3.1`          | |  | ||||||
| | Llama 3.1          | 70B        | 40GB  | `ollama run llama3.1:70b`      | |  | ||||||
| | Llama 3.1          | 405B       | 231GB | `ollama run llama3.1:405b`     | |  | ||||||
| | Phi 3 Mini         | 3.8B       | 2.3GB | `ollama run phi3`              | |  | ||||||
| | Phi 3 Medium       | 14B        | 7.9GB | `ollama run phi3:medium`       | |  | ||||||
| | Gemma 2            | 2B         | 1.6GB | `ollama run gemma2:2b`         | |  | ||||||
| | Gemma 2            | 9B         | 5.5GB | `ollama run gemma2`            | |  | ||||||
| | Gemma 2            | 27B        | 16GB  | `ollama run gemma2:27b`        | |  | ||||||
| | Mistral            | 7B         | 4.1GB | `ollama run mistral`           | |  | ||||||
| | Moondream 2        | 1.4B       | 829MB | `ollama run moondream`         | |  | ||||||
| | Neural Chat        | 7B         | 4.1GB | `ollama run neural-chat`       | | | Neural Chat        | 7B         | 4.1GB | `ollama run neural-chat`       | | ||||||
| | Starling           | 7B         | 4.1GB | `ollama run starling-lm`       | | | Starling           | 7B         | 4.1GB | `ollama run starling-lm`       | | ||||||
|  | | Mistral            | 7B         | 4.1GB | `ollama run mistral`           | | ||||||
|  | | Llama 2            | 7B         | 3.8GB | `ollama run llama2`            | | ||||||
| | Code Llama         | 7B         | 3.8GB | `ollama run codellama`         | | | Code Llama         | 7B         | 3.8GB | `ollama run codellama`         | | ||||||
| | Llama 2 Uncensored | 7B         | 3.8GB | `ollama run llama2-uncensored` | | | Llama 2 Uncensored | 7B         | 3.8GB | `ollama run llama2-uncensored` | | ||||||
| | LLaVA              | 7B         | 4.5GB | `ollama run llava`             | | | Llama 2 13B        | 13B        | 7.3GB | `ollama run llama2:13b`        | | ||||||
| | Solar              | 10.7B      | 6.1GB | `ollama run solar`             | | | Llama 2 70B        | 70B        | 39GB  | `ollama run llama2:70b`        | | ||||||
|  | | Orca Mini          | 3B         | 1.9GB | `ollama run orca-mini`         | | ||||||
|  | | Vicuna             | 7B         | 3.8GB | `ollama run vicuna`            | | ||||||
|  |  | ||||||
| > [!NOTE] | > Note: You should have at least 8 GB of RAM to run the 3B models, 16 GB to run the 7B models, and 32 GB to run the 13B models. | ||||||
| > You should have at least 8 GB of RAM available to run the 7B models, 16 GB to run the 13B models, and 32 GB to run the 33B models. |  | ||||||
|  |  | ||||||
| ## Customize a model | ## Customize your own model | ||||||
|  |  | ||||||
| ### Import from GGUF | ### Import from GGUF | ||||||
|  |  | ||||||
| @@ -101,21 +90,21 @@ See the [guide](docs/import.md) on importing models for more information. | |||||||
|  |  | ||||||
| ### Customize a prompt | ### Customize a prompt | ||||||
|  |  | ||||||
| Models from the Ollama library can be customized with a prompt. For example, to customize the `llama3.2` model: | Models from the Ollama library can be customized with a prompt. For example, to customize the `llama2` model: | ||||||
|  |  | ||||||
| ``` | ``` | ||||||
| ollama pull llama3.2 | ollama pull llama2 | ||||||
| ``` | ``` | ||||||
|  |  | ||||||
| Create a `Modelfile`: | Create a `Modelfile`: | ||||||
|  |  | ||||||
| ``` | ``` | ||||||
| FROM llama3.2 | FROM llama2 | ||||||
|  |  | ||||||
| # set the temperature to 1 [higher is more creative, lower is more coherent] | # set the temperature to 1 [higher is more creative, lower is more coherent] | ||||||
| PARAMETER temperature 1 | PARAMETER temperature 1 | ||||||
|  |  | ||||||
| # set the system message | # set the system prompt | ||||||
| SYSTEM """ | SYSTEM """ | ||||||
| You are Mario from Super Mario Bros. Answer as Mario, the assistant, only. | You are Mario from Super Mario Bros. Answer as Mario, the assistant, only. | ||||||
| """ | """ | ||||||
| @@ -138,14 +127,10 @@ For more examples, see the [examples](examples) directory. For more information | |||||||
|  |  | ||||||
| `ollama create` is used to create a model from a Modelfile. | `ollama create` is used to create a model from a Modelfile. | ||||||
|  |  | ||||||
| ``` |  | ||||||
| ollama create mymodel -f ./Modelfile |  | ||||||
| ``` |  | ||||||
|  |  | ||||||
| ### Pull a model | ### Pull a model | ||||||
|  |  | ||||||
| ``` | ``` | ||||||
| ollama pull llama3.2 | ollama pull llama2 | ||||||
| ``` | ``` | ||||||
|  |  | ||||||
| > This command can also be used to update a local model. Only the diff will be pulled. | > This command can also be used to update a local model. Only the diff will be pulled. | ||||||
| @@ -153,13 +138,13 @@ ollama pull llama3.2 | |||||||
| ### Remove a model | ### Remove a model | ||||||
|  |  | ||||||
| ``` | ``` | ||||||
| ollama rm llama3.2 | ollama rm llama2 | ||||||
| ``` | ``` | ||||||
|  |  | ||||||
| ### Copy a model | ### Copy a model | ||||||
|  |  | ||||||
| ``` | ``` | ||||||
| ollama cp llama3.2 my-model | ollama cp llama2 my-llama2 | ||||||
| ``` | ``` | ||||||
|  |  | ||||||
| ### Multiline input | ### Multiline input | ||||||
| @@ -173,53 +158,37 @@ For multiline input, you can wrap text with `"""`: | |||||||
| I'm a basic program that prints the famous "Hello, world!" message to the console. | I'm a basic program that prints the famous "Hello, world!" message to the console. | ||||||
| ``` | ``` | ||||||
|  |  | ||||||
| ### Multimodal models | ### Pass in prompt as arguments | ||||||
|  |  | ||||||
| ``` | ``` | ||||||
| ollama run llava "What's in this image? /Users/jmorgan/Desktop/smile.png" | $ ollama run llama2 "Summarize this file: $(cat README.md)" | ||||||
| The image features a yellow smiley face, which is likely the central focus of the picture. |  | ||||||
| ``` |  | ||||||
|  |  | ||||||
| ### Pass the prompt as an argument |  | ||||||
|  |  | ||||||
| ``` |  | ||||||
| $ ollama run llama3.2 "Summarize this file: $(cat README.md)" |  | ||||||
|  Ollama is a lightweight, extensible framework for building and running language models on the local machine. It provides a simple API for creating, running, and managing models, as well as a library of pre-built models that can be easily used in a variety of applications. |  Ollama is a lightweight, extensible framework for building and running language models on the local machine. It provides a simple API for creating, running, and managing models, as well as a library of pre-built models that can be easily used in a variety of applications. | ||||||
| ``` | ``` | ||||||
|  |  | ||||||
| ### Show model information |  | ||||||
|  |  | ||||||
| ``` |  | ||||||
| ollama show llama3.2 |  | ||||||
| ``` |  | ||||||
|  |  | ||||||
| ### List models on your computer | ### List models on your computer | ||||||
|  |  | ||||||
| ``` | ``` | ||||||
| ollama list | ollama list | ||||||
| ``` | ``` | ||||||
|  |  | ||||||
| ### List which models are currently loaded |  | ||||||
|  |  | ||||||
| ``` |  | ||||||
| ollama ps |  | ||||||
| ``` |  | ||||||
|  |  | ||||||
| ### Stop a model which is currently running |  | ||||||
|  |  | ||||||
| ``` |  | ||||||
| ollama stop llama3.2 |  | ||||||
| ``` |  | ||||||
|  |  | ||||||
| ### Start Ollama | ### Start Ollama | ||||||
|  |  | ||||||
| `ollama serve` is used when you want to start ollama without running the desktop application. | `ollama serve` is used when you want to start ollama without running the desktop application. | ||||||
|  |  | ||||||
| ## Building | ## Building | ||||||
|  |  | ||||||
| See the [developer guide](https://github.com/ollama/ollama/blob/main/docs/development.md) | Install `cmake` and `go`: | ||||||
|  |  | ||||||
| ### Running local builds | ``` | ||||||
|  | brew install cmake go | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | Then generate dependencies and build: | ||||||
|  |  | ||||||
|  | ``` | ||||||
|  | go generate ./... | ||||||
|  | go build . | ||||||
|  | ``` | ||||||
|  |  | ||||||
| Next, start the server: | Next, start the server: | ||||||
|  |  | ||||||
| @@ -230,107 +199,40 @@ Next, start the server: | |||||||
| Finally, in a separate shell, run a model: | Finally, in a separate shell, run a model: | ||||||
|  |  | ||||||
| ``` | ``` | ||||||
| ./ollama run llama3.2 | ./ollama run llama2 | ||||||
| ``` | ``` | ||||||
|  |  | ||||||
| ## REST API | ## REST API | ||||||
|  |  | ||||||
| Ollama has a REST API for running and managing models. | Ollama has a REST API for running and managing models. | ||||||
|  | For example, to generate text from a model: | ||||||
| ### Generate a response |  | ||||||
|  |  | ||||||
| ``` | ``` | ||||||
| curl http://localhost:11434/api/generate -d '{ | curl http://localhost:11434/api/generate -d '{ | ||||||
|   "model": "llama3.2", |   "model": "llama2", | ||||||
|   "prompt":"Why is the sky blue?" |   "prompt":"Why is the sky blue?" | ||||||
| }' | }' | ||||||
| ``` | ``` | ||||||
|  |  | ||||||
| ### Chat with a model |  | ||||||
|  |  | ||||||
| ``` |  | ||||||
| curl http://localhost:11434/api/chat -d '{ |  | ||||||
|   "model": "llama3.2", |  | ||||||
|   "messages": [ |  | ||||||
|     { "role": "user", "content": "why is the sky blue?" } |  | ||||||
|   ] |  | ||||||
| }' |  | ||||||
| ``` |  | ||||||
|  |  | ||||||
| See the [API documentation](./docs/api.md) for all endpoints. | See the [API documentation](./docs/api.md) for all endpoints. | ||||||
|  |  | ||||||
| ## Community Integrations | ## Community Integrations | ||||||
|  |  | ||||||
|  | ### Mobile | ||||||
|  |  | ||||||
|  | - [Mobile Artificial Intelligence Distribution](https://github.com/MaidFoundation/Maid) (Maid) | ||||||
|  |  | ||||||
| ### Web & Desktop | ### Web & Desktop | ||||||
|  |  | ||||||
| - [Open WebUI](https://github.com/open-webui/open-webui) |  | ||||||
| - [Enchanted (macOS native)](https://github.com/AugustDev/enchanted) |  | ||||||
| - [Hollama](https://github.com/fmaclen/hollama) |  | ||||||
| - [Lollms-Webui](https://github.com/ParisNeo/lollms-webui) |  | ||||||
| - [LibreChat](https://github.com/danny-avila/LibreChat) |  | ||||||
| - [Bionic GPT](https://github.com/bionic-gpt/bionic-gpt) |  | ||||||
| - [HTML UI](https://github.com/rtcfirefly/ollama-ui) | - [HTML UI](https://github.com/rtcfirefly/ollama-ui) | ||||||
| - [Saddle](https://github.com/jikkuatwork/saddle) |  | ||||||
| - [Chatbot UI](https://github.com/ivanfioravanti/chatbot-ollama) | - [Chatbot UI](https://github.com/ivanfioravanti/chatbot-ollama) | ||||||
| - [Chatbot UI v2](https://github.com/mckaywrigley/chatbot-ui) |  | ||||||
| - [Typescript UI](https://github.com/ollama-interface/Ollama-Gui?tab=readme-ov-file) | - [Typescript UI](https://github.com/ollama-interface/Ollama-Gui?tab=readme-ov-file) | ||||||
| - [Minimalistic React UI for Ollama Models](https://github.com/richawo/minimal-llm-ui) | - [Minimalistic React UI for Ollama Models](https://github.com/richawo/minimal-llm-ui) | ||||||
|  | - [Web UI](https://github.com/ollama-webui/ollama-webui) | ||||||
| - [Ollamac](https://github.com/kevinhermawan/Ollamac) | - [Ollamac](https://github.com/kevinhermawan/Ollamac) | ||||||
| - [big-AGI](https://github.com/enricoros/big-AGI/blob/main/docs/config-local-ollama.md) | - [big-AGI](https://github.com/enricoros/big-agi/blob/main/docs/config-ollama.md) | ||||||
| - [Cheshire Cat assistant framework](https://github.com/cheshire-cat-ai/core) | - [Cheshire Cat assistant framework](https://github.com/cheshire-cat-ai/core) | ||||||
| - [Amica](https://github.com/semperai/amica) | - [Amica](https://github.com/semperai/amica) | ||||||
| - [chatd](https://github.com/BruceMacD/chatd) |  | ||||||
| - [Ollama-SwiftUI](https://github.com/kghandour/Ollama-SwiftUI) |  | ||||||
| - [Dify.AI](https://github.com/langgenius/dify) |  | ||||||
| - [MindMac](https://mindmac.app) |  | ||||||
| - [NextJS Web Interface for Ollama](https://github.com/jakobhoeg/nextjs-ollama-llm-ui) |  | ||||||
| - [Msty](https://msty.app) |  | ||||||
| - [Chatbox](https://github.com/Bin-Huang/Chatbox) |  | ||||||
| - [WinForm Ollama Copilot](https://github.com/tgraupmann/WinForm_Ollama_Copilot) |  | ||||||
| - [NextChat](https://github.com/ChatGPTNextWeb/ChatGPT-Next-Web) with [Get Started Doc](https://docs.nextchat.dev/models/ollama) |  | ||||||
| - [Alpaca WebUI](https://github.com/mmo80/alpaca-webui) |  | ||||||
| - [OllamaGUI](https://github.com/enoch1118/ollamaGUI) |  | ||||||
| - [OpenAOE](https://github.com/InternLM/OpenAOE) |  | ||||||
| - [Odin Runes](https://github.com/leonid20000/OdinRunes) |  | ||||||
| - [LLM-X](https://github.com/mrdjohnson/llm-x) (Progressive Web App) |  | ||||||
| - [AnythingLLM (Docker + MacOs/Windows/Linux native app)](https://github.com/Mintplex-Labs/anything-llm) |  | ||||||
| - [Ollama Basic Chat: Uses HyperDiv Reactive UI](https://github.com/rapidarchitect/ollama_basic_chat) |  | ||||||
| - [Ollama-chats RPG](https://github.com/drazdra/ollama-chats) |  | ||||||
| - [QA-Pilot](https://github.com/reid41/QA-Pilot) (Chat with Code Repository) |  | ||||||
| - [ChatOllama](https://github.com/sugarforever/chat-ollama) (Open Source Chatbot based on Ollama with Knowledge Bases) |  | ||||||
| - [CRAG Ollama Chat](https://github.com/Nagi-ovo/CRAG-Ollama-Chat) (Simple Web Search with Corrective RAG) |  | ||||||
| - [RAGFlow](https://github.com/infiniflow/ragflow) (Open-source Retrieval-Augmented Generation engine based on deep document understanding) |  | ||||||
| - [StreamDeploy](https://github.com/StreamDeploy-DevRel/streamdeploy-llm-app-scaffold) (LLM Application Scaffold) |  | ||||||
| - [chat](https://github.com/swuecho/chat) (chat web app for teams) |  | ||||||
| - [Lobe Chat](https://github.com/lobehub/lobe-chat) with [Integrating Doc](https://lobehub.com/docs/self-hosting/examples/ollama) |  | ||||||
| - [Ollama RAG Chatbot](https://github.com/datvodinh/rag-chatbot.git) (Local Chat with multiple PDFs using Ollama and RAG) |  | ||||||
| - [BrainSoup](https://www.nurgo-software.com/products/brainsoup) (Flexible native client with RAG & multi-agent automation) |  | ||||||
| - [macai](https://github.com/Renset/macai) (macOS client for Ollama, ChatGPT, and other compatible API back-ends) |  | ||||||
| - [Olpaka](https://github.com/Otacon/olpaka) (User-friendly Flutter Web App for Ollama) |  | ||||||
| - [OllamaSpring](https://github.com/CrazyNeil/OllamaSpring) (Ollama Client for macOS) |  | ||||||
| - [LLocal.in](https://github.com/kartikm7/llocal) (Easy to use Electron Desktop Client for Ollama) |  | ||||||
| - [AiLama](https://github.com/zeyoyt/ailama) (A Discord User App that allows you to interact with Ollama anywhere in discord ) |  | ||||||
| - [Ollama with Google Mesop](https://github.com/rapidarchitect/ollama_mesop/) (Mesop Chat Client implementation with Ollama) |  | ||||||
| - [Painting Droid](https://github.com/mateuszmigas/painting-droid) (Painting app with AI integrations) |  | ||||||
| - [Kerlig AI](https://www.kerlig.com/) (AI writing assistant for macOS) |  | ||||||
| - [AI Studio](https://github.com/MindWorkAI/AI-Studio) |  | ||||||
| - [Sidellama](https://github.com/gyopak/sidellama) (browser-based LLM client) |  | ||||||
| - [LLMStack](https://github.com/trypromptly/LLMStack) (No-code multi-agent framework to build LLM agents and workflows) |  | ||||||
| - [BoltAI for Mac](https://boltai.com) (AI Chat Client for Mac) |  | ||||||
| - [Harbor](https://github.com/av/harbor) (Containerized LLM Toolkit with Ollama as default backend) |  | ||||||
| - [Go-CREW](https://www.jonathanhecl.com/go-crew/) (Powerful Offline RAG in Golang) |  | ||||||
| - [PartCAD](https://github.com/openvmp/partcad/) (CAD model generation with OpenSCAD and CadQuery) |  | ||||||
| - [Ollama4j Web UI](https://github.com/ollama4j/ollama4j-web-ui) - Java-based Web UI for Ollama built with Vaadin, Spring Boot and Ollama4j |  | ||||||
| - [PyOllaMx](https://github.com/kspviswa/pyOllaMx) - macOS application capable of chatting with both Ollama and Apple MLX models. |  | ||||||
| - [Claude Dev](https://github.com/saoudrizwan/claude-dev) - VSCode extension for multi-file/whole-repo coding |  | ||||||
| - [Cherry Studio](https://github.com/kangfenmao/cherry-studio) (Desktop client with Ollama support) |  | ||||||
| - [ConfiChat](https://github.com/1runeberg/confichat) (Lightweight, standalone, multi-platform, and privacy focused LLM chat interface with optional encryption) |  | ||||||
| - [Archyve](https://github.com/nickthecook/archyve) (RAG-enabling document library) |  | ||||||
| - [crewAI with Mesop](https://github.com/rapidarchitect/ollama-crew-mesop) (Mesop Web Interface to run crewAI with Ollama) |  | ||||||
| - [LLMChat](https://github.com/trendy-design/llmchat) (Privacy focused, 100% local, intuitive all-in-one chat interface) |  | ||||||
| - [ARGO](https://github.com/xark-argo/argo) (Locally download and run Ollama and Huggingface models with RAG on Mac/Windows/Linux) |  | ||||||
| - [G1](https://github.com/bklieger-groq/g1) (Prototype of using prompting strategies to improve the LLM's reasoning through o1-like reasoning chains.) |  | ||||||
| - [Ollama App](https://github.com/JHubi1/ollama-app) (Modern and easy-to-use multi-platform client for Ollama) |  | ||||||
|  |  | ||||||
| ### Terminal | ### Terminal | ||||||
|  |  | ||||||
| @@ -339,86 +241,31 @@ See the [API documentation](./docs/api.md) for all endpoints. | |||||||
| - [Emacs client](https://github.com/zweifisch/ollama) | - [Emacs client](https://github.com/zweifisch/ollama) | ||||||
| - [gen.nvim](https://github.com/David-Kunz/gen.nvim) | - [gen.nvim](https://github.com/David-Kunz/gen.nvim) | ||||||
| - [ollama.nvim](https://github.com/nomnivore/ollama.nvim) | - [ollama.nvim](https://github.com/nomnivore/ollama.nvim) | ||||||
| - [ollero.nvim](https://github.com/marco-souza/ollero.nvim) |  | ||||||
| - [ollama-chat.nvim](https://github.com/gerazov/ollama-chat.nvim) |  | ||||||
| - [ogpt.nvim](https://github.com/huynle/ogpt.nvim) | - [ogpt.nvim](https://github.com/huynle/ogpt.nvim) | ||||||
| - [gptel Emacs client](https://github.com/karthink/gptel) | - [gptel Emacs client](https://github.com/karthink/gptel) | ||||||
| - [Oatmeal](https://github.com/dustinblackman/oatmeal) | - [Oatmeal](https://github.com/dustinblackman/oatmeal) | ||||||
| - [cmdh](https://github.com/pgibler/cmdh) |  | ||||||
| - [ooo](https://github.com/npahlfer/ooo) |  | ||||||
| - [shell-pilot](https://github.com/reid41/shell-pilot) |  | ||||||
| - [tenere](https://github.com/pythops/tenere) |  | ||||||
| - [llm-ollama](https://github.com/taketwo/llm-ollama) for [Datasette's LLM CLI](https://llm.datasette.io/en/stable/). |  | ||||||
| - [typechat-cli](https://github.com/anaisbetts/typechat-cli) |  | ||||||
| - [ShellOracle](https://github.com/djcopley/ShellOracle) |  | ||||||
| - [tlm](https://github.com/yusufcanb/tlm) |  | ||||||
| - [podman-ollama](https://github.com/ericcurtin/podman-ollama) |  | ||||||
| - [gollama](https://github.com/sammcj/gollama) |  | ||||||
| - [Ollama eBook Summary](https://github.com/cognitivetech/ollama-ebook-summary/) |  | ||||||
| - [Ollama Mixture of Experts (MOE) in 50 lines of code](https://github.com/rapidarchitect/ollama_moe) |  | ||||||
| - [vim-intelligence-bridge](https://github.com/pepo-ec/vim-intelligence-bridge) Simple interaction of "Ollama" with the Vim editor |  | ||||||
|  |  | ||||||
| ### Apple Vision Pro |  | ||||||
| - [Enchanted](https://github.com/AugustDev/enchanted) |  | ||||||
|  |  | ||||||
| ### Database |  | ||||||
|  |  | ||||||
| - [MindsDB](https://github.com/mindsdb/mindsdb/blob/staging/mindsdb/integrations/handlers/ollama_handler/README.md) (Connects Ollama models with nearly 200 data platforms and apps) |  | ||||||
| - [chromem-go](https://github.com/philippgille/chromem-go/blob/v0.5.0/embed_ollama.go) with [example](https://github.com/philippgille/chromem-go/tree/v0.5.0/examples/rag-wikipedia-ollama) |  | ||||||
|  |  | ||||||
| ### Package managers | ### Package managers | ||||||
|  |  | ||||||
| - [Pacman](https://archlinux.org/packages/extra/x86_64/ollama/) | - [Pacman](https://archlinux.org/packages/extra/x86_64/ollama/) | ||||||
| - [Gentoo](https://github.com/gentoo/guru/tree/master/app-misc/ollama) |  | ||||||
| - [Helm Chart](https://artifacthub.io/packages/helm/ollama-helm/ollama) |  | ||||||
| - [Guix channel](https://codeberg.org/tusharhero/ollama-guix) |  | ||||||
| - [Nix package](https://search.nixos.org/packages?channel=24.05&show=ollama&from=0&size=50&sort=relevance&type=packages&query=ollama) |  | ||||||
| - [Flox](https://flox.dev/blog/ollama-part-one) |  | ||||||
|  |  | ||||||
| ### Libraries | ### Libraries | ||||||
|  |  | ||||||
| - [LangChain](https://python.langchain.com/docs/integrations/llms/ollama) and [LangChain.js](https://js.langchain.com/docs/integrations/chat/ollama/) with [example](https://js.langchain.com/docs/tutorials/local_rag/) | - [LangChain](https://python.langchain.com/docs/integrations/llms/ollama) and [LangChain.js](https://js.langchain.com/docs/modules/model_io/models/llms/integrations/ollama) with [example](https://js.langchain.com/docs/use_cases/question_answering/local_retrieval_qa) | ||||||
| - [Firebase Genkit](https://firebase.google.com/docs/genkit/plugins/ollama) |  | ||||||
| - [crewAI](https://github.com/crewAIInc/crewAI) |  | ||||||
| - [LangChainGo](https://github.com/tmc/langchaingo/) with [example](https://github.com/tmc/langchaingo/tree/main/examples/ollama-completion-example) | - [LangChainGo](https://github.com/tmc/langchaingo/) with [example](https://github.com/tmc/langchaingo/tree/main/examples/ollama-completion-example) | ||||||
| - [LangChain4j](https://github.com/langchain4j/langchain4j) with [example](https://github.com/langchain4j/langchain4j-examples/tree/main/ollama-examples/src/main/java) | - [LlamaIndex](https://gpt-index.readthedocs.io/en/stable/examples/llm/ollama.html) | ||||||
| - [LangChainRust](https://github.com/Abraxas-365/langchain-rust) with [example](https://github.com/Abraxas-365/langchain-rust/blob/main/examples/llm_ollama.rs) |  | ||||||
| - [LlamaIndex](https://docs.llamaindex.ai/en/stable/examples/llm/ollama/) and [LlamaIndexTS](https://ts.llamaindex.ai/modules/llms/available_llms/ollama) |  | ||||||
| - [LiteLLM](https://github.com/BerriAI/litellm) | - [LiteLLM](https://github.com/BerriAI/litellm) | ||||||
| - [OllamaFarm for Go](https://github.com/presbrey/ollamafarm) |  | ||||||
| - [OllamaSharp for .NET](https://github.com/awaescher/OllamaSharp) | - [OllamaSharp for .NET](https://github.com/awaescher/OllamaSharp) | ||||||
| - [Ollama for Ruby](https://github.com/gbaptista/ollama-ai) |  | ||||||
| - [Ollama-rs for Rust](https://github.com/pepperoni21/ollama-rs) | - [Ollama-rs for Rust](https://github.com/pepperoni21/ollama-rs) | ||||||
| - [Ollama-hpp for C++](https://github.com/jmont-dev/ollama-hpp) | - [Ollama4j for Java](https://github.com/amithkoujalgi/ollama4j) | ||||||
| - [Ollama4j for Java](https://github.com/ollama4j/ollama4j) |  | ||||||
| - [ModelFusion Typescript Library](https://modelfusion.dev/integration/model-provider/ollama) | - [ModelFusion Typescript Library](https://modelfusion.dev/integration/model-provider/ollama) | ||||||
| - [OllamaKit for Swift](https://github.com/kevinhermawan/OllamaKit) | - [OllamaKit for Swift](https://github.com/kevinhermawan/OllamaKit) | ||||||
| - [Ollama for Dart](https://github.com/breitburg/dart-ollama) | - [Ollama for Dart](https://github.com/breitburg/dart-ollama) | ||||||
| - [Ollama for Laravel](https://github.com/cloudstudio/ollama-laravel) | - [Ollama for Laravel](https://github.com/cloudstudio/ollama-laravel) | ||||||
| - [LangChainDart](https://github.com/davidmigloz/langchain_dart) |  | ||||||
| - [Semantic Kernel - Python](https://github.com/microsoft/semantic-kernel/tree/main/python/semantic_kernel/connectors/ai/ollama) |  | ||||||
| - [Haystack](https://github.com/deepset-ai/haystack-integrations/blob/main/integrations/ollama.md) |  | ||||||
| - [Elixir LangChain](https://github.com/brainlid/langchain) |  | ||||||
| - [Ollama for R - rollama](https://github.com/JBGruber/rollama) |  | ||||||
| - [Ollama for R - ollama-r](https://github.com/hauselin/ollama-r) |  | ||||||
| - [Ollama-ex for Elixir](https://github.com/lebrunel/ollama-ex) |  | ||||||
| - [Ollama Connector for SAP ABAP](https://github.com/b-tocs/abap_btocs_ollama) |  | ||||||
| - [Testcontainers](https://testcontainers.com/modules/ollama/) |  | ||||||
| - [Portkey](https://portkey.ai/docs/welcome/integration-guides/ollama) |  | ||||||
| - [PromptingTools.jl](https://github.com/svilupp/PromptingTools.jl) with an [example](https://svilupp.github.io/PromptingTools.jl/dev/examples/working_with_ollama) |  | ||||||
| - [LlamaScript](https://github.com/Project-Llama/llamascript) |  | ||||||
| - [Gollm](https://docs.gollm.co/examples/ollama-example) |  | ||||||
| - [Ollamaclient for Golang](https://github.com/xyproto/ollamaclient) |  | ||||||
| - [High-level function abstraction in Go](https://gitlab.com/tozd/go/fun) |  | ||||||
| - [Ollama PHP](https://github.com/ArdaGnsrn/ollama-php) |  | ||||||
| - [Agents-Flex for Java](https://github.com/agents-flex/agents-flex) with [example](https://github.com/agents-flex/agents-flex/tree/main/agents-flex-llm/agents-flex-llm-ollama/src/test/java/com/agentsflex/llm/ollama) |  | ||||||
|  |  | ||||||
| ### Mobile | ### Mobile | ||||||
|  |  | ||||||
| - [Enchanted](https://github.com/AugustDev/enchanted) | - [Maid](https://github.com/danemadsen/Maid) (Mobile Artificial Intelligence Distribution) | ||||||
| - [Maid](https://github.com/Mobile-Artificial-Intelligence/maid) |  | ||||||
| - [Ollama App](https://github.com/JHubi1/ollama-app) (Modern and easy-to-use multi-platform client for Ollama) |  | ||||||
| - [ConfiChat](https://github.com/1runeberg/confichat) (Lightweight, standalone, multi-platform, and privacy focused LLM chat interface with optional encryption) |  | ||||||
|  |  | ||||||
| ### Extensions & Plugins | ### Extensions & Plugins | ||||||
|  |  | ||||||
| @@ -427,34 +274,9 @@ See the [API documentation](./docs/api.md) for all endpoints. | |||||||
| - [Continue](https://github.com/continuedev/continue) | - [Continue](https://github.com/continuedev/continue) | ||||||
| - [Obsidian Ollama plugin](https://github.com/hinterdupfinger/obsidian-ollama) | - [Obsidian Ollama plugin](https://github.com/hinterdupfinger/obsidian-ollama) | ||||||
| - [Logseq Ollama plugin](https://github.com/omagdy7/ollama-logseq) | - [Logseq Ollama plugin](https://github.com/omagdy7/ollama-logseq) | ||||||
| - [NotesOllama](https://github.com/andersrex/notesollama) (Apple Notes Ollama plugin) |  | ||||||
| - [Dagger Chatbot](https://github.com/samalba/dagger-chatbot) | - [Dagger Chatbot](https://github.com/samalba/dagger-chatbot) | ||||||
| - [Discord AI Bot](https://github.com/mekb-turtle/discord-ai-bot) | - [Discord AI Bot](https://github.com/mekb-turtle/discord-ai-bot) | ||||||
| - [Ollama Telegram Bot](https://github.com/ruecat/ollama-telegram) |  | ||||||
| - [Hass Ollama Conversation](https://github.com/ej52/hass-ollama-conversation) | - [Hass Ollama Conversation](https://github.com/ej52/hass-ollama-conversation) | ||||||
| - [Rivet plugin](https://github.com/abrenneke/rivet-plugin-ollama) | - [Rivet plugin](https://github.com/abrenneke/rivet-plugin-ollama) | ||||||
| - [Obsidian BMO Chatbot plugin](https://github.com/longy2k/obsidian-bmo-chatbot) |  | ||||||
| - [Cliobot](https://github.com/herval/cliobot) (Telegram bot with Ollama support) |  | ||||||
| - [Copilot for Obsidian plugin](https://github.com/logancyang/obsidian-copilot) |  | ||||||
| - [Obsidian Local GPT plugin](https://github.com/pfrankov/obsidian-local-gpt) |  | ||||||
| - [Open Interpreter](https://docs.openinterpreter.com/language-model-setup/local-models/ollama) |  | ||||||
| - [Llama Coder](https://github.com/ex3ndr/llama-coder) (Copilot alternative using Ollama) | - [Llama Coder](https://github.com/ex3ndr/llama-coder) (Copilot alternative using Ollama) | ||||||
| - [Ollama Copilot](https://github.com/bernardo-bruning/ollama-copilot) (Proxy that allows you to use ollama as a copilot like Github copilot) | - [Obsidian BMO Chatbot plugin](https://github.com/longy2k/obsidian-bmo-chatbot) | ||||||
| - [twinny](https://github.com/rjmacarthy/twinny) (Copilot and Copilot chat alternative using Ollama) |  | ||||||
| - [Wingman-AI](https://github.com/RussellCanfield/wingman-ai) (Copilot code and chat alternative using Ollama and Hugging Face) |  | ||||||
| - [Page Assist](https://github.com/n4ze3m/page-assist) (Chrome Extension) |  | ||||||
| - [Plasmoid Ollama Control](https://github.com/imoize/plasmoid-ollamacontrol) (KDE Plasma extension that allows you to quickly manage/control Ollama model) |  | ||||||
| - [AI Telegram Bot](https://github.com/tusharhero/aitelegrambot) (Telegram bot using Ollama in backend) |  | ||||||
| - [AI ST Completion](https://github.com/yaroslavyaroslav/OpenAI-sublime-text) (Sublime Text 4 AI assistant plugin with Ollama support) |  | ||||||
| - [Discord-Ollama Chat Bot](https://github.com/kevinthedang/discord-ollama) (Generalized TypeScript Discord Bot w/ Tuning Documentation) |  | ||||||
| - [Discord AI chat/moderation bot](https://github.com/rapmd73/Companion) Chat/moderation bot written in python. Uses Ollama to create personalities. |  | ||||||
| - [Headless Ollama](https://github.com/nischalj10/headless-ollama) (Scripts to automatically install ollama client & models on any OS for apps that depends on ollama server) |  | ||||||
| - [vnc-lm](https://github.com/jk011ru/vnc-lm) (A containerized Discord bot with support for attachments and web links) |  | ||||||
| - [LSP-AI](https://github.com/SilasMarvin/lsp-ai) (Open-source language server for AI-powered functionality) |  | ||||||
| - [QodeAssist](https://github.com/Palm1r/QodeAssist) (AI-powered coding assistant plugin for Qt Creator) |  | ||||||
| - [Obsidian Quiz Generator plugin](https://github.com/ECuiDev/obsidian-quiz-generator) |  | ||||||
|  |  | ||||||
| ### Supported backends |  | ||||||
|  |  | ||||||
| - [llama.cpp](https://github.com/ggerganov/llama.cpp) project founded by Georgi Gerganov. |  | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										25
									
								
								SECURITY.md
									
									
									
									
									
								
							
							
						
						| @@ -1,25 +0,0 @@ | |||||||
| # Security |  | ||||||
|  |  | ||||||
| The Ollama maintainer team takes security seriously and will actively work to resolve security issues. |  | ||||||
|  |  | ||||||
| ## Reporting a vulnerability |  | ||||||
|  |  | ||||||
| If you discover a security vulnerability, please do not open a public issue. Instead, please report it by emailing hello@ollama.com. We ask that you give us sufficient time to investigate and address the vulnerability before disclosing it publicly. |  | ||||||
|  |  | ||||||
| Please include the following details in your report: |  | ||||||
| - A description of the vulnerability |  | ||||||
| - Steps to reproduce the issue |  | ||||||
| - Your assessment of the potential impact |  | ||||||
| - Any possible mitigations |  | ||||||
|  |  | ||||||
| ## Security best practices |  | ||||||
|  |  | ||||||
| While the maintainer team does their best to secure Ollama, users are encouraged to implement their own security best practices, such as: |  | ||||||
|  |  | ||||||
| - Regularly updating to the latest version of Ollama |  | ||||||
| - Securing access to hosted instances of Ollama |  | ||||||
| - Monitoring systems for unusual activity |  | ||||||
|  |  | ||||||
| ## Contact |  | ||||||
|  |  | ||||||
| For any other questions or concerns related to security, please contact us at hello@ollama.com |  | ||||||
							
								
								
									
										193
									
								
								api/client.go
									
									
									
									
									
								
							
							
						
						| @@ -1,16 +1,3 @@ | |||||||
| // Package api implements the client-side API for code wishing to interact |  | ||||||
| // with the ollama service. The methods of the [Client] type correspond to |  | ||||||
| // the ollama REST API as described in [the API documentation]. |  | ||||||
| // The ollama command-line client itself uses this package to interact with |  | ||||||
| // the backend service. |  | ||||||
| // |  | ||||||
| // # Examples |  | ||||||
| // |  | ||||||
| // Several examples of using this package are available [in the GitHub |  | ||||||
| // repository]. |  | ||||||
| // |  | ||||||
| // [the API documentation]: https://github.com/ollama/ollama/blob/main/docs/api.md |  | ||||||
| // [in the GitHub repository]: https://github.com/ollama/ollama/tree/main/examples |  | ||||||
| package api | package api | ||||||
|  |  | ||||||
| import ( | import ( | ||||||
| @@ -21,20 +8,20 @@ import ( | |||||||
| 	"errors" | 	"errors" | ||||||
| 	"fmt" | 	"fmt" | ||||||
| 	"io" | 	"io" | ||||||
|  | 	"net" | ||||||
| 	"net/http" | 	"net/http" | ||||||
| 	"net/url" | 	"net/url" | ||||||
|  | 	"os" | ||||||
| 	"runtime" | 	"runtime" | ||||||
|  | 	"strings" | ||||||
|  |  | ||||||
| 	"github.com/ollama/ollama/envconfig" | 	"github.com/jmorganca/ollama/format" | ||||||
| 	"github.com/ollama/ollama/format" | 	"github.com/jmorganca/ollama/version" | ||||||
| 	"github.com/ollama/ollama/version" |  | ||||||
| ) | ) | ||||||
|  |  | ||||||
| // Client encapsulates client state for interacting with the ollama |  | ||||||
| // service. Use [ClientFromEnvironment] to create new Clients. |  | ||||||
| type Client struct { | type Client struct { | ||||||
| 	base *url.URL | 	base *url.URL | ||||||
| 	http *http.Client | 	http http.Client | ||||||
| } | } | ||||||
|  |  | ||||||
| func checkError(resp *http.Response, body []byte) error { | func checkError(resp *http.Response, body []byte) error { | ||||||
| @@ -53,27 +40,56 @@ func checkError(resp *http.Response, body []byte) error { | |||||||
| 	return apiError | 	return apiError | ||||||
| } | } | ||||||
|  |  | ||||||
| // ClientFromEnvironment creates a new [Client] using configuration from the |  | ||||||
| // environment variable OLLAMA_HOST, which points to the network host and |  | ||||||
| // port on which the ollama service is listenting. The format of this variable |  | ||||||
| // is: |  | ||||||
| // |  | ||||||
| //	<scheme>://<host>:<port> |  | ||||||
| // |  | ||||||
| // If the variable is not specified, a default ollama host and port will be |  | ||||||
| // used. |  | ||||||
| func ClientFromEnvironment() (*Client, error) { | func ClientFromEnvironment() (*Client, error) { | ||||||
| 	return &Client{ | 	defaultPort := "11434" | ||||||
| 		base: envconfig.Host(), |  | ||||||
| 		http: http.DefaultClient, |  | ||||||
| 	}, nil |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func NewClient(base *url.URL, http *http.Client) *Client { | 	scheme, hostport, ok := strings.Cut(os.Getenv("OLLAMA_HOST"), "://") | ||||||
| 	return &Client{ | 	switch { | ||||||
| 		base: base, | 	case !ok: | ||||||
| 		http: http, | 		scheme, hostport = "http", os.Getenv("OLLAMA_HOST") | ||||||
|  | 	case scheme == "http": | ||||||
|  | 		defaultPort = "80" | ||||||
|  | 	case scheme == "https": | ||||||
|  | 		defaultPort = "443" | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
|  | 	// trim trailing slashes | ||||||
|  | 	hostport = strings.TrimRight(hostport, "/") | ||||||
|  |  | ||||||
|  | 	host, port, err := net.SplitHostPort(hostport) | ||||||
|  | 	if err != nil { | ||||||
|  | 		host, port = "127.0.0.1", defaultPort | ||||||
|  | 		if ip := net.ParseIP(strings.Trim(hostport, "[]")); ip != nil { | ||||||
|  | 			host = ip.String() | ||||||
|  | 		} else if hostport != "" { | ||||||
|  | 			host = hostport | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	client := Client{ | ||||||
|  | 		base: &url.URL{ | ||||||
|  | 			Scheme: scheme, | ||||||
|  | 			Host:   net.JoinHostPort(host, port), | ||||||
|  | 		}, | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	mockRequest, err := http.NewRequest(http.MethodHead, client.base.String(), nil) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	proxyURL, err := http.ProxyFromEnvironment(mockRequest) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	client.http = http.Client{ | ||||||
|  | 		Transport: &http.Transport{ | ||||||
|  | 			Proxy: http.ProxyURL(proxyURL), | ||||||
|  | 		}, | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return &client, nil | ||||||
| } | } | ||||||
|  |  | ||||||
| func (c *Client) do(ctx context.Context, method, path string, reqData, respData any) error { | func (c *Client) do(ctx context.Context, method, path string, reqData, respData any) error { | ||||||
| @@ -173,7 +189,7 @@ func (c *Client) stream(ctx context.Context, method, path string, data any, fn f | |||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		if errorResponse.Error != "" { | 		if errorResponse.Error != "" { | ||||||
| 			return errors.New(errorResponse.Error) | 			return fmt.Errorf(errorResponse.Error) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		if response.StatusCode >= http.StatusBadRequest { | 		if response.StatusCode >= http.StatusBadRequest { | ||||||
| @@ -192,14 +208,8 @@ func (c *Client) stream(ctx context.Context, method, path string, data any, fn f | |||||||
| 	return nil | 	return nil | ||||||
| } | } | ||||||
|  |  | ||||||
| // GenerateResponseFunc is a function that [Client.Generate] invokes every time |  | ||||||
| // a response is received from the service. If this function returns an error, |  | ||||||
| // [Client.Generate] will stop generating and return this error. |  | ||||||
| type GenerateResponseFunc func(GenerateResponse) error | type GenerateResponseFunc func(GenerateResponse) error | ||||||
|  |  | ||||||
| // Generate generates a response for a given prompt. The req parameter should |  | ||||||
| // be populated with prompt details. fn is called for each response (there may |  | ||||||
| // be multiple responses, e.g. in case streaming is enabled). |  | ||||||
| func (c *Client) Generate(ctx context.Context, req *GenerateRequest, fn GenerateResponseFunc) error { | func (c *Client) Generate(ctx context.Context, req *GenerateRequest, fn GenerateResponseFunc) error { | ||||||
| 	return c.stream(ctx, http.MethodPost, "/api/generate", req, func(bts []byte) error { | 	return c.stream(ctx, http.MethodPost, "/api/generate", req, func(bts []byte) error { | ||||||
| 		var resp GenerateResponse | 		var resp GenerateResponse | ||||||
| @@ -211,34 +221,8 @@ func (c *Client) Generate(ctx context.Context, req *GenerateRequest, fn Generate | |||||||
| 	}) | 	}) | ||||||
| } | } | ||||||
|  |  | ||||||
| // ChatResponseFunc is a function that [Client.Chat] invokes every time |  | ||||||
| // a response is received from the service. If this function returns an error, |  | ||||||
| // [Client.Chat] will stop generating and return this error. |  | ||||||
| type ChatResponseFunc func(ChatResponse) error |  | ||||||
|  |  | ||||||
| // Chat generates the next message in a chat. [ChatRequest] may contain a |  | ||||||
| // sequence of messages which can be used to maintain chat history with a model. |  | ||||||
| // fn is called for each response (there may be multiple responses, e.g. if case |  | ||||||
| // streaming is enabled). |  | ||||||
| func (c *Client) Chat(ctx context.Context, req *ChatRequest, fn ChatResponseFunc) error { |  | ||||||
| 	return c.stream(ctx, http.MethodPost, "/api/chat", req, func(bts []byte) error { |  | ||||||
| 		var resp ChatResponse |  | ||||||
| 		if err := json.Unmarshal(bts, &resp); err != nil { |  | ||||||
| 			return err |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		return fn(resp) |  | ||||||
| 	}) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // PullProgressFunc is a function that [Client.Pull] invokes every time there |  | ||||||
| // is progress with a "pull" request sent to the service. If this function |  | ||||||
| // returns an error, [Client.Pull] will stop the process and return this error. |  | ||||||
| type PullProgressFunc func(ProgressResponse) error | type PullProgressFunc func(ProgressResponse) error | ||||||
|  |  | ||||||
| // Pull downloads a model from the ollama library. fn is called each time |  | ||||||
| // progress is made on the request and can be used to display a progress bar, |  | ||||||
| // etc. |  | ||||||
| func (c *Client) Pull(ctx context.Context, req *PullRequest, fn PullProgressFunc) error { | func (c *Client) Pull(ctx context.Context, req *PullRequest, fn PullProgressFunc) error { | ||||||
| 	return c.stream(ctx, http.MethodPost, "/api/pull", req, func(bts []byte) error { | 	return c.stream(ctx, http.MethodPost, "/api/pull", req, func(bts []byte) error { | ||||||
| 		var resp ProgressResponse | 		var resp ProgressResponse | ||||||
| @@ -250,14 +234,8 @@ func (c *Client) Pull(ctx context.Context, req *PullRequest, fn PullProgressFunc | |||||||
| 	}) | 	}) | ||||||
| } | } | ||||||
|  |  | ||||||
| // PushProgressFunc is a function that [Client.Push] invokes when progress is |  | ||||||
| // made. |  | ||||||
| // It's similar to other progress function types like [PullProgressFunc]. |  | ||||||
| type PushProgressFunc func(ProgressResponse) error | type PushProgressFunc func(ProgressResponse) error | ||||||
|  |  | ||||||
| // Push uploads a model to the model library; requires registering for ollama.ai |  | ||||||
| // and adding a public key first. fn is called each time progress is made on |  | ||||||
| // the request and can be used to display a progress bar, etc. |  | ||||||
| func (c *Client) Push(ctx context.Context, req *PushRequest, fn PushProgressFunc) error { | func (c *Client) Push(ctx context.Context, req *PushRequest, fn PushProgressFunc) error { | ||||||
| 	return c.stream(ctx, http.MethodPost, "/api/push", req, func(bts []byte) error { | 	return c.stream(ctx, http.MethodPost, "/api/push", req, func(bts []byte) error { | ||||||
| 		var resp ProgressResponse | 		var resp ProgressResponse | ||||||
| @@ -269,15 +247,8 @@ func (c *Client) Push(ctx context.Context, req *PushRequest, fn PushProgressFunc | |||||||
| 	}) | 	}) | ||||||
| } | } | ||||||
|  |  | ||||||
| // CreateProgressFunc is a function that [Client.Create] invokes when progress |  | ||||||
| // is made. |  | ||||||
| // It's similar to other progress function types like [PullProgressFunc]. |  | ||||||
| type CreateProgressFunc func(ProgressResponse) error | type CreateProgressFunc func(ProgressResponse) error | ||||||
|  |  | ||||||
| // Create creates a model from a [Modelfile]. fn is a progress function that |  | ||||||
| // behaves similarly to other methods (see [Client.Pull]). |  | ||||||
| // |  | ||||||
| // [Modelfile]: https://github.com/ollama/ollama/blob/main/docs/modelfile.md |  | ||||||
| func (c *Client) Create(ctx context.Context, req *CreateRequest, fn CreateProgressFunc) error { | func (c *Client) Create(ctx context.Context, req *CreateRequest, fn CreateProgressFunc) error { | ||||||
| 	return c.stream(ctx, http.MethodPost, "/api/create", req, func(bts []byte) error { | 	return c.stream(ctx, http.MethodPost, "/api/create", req, func(bts []byte) error { | ||||||
| 		var resp ProgressResponse | 		var resp ProgressResponse | ||||||
| @@ -289,7 +260,6 @@ func (c *Client) Create(ctx context.Context, req *CreateRequest, fn CreateProgre | |||||||
| 	}) | 	}) | ||||||
| } | } | ||||||
|  |  | ||||||
| // List lists models that are available locally. |  | ||||||
| func (c *Client) List(ctx context.Context) (*ListResponse, error) { | func (c *Client) List(ctx context.Context) (*ListResponse, error) { | ||||||
| 	var lr ListResponse | 	var lr ListResponse | ||||||
| 	if err := c.do(ctx, http.MethodGet, "/api/tags", nil, &lr); err != nil { | 	if err := c.do(ctx, http.MethodGet, "/api/tags", nil, &lr); err != nil { | ||||||
| @@ -298,17 +268,6 @@ func (c *Client) List(ctx context.Context) (*ListResponse, error) { | |||||||
| 	return &lr, nil | 	return &lr, nil | ||||||
| } | } | ||||||
|  |  | ||||||
| // ListRunning lists running models. |  | ||||||
| func (c *Client) ListRunning(ctx context.Context) (*ProcessResponse, error) { |  | ||||||
| 	var lr ProcessResponse |  | ||||||
| 	if err := c.do(ctx, http.MethodGet, "/api/ps", nil, &lr); err != nil { |  | ||||||
| 		return nil, err |  | ||||||
| 	} |  | ||||||
| 	return &lr, nil |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // Copy copies a model - creating a model with another name from an existing |  | ||||||
| // model. |  | ||||||
| func (c *Client) Copy(ctx context.Context, req *CopyRequest) error { | func (c *Client) Copy(ctx context.Context, req *CopyRequest) error { | ||||||
| 	if err := c.do(ctx, http.MethodPost, "/api/copy", req, nil); err != nil { | 	if err := c.do(ctx, http.MethodPost, "/api/copy", req, nil); err != nil { | ||||||
| 		return err | 		return err | ||||||
| @@ -316,7 +275,6 @@ func (c *Client) Copy(ctx context.Context, req *CopyRequest) error { | |||||||
| 	return nil | 	return nil | ||||||
| } | } | ||||||
|  |  | ||||||
| // Delete deletes a model and its data. |  | ||||||
| func (c *Client) Delete(ctx context.Context, req *DeleteRequest) error { | func (c *Client) Delete(ctx context.Context, req *DeleteRequest) error { | ||||||
| 	if err := c.do(ctx, http.MethodDelete, "/api/delete", req, nil); err != nil { | 	if err := c.do(ctx, http.MethodDelete, "/api/delete", req, nil); err != nil { | ||||||
| 		return err | 		return err | ||||||
| @@ -324,7 +282,6 @@ func (c *Client) Delete(ctx context.Context, req *DeleteRequest) error { | |||||||
| 	return nil | 	return nil | ||||||
| } | } | ||||||
|  |  | ||||||
| // Show obtains model information, including details, modelfile, license etc. |  | ||||||
| func (c *Client) Show(ctx context.Context, req *ShowRequest) (*ShowResponse, error) { | func (c *Client) Show(ctx context.Context, req *ShowRequest) (*ShowResponse, error) { | ||||||
| 	var resp ShowResponse | 	var resp ShowResponse | ||||||
| 	if err := c.do(ctx, http.MethodPost, "/api/show", req, &resp); err != nil { | 	if err := c.do(ctx, http.MethodPost, "/api/show", req, &resp); err != nil { | ||||||
| @@ -333,8 +290,6 @@ func (c *Client) Show(ctx context.Context, req *ShowRequest) (*ShowResponse, err | |||||||
| 	return &resp, nil | 	return &resp, nil | ||||||
| } | } | ||||||
|  |  | ||||||
| // Heartbeat checks if the server has started and is responsive; if yes, it |  | ||||||
| // returns nil, otherwise an error. |  | ||||||
| func (c *Client) Heartbeat(ctx context.Context) error { | func (c *Client) Heartbeat(ctx context.Context) error { | ||||||
| 	if err := c.do(ctx, http.MethodHead, "/", nil, nil); err != nil { | 	if err := c.do(ctx, http.MethodHead, "/", nil, nil); err != nil { | ||||||
| 		return err | 		return err | ||||||
| @@ -342,39 +297,17 @@ func (c *Client) Heartbeat(ctx context.Context) error { | |||||||
| 	return nil | 	return nil | ||||||
| } | } | ||||||
|  |  | ||||||
| // Embed generates embeddings from a model. |  | ||||||
| func (c *Client) Embed(ctx context.Context, req *EmbedRequest) (*EmbedResponse, error) { |  | ||||||
| 	var resp EmbedResponse |  | ||||||
| 	if err := c.do(ctx, http.MethodPost, "/api/embed", req, &resp); err != nil { |  | ||||||
| 		return nil, err |  | ||||||
| 	} |  | ||||||
| 	return &resp, nil |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // Embeddings generates an embedding from a model. |  | ||||||
| func (c *Client) Embeddings(ctx context.Context, req *EmbeddingRequest) (*EmbeddingResponse, error) { |  | ||||||
| 	var resp EmbeddingResponse |  | ||||||
| 	if err := c.do(ctx, http.MethodPost, "/api/embeddings", req, &resp); err != nil { |  | ||||||
| 		return nil, err |  | ||||||
| 	} |  | ||||||
| 	return &resp, nil |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // CreateBlob creates a blob from a file on the server. digest is the |  | ||||||
| // expected SHA256 digest of the file, and r represents the file. |  | ||||||
| func (c *Client) CreateBlob(ctx context.Context, digest string, r io.Reader) error { | func (c *Client) CreateBlob(ctx context.Context, digest string, r io.Reader) error { | ||||||
| 	return c.do(ctx, http.MethodPost, fmt.Sprintf("/api/blobs/%s", digest), r, nil) | 	if err := c.do(ctx, http.MethodHead, fmt.Sprintf("/api/blobs/%s", digest), nil, nil); err != nil { | ||||||
| } | 		var statusError StatusError | ||||||
|  | 		if !errors.As(err, &statusError) || statusError.StatusCode != http.StatusNotFound { | ||||||
|  | 			return err | ||||||
|  | 		} | ||||||
|  |  | ||||||
| // Version returns the Ollama server version as a string. | 		if err := c.do(ctx, http.MethodPost, fmt.Sprintf("/api/blobs/%s", digest), r, nil); err != nil { | ||||||
| func (c *Client) Version(ctx context.Context) (string, error) { | 			return err | ||||||
| 	var version struct { | 		} | ||||||
| 		Version string `json:"version"` |  | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	if err := c.do(ctx, http.MethodGet, "/api/version", nil, &version); err != nil { | 	return nil | ||||||
| 		return "", err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	return version.Version, nil |  | ||||||
| } | } | ||||||
|   | |||||||
							
								
								
									
										284
									
								
								api/client.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,284 @@ | |||||||
|  | import os | ||||||
|  | import json | ||||||
|  | import requests | ||||||
|  | import os | ||||||
|  | import hashlib | ||||||
|  | import json | ||||||
|  | from pathlib import Path | ||||||
|  |  | ||||||
|  | BASE_URL = os.environ.get('OLLAMA_HOST', 'http://localhost:11434') | ||||||
|  |  | ||||||
|  | # Generate a response for a given prompt with a provided model. This is a streaming endpoint, so will be a series of responses. | ||||||
|  | # The final response object will include statistics and additional data from the request. Use the callback function to override | ||||||
|  | # the default handler. | ||||||
|  | def generate(model_name, prompt, system=None, template=None, format="", context=None, options=None, callback=None): | ||||||
|  |     try: | ||||||
|  |         url = f"{BASE_URL}/api/generate" | ||||||
|  |         payload = { | ||||||
|  |             "model": model_name,  | ||||||
|  |             "prompt": prompt,  | ||||||
|  |             "system": system,  | ||||||
|  |             "template": template,  | ||||||
|  |             "context": context,  | ||||||
|  |             "options": options, | ||||||
|  |             "format": format, | ||||||
|  |         } | ||||||
|  |          | ||||||
|  |         # Remove keys with None values | ||||||
|  |         payload = {k: v for k, v in payload.items() if v is not None} | ||||||
|  |          | ||||||
|  |         with requests.post(url, json=payload, stream=True) as response: | ||||||
|  |             response.raise_for_status() | ||||||
|  |              | ||||||
|  |             # Creating a variable to hold the context history of the final chunk | ||||||
|  |             final_context = None | ||||||
|  |              | ||||||
|  |             # Variable to hold concatenated response strings if no callback is provided | ||||||
|  |             full_response = "" | ||||||
|  |  | ||||||
|  |             # Iterating over the response line by line and displaying the details | ||||||
|  |             for line in response.iter_lines(): | ||||||
|  |                 if line: | ||||||
|  |                     # Parsing each line (JSON chunk) and extracting the details | ||||||
|  |                     chunk = json.loads(line) | ||||||
|  |                      | ||||||
|  |                     # If a callback function is provided, call it with the chunk | ||||||
|  |                     if callback: | ||||||
|  |                         callback(chunk) | ||||||
|  |                     else: | ||||||
|  |                         # If this is not the last chunk, add the "response" field value to full_response and print it | ||||||
|  |                         if not chunk.get("done"): | ||||||
|  |                             response_piece = chunk.get("response", "") | ||||||
|  |                             full_response += response_piece | ||||||
|  |                             print(response_piece, end="", flush=True) | ||||||
|  |                      | ||||||
|  |                     # Check if it's the last chunk (done is true) | ||||||
|  |                     if chunk.get("done"): | ||||||
|  |                         final_context = chunk.get("context") | ||||||
|  |              | ||||||
|  |             # Return the full response and the final context | ||||||
|  |             return full_response, final_context | ||||||
|  |     except requests.exceptions.RequestException as e: | ||||||
|  |         print(f"An error occurred: {e}") | ||||||
|  |         return None, None | ||||||
|  |      | ||||||
|  |  | ||||||
|  | # Create a blob file on the server if it doesn't exist. | ||||||
|  | def create_blob(digest, file_path): | ||||||
|  |     url = f"{BASE_URL}/api/blobs/{digest}" | ||||||
|  |  | ||||||
|  |     # Check if the blob exists | ||||||
|  |     response = requests.head(url) | ||||||
|  |     if response.status_code != 404: | ||||||
|  |         return  # Blob already exists, no need to upload | ||||||
|  |     response.raise_for_status() | ||||||
|  |  | ||||||
|  |     # Upload the blob | ||||||
|  |     with open(file_path, 'rb') as file_data: | ||||||
|  |         requests.post(url, data=file_data) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | # Create a model from a Modelfile. Use the callback function to override the default handler. | ||||||
|  | def create(model_name, filename, callback=None): | ||||||
|  |     try: | ||||||
|  |         file_path = Path(filename).expanduser().resolve() | ||||||
|  |         processed_lines = [] | ||||||
|  |  | ||||||
|  |         # Read and process the modelfile | ||||||
|  |         with open(file_path, 'r') as f: | ||||||
|  |             for line in f:             | ||||||
|  |                 # Skip empty or whitespace-only lines | ||||||
|  |                 if not line.strip(): | ||||||
|  |                     continue | ||||||
|  |              | ||||||
|  |                 command, args = line.split(maxsplit=1) | ||||||
|  |  | ||||||
|  |                 if command.upper() in ["FROM", "ADAPTER"]: | ||||||
|  |                     path = Path(args.strip()).expanduser() | ||||||
|  |  | ||||||
|  |                     # Check if path is relative and resolve it | ||||||
|  |                     if not path.is_absolute(): | ||||||
|  |                         path = (file_path.parent / path) | ||||||
|  |  | ||||||
|  |                     # Skip if file does not exist for "model", this is handled by the server | ||||||
|  |                     if not path.exists(): | ||||||
|  |                         processed_lines.append(line) | ||||||
|  |                         continue | ||||||
|  |  | ||||||
|  |                     # Calculate SHA-256 hash | ||||||
|  |                     with open(path, 'rb') as bin_file: | ||||||
|  |                         hash = hashlib.sha256() | ||||||
|  |                         hash.update(bin_file.read()) | ||||||
|  |                         blob = f"sha256:{hash.hexdigest()}" | ||||||
|  |                  | ||||||
|  |                     # Add the file to the remote server | ||||||
|  |                     create_blob(blob, path) | ||||||
|  |  | ||||||
|  |                     # Replace path with digest in the line | ||||||
|  |                     line = f"{command} @{blob}\n" | ||||||
|  |  | ||||||
|  |                 processed_lines.append(line) | ||||||
|  |  | ||||||
|  |         # Combine processed lines back into a single string | ||||||
|  |         modelfile_content = '\n'.join(processed_lines) | ||||||
|  |  | ||||||
|  |         url = f"{BASE_URL}/api/create" | ||||||
|  |         payload = {"name": model_name, "modelfile": modelfile_content} | ||||||
|  |  | ||||||
|  |         # Making a POST request with the stream parameter set to True to handle streaming responses | ||||||
|  |         with requests.post(url, json=payload, stream=True) as response: | ||||||
|  |             response.raise_for_status() | ||||||
|  |             # Iterating over the response line by line and displaying the status | ||||||
|  |             for line in response.iter_lines(): | ||||||
|  |                 if line: | ||||||
|  |                     chunk = json.loads(line) | ||||||
|  |                     if callback: | ||||||
|  |                         callback(chunk) | ||||||
|  |                     else: | ||||||
|  |                         print(f"Status: {chunk.get('status')}") | ||||||
|  |  | ||||||
|  |     except Exception as e: | ||||||
|  |         print(f"An error occurred: {e}") | ||||||
|  |  | ||||||
|  |  | ||||||
|  | # Pull a model from a the model registry. Cancelled pulls are resumed from where they left off, and multiple | ||||||
|  | # calls to will share the same download progress. Use the callback function to override the default handler. | ||||||
|  | def pull(model_name, insecure=False, callback=None): | ||||||
|  |     try: | ||||||
|  |         url = f"{BASE_URL}/api/pull" | ||||||
|  |         payload = { | ||||||
|  |             "name": model_name, | ||||||
|  |             "insecure": insecure | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         # Making a POST request with the stream parameter set to True to handle streaming responses | ||||||
|  |         with requests.post(url, json=payload, stream=True) as response: | ||||||
|  |             response.raise_for_status() | ||||||
|  |  | ||||||
|  |             # Iterating over the response line by line and displaying the details | ||||||
|  |             for line in response.iter_lines(): | ||||||
|  |                 if line: | ||||||
|  |                     # Parsing each line (JSON chunk) and extracting the details | ||||||
|  |                     chunk = json.loads(line) | ||||||
|  |  | ||||||
|  |                     # If a callback function is provided, call it with the chunk | ||||||
|  |                     if callback: | ||||||
|  |                         callback(chunk) | ||||||
|  |                     else: | ||||||
|  |                         # Print the status message directly to the console | ||||||
|  |                         print(chunk.get('status', ''), end='', flush=True) | ||||||
|  |                      | ||||||
|  |                     # If there's layer data, you might also want to print that (adjust as necessary) | ||||||
|  |                     if 'digest' in chunk: | ||||||
|  |                         print(f" - Digest: {chunk['digest']}", end='', flush=True) | ||||||
|  |                         print(f" - Total: {chunk['total']}", end='', flush=True) | ||||||
|  |                         print(f" - Completed: {chunk['completed']}", end='\n', flush=True) | ||||||
|  |                     else: | ||||||
|  |                         print() | ||||||
|  |     except requests.exceptions.RequestException as e: | ||||||
|  |         print(f"An error occurred: {e}") | ||||||
|  |  | ||||||
|  | # Push a model to the model registry. Use the callback function to override the default handler. | ||||||
|  | def push(model_name, insecure=False, callback=None): | ||||||
|  |     try: | ||||||
|  |         url = f"{BASE_URL}/api/push" | ||||||
|  |         payload = { | ||||||
|  |             "name": model_name, | ||||||
|  |             "insecure": insecure | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         # Making a POST request with the stream parameter set to True to handle streaming responses | ||||||
|  |         with requests.post(url, json=payload, stream=True) as response: | ||||||
|  |             response.raise_for_status() | ||||||
|  |  | ||||||
|  |             # Iterating over the response line by line and displaying the details | ||||||
|  |             for line in response.iter_lines(): | ||||||
|  |                 if line: | ||||||
|  |                     # Parsing each line (JSON chunk) and extracting the details | ||||||
|  |                     chunk = json.loads(line) | ||||||
|  |  | ||||||
|  |                     # If a callback function is provided, call it with the chunk | ||||||
|  |                     if callback: | ||||||
|  |                         callback(chunk) | ||||||
|  |                     else: | ||||||
|  |                         # Print the status message directly to the console | ||||||
|  |                         print(chunk.get('status', ''), end='', flush=True) | ||||||
|  |                      | ||||||
|  |                     # If there's layer data, you might also want to print that (adjust as necessary) | ||||||
|  |                     if 'digest' in chunk: | ||||||
|  |                         print(f" - Digest: {chunk['digest']}", end='', flush=True) | ||||||
|  |                         print(f" - Total: {chunk['total']}", end='', flush=True) | ||||||
|  |                         print(f" - Completed: {chunk['completed']}", end='\n', flush=True) | ||||||
|  |                     else: | ||||||
|  |                         print() | ||||||
|  |     except requests.exceptions.RequestException as e: | ||||||
|  |         print(f"An error occurred: {e}") | ||||||
|  |  | ||||||
|  | # List models that are available locally. | ||||||
|  | def list(): | ||||||
|  |     try: | ||||||
|  |         response = requests.get(f"{BASE_URL}/api/tags") | ||||||
|  |         response.raise_for_status() | ||||||
|  |         data = response.json() | ||||||
|  |         models = data.get('models', []) | ||||||
|  |         return models | ||||||
|  |  | ||||||
|  |     except requests.exceptions.RequestException as e: | ||||||
|  |         print(f"An error occurred: {e}") | ||||||
|  |         return None | ||||||
|  |  | ||||||
|  | # Copy a model. Creates a model with another name from an existing model. | ||||||
|  | def copy(source, destination): | ||||||
|  |     try: | ||||||
|  |         # Create the JSON payload | ||||||
|  |         payload = { | ||||||
|  |             "source": source, | ||||||
|  |             "destination": destination | ||||||
|  |         } | ||||||
|  |          | ||||||
|  |         response = requests.post(f"{BASE_URL}/api/copy", json=payload) | ||||||
|  |         response.raise_for_status() | ||||||
|  |          | ||||||
|  |         # If the request was successful, return a message indicating that the copy was successful | ||||||
|  |         return "Copy successful" | ||||||
|  |  | ||||||
|  |     except requests.exceptions.RequestException as e: | ||||||
|  |         print(f"An error occurred: {e}") | ||||||
|  |         return None | ||||||
|  |  | ||||||
|  | # Delete a model and its data. | ||||||
|  | def delete(model_name): | ||||||
|  |     try: | ||||||
|  |         url = f"{BASE_URL}/api/delete" | ||||||
|  |         payload = {"name": model_name} | ||||||
|  |         response = requests.delete(url, json=payload) | ||||||
|  |         response.raise_for_status() | ||||||
|  |         return "Delete successful" | ||||||
|  |     except requests.exceptions.RequestException as e: | ||||||
|  |         print(f"An error occurred: {e}") | ||||||
|  |         return None | ||||||
|  |  | ||||||
|  | # Show info about a model. | ||||||
|  | def show(model_name): | ||||||
|  |     try: | ||||||
|  |         url = f"{BASE_URL}/api/show" | ||||||
|  |         payload = {"name": model_name} | ||||||
|  |         response = requests.post(url, json=payload) | ||||||
|  |         response.raise_for_status() | ||||||
|  |          | ||||||
|  |         # Parse the JSON response and return it | ||||||
|  |         data = response.json() | ||||||
|  |         return data | ||||||
|  |     except requests.exceptions.RequestException as e: | ||||||
|  |         print(f"An error occurred: {e}") | ||||||
|  |         return None | ||||||
|  |  | ||||||
|  | def heartbeat(): | ||||||
|  |     try: | ||||||
|  |         url = f"{BASE_URL}/" | ||||||
|  |         response = requests.head(url) | ||||||
|  |         response.raise_for_status() | ||||||
|  |         return "Ollama is running" | ||||||
|  |     except requests.exceptions.RequestException as e: | ||||||
|  |         print(f"An error occurred: {e}") | ||||||
|  |         return "Ollama is not running" | ||||||
| @@ -1,8 +1,6 @@ | |||||||
| package api | package api | ||||||
|  |  | ||||||
| import ( | import "testing" | ||||||
| 	"testing" |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| func TestClientFromEnvironment(t *testing.T) { | func TestClientFromEnvironment(t *testing.T) { | ||||||
| 	type testCase struct { | 	type testCase struct { | ||||||
|   | |||||||
							
								
								
									
										649
									
								
								api/types.go
									
									
									
									
									
								
							
							
						
						| @@ -3,16 +3,13 @@ package api | |||||||
| import ( | import ( | ||||||
| 	"encoding/json" | 	"encoding/json" | ||||||
| 	"fmt" | 	"fmt" | ||||||
| 	"log/slog" |  | ||||||
| 	"math" | 	"math" | ||||||
| 	"os" | 	"os" | ||||||
| 	"reflect" | 	"reflect" | ||||||
| 	"strconv" |  | ||||||
| 	"strings" | 	"strings" | ||||||
| 	"time" | 	"time" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| // StatusError is an error with and HTTP status code. |  | ||||||
| type StatusError struct { | type StatusError struct { | ||||||
| 	StatusCode   int | 	StatusCode   int | ||||||
| 	Status       string | 	Status       string | ||||||
| @@ -33,178 +30,20 @@ func (e StatusError) Error() string { | |||||||
| 	} | 	} | ||||||
| } | } | ||||||
|  |  | ||||||
| // ImageData represents the raw binary data of an image file. |  | ||||||
| type ImageData []byte |  | ||||||
|  |  | ||||||
| // GenerateRequest describes a request sent by [Client.Generate]. While you |  | ||||||
| // have to specify the Model and Prompt fields, all the other fields have |  | ||||||
| // reasonable defaults for basic uses. |  | ||||||
| type GenerateRequest struct { | type GenerateRequest struct { | ||||||
| 	// Model is the model name; it should be a name familiar to Ollama from | 	Model    string `json:"model"` | ||||||
| 	// the library at https://ollama.com/library | 	Prompt   string `json:"prompt"` | ||||||
| 	Model string `json:"model"` | 	System   string `json:"system"` | ||||||
|  |  | ||||||
| 	// Prompt is the textual prompt to send to the model. |  | ||||||
| 	Prompt string `json:"prompt"` |  | ||||||
|  |  | ||||||
| 	// Suffix is the text that comes after the inserted text. |  | ||||||
| 	Suffix string `json:"suffix"` |  | ||||||
|  |  | ||||||
| 	// System overrides the model's default system message/prompt. |  | ||||||
| 	System string `json:"system"` |  | ||||||
|  |  | ||||||
| 	// Template overrides the model's default prompt template. |  | ||||||
| 	Template string `json:"template"` | 	Template string `json:"template"` | ||||||
|  | 	Context  []int  `json:"context,omitempty"` | ||||||
|  | 	Stream   *bool  `json:"stream,omitempty"` | ||||||
|  | 	Raw      bool   `json:"raw,omitempty"` | ||||||
|  | 	Format   string `json:"format"` | ||||||
|  |  | ||||||
| 	// Context is the context parameter returned from a previous call to |  | ||||||
| 	// Generate call. It can be used to keep a short conversational memory. |  | ||||||
| 	Context []int `json:"context,omitempty"` |  | ||||||
|  |  | ||||||
| 	// Stream specifies whether the response is streaming; it is true by default. |  | ||||||
| 	Stream *bool `json:"stream,omitempty"` |  | ||||||
|  |  | ||||||
| 	// Raw set to true means that no formatting will be applied to the prompt. |  | ||||||
| 	Raw bool `json:"raw,omitempty"` |  | ||||||
|  |  | ||||||
| 	// Format specifies the format to return a response in. |  | ||||||
| 	Format string `json:"format"` |  | ||||||
|  |  | ||||||
| 	// KeepAlive controls how long the model will stay loaded in memory following |  | ||||||
| 	// this request. |  | ||||||
| 	KeepAlive *Duration `json:"keep_alive,omitempty"` |  | ||||||
|  |  | ||||||
| 	// Images is an optional list of base64-encoded images accompanying this |  | ||||||
| 	// request, for multimodal models. |  | ||||||
| 	Images []ImageData `json:"images,omitempty"` |  | ||||||
|  |  | ||||||
| 	// Options lists model-specific options. For example, temperature can be |  | ||||||
| 	// set through this field, if the model supports it. |  | ||||||
| 	Options map[string]interface{} `json:"options"` | 	Options map[string]interface{} `json:"options"` | ||||||
| } | } | ||||||
|  |  | ||||||
| // ChatRequest describes a request sent by [Client.Chat]. | // Options specfied in GenerateRequest, if you add a new option here add it to the API docs also | ||||||
| type ChatRequest struct { |  | ||||||
| 	// Model is the model name, as in [GenerateRequest]. |  | ||||||
| 	Model string `json:"model"` |  | ||||||
|  |  | ||||||
| 	// Messages is the messages of the chat - can be used to keep a chat memory. |  | ||||||
| 	Messages []Message `json:"messages"` |  | ||||||
|  |  | ||||||
| 	// Stream enable streaming of returned response; true by default. |  | ||||||
| 	Stream *bool `json:"stream,omitempty"` |  | ||||||
|  |  | ||||||
| 	// Format is the format to return the response in (e.g. "json"). |  | ||||||
| 	Format string `json:"format"` |  | ||||||
|  |  | ||||||
| 	// KeepAlive controls how long the model will stay loaded into memory |  | ||||||
| 	// followin the request. |  | ||||||
| 	KeepAlive *Duration `json:"keep_alive,omitempty"` |  | ||||||
|  |  | ||||||
| 	// Tools is an optional list of tools the model has access to. |  | ||||||
| 	Tools `json:"tools,omitempty"` |  | ||||||
|  |  | ||||||
| 	// Options lists model-specific options. |  | ||||||
| 	Options map[string]interface{} `json:"options"` |  | ||||||
| } |  | ||||||
|  |  | ||||||
| type Tools []Tool |  | ||||||
|  |  | ||||||
| func (t Tools) String() string { |  | ||||||
| 	bts, _ := json.Marshal(t) |  | ||||||
| 	return string(bts) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (t Tool) String() string { |  | ||||||
| 	bts, _ := json.Marshal(t) |  | ||||||
| 	return string(bts) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // Message is a single message in a chat sequence. The message contains the |  | ||||||
| // role ("system", "user", or "assistant"), the content and an optional list |  | ||||||
| // of images. |  | ||||||
| type Message struct { |  | ||||||
| 	Role      string      `json:"role"` |  | ||||||
| 	Content   string      `json:"content"` |  | ||||||
| 	Images    []ImageData `json:"images,omitempty"` |  | ||||||
| 	ToolCalls []ToolCall  `json:"tool_calls,omitempty"` |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (m *Message) UnmarshalJSON(b []byte) error { |  | ||||||
| 	type Alias Message |  | ||||||
| 	var a Alias |  | ||||||
| 	if err := json.Unmarshal(b, &a); err != nil { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	*m = Message(a) |  | ||||||
| 	m.Role = strings.ToLower(m.Role) |  | ||||||
| 	return nil |  | ||||||
| } |  | ||||||
|  |  | ||||||
| type ToolCall struct { |  | ||||||
| 	Function ToolCallFunction `json:"function"` |  | ||||||
| } |  | ||||||
|  |  | ||||||
| type ToolCallFunction struct { |  | ||||||
| 	Name      string                    `json:"name"` |  | ||||||
| 	Arguments ToolCallFunctionArguments `json:"arguments"` |  | ||||||
| } |  | ||||||
|  |  | ||||||
| type ToolCallFunctionArguments map[string]any |  | ||||||
|  |  | ||||||
| func (t *ToolCallFunctionArguments) String() string { |  | ||||||
| 	bts, _ := json.Marshal(t) |  | ||||||
| 	return string(bts) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| type Tool struct { |  | ||||||
| 	Type     string       `json:"type"` |  | ||||||
| 	Function ToolFunction `json:"function"` |  | ||||||
| } |  | ||||||
|  |  | ||||||
| type ToolFunction struct { |  | ||||||
| 	Name        string `json:"name"` |  | ||||||
| 	Description string `json:"description"` |  | ||||||
| 	Parameters  struct { |  | ||||||
| 		Type       string   `json:"type"` |  | ||||||
| 		Required   []string `json:"required"` |  | ||||||
| 		Properties map[string]struct { |  | ||||||
| 			Type        string   `json:"type"` |  | ||||||
| 			Description string   `json:"description"` |  | ||||||
| 			Enum        []string `json:"enum,omitempty"` |  | ||||||
| 		} `json:"properties"` |  | ||||||
| 	} `json:"parameters"` |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (t *ToolFunction) String() string { |  | ||||||
| 	bts, _ := json.Marshal(t) |  | ||||||
| 	return string(bts) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // ChatResponse is the response returned by [Client.Chat]. Its fields are |  | ||||||
| // similar to [GenerateResponse]. |  | ||||||
| type ChatResponse struct { |  | ||||||
| 	Model      string    `json:"model"` |  | ||||||
| 	CreatedAt  time.Time `json:"created_at"` |  | ||||||
| 	Message    Message   `json:"message"` |  | ||||||
| 	DoneReason string    `json:"done_reason,omitempty"` |  | ||||||
|  |  | ||||||
| 	Done bool `json:"done"` |  | ||||||
|  |  | ||||||
| 	Metrics |  | ||||||
| } |  | ||||||
|  |  | ||||||
| type Metrics struct { |  | ||||||
| 	TotalDuration      time.Duration `json:"total_duration,omitempty"` |  | ||||||
| 	LoadDuration       time.Duration `json:"load_duration,omitempty"` |  | ||||||
| 	PromptEvalCount    int           `json:"prompt_eval_count,omitempty"` |  | ||||||
| 	PromptEvalDuration time.Duration `json:"prompt_eval_duration,omitempty"` |  | ||||||
| 	EvalCount          int           `json:"eval_count,omitempty"` |  | ||||||
| 	EvalDuration       time.Duration `json:"eval_duration,omitempty"` |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // Options specified in [GenerateRequest], if you add a new option here add it |  | ||||||
| // to the API docs also. |  | ||||||
| type Options struct { | type Options struct { | ||||||
| 	Runner | 	Runner | ||||||
|  |  | ||||||
| @@ -214,7 +53,6 @@ type Options struct { | |||||||
| 	NumPredict       int      `json:"num_predict,omitempty"` | 	NumPredict       int      `json:"num_predict,omitempty"` | ||||||
| 	TopK             int      `json:"top_k,omitempty"` | 	TopK             int      `json:"top_k,omitempty"` | ||||||
| 	TopP             float32  `json:"top_p,omitempty"` | 	TopP             float32  `json:"top_p,omitempty"` | ||||||
| 	MinP             float32  `json:"min_p,omitempty"` |  | ||||||
| 	TFSZ             float32  `json:"tfs_z,omitempty"` | 	TFSZ             float32  `json:"tfs_z,omitempty"` | ||||||
| 	TypicalP         float32  `json:"typical_p,omitempty"` | 	TypicalP         float32  `json:"typical_p,omitempty"` | ||||||
| 	RepeatLastN      int      `json:"repeat_last_n,omitempty"` | 	RepeatLastN      int      `json:"repeat_last_n,omitempty"` | ||||||
| @@ -231,142 +69,71 @@ type Options struct { | |||||||
|  |  | ||||||
| // Runner options which must be set when the model is loaded into memory | // Runner options which must be set when the model is loaded into memory | ||||||
| type Runner struct { | type Runner struct { | ||||||
| 	NumCtx    int   `json:"num_ctx,omitempty"` | 	UseNUMA            bool    `json:"numa,omitempty"` | ||||||
| 	NumBatch  int   `json:"num_batch,omitempty"` | 	NumCtx             int     `json:"num_ctx,omitempty"` | ||||||
| 	NumGPU    int   `json:"num_gpu,omitempty"` | 	NumBatch           int     `json:"num_batch,omitempty"` | ||||||
| 	MainGPU   int   `json:"main_gpu,omitempty"` | 	NumGQA             int     `json:"num_gqa,omitempty"` | ||||||
| 	LowVRAM   bool  `json:"low_vram,omitempty"` | 	NumGPU             int     `json:"num_gpu,omitempty"` | ||||||
| 	F16KV     bool  `json:"f16_kv,omitempty"` | 	MainGPU            int     `json:"main_gpu,omitempty"` | ||||||
| 	LogitsAll bool  `json:"logits_all,omitempty"` | 	LowVRAM            bool    `json:"low_vram,omitempty"` | ||||||
| 	VocabOnly bool  `json:"vocab_only,omitempty"` | 	F16KV              bool    `json:"f16_kv,omitempty"` | ||||||
| 	UseMMap   *bool `json:"use_mmap,omitempty"` | 	LogitsAll          bool    `json:"logits_all,omitempty"` | ||||||
| 	UseMLock  bool  `json:"use_mlock,omitempty"` | 	VocabOnly          bool    `json:"vocab_only,omitempty"` | ||||||
| 	NumThread int   `json:"num_thread,omitempty"` | 	UseMMap            bool    `json:"use_mmap,omitempty"` | ||||||
|  | 	UseMLock           bool    `json:"use_mlock,omitempty"` | ||||||
|  | 	EmbeddingOnly      bool    `json:"embedding_only,omitempty"` | ||||||
|  | 	RopeFrequencyBase  float32 `json:"rope_frequency_base,omitempty"` | ||||||
|  | 	RopeFrequencyScale float32 `json:"rope_frequency_scale,omitempty"` | ||||||
|  | 	NumThread          int     `json:"num_thread,omitempty"` | ||||||
| } | } | ||||||
|  |  | ||||||
| // EmbedRequest is the request passed to [Client.Embed]. |  | ||||||
| type EmbedRequest struct { |  | ||||||
| 	// Model is the model name. |  | ||||||
| 	Model string `json:"model"` |  | ||||||
|  |  | ||||||
| 	// Input is the input to embed. |  | ||||||
| 	Input any `json:"input"` |  | ||||||
|  |  | ||||||
| 	// KeepAlive controls how long the model will stay loaded in memory following |  | ||||||
| 	// this request. |  | ||||||
| 	KeepAlive *Duration `json:"keep_alive,omitempty"` |  | ||||||
|  |  | ||||||
| 	Truncate *bool `json:"truncate,omitempty"` |  | ||||||
|  |  | ||||||
| 	// Options lists model-specific options. |  | ||||||
| 	Options map[string]interface{} `json:"options"` |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // EmbedResponse is the response from [Client.Embed]. |  | ||||||
| type EmbedResponse struct { |  | ||||||
| 	Model      string      `json:"model"` |  | ||||||
| 	Embeddings [][]float32 `json:"embeddings"` |  | ||||||
|  |  | ||||||
| 	TotalDuration   time.Duration `json:"total_duration,omitempty"` |  | ||||||
| 	LoadDuration    time.Duration `json:"load_duration,omitempty"` |  | ||||||
| 	PromptEvalCount int           `json:"prompt_eval_count,omitempty"` |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // EmbeddingRequest is the request passed to [Client.Embeddings]. |  | ||||||
| type EmbeddingRequest struct { | type EmbeddingRequest struct { | ||||||
| 	// Model is the model name. | 	Model  string `json:"model"` | ||||||
| 	Model string `json:"model"` |  | ||||||
|  |  | ||||||
| 	// Prompt is the textual prompt to embed. |  | ||||||
| 	Prompt string `json:"prompt"` | 	Prompt string `json:"prompt"` | ||||||
|  |  | ||||||
| 	// KeepAlive controls how long the model will stay loaded in memory following |  | ||||||
| 	// this request. |  | ||||||
| 	KeepAlive *Duration `json:"keep_alive,omitempty"` |  | ||||||
|  |  | ||||||
| 	// Options lists model-specific options. |  | ||||||
| 	Options map[string]interface{} `json:"options"` | 	Options map[string]interface{} `json:"options"` | ||||||
| } | } | ||||||
|  |  | ||||||
| // EmbeddingResponse is the response from [Client.Embeddings]. |  | ||||||
| type EmbeddingResponse struct { | type EmbeddingResponse struct { | ||||||
| 	Embedding []float64 `json:"embedding"` | 	Embedding []float64 `json:"embedding"` | ||||||
| } | } | ||||||
|  |  | ||||||
| // CreateRequest is the request passed to [Client.Create]. |  | ||||||
| type CreateRequest struct { | type CreateRequest struct { | ||||||
| 	Model     string `json:"model"` | 	Name      string `json:"name"` | ||||||
|  | 	Path      string `json:"path"` | ||||||
| 	Modelfile string `json:"modelfile"` | 	Modelfile string `json:"modelfile"` | ||||||
| 	Stream    *bool  `json:"stream,omitempty"` | 	Stream    *bool  `json:"stream,omitempty"` | ||||||
| 	Quantize  string `json:"quantize,omitempty"` |  | ||||||
|  |  | ||||||
| 	// Deprecated: set the model name with Model instead |  | ||||||
| 	Name string `json:"name"` |  | ||||||
|  |  | ||||||
| 	// Deprecated: set the file content with Modelfile instead |  | ||||||
| 	Path string `json:"path"` |  | ||||||
|  |  | ||||||
| 	// Deprecated: use Quantize instead |  | ||||||
| 	Quantization string `json:"quantization,omitempty"` |  | ||||||
| } | } | ||||||
|  |  | ||||||
| // DeleteRequest is the request passed to [Client.Delete]. |  | ||||||
| type DeleteRequest struct { | type DeleteRequest struct { | ||||||
| 	Model string `json:"model"` |  | ||||||
|  |  | ||||||
| 	// Deprecated: set the model name with Model instead |  | ||||||
| 	Name string `json:"name"` | 	Name string `json:"name"` | ||||||
| } | } | ||||||
|  |  | ||||||
| // ShowRequest is the request passed to [Client.Show]. |  | ||||||
| type ShowRequest struct { | type ShowRequest struct { | ||||||
| 	Model  string `json:"model"` |  | ||||||
| 	System string `json:"system"` |  | ||||||
|  |  | ||||||
| 	// Template is deprecated |  | ||||||
| 	Template string `json:"template"` |  | ||||||
| 	Verbose  bool   `json:"verbose"` |  | ||||||
|  |  | ||||||
| 	Options map[string]interface{} `json:"options"` |  | ||||||
|  |  | ||||||
| 	// Deprecated: set the model name with Model instead |  | ||||||
| 	Name string `json:"name"` | 	Name string `json:"name"` | ||||||
| } | } | ||||||
|  |  | ||||||
| // ShowResponse is the response returned from [Client.Show]. |  | ||||||
| type ShowResponse struct { | type ShowResponse struct { | ||||||
| 	License       string         `json:"license,omitempty"` | 	License    string `json:"license,omitempty"` | ||||||
| 	Modelfile     string         `json:"modelfile,omitempty"` | 	Modelfile  string `json:"modelfile,omitempty"` | ||||||
| 	Parameters    string         `json:"parameters,omitempty"` | 	Parameters string `json:"parameters,omitempty"` | ||||||
| 	Template      string         `json:"template,omitempty"` | 	Template   string `json:"template,omitempty"` | ||||||
| 	System        string         `json:"system,omitempty"` | 	System     string `json:"system,omitempty"` | ||||||
| 	Details       ModelDetails   `json:"details,omitempty"` |  | ||||||
| 	Messages      []Message      `json:"messages,omitempty"` |  | ||||||
| 	ModelInfo     map[string]any `json:"model_info,omitempty"` |  | ||||||
| 	ProjectorInfo map[string]any `json:"projector_info,omitempty"` |  | ||||||
| 	ModifiedAt    time.Time      `json:"modified_at,omitempty"` |  | ||||||
| } | } | ||||||
|  |  | ||||||
| // CopyRequest is the request passed to [Client.Copy]. |  | ||||||
| type CopyRequest struct { | type CopyRequest struct { | ||||||
| 	Source      string `json:"source"` | 	Source      string `json:"source"` | ||||||
| 	Destination string `json:"destination"` | 	Destination string `json:"destination"` | ||||||
| } | } | ||||||
|  |  | ||||||
| // PullRequest is the request passed to [Client.Pull]. |  | ||||||
| type PullRequest struct { | type PullRequest struct { | ||||||
| 	Model    string `json:"model"` | 	Name     string `json:"name"` | ||||||
| 	Insecure bool   `json:"insecure,omitempty"` | 	Insecure bool   `json:"insecure,omitempty"` | ||||||
| 	Username string `json:"username"` | 	Username string `json:"username"` | ||||||
| 	Password string `json:"password"` | 	Password string `json:"password"` | ||||||
| 	Stream   *bool  `json:"stream,omitempty"` | 	Stream   *bool  `json:"stream,omitempty"` | ||||||
|  |  | ||||||
| 	// Deprecated: set the model name with Model instead |  | ||||||
| 	Name string `json:"name"` |  | ||||||
| } | } | ||||||
|  |  | ||||||
| // ProgressResponse is the response passed to progress functions like |  | ||||||
| // [PullProgressFunc] and [PushProgressFunc]. |  | ||||||
| type ProgressResponse struct { | type ProgressResponse struct { | ||||||
| 	Status    string `json:"status"` | 	Status    string `json:"status"` | ||||||
| 	Digest    string `json:"digest,omitempty"` | 	Digest    string `json:"digest,omitempty"` | ||||||
| @@ -374,122 +141,75 @@ type ProgressResponse struct { | |||||||
| 	Completed int64  `json:"completed,omitempty"` | 	Completed int64  `json:"completed,omitempty"` | ||||||
| } | } | ||||||
|  |  | ||||||
| // PushRequest is the request passed to [Client.Push]. |  | ||||||
| type PushRequest struct { | type PushRequest struct { | ||||||
| 	Model    string `json:"model"` | 	Name     string `json:"name"` | ||||||
| 	Insecure bool   `json:"insecure,omitempty"` | 	Insecure bool   `json:"insecure,omitempty"` | ||||||
| 	Username string `json:"username"` | 	Username string `json:"username"` | ||||||
| 	Password string `json:"password"` | 	Password string `json:"password"` | ||||||
| 	Stream   *bool  `json:"stream,omitempty"` | 	Stream   *bool  `json:"stream,omitempty"` | ||||||
|  |  | ||||||
| 	// Deprecated: set the model name with Model instead |  | ||||||
| 	Name string `json:"name"` |  | ||||||
| } | } | ||||||
|  |  | ||||||
| // ListResponse is the response from [Client.List]. |  | ||||||
| type ListResponse struct { | type ListResponse struct { | ||||||
| 	Models []ListModelResponse `json:"models"` | 	Models []ModelResponse `json:"models"` | ||||||
| } | } | ||||||
|  |  | ||||||
| // ProcessResponse is the response from [Client.Process]. | type ModelResponse struct { | ||||||
| type ProcessResponse struct { | 	Name       string    `json:"name"` | ||||||
| 	Models []ProcessModelResponse `json:"models"` | 	ModifiedAt time.Time `json:"modified_at"` | ||||||
| } | 	Size       int64     `json:"size"` | ||||||
|  | 	Digest     string    `json:"digest"` | ||||||
| // ListModelResponse is a single model description in [ListResponse]. |  | ||||||
| type ListModelResponse struct { |  | ||||||
| 	Name       string       `json:"name"` |  | ||||||
| 	Model      string       `json:"model"` |  | ||||||
| 	ModifiedAt time.Time    `json:"modified_at"` |  | ||||||
| 	Size       int64        `json:"size"` |  | ||||||
| 	Digest     string       `json:"digest"` |  | ||||||
| 	Details    ModelDetails `json:"details,omitempty"` |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // ProcessModelResponse is a single model description in [ProcessResponse]. |  | ||||||
| type ProcessModelResponse struct { |  | ||||||
| 	Name      string       `json:"name"` |  | ||||||
| 	Model     string       `json:"model"` |  | ||||||
| 	Size      int64        `json:"size"` |  | ||||||
| 	Digest    string       `json:"digest"` |  | ||||||
| 	Details   ModelDetails `json:"details,omitempty"` |  | ||||||
| 	ExpiresAt time.Time    `json:"expires_at"` |  | ||||||
| 	SizeVRAM  int64        `json:"size_vram"` |  | ||||||
| } |  | ||||||
|  |  | ||||||
| type RetrieveModelResponse struct { |  | ||||||
| 	Id      string `json:"id"` |  | ||||||
| 	Object  string `json:"object"` |  | ||||||
| 	Created int64  `json:"created"` |  | ||||||
| 	OwnedBy string `json:"owned_by"` |  | ||||||
| } | } | ||||||
|  |  | ||||||
| type TokenResponse struct { | type TokenResponse struct { | ||||||
| 	Token string `json:"token"` | 	Token string `json:"token"` | ||||||
| } | } | ||||||
|  |  | ||||||
| // GenerateResponse is the response passed into [GenerateResponseFunc]. |  | ||||||
| type GenerateResponse struct { | type GenerateResponse struct { | ||||||
| 	// Model is the model name that generated the response. | 	Model     string    `json:"model"` | ||||||
| 	Model string `json:"model"` |  | ||||||
|  |  | ||||||
| 	// CreatedAt is the timestamp of the response. |  | ||||||
| 	CreatedAt time.Time `json:"created_at"` | 	CreatedAt time.Time `json:"created_at"` | ||||||
|  | 	Response  string    `json:"response"` | ||||||
|  |  | ||||||
| 	// Response is the textual response itself. | 	Done    bool  `json:"done"` | ||||||
| 	Response string `json:"response"` |  | ||||||
|  |  | ||||||
| 	// Done specifies if the response is complete. |  | ||||||
| 	Done bool `json:"done"` |  | ||||||
|  |  | ||||||
| 	// DoneReason is the reason the model stopped generating text. |  | ||||||
| 	DoneReason string `json:"done_reason,omitempty"` |  | ||||||
|  |  | ||||||
| 	// Context is an encoding of the conversation used in this response; this |  | ||||||
| 	// can be sent in the next request to keep a conversational memory. |  | ||||||
| 	Context []int `json:"context,omitempty"` | 	Context []int `json:"context,omitempty"` | ||||||
|  |  | ||||||
| 	Metrics | 	TotalDuration      time.Duration `json:"total_duration,omitempty"` | ||||||
|  | 	LoadDuration       time.Duration `json:"load_duration,omitempty"` | ||||||
|  | 	PromptEvalCount    int           `json:"prompt_eval_count,omitempty"` | ||||||
|  | 	PromptEvalDuration time.Duration `json:"prompt_eval_duration,omitempty"` | ||||||
|  | 	EvalCount          int           `json:"eval_count,omitempty"` | ||||||
|  | 	EvalDuration       time.Duration `json:"eval_duration,omitempty"` | ||||||
| } | } | ||||||
|  |  | ||||||
| // ModelDetails provides details about a model. | func (r *GenerateResponse) Summary() { | ||||||
| type ModelDetails struct { | 	if r.TotalDuration > 0 { | ||||||
| 	ParentModel       string   `json:"parent_model"` | 		fmt.Fprintf(os.Stderr, "total duration:       %v\n", r.TotalDuration) | ||||||
| 	Format            string   `json:"format"` |  | ||||||
| 	Family            string   `json:"family"` |  | ||||||
| 	Families          []string `json:"families"` |  | ||||||
| 	ParameterSize     string   `json:"parameter_size"` |  | ||||||
| 	QuantizationLevel string   `json:"quantization_level"` |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (m *Metrics) Summary() { |  | ||||||
| 	if m.TotalDuration > 0 { |  | ||||||
| 		fmt.Fprintf(os.Stderr, "total duration:       %v\n", m.TotalDuration) |  | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	if m.LoadDuration > 0 { | 	if r.LoadDuration > 0 { | ||||||
| 		fmt.Fprintf(os.Stderr, "load duration:        %v\n", m.LoadDuration) | 		fmt.Fprintf(os.Stderr, "load duration:        %v\n", r.LoadDuration) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	if m.PromptEvalCount > 0 { | 	if r.PromptEvalCount > 0 { | ||||||
| 		fmt.Fprintf(os.Stderr, "prompt eval count:    %d token(s)\n", m.PromptEvalCount) | 		fmt.Fprintf(os.Stderr, "prompt eval count:    %d token(s)\n", r.PromptEvalCount) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	if m.PromptEvalDuration > 0 { | 	if r.PromptEvalDuration > 0 { | ||||||
| 		fmt.Fprintf(os.Stderr, "prompt eval duration: %s\n", m.PromptEvalDuration) | 		fmt.Fprintf(os.Stderr, "prompt eval duration: %s\n", r.PromptEvalDuration) | ||||||
| 		fmt.Fprintf(os.Stderr, "prompt eval rate:     %.2f tokens/s\n", float64(m.PromptEvalCount)/m.PromptEvalDuration.Seconds()) | 		fmt.Fprintf(os.Stderr, "prompt eval rate:     %.2f tokens/s\n", float64(r.PromptEvalCount)/r.PromptEvalDuration.Seconds()) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	if m.EvalCount > 0 { | 	if r.EvalCount > 0 { | ||||||
| 		fmt.Fprintf(os.Stderr, "eval count:           %d token(s)\n", m.EvalCount) | 		fmt.Fprintf(os.Stderr, "eval count:           %d token(s)\n", r.EvalCount) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	if m.EvalDuration > 0 { | 	if r.EvalDuration > 0 { | ||||||
| 		fmt.Fprintf(os.Stderr, "eval duration:        %s\n", m.EvalDuration) | 		fmt.Fprintf(os.Stderr, "eval duration:        %s\n", r.EvalDuration) | ||||||
| 		fmt.Fprintf(os.Stderr, "eval rate:            %.2f tokens/s\n", float64(m.EvalCount)/m.EvalDuration.Seconds()) | 		fmt.Fprintf(os.Stderr, "eval rate:            %.2f tokens/s\n", float64(r.EvalCount)/r.EvalDuration.Seconds()) | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
|  |  | ||||||
|  | var ErrInvalidOpts = fmt.Errorf("invalid options") | ||||||
|  |  | ||||||
| func (opts *Options) FromMap(m map[string]interface{}) error { | func (opts *Options) FromMap(m map[string]interface{}) error { | ||||||
| 	valueOpts := reflect.ValueOf(opts).Elem() // names of the fields in the options struct | 	valueOpts := reflect.ValueOf(opts).Elem() // names of the fields in the options struct | ||||||
| 	typeOpts := reflect.TypeOf(opts).Elem()   // types of the fields in the options struct | 	typeOpts := reflect.TypeOf(opts).Elem()   // types of the fields in the options struct | ||||||
| @@ -503,94 +223,81 @@ func (opts *Options) FromMap(m map[string]interface{}) error { | |||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
|  | 	invalidOpts := []string{} | ||||||
| 	for key, val := range m { | 	for key, val := range m { | ||||||
| 		opt, ok := jsonOpts[key] | 		if opt, ok := jsonOpts[key]; ok { | ||||||
| 		if !ok { | 			field := valueOpts.FieldByName(opt.Name) | ||||||
| 			slog.Warn("invalid option provided", "option", key) | 			if field.IsValid() && field.CanSet() { | ||||||
| 			continue | 				if val == nil { | ||||||
| 		} | 					continue | ||||||
|  | 				} | ||||||
|  |  | ||||||
| 		field := valueOpts.FieldByName(opt.Name) | 				switch field.Kind() { | ||||||
| 		if field.IsValid() && field.CanSet() { | 				case reflect.Int: | ||||||
| 			if val == nil { | 					switch t := val.(type) { | ||||||
| 				continue | 					case int64: | ||||||
| 			} | 						field.SetInt(t) | ||||||
|  | 					case float64: | ||||||
| 			switch field.Kind() { | 						// when JSON unmarshals numbers, it uses float64, not int | ||||||
| 			case reflect.Int: | 						field.SetInt(int64(t)) | ||||||
| 				switch t := val.(type) { | 					default: | ||||||
| 				case int64: | 						return fmt.Errorf("option %q must be of type integer", key) | ||||||
| 					field.SetInt(t) |  | ||||||
| 				case float64: |  | ||||||
| 					// when JSON unmarshals numbers, it uses float64, not int |  | ||||||
| 					field.SetInt(int64(t)) |  | ||||||
| 				default: |  | ||||||
| 					return fmt.Errorf("option %q must be of type integer", key) |  | ||||||
| 				} |  | ||||||
| 			case reflect.Bool: |  | ||||||
| 				val, ok := val.(bool) |  | ||||||
| 				if !ok { |  | ||||||
| 					return fmt.Errorf("option %q must be of type boolean", key) |  | ||||||
| 				} |  | ||||||
| 				field.SetBool(val) |  | ||||||
| 			case reflect.Float32: |  | ||||||
| 				// JSON unmarshals to float64 |  | ||||||
| 				val, ok := val.(float64) |  | ||||||
| 				if !ok { |  | ||||||
| 					return fmt.Errorf("option %q must be of type float32", key) |  | ||||||
| 				} |  | ||||||
| 				field.SetFloat(val) |  | ||||||
| 			case reflect.String: |  | ||||||
| 				val, ok := val.(string) |  | ||||||
| 				if !ok { |  | ||||||
| 					return fmt.Errorf("option %q must be of type string", key) |  | ||||||
| 				} |  | ||||||
| 				field.SetString(val) |  | ||||||
| 			case reflect.Slice: |  | ||||||
| 				// JSON unmarshals to []interface{}, not []string |  | ||||||
| 				val, ok := val.([]interface{}) |  | ||||||
| 				if !ok { |  | ||||||
| 					return fmt.Errorf("option %q must be of type array", key) |  | ||||||
| 				} |  | ||||||
| 				// convert []interface{} to []string |  | ||||||
| 				slice := make([]string, len(val)) |  | ||||||
| 				for i, item := range val { |  | ||||||
| 					str, ok := item.(string) |  | ||||||
| 					if !ok { |  | ||||||
| 						return fmt.Errorf("option %q must be of an array of strings", key) |  | ||||||
| 					} | 					} | ||||||
| 					slice[i] = str | 				case reflect.Bool: | ||||||
| 				} |  | ||||||
| 				field.Set(reflect.ValueOf(slice)) |  | ||||||
| 			case reflect.Pointer: |  | ||||||
| 				var b bool |  | ||||||
| 				if field.Type() == reflect.TypeOf(&b) { |  | ||||||
| 					val, ok := val.(bool) | 					val, ok := val.(bool) | ||||||
| 					if !ok { | 					if !ok { | ||||||
| 						return fmt.Errorf("option %q must be of type boolean", key) | 						return fmt.Errorf("option %q must be of type boolean", key) | ||||||
| 					} | 					} | ||||||
| 					field.Set(reflect.ValueOf(&val)) | 					field.SetBool(val) | ||||||
| 				} else { | 				case reflect.Float32: | ||||||
| 					return fmt.Errorf("unknown type loading config params: %v %v", field.Kind(), field.Type()) | 					// JSON unmarshals to float64 | ||||||
|  | 					val, ok := val.(float64) | ||||||
|  | 					if !ok { | ||||||
|  | 						return fmt.Errorf("option %q must be of type float32", key) | ||||||
|  | 					} | ||||||
|  | 					field.SetFloat(val) | ||||||
|  | 				case reflect.String: | ||||||
|  | 					val, ok := val.(string) | ||||||
|  | 					if !ok { | ||||||
|  | 						return fmt.Errorf("option %q must be of type string", key) | ||||||
|  | 					} | ||||||
|  | 					field.SetString(val) | ||||||
|  | 				case reflect.Slice: | ||||||
|  | 					// JSON unmarshals to []interface{}, not []string | ||||||
|  | 					val, ok := val.([]interface{}) | ||||||
|  | 					if !ok { | ||||||
|  | 						return fmt.Errorf("option %q must be of type array", key) | ||||||
|  | 					} | ||||||
|  | 					// convert []interface{} to []string | ||||||
|  | 					slice := make([]string, len(val)) | ||||||
|  | 					for i, item := range val { | ||||||
|  | 						str, ok := item.(string) | ||||||
|  | 						if !ok { | ||||||
|  | 							return fmt.Errorf("option %q must be of an array of strings", key) | ||||||
|  | 						} | ||||||
|  | 						slice[i] = str | ||||||
|  | 					} | ||||||
|  | 					field.Set(reflect.ValueOf(slice)) | ||||||
|  | 				default: | ||||||
|  | 					return fmt.Errorf("unknown type loading config params: %v", field.Kind()) | ||||||
| 				} | 				} | ||||||
| 			default: |  | ||||||
| 				return fmt.Errorf("unknown type loading config params: %v", field.Kind()) |  | ||||||
| 			} | 			} | ||||||
|  | 		} else { | ||||||
|  | 			invalidOpts = append(invalidOpts, key) | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
|  | 	if len(invalidOpts) > 0 { | ||||||
|  | 		return fmt.Errorf("%w: %v", ErrInvalidOpts, strings.Join(invalidOpts, ", ")) | ||||||
|  | 	} | ||||||
| 	return nil | 	return nil | ||||||
| } | } | ||||||
|  |  | ||||||
| // DefaultOptions is the default set of options for [GenerateRequest]; these |  | ||||||
| // values are used unless the user specifies other values explicitly. |  | ||||||
| func DefaultOptions() Options { | func DefaultOptions() Options { | ||||||
| 	return Options{ | 	return Options{ | ||||||
| 		// options set on request to runner | 		// options set on request to runner | ||||||
| 		NumPredict: -1, | 		NumPredict:       -1, | ||||||
|  | 		NumKeep:          0, | ||||||
| 		// set a minimal num_keep to avoid issues on context shifts |  | ||||||
| 		NumKeep:          4, |  | ||||||
| 		Temperature:      0.8, | 		Temperature:      0.8, | ||||||
| 		TopK:             40, | 		TopK:             40, | ||||||
| 		TopP:             0.9, | 		TopP:             0.9, | ||||||
| @@ -608,14 +315,19 @@ func DefaultOptions() Options { | |||||||
|  |  | ||||||
| 		Runner: Runner{ | 		Runner: Runner{ | ||||||
| 			// options set when the model is loaded | 			// options set when the model is loaded | ||||||
| 			NumCtx:    2048, | 			NumCtx:             2048, | ||||||
| 			NumBatch:  512, | 			RopeFrequencyBase:  10000.0, | ||||||
| 			NumGPU:    -1, // -1 here indicates that NumGPU should be set dynamically | 			RopeFrequencyScale: 1.0, | ||||||
| 			NumThread: 0,  // let the runtime decide | 			NumBatch:           512, | ||||||
| 			LowVRAM:   false, | 			NumGPU:             -1, // -1 here indicates that NumGPU should be set dynamically | ||||||
| 			F16KV:     true, | 			NumGQA:             1, | ||||||
| 			UseMLock:  false, | 			NumThread:          0, // let the runtime decide | ||||||
| 			UseMMap:   nil, | 			LowVRAM:            false, | ||||||
|  | 			F16KV:              true, | ||||||
|  | 			UseMLock:           false, | ||||||
|  | 			UseMMap:            true, | ||||||
|  | 			UseNUMA:            false, | ||||||
|  | 			EmbeddingOnly:      true, | ||||||
| 		}, | 		}, | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
| @@ -624,13 +336,6 @@ type Duration struct { | |||||||
| 	time.Duration | 	time.Duration | ||||||
| } | } | ||||||
|  |  | ||||||
| func (d Duration) MarshalJSON() ([]byte, error) { |  | ||||||
| 	if d.Duration < 0 { |  | ||||||
| 		return []byte("-1"), nil |  | ||||||
| 	} |  | ||||||
| 	return []byte("\"" + d.Duration.String() + "\""), nil |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (d *Duration) UnmarshalJSON(b []byte) (err error) { | func (d *Duration) UnmarshalJSON(b []byte) (err error) { | ||||||
| 	var v any | 	var v any | ||||||
| 	if err := json.Unmarshal(b, &v); err != nil { | 	if err := json.Unmarshal(b, &v); err != nil { | ||||||
| @@ -642,92 +347,16 @@ func (d *Duration) UnmarshalJSON(b []byte) (err error) { | |||||||
| 	switch t := v.(type) { | 	switch t := v.(type) { | ||||||
| 	case float64: | 	case float64: | ||||||
| 		if t < 0 { | 		if t < 0 { | ||||||
| 			d.Duration = time.Duration(math.MaxInt64) | 			t = math.MaxFloat64 | ||||||
| 		} else { |  | ||||||
| 			d.Duration = time.Duration(int(t) * int(time.Second)) |  | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
|  | 		d.Duration = time.Duration(t) | ||||||
| 	case string: | 	case string: | ||||||
| 		d.Duration, err = time.ParseDuration(t) | 		d.Duration, err = time.ParseDuration(t) | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			return err | 			return err | ||||||
| 		} | 		} | ||||||
| 		if d.Duration < 0 { |  | ||||||
| 			d.Duration = time.Duration(math.MaxInt64) |  | ||||||
| 		} |  | ||||||
| 	default: |  | ||||||
| 		return fmt.Errorf("Unsupported type: '%s'", reflect.TypeOf(v)) |  | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	return nil | 	return nil | ||||||
| } | } | ||||||
|  |  | ||||||
| // FormatParams converts specified parameter options to their correct types |  | ||||||
| func FormatParams(params map[string][]string) (map[string]interface{}, error) { |  | ||||||
| 	opts := Options{} |  | ||||||
| 	valueOpts := reflect.ValueOf(&opts).Elem() // names of the fields in the options struct |  | ||||||
| 	typeOpts := reflect.TypeOf(opts)           // types of the fields in the options struct |  | ||||||
|  |  | ||||||
| 	// build map of json struct tags to their types |  | ||||||
| 	jsonOpts := make(map[string]reflect.StructField) |  | ||||||
| 	for _, field := range reflect.VisibleFields(typeOpts) { |  | ||||||
| 		jsonTag := strings.Split(field.Tag.Get("json"), ",")[0] |  | ||||||
| 		if jsonTag != "" { |  | ||||||
| 			jsonOpts[jsonTag] = field |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	out := make(map[string]interface{}) |  | ||||||
| 	// iterate params and set values based on json struct tags |  | ||||||
| 	for key, vals := range params { |  | ||||||
| 		if opt, ok := jsonOpts[key]; !ok { |  | ||||||
| 			return nil, fmt.Errorf("unknown parameter '%s'", key) |  | ||||||
| 		} else { |  | ||||||
| 			field := valueOpts.FieldByName(opt.Name) |  | ||||||
| 			if field.IsValid() && field.CanSet() { |  | ||||||
| 				switch field.Kind() { |  | ||||||
| 				case reflect.Float32: |  | ||||||
| 					floatVal, err := strconv.ParseFloat(vals[0], 32) |  | ||||||
| 					if err != nil { |  | ||||||
| 						return nil, fmt.Errorf("invalid float value %s", vals) |  | ||||||
| 					} |  | ||||||
|  |  | ||||||
| 					out[key] = float32(floatVal) |  | ||||||
| 				case reflect.Int: |  | ||||||
| 					intVal, err := strconv.ParseInt(vals[0], 10, 64) |  | ||||||
| 					if err != nil { |  | ||||||
| 						return nil, fmt.Errorf("invalid int value %s", vals) |  | ||||||
| 					} |  | ||||||
|  |  | ||||||
| 					out[key] = intVal |  | ||||||
| 				case reflect.Bool: |  | ||||||
| 					boolVal, err := strconv.ParseBool(vals[0]) |  | ||||||
| 					if err != nil { |  | ||||||
| 						return nil, fmt.Errorf("invalid bool value %s", vals) |  | ||||||
| 					} |  | ||||||
|  |  | ||||||
| 					out[key] = boolVal |  | ||||||
| 				case reflect.String: |  | ||||||
| 					out[key] = vals[0] |  | ||||||
| 				case reflect.Slice: |  | ||||||
| 					// TODO: only string slices are supported right now |  | ||||||
| 					out[key] = vals |  | ||||||
| 				case reflect.Pointer: |  | ||||||
| 					var b bool |  | ||||||
| 					if field.Type() == reflect.TypeOf(&b) { |  | ||||||
| 						boolVal, err := strconv.ParseBool(vals[0]) |  | ||||||
| 						if err != nil { |  | ||||||
| 							return nil, fmt.Errorf("invalid bool value %s", vals) |  | ||||||
| 						} |  | ||||||
| 						out[key] = &boolVal |  | ||||||
| 					} else { |  | ||||||
| 						return nil, fmt.Errorf("unknown type %s for %s", field.Kind(), key) |  | ||||||
| 					} |  | ||||||
| 				default: |  | ||||||
| 					return nil, fmt.Errorf("unknown type %s for %s", field.Kind(), key) |  | ||||||
| 				} |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	return out, nil |  | ||||||
| } |  | ||||||
|   | |||||||
| @@ -1,233 +0,0 @@ | |||||||
| package api |  | ||||||
|  |  | ||||||
| import ( |  | ||||||
| 	"encoding/json" |  | ||||||
| 	"errors" |  | ||||||
| 	"math" |  | ||||||
| 	"testing" |  | ||||||
| 	"time" |  | ||||||
|  |  | ||||||
| 	"github.com/stretchr/testify/assert" |  | ||||||
| 	"github.com/stretchr/testify/require" |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| func TestKeepAliveParsingFromJSON(t *testing.T) { |  | ||||||
| 	tests := []struct { |  | ||||||
| 		name string |  | ||||||
| 		req  string |  | ||||||
| 		exp  *Duration |  | ||||||
| 	}{ |  | ||||||
| 		{ |  | ||||||
| 			name: "Positive Integer", |  | ||||||
| 			req:  `{ "keep_alive": 42 }`, |  | ||||||
| 			exp:  &Duration{42 * time.Second}, |  | ||||||
| 		}, |  | ||||||
| 		{ |  | ||||||
| 			name: "Positive Float", |  | ||||||
| 			req:  `{ "keep_alive": 42.5 }`, |  | ||||||
| 			exp:  &Duration{42 * time.Second}, |  | ||||||
| 		}, |  | ||||||
| 		{ |  | ||||||
| 			name: "Positive Integer String", |  | ||||||
| 			req:  `{ "keep_alive": "42m" }`, |  | ||||||
| 			exp:  &Duration{42 * time.Minute}, |  | ||||||
| 		}, |  | ||||||
| 		{ |  | ||||||
| 			name: "Negative Integer", |  | ||||||
| 			req:  `{ "keep_alive": -1 }`, |  | ||||||
| 			exp:  &Duration{math.MaxInt64}, |  | ||||||
| 		}, |  | ||||||
| 		{ |  | ||||||
| 			name: "Negative Float", |  | ||||||
| 			req:  `{ "keep_alive": -3.14 }`, |  | ||||||
| 			exp:  &Duration{math.MaxInt64}, |  | ||||||
| 		}, |  | ||||||
| 		{ |  | ||||||
| 			name: "Negative Integer String", |  | ||||||
| 			req:  `{ "keep_alive": "-1m" }`, |  | ||||||
| 			exp:  &Duration{math.MaxInt64}, |  | ||||||
| 		}, |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	for _, test := range tests { |  | ||||||
| 		t.Run(test.name, func(t *testing.T) { |  | ||||||
| 			var dec ChatRequest |  | ||||||
| 			err := json.Unmarshal([]byte(test.req), &dec) |  | ||||||
| 			require.NoError(t, err) |  | ||||||
|  |  | ||||||
| 			assert.Equal(t, test.exp, dec.KeepAlive) |  | ||||||
| 		}) |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func TestDurationMarshalUnmarshal(t *testing.T) { |  | ||||||
| 	tests := []struct { |  | ||||||
| 		name     string |  | ||||||
| 		input    time.Duration |  | ||||||
| 		expected time.Duration |  | ||||||
| 	}{ |  | ||||||
| 		{ |  | ||||||
| 			"negative duration", |  | ||||||
| 			time.Duration(-1), |  | ||||||
| 			time.Duration(math.MaxInt64), |  | ||||||
| 		}, |  | ||||||
| 		{ |  | ||||||
| 			"positive duration", |  | ||||||
| 			42 * time.Second, |  | ||||||
| 			42 * time.Second, |  | ||||||
| 		}, |  | ||||||
| 		{ |  | ||||||
| 			"another positive duration", |  | ||||||
| 			42 * time.Minute, |  | ||||||
| 			42 * time.Minute, |  | ||||||
| 		}, |  | ||||||
| 		{ |  | ||||||
| 			"zero duration", |  | ||||||
| 			time.Duration(0), |  | ||||||
| 			time.Duration(0), |  | ||||||
| 		}, |  | ||||||
| 		{ |  | ||||||
| 			"max duration", |  | ||||||
| 			time.Duration(math.MaxInt64), |  | ||||||
| 			time.Duration(math.MaxInt64), |  | ||||||
| 		}, |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	for _, test := range tests { |  | ||||||
| 		t.Run(test.name, func(t *testing.T) { |  | ||||||
| 			b, err := json.Marshal(Duration{test.input}) |  | ||||||
| 			require.NoError(t, err) |  | ||||||
|  |  | ||||||
| 			var d Duration |  | ||||||
| 			err = json.Unmarshal(b, &d) |  | ||||||
| 			require.NoError(t, err) |  | ||||||
|  |  | ||||||
| 			assert.Equal(t, test.expected, d.Duration, "input %v, marshalled %v, got %v", test.input, string(b), d.Duration) |  | ||||||
| 		}) |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func TestUseMmapParsingFromJSON(t *testing.T) { |  | ||||||
| 	tr := true |  | ||||||
| 	fa := false |  | ||||||
| 	tests := []struct { |  | ||||||
| 		name string |  | ||||||
| 		req  string |  | ||||||
| 		exp  *bool |  | ||||||
| 	}{ |  | ||||||
| 		{ |  | ||||||
| 			name: "Undefined", |  | ||||||
| 			req:  `{ }`, |  | ||||||
| 			exp:  nil, |  | ||||||
| 		}, |  | ||||||
| 		{ |  | ||||||
| 			name: "True", |  | ||||||
| 			req:  `{ "use_mmap": true }`, |  | ||||||
| 			exp:  &tr, |  | ||||||
| 		}, |  | ||||||
| 		{ |  | ||||||
| 			name: "False", |  | ||||||
| 			req:  `{ "use_mmap": false }`, |  | ||||||
| 			exp:  &fa, |  | ||||||
| 		}, |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	for _, test := range tests { |  | ||||||
| 		t.Run(test.name, func(t *testing.T) { |  | ||||||
| 			var oMap map[string]interface{} |  | ||||||
| 			err := json.Unmarshal([]byte(test.req), &oMap) |  | ||||||
| 			require.NoError(t, err) |  | ||||||
| 			opts := DefaultOptions() |  | ||||||
| 			err = opts.FromMap(oMap) |  | ||||||
| 			require.NoError(t, err) |  | ||||||
| 			assert.Equal(t, test.exp, opts.UseMMap) |  | ||||||
| 		}) |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func TestUseMmapFormatParams(t *testing.T) { |  | ||||||
| 	tr := true |  | ||||||
| 	fa := false |  | ||||||
| 	tests := []struct { |  | ||||||
| 		name string |  | ||||||
| 		req  map[string][]string |  | ||||||
| 		exp  *bool |  | ||||||
| 		err  error |  | ||||||
| 	}{ |  | ||||||
| 		{ |  | ||||||
| 			name: "True", |  | ||||||
| 			req: map[string][]string{ |  | ||||||
| 				"use_mmap": {"true"}, |  | ||||||
| 			}, |  | ||||||
| 			exp: &tr, |  | ||||||
| 			err: nil, |  | ||||||
| 		}, |  | ||||||
| 		{ |  | ||||||
| 			name: "False", |  | ||||||
| 			req: map[string][]string{ |  | ||||||
| 				"use_mmap": {"false"}, |  | ||||||
| 			}, |  | ||||||
| 			exp: &fa, |  | ||||||
| 			err: nil, |  | ||||||
| 		}, |  | ||||||
| 		{ |  | ||||||
| 			name: "Numeric True", |  | ||||||
| 			req: map[string][]string{ |  | ||||||
| 				"use_mmap": {"1"}, |  | ||||||
| 			}, |  | ||||||
| 			exp: &tr, |  | ||||||
| 			err: nil, |  | ||||||
| 		}, |  | ||||||
| 		{ |  | ||||||
| 			name: "Numeric False", |  | ||||||
| 			req: map[string][]string{ |  | ||||||
| 				"use_mmap": {"0"}, |  | ||||||
| 			}, |  | ||||||
| 			exp: &fa, |  | ||||||
| 			err: nil, |  | ||||||
| 		}, |  | ||||||
| 		{ |  | ||||||
| 			name: "invalid string", |  | ||||||
| 			req: map[string][]string{ |  | ||||||
| 				"use_mmap": {"foo"}, |  | ||||||
| 			}, |  | ||||||
| 			exp: nil, |  | ||||||
| 			err: errors.New("invalid bool value [foo]"), |  | ||||||
| 		}, |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	for _, test := range tests { |  | ||||||
| 		t.Run(test.name, func(t *testing.T) { |  | ||||||
| 			resp, err := FormatParams(test.req) |  | ||||||
| 			require.Equal(t, test.err, err) |  | ||||||
| 			respVal, ok := resp["use_mmap"] |  | ||||||
| 			if test.exp != nil { |  | ||||||
| 				assert.True(t, ok, "resp: %v", resp) |  | ||||||
| 				assert.Equal(t, *test.exp, *respVal.(*bool)) |  | ||||||
| 			} |  | ||||||
| 		}) |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func TestMessage_UnmarshalJSON(t *testing.T) { |  | ||||||
| 	tests := []struct { |  | ||||||
| 		input    string |  | ||||||
| 		expected string |  | ||||||
| 	}{ |  | ||||||
| 		{`{"role": "USER", "content": "Hello!"}`, "user"}, |  | ||||||
| 		{`{"role": "System", "content": "Initialization complete."}`, "system"}, |  | ||||||
| 		{`{"role": "assistant", "content": "How can I help you?"}`, "assistant"}, |  | ||||||
| 		{`{"role": "TOOl", "content": "Access granted."}`, "tool"}, |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	for _, test := range tests { |  | ||||||
| 		var msg Message |  | ||||||
| 		if err := json.Unmarshal([]byte(test.input), &msg); err != nil { |  | ||||||
| 			t.Errorf("Unexpected error: %v", err) |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		if msg.Role != test.expected { |  | ||||||
| 			t.Errorf("role not lowercased: got %v, expected %v", msg.Role, test.expected) |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
							
								
								
									
										93
									
								
								app/.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -1 +1,92 @@ | |||||||
| ollama.syso | # Logs | ||||||
|  | logs | ||||||
|  | *.log | ||||||
|  | npm-debug.log* | ||||||
|  | yarn-debug.log* | ||||||
|  | yarn-error.log* | ||||||
|  | lerna-debug.log* | ||||||
|  |  | ||||||
|  | # Diagnostic reports (https://nodejs.org/api/report.html) | ||||||
|  | report.[0-9]*.[0-9]*.[0-9]*.[0-9]*.json | ||||||
|  |  | ||||||
|  | # Runtime data | ||||||
|  | pids | ||||||
|  | *.pid | ||||||
|  | *.seed | ||||||
|  | *.pid.lock | ||||||
|  | .DS_Store | ||||||
|  |  | ||||||
|  | # Directory for instrumented libs generated by jscoverage/JSCover | ||||||
|  | lib-cov | ||||||
|  |  | ||||||
|  | # Coverage directory used by tools like istanbul | ||||||
|  | coverage | ||||||
|  | *.lcov | ||||||
|  |  | ||||||
|  | # nyc test coverage | ||||||
|  | .nyc_output | ||||||
|  |  | ||||||
|  | # node-waf configuration | ||||||
|  | .lock-wscript | ||||||
|  |  | ||||||
|  | # Compiled binary addons (https://nodejs.org/api/addons.html) | ||||||
|  | build/Release | ||||||
|  |  | ||||||
|  | # Dependency directories | ||||||
|  | node_modules/ | ||||||
|  | jspm_packages/ | ||||||
|  |  | ||||||
|  | # TypeScript v1 declaration files | ||||||
|  | typings/ | ||||||
|  |  | ||||||
|  | # TypeScript cache | ||||||
|  | *.tsbuildinfo | ||||||
|  |  | ||||||
|  | # Optional npm cache directory | ||||||
|  | .npm | ||||||
|  |  | ||||||
|  | # Optional eslint cache | ||||||
|  | .eslintcache | ||||||
|  |  | ||||||
|  | # Optional REPL history | ||||||
|  | .node_repl_history | ||||||
|  |  | ||||||
|  | # Output of 'npm pack' | ||||||
|  | *.tgz | ||||||
|  |  | ||||||
|  | # Yarn Integrity file | ||||||
|  | .yarn-integrity | ||||||
|  |  | ||||||
|  | # dotenv environment variables file | ||||||
|  | .env | ||||||
|  | .env.test | ||||||
|  |  | ||||||
|  | # parcel-bundler cache (https://parceljs.org/) | ||||||
|  | .cache | ||||||
|  |  | ||||||
|  | # next.js build output | ||||||
|  | .next | ||||||
|  |  | ||||||
|  | # nuxt.js build output | ||||||
|  | .nuxt | ||||||
|  |  | ||||||
|  | # vuepress build output | ||||||
|  | .vuepress/dist | ||||||
|  |  | ||||||
|  | # Serverless directories | ||||||
|  | .serverless/ | ||||||
|  |  | ||||||
|  | # FuseBox cache | ||||||
|  | .fusebox/ | ||||||
|  |  | ||||||
|  | # DynamoDB Local files | ||||||
|  | .dynamodb/ | ||||||
|  |  | ||||||
|  | # Webpack | ||||||
|  | .webpack/ | ||||||
|  |  | ||||||
|  | # Vite | ||||||
|  | .vite/ | ||||||
|  |  | ||||||
|  | # Electron-Forge | ||||||
|  | out/ | ||||||
|   | |||||||
| @@ -1,22 +1,21 @@ | |||||||
| # Ollama App | # Desktop | ||||||
|  |  | ||||||
| ## Linux | This app builds upon Ollama to provide a desktop experience for running models. | ||||||
|  |  | ||||||
| TODO | ## Developing | ||||||
|  |  | ||||||
| ## MacOS | First, build the `ollama` binary: | ||||||
|  |  | ||||||
| TODO |  | ||||||
|  |  | ||||||
| ## Windows |  | ||||||
|  |  | ||||||
| If you want to build the installer, youll need to install |  | ||||||
| - https://jrsoftware.org/isinfo.php |  | ||||||
|  |  | ||||||
|  |  | ||||||
| In the top directory of this repo, run the following powershell script |  | ||||||
| to build the ollama CLI, ollama app, and ollama installer. |  | ||||||
|  |  | ||||||
| ``` | ``` | ||||||
| powershell -ExecutionPolicy Bypass -File .\scripts\build_windows.ps1 | cd .. | ||||||
|  | go build . | ||||||
| ``` | ``` | ||||||
|  |  | ||||||
|  | Then run the desktop app with `npm start`: | ||||||
|  |  | ||||||
|  | ``` | ||||||
|  | cd app | ||||||
|  | npm install | ||||||
|  | npm start | ||||||
|  | ``` | ||||||
|  |  | ||||||
|   | |||||||
| Before Width: | Height: | Size: 7.3 KiB | 
| @@ -1,17 +0,0 @@ | |||||||
| package assets |  | ||||||
|  |  | ||||||
| import ( |  | ||||||
| 	"embed" |  | ||||||
| 	"io/fs" |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| //go:embed *.ico |  | ||||||
| var icons embed.FS |  | ||||||
|  |  | ||||||
| func ListIcons() ([]string, error) { |  | ||||||
| 	return fs.Glob(icons, "*") |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func GetIcon(filename string) ([]byte, error) { |  | ||||||
| 	return icons.ReadFile(filename) |  | ||||||
| } |  | ||||||
| Before Width: | Height: | Size: 402 B After Width: | Height: | Size: 402 B | 
| Before Width: | Height: | Size: 741 B After Width: | Height: | Size: 741 B | 
| Before Width: | Height: | Size: 440 B After Width: | Height: | Size: 440 B | 
| Before Width: | Height: | Size: 763 B After Width: | Height: | Size: 763 B | 
| Before Width: | Height: | Size: 447 B After Width: | Height: | Size: 447 B | 
| Before Width: | Height: | Size: 891 B After Width: | Height: | Size: 891 B | 
| Before Width: | Height: | Size: 443 B After Width: | Height: | Size: 443 B | 
| Before Width: | Height: | Size: 844 B After Width: | Height: | Size: 844 B | 
| Before Width: | Height: | Size: 76 KiB | 
| Before Width: | Height: | Size: 89 KiB | 
| Before Width: | Height: | Size: 91 KiB | 
| @@ -1,9 +0,0 @@ | |||||||
| //go:build !windows |  | ||||||
|  |  | ||||||
| package lifecycle |  | ||||||
|  |  | ||||||
| import "errors" |  | ||||||
|  |  | ||||||
| func GetStarted() error { |  | ||||||
| 	return errors.New("not implemented") |  | ||||||
| } |  | ||||||
| @@ -1,43 +0,0 @@ | |||||||
| package lifecycle |  | ||||||
|  |  | ||||||
| import ( |  | ||||||
| 	"fmt" |  | ||||||
| 	"log/slog" |  | ||||||
| 	"os" |  | ||||||
| 	"os/exec" |  | ||||||
| 	"path/filepath" |  | ||||||
| 	"syscall" |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| func GetStarted() error { |  | ||||||
| 	const CREATE_NEW_CONSOLE = 0x00000010 |  | ||||||
| 	var err error |  | ||||||
| 	bannerScript := filepath.Join(AppDir, "ollama_welcome.ps1") |  | ||||||
| 	args := []string{ |  | ||||||
| 		// TODO once we're signed, the execution policy bypass should be removed |  | ||||||
| 		"powershell", "-noexit", "-ExecutionPolicy", "Bypass", "-nologo", "-file", bannerScript, |  | ||||||
| 	} |  | ||||||
| 	args[0], err = exec.LookPath(args[0]) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	// Make sure the script actually exists |  | ||||||
| 	_, err = os.Stat(bannerScript) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return fmt.Errorf("getting started banner script error %s", err) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	slog.Info(fmt.Sprintf("opening getting started terminal with %v", args)) |  | ||||||
| 	attrs := &os.ProcAttr{ |  | ||||||
| 		Files: []*os.File{os.Stdin, os.Stdout, os.Stderr}, |  | ||||||
| 		Sys:   &syscall.SysProcAttr{CreationFlags: CREATE_NEW_CONSOLE, HideWindow: false}, |  | ||||||
| 	} |  | ||||||
| 	proc, err := os.StartProcess(args[0], args, attrs) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return fmt.Errorf("unable to start getting started shell %w", err) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	slog.Debug(fmt.Sprintf("getting started terminal PID: %d", proc.Pid)) |  | ||||||
| 	return proc.Release() |  | ||||||
| } |  | ||||||
| @@ -1,92 +0,0 @@ | |||||||
| package lifecycle |  | ||||||
|  |  | ||||||
| import ( |  | ||||||
| 	"context" |  | ||||||
| 	"fmt" |  | ||||||
| 	"log" |  | ||||||
| 	"log/slog" |  | ||||||
| 	"os" |  | ||||||
| 	"os/signal" |  | ||||||
| 	"syscall" |  | ||||||
|  |  | ||||||
| 	"github.com/ollama/ollama/app/store" |  | ||||||
| 	"github.com/ollama/ollama/app/tray" |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| func Run() { |  | ||||||
| 	InitLogging() |  | ||||||
|  |  | ||||||
| 	ctx, cancel := context.WithCancel(context.Background()) |  | ||||||
| 	var done chan int |  | ||||||
|  |  | ||||||
| 	t, err := tray.NewTray() |  | ||||||
| 	if err != nil { |  | ||||||
| 		log.Fatalf("Failed to start: %s", err) |  | ||||||
| 	} |  | ||||||
| 	callbacks := t.GetCallbacks() |  | ||||||
|  |  | ||||||
| 	signals := make(chan os.Signal, 1) |  | ||||||
| 	signal.Notify(signals, syscall.SIGINT, syscall.SIGTERM) |  | ||||||
|  |  | ||||||
| 	go func() { |  | ||||||
| 		slog.Debug("starting callback loop") |  | ||||||
| 		for { |  | ||||||
| 			select { |  | ||||||
| 			case <-callbacks.Quit: |  | ||||||
| 				slog.Debug("quit called") |  | ||||||
| 				t.Quit() |  | ||||||
| 			case <-signals: |  | ||||||
| 				slog.Debug("shutting down due to signal") |  | ||||||
| 				t.Quit() |  | ||||||
| 			case <-callbacks.Update: |  | ||||||
| 				err := DoUpgrade(cancel, done) |  | ||||||
| 				if err != nil { |  | ||||||
| 					slog.Warn(fmt.Sprintf("upgrade attempt failed: %s", err)) |  | ||||||
| 				} |  | ||||||
| 			case <-callbacks.ShowLogs: |  | ||||||
| 				ShowLogs() |  | ||||||
| 			case <-callbacks.DoFirstUse: |  | ||||||
| 				err := GetStarted() |  | ||||||
| 				if err != nil { |  | ||||||
| 					slog.Warn(fmt.Sprintf("Failed to launch getting started shell: %s", err)) |  | ||||||
| 				} |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 	}() |  | ||||||
|  |  | ||||||
| 	// Are we first use? |  | ||||||
| 	if !store.GetFirstTimeRun() { |  | ||||||
| 		slog.Debug("First time run") |  | ||||||
| 		err = t.DisplayFirstUseNotification() |  | ||||||
| 		if err != nil { |  | ||||||
| 			slog.Debug(fmt.Sprintf("XXX failed to display first use notification %v", err)) |  | ||||||
| 		} |  | ||||||
| 		store.SetFirstTimeRun(true) |  | ||||||
| 	} else { |  | ||||||
| 		slog.Debug("Not first time, skipping first run notification") |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	if IsServerRunning(ctx) { |  | ||||||
| 		slog.Info("Detected another instance of ollama running, exiting") |  | ||||||
| 		os.Exit(1) |  | ||||||
| 	} else { |  | ||||||
| 		done, err = SpawnServer(ctx, CLIName) |  | ||||||
| 		if err != nil { |  | ||||||
| 			// TODO - should we retry in a backoff loop? |  | ||||||
| 			// TODO - should we pop up a warning and maybe add a menu item to view application logs? |  | ||||||
| 			slog.Error(fmt.Sprintf("Failed to spawn ollama server %s", err)) |  | ||||||
| 			done = make(chan int, 1) |  | ||||||
| 			done <- 1 |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	StartBackgroundUpdaterChecker(ctx, t.UpdateAvailable) |  | ||||||
|  |  | ||||||
| 	t.Run() |  | ||||||
| 	cancel() |  | ||||||
| 	slog.Info("Waiting for ollama server to shutdown...") |  | ||||||
| 	if done != nil { |  | ||||||
| 		<-done |  | ||||||
| 	} |  | ||||||
| 	slog.Info("Ollama app exiting") |  | ||||||
| } |  | ||||||
| @@ -1,80 +0,0 @@ | |||||||
| package lifecycle |  | ||||||
|  |  | ||||||
| import ( |  | ||||||
| 	"fmt" |  | ||||||
| 	"log/slog" |  | ||||||
| 	"os" |  | ||||||
| 	"path/filepath" |  | ||||||
| 	"strconv" |  | ||||||
| 	"strings" |  | ||||||
|  |  | ||||||
| 	"github.com/ollama/ollama/envconfig" |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| func InitLogging() { |  | ||||||
| 	level := slog.LevelInfo |  | ||||||
|  |  | ||||||
| 	if envconfig.Debug() { |  | ||||||
| 		level = slog.LevelDebug |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	var logFile *os.File |  | ||||||
| 	var err error |  | ||||||
| 	// Detect if we're a GUI app on windows, and if not, send logs to console |  | ||||||
| 	if os.Stderr.Fd() != 0 { |  | ||||||
| 		// Console app detected |  | ||||||
| 		logFile = os.Stderr |  | ||||||
| 		// TODO - write one-line to the app.log file saying we're running in console mode to help avoid confusion |  | ||||||
| 	} else { |  | ||||||
| 		rotateLogs(AppLogFile) |  | ||||||
| 		logFile, err = os.OpenFile(AppLogFile, os.O_APPEND|os.O_WRONLY|os.O_CREATE, 0o755) |  | ||||||
| 		if err != nil { |  | ||||||
| 			slog.Error(fmt.Sprintf("failed to create server log %v", err)) |  | ||||||
| 			return |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| 	handler := slog.NewTextHandler(logFile, &slog.HandlerOptions{ |  | ||||||
| 		Level:     level, |  | ||||||
| 		AddSource: true, |  | ||||||
| 		ReplaceAttr: func(_ []string, attr slog.Attr) slog.Attr { |  | ||||||
| 			if attr.Key == slog.SourceKey { |  | ||||||
| 				source := attr.Value.Any().(*slog.Source) |  | ||||||
| 				source.File = filepath.Base(source.File) |  | ||||||
| 			} |  | ||||||
| 			return attr |  | ||||||
| 		}, |  | ||||||
| 	}) |  | ||||||
|  |  | ||||||
| 	slog.SetDefault(slog.New(handler)) |  | ||||||
|  |  | ||||||
| 	slog.Info("ollama app started") |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func rotateLogs(logFile string) { |  | ||||||
| 	if _, err := os.Stat(logFile); os.IsNotExist(err) { |  | ||||||
| 		return |  | ||||||
| 	} |  | ||||||
| 	index := strings.LastIndex(logFile, ".") |  | ||||||
| 	pre := logFile[:index] |  | ||||||
| 	post := "." + logFile[index+1:] |  | ||||||
| 	for i := LogRotationCount; i > 0; i-- { |  | ||||||
| 		older := pre + "-" + strconv.Itoa(i) + post |  | ||||||
| 		newer := pre + "-" + strconv.Itoa(i-1) + post |  | ||||||
| 		if i == 1 { |  | ||||||
| 			newer = pre + post |  | ||||||
| 		} |  | ||||||
| 		if _, err := os.Stat(newer); err == nil { |  | ||||||
| 			if _, err := os.Stat(older); err == nil { |  | ||||||
| 				err := os.Remove(older) |  | ||||||
| 				if err != nil { |  | ||||||
| 					slog.Warn("Failed to remove older log", "older", older, "error", err) |  | ||||||
| 					continue |  | ||||||
| 				} |  | ||||||
| 			} |  | ||||||
| 			err := os.Rename(newer, older) |  | ||||||
| 			if err != nil { |  | ||||||
| 				slog.Warn("Failed to rotate log", "older", older, "newer", newer, "error", err) |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
| @@ -1,9 +0,0 @@ | |||||||
| //go:build !windows |  | ||||||
|  |  | ||||||
| package lifecycle |  | ||||||
|  |  | ||||||
| import "log/slog" |  | ||||||
|  |  | ||||||
| func ShowLogs() { |  | ||||||
| 	slog.Warn("not implemented") |  | ||||||
| } |  | ||||||
| @@ -1,44 +0,0 @@ | |||||||
| package lifecycle |  | ||||||
|  |  | ||||||
| import ( |  | ||||||
| 	"os" |  | ||||||
| 	"path/filepath" |  | ||||||
| 	"strconv" |  | ||||||
| 	"testing" |  | ||||||
|  |  | ||||||
| 	"github.com/stretchr/testify/assert" |  | ||||||
| 	"github.com/stretchr/testify/require" |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| func TestRotateLogs(t *testing.T) { |  | ||||||
| 	logDir := t.TempDir() |  | ||||||
| 	logFile := filepath.Join(logDir, "testlog.log") |  | ||||||
|  |  | ||||||
| 	// No log exists |  | ||||||
| 	rotateLogs(logFile) |  | ||||||
|  |  | ||||||
| 	require.NoError(t, os.WriteFile(logFile, []byte("1"), 0o644)) |  | ||||||
| 	assert.FileExists(t, logFile) |  | ||||||
| 	// First rotation |  | ||||||
| 	rotateLogs(logFile) |  | ||||||
| 	assert.FileExists(t, filepath.Join(logDir, "testlog-1.log")) |  | ||||||
| 	assert.NoFileExists(t, filepath.Join(logDir, "testlog-2.log")) |  | ||||||
| 	assert.NoFileExists(t, logFile) |  | ||||||
|  |  | ||||||
| 	// Should be a no-op without a new log |  | ||||||
| 	rotateLogs(logFile) |  | ||||||
| 	assert.FileExists(t, filepath.Join(logDir, "testlog-1.log")) |  | ||||||
| 	assert.NoFileExists(t, filepath.Join(logDir, "testlog-2.log")) |  | ||||||
| 	assert.NoFileExists(t, logFile) |  | ||||||
|  |  | ||||||
| 	for i := 2; i <= LogRotationCount+1; i++ { |  | ||||||
| 		require.NoError(t, os.WriteFile(logFile, []byte(strconv.Itoa(i)), 0o644)) |  | ||||||
| 		assert.FileExists(t, logFile) |  | ||||||
| 		rotateLogs(logFile) |  | ||||||
| 		assert.NoFileExists(t, logFile) |  | ||||||
| 		for j := 1; j < i; j++ { |  | ||||||
| 			assert.FileExists(t, filepath.Join(logDir, "testlog-"+strconv.Itoa(j)+".log")) |  | ||||||
| 		} |  | ||||||
| 		assert.NoFileExists(t, filepath.Join(logDir, "testlog-"+strconv.Itoa(i+1)+".log")) |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
| @@ -1,19 +0,0 @@ | |||||||
| package lifecycle |  | ||||||
|  |  | ||||||
| import ( |  | ||||||
| 	"fmt" |  | ||||||
| 	"log/slog" |  | ||||||
| 	"os/exec" |  | ||||||
| 	"syscall" |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| func ShowLogs() { |  | ||||||
| 	cmd_path := "c:\\Windows\\system32\\cmd.exe" |  | ||||||
| 	slog.Debug(fmt.Sprintf("viewing logs with start %s", AppDataDir)) |  | ||||||
| 	cmd := exec.Command(cmd_path, "/c", "start", AppDataDir) |  | ||||||
| 	cmd.SysProcAttr = &syscall.SysProcAttr{HideWindow: false, CreationFlags: 0x08000000} |  | ||||||
| 	err := cmd.Start() |  | ||||||
| 	if err != nil { |  | ||||||
| 		slog.Error(fmt.Sprintf("Failed to open log dir: %s", err)) |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
| @@ -1,79 +0,0 @@ | |||||||
| package lifecycle |  | ||||||
|  |  | ||||||
| import ( |  | ||||||
| 	"errors" |  | ||||||
| 	"fmt" |  | ||||||
| 	"log/slog" |  | ||||||
| 	"os" |  | ||||||
| 	"path/filepath" |  | ||||||
| 	"runtime" |  | ||||||
| 	"strings" |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| var ( |  | ||||||
| 	AppName    = "ollama app" |  | ||||||
| 	CLIName    = "ollama" |  | ||||||
| 	AppDir     = "/opt/Ollama" |  | ||||||
| 	AppDataDir = "/opt/Ollama" |  | ||||||
| 	// TODO - should there be a distinct log dir? |  | ||||||
| 	UpdateStageDir   = "/tmp" |  | ||||||
| 	AppLogFile       = "/tmp/ollama_app.log" |  | ||||||
| 	ServerLogFile    = "/tmp/ollama.log" |  | ||||||
| 	UpgradeLogFile   = "/tmp/ollama_update.log" |  | ||||||
| 	Installer        = "OllamaSetup.exe" |  | ||||||
| 	LogRotationCount = 5 |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| func init() { |  | ||||||
| 	if runtime.GOOS == "windows" { |  | ||||||
| 		AppName += ".exe" |  | ||||||
| 		CLIName += ".exe" |  | ||||||
| 		// Logs, configs, downloads go to LOCALAPPDATA |  | ||||||
| 		localAppData := os.Getenv("LOCALAPPDATA") |  | ||||||
| 		AppDataDir = filepath.Join(localAppData, "Ollama") |  | ||||||
| 		UpdateStageDir = filepath.Join(AppDataDir, "updates") |  | ||||||
| 		AppLogFile = filepath.Join(AppDataDir, "app.log") |  | ||||||
| 		ServerLogFile = filepath.Join(AppDataDir, "server.log") |  | ||||||
| 		UpgradeLogFile = filepath.Join(AppDataDir, "upgrade.log") |  | ||||||
|  |  | ||||||
| 		// Executables are stored in APPDATA |  | ||||||
| 		AppDir = filepath.Join(localAppData, "Programs", "Ollama") |  | ||||||
|  |  | ||||||
| 		// Make sure we have PATH set correctly for any spawned children |  | ||||||
| 		paths := strings.Split(os.Getenv("PATH"), ";") |  | ||||||
| 		// Start with whatever we find in the PATH/LD_LIBRARY_PATH |  | ||||||
| 		found := false |  | ||||||
| 		for _, path := range paths { |  | ||||||
| 			d, err := filepath.Abs(path) |  | ||||||
| 			if err != nil { |  | ||||||
| 				continue |  | ||||||
| 			} |  | ||||||
| 			if strings.EqualFold(AppDir, d) { |  | ||||||
| 				found = true |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 		if !found { |  | ||||||
| 			paths = append(paths, AppDir) |  | ||||||
|  |  | ||||||
| 			pathVal := strings.Join(paths, ";") |  | ||||||
| 			slog.Debug("setting PATH=" + pathVal) |  | ||||||
| 			err := os.Setenv("PATH", pathVal) |  | ||||||
| 			if err != nil { |  | ||||||
| 				slog.Error(fmt.Sprintf("failed to update PATH: %s", err)) |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		// Make sure our logging dir exists |  | ||||||
| 		_, err := os.Stat(AppDataDir) |  | ||||||
| 		if errors.Is(err, os.ErrNotExist) { |  | ||||||
| 			if err := os.MkdirAll(AppDataDir, 0o755); err != nil { |  | ||||||
| 				slog.Error(fmt.Sprintf("create ollama dir %s: %v", AppDataDir, err)) |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 	} else if runtime.GOOS == "darwin" { |  | ||||||
| 		// TODO |  | ||||||
| 		AppName += ".app" |  | ||||||
| 		// } else if runtime.GOOS == "linux" { |  | ||||||
| 		// TODO |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
| @@ -1,180 +0,0 @@ | |||||||
| package lifecycle |  | ||||||
|  |  | ||||||
| import ( |  | ||||||
| 	"context" |  | ||||||
| 	"errors" |  | ||||||
| 	"fmt" |  | ||||||
| 	"io" |  | ||||||
| 	"log/slog" |  | ||||||
| 	"os" |  | ||||||
| 	"os/exec" |  | ||||||
| 	"path/filepath" |  | ||||||
| 	"time" |  | ||||||
|  |  | ||||||
| 	"github.com/ollama/ollama/api" |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| func getCLIFullPath(command string) string { |  | ||||||
| 	var cmdPath string |  | ||||||
| 	appExe, err := os.Executable() |  | ||||||
| 	if err == nil { |  | ||||||
| 		cmdPath = filepath.Join(filepath.Dir(appExe), command) |  | ||||||
| 		_, err := os.Stat(cmdPath) |  | ||||||
| 		if err == nil { |  | ||||||
| 			return cmdPath |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| 	cmdPath, err = exec.LookPath(command) |  | ||||||
| 	if err == nil { |  | ||||||
| 		_, err := os.Stat(cmdPath) |  | ||||||
| 		if err == nil { |  | ||||||
| 			return cmdPath |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| 	pwd, err := os.Getwd() |  | ||||||
| 	if err == nil { |  | ||||||
| 		cmdPath = filepath.Join(pwd, command) |  | ||||||
| 		_, err = os.Stat(cmdPath) |  | ||||||
| 		if err == nil { |  | ||||||
| 			return cmdPath |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	return command |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func start(ctx context.Context, command string) (*exec.Cmd, error) { |  | ||||||
| 	cmd := getCmd(ctx, getCLIFullPath(command)) |  | ||||||
| 	stdout, err := cmd.StdoutPipe() |  | ||||||
| 	if err != nil { |  | ||||||
| 		return nil, fmt.Errorf("failed to spawn server stdout pipe: %w", err) |  | ||||||
| 	} |  | ||||||
| 	stderr, err := cmd.StderrPipe() |  | ||||||
| 	if err != nil { |  | ||||||
| 		return nil, fmt.Errorf("failed to spawn server stderr pipe: %w", err) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	rotateLogs(ServerLogFile) |  | ||||||
| 	logFile, err := os.OpenFile(ServerLogFile, os.O_APPEND|os.O_WRONLY|os.O_CREATE, 0o755) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return nil, fmt.Errorf("failed to create server log: %w", err) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	logDir := filepath.Dir(ServerLogFile) |  | ||||||
| 	_, err = os.Stat(logDir) |  | ||||||
| 	if err != nil { |  | ||||||
| 		if !errors.Is(err, os.ErrNotExist) { |  | ||||||
| 			return nil, fmt.Errorf("stat ollama server log dir %s: %v", logDir, err) |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		if err := os.MkdirAll(logDir, 0o755); err != nil { |  | ||||||
| 			return nil, fmt.Errorf("create ollama server log dir %s: %v", logDir, err) |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	go func() { |  | ||||||
| 		defer logFile.Close() |  | ||||||
| 		io.Copy(logFile, stdout) //nolint:errcheck |  | ||||||
| 	}() |  | ||||||
| 	go func() { |  | ||||||
| 		defer logFile.Close() |  | ||||||
| 		io.Copy(logFile, stderr) //nolint:errcheck |  | ||||||
| 	}() |  | ||||||
|  |  | ||||||
| 	// Re-wire context done behavior to attempt a graceful shutdown of the server |  | ||||||
| 	cmd.Cancel = func() error { |  | ||||||
| 		if cmd.Process != nil { |  | ||||||
| 			err := terminate(cmd) |  | ||||||
| 			if err != nil { |  | ||||||
| 				slog.Warn("error trying to gracefully terminate server", "err", err) |  | ||||||
| 				return cmd.Process.Kill() |  | ||||||
| 			} |  | ||||||
|  |  | ||||||
| 			tick := time.NewTicker(10 * time.Millisecond) |  | ||||||
| 			defer tick.Stop() |  | ||||||
|  |  | ||||||
| 			for { |  | ||||||
| 				select { |  | ||||||
| 				case <-tick.C: |  | ||||||
| 					exited, err := isProcessExited(cmd.Process.Pid) |  | ||||||
| 					if err != nil { |  | ||||||
| 						return err |  | ||||||
| 					} |  | ||||||
|  |  | ||||||
| 					if exited { |  | ||||||
| 						return nil |  | ||||||
| 					} |  | ||||||
| 				case <-time.After(5 * time.Second): |  | ||||||
| 					slog.Warn("graceful server shutdown timeout, killing", "pid", cmd.Process.Pid) |  | ||||||
| 					return cmd.Process.Kill() |  | ||||||
| 				} |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 		return nil |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	// run the command and wait for it to finish |  | ||||||
| 	if err := cmd.Start(); err != nil { |  | ||||||
| 		return nil, fmt.Errorf("failed to start server %w", err) |  | ||||||
| 	} |  | ||||||
| 	if cmd.Process != nil { |  | ||||||
| 		slog.Info(fmt.Sprintf("started ollama server with pid %d", cmd.Process.Pid)) |  | ||||||
| 	} |  | ||||||
| 	slog.Info(fmt.Sprintf("ollama server logs %s", ServerLogFile)) |  | ||||||
|  |  | ||||||
| 	return cmd, nil |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func SpawnServer(ctx context.Context, command string) (chan int, error) { |  | ||||||
| 	done := make(chan int) |  | ||||||
|  |  | ||||||
| 	go func() { |  | ||||||
| 		// Keep the server running unless we're shuttind down the app |  | ||||||
| 		crashCount := 0 |  | ||||||
| 		for { |  | ||||||
| 			slog.Info("starting server...") |  | ||||||
| 			cmd, err := start(ctx, command) |  | ||||||
| 			if err != nil { |  | ||||||
| 				crashCount++ |  | ||||||
| 				slog.Error(fmt.Sprintf("failed to start server %s", err)) |  | ||||||
| 				time.Sleep(500 * time.Millisecond * time.Duration(crashCount)) |  | ||||||
| 				continue |  | ||||||
| 			} |  | ||||||
|  |  | ||||||
| 			cmd.Wait() //nolint:errcheck |  | ||||||
| 			var code int |  | ||||||
| 			if cmd.ProcessState != nil { |  | ||||||
| 				code = cmd.ProcessState.ExitCode() |  | ||||||
| 			} |  | ||||||
|  |  | ||||||
| 			select { |  | ||||||
| 			case <-ctx.Done(): |  | ||||||
| 				slog.Info(fmt.Sprintf("server shutdown with exit code %d", code)) |  | ||||||
| 				done <- code |  | ||||||
| 				return |  | ||||||
| 			default: |  | ||||||
| 				crashCount++ |  | ||||||
| 				slog.Warn(fmt.Sprintf("server crash %d - exit code %d - respawning", crashCount, code)) |  | ||||||
| 				time.Sleep(500 * time.Millisecond * time.Duration(crashCount)) |  | ||||||
| 				break |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 	}() |  | ||||||
|  |  | ||||||
| 	return done, nil |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func IsServerRunning(ctx context.Context) bool { |  | ||||||
| 	client, err := api.ClientFromEnvironment() |  | ||||||
| 	if err != nil { |  | ||||||
| 		slog.Info("unable to connect to server") |  | ||||||
| 		return false |  | ||||||
| 	} |  | ||||||
| 	err = client.Heartbeat(ctx) |  | ||||||
| 	if err != nil { |  | ||||||
| 		slog.Debug(fmt.Sprintf("heartbeat from server: %s", err)) |  | ||||||
| 		slog.Info("unable to connect to server") |  | ||||||
| 		return false |  | ||||||
| 	} |  | ||||||
| 	return true |  | ||||||
| } |  | ||||||
| @@ -1,38 +0,0 @@ | |||||||
| //go:build !windows |  | ||||||
|  |  | ||||||
| package lifecycle |  | ||||||
|  |  | ||||||
| import ( |  | ||||||
| 	"context" |  | ||||||
| 	"errors" |  | ||||||
| 	"fmt" |  | ||||||
| 	"os" |  | ||||||
| 	"os/exec" |  | ||||||
| 	"syscall" |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| func getCmd(ctx context.Context, cmd string) *exec.Cmd { |  | ||||||
| 	return exec.CommandContext(ctx, cmd, "serve") |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func terminate(cmd *exec.Cmd) error { |  | ||||||
| 	return cmd.Process.Signal(os.Interrupt) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func isProcessExited(pid int) (bool, error) { |  | ||||||
| 	proc, err := os.FindProcess(pid) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return false, fmt.Errorf("failed to find process: %v", err) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	err = proc.Signal(syscall.Signal(0)) |  | ||||||
| 	if err != nil { |  | ||||||
| 		if errors.Is(err, os.ErrProcessDone) || errors.Is(err, syscall.ESRCH) { |  | ||||||
| 			return true, nil |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		return false, fmt.Errorf("error signaling process: %v", err) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	return false, nil |  | ||||||
| } |  | ||||||
| @@ -1,91 +0,0 @@ | |||||||
| package lifecycle |  | ||||||
|  |  | ||||||
| import ( |  | ||||||
| 	"context" |  | ||||||
| 	"fmt" |  | ||||||
| 	"os/exec" |  | ||||||
| 	"syscall" |  | ||||||
|  |  | ||||||
| 	"golang.org/x/sys/windows" |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| func getCmd(ctx context.Context, exePath string) *exec.Cmd { |  | ||||||
| 	cmd := exec.CommandContext(ctx, exePath, "serve") |  | ||||||
| 	cmd.SysProcAttr = &syscall.SysProcAttr{ |  | ||||||
| 		HideWindow:    true, |  | ||||||
| 		CreationFlags: windows.CREATE_NEW_PROCESS_GROUP, |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	return cmd |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func terminate(cmd *exec.Cmd) error { |  | ||||||
| 	dll, err := windows.LoadDLL("kernel32.dll") |  | ||||||
| 	if err != nil { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
| 	//nolint:errcheck |  | ||||||
| 	defer dll.Release() |  | ||||||
|  |  | ||||||
| 	pid := cmd.Process.Pid |  | ||||||
|  |  | ||||||
| 	f, err := dll.FindProc("AttachConsole") |  | ||||||
| 	if err != nil { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	r1, _, err := f.Call(uintptr(pid)) |  | ||||||
| 	if r1 == 0 && err != syscall.ERROR_ACCESS_DENIED { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	f, err = dll.FindProc("SetConsoleCtrlHandler") |  | ||||||
| 	if err != nil { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	r1, _, err = f.Call(0, 1) |  | ||||||
| 	if r1 == 0 { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	f, err = dll.FindProc("GenerateConsoleCtrlEvent") |  | ||||||
| 	if err != nil { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	r1, _, err = f.Call(windows.CTRL_BREAK_EVENT, uintptr(pid)) |  | ||||||
| 	if r1 == 0 { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	r1, _, err = f.Call(windows.CTRL_C_EVENT, uintptr(pid)) |  | ||||||
| 	if r1 == 0 { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	return nil |  | ||||||
| } |  | ||||||
|  |  | ||||||
| const STILL_ACTIVE = 259 |  | ||||||
|  |  | ||||||
| func isProcessExited(pid int) (bool, error) { |  | ||||||
| 	hProcess, err := windows.OpenProcess(windows.PROCESS_QUERY_INFORMATION, false, uint32(pid)) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return false, fmt.Errorf("failed to open process: %v", err) |  | ||||||
| 	} |  | ||||||
| 	//nolint:errcheck |  | ||||||
| 	defer windows.CloseHandle(hProcess) |  | ||||||
|  |  | ||||||
| 	var exitCode uint32 |  | ||||||
| 	err = windows.GetExitCodeProcess(hProcess, &exitCode) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return false, fmt.Errorf("failed to get exit code: %v", err) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	if exitCode == STILL_ACTIVE { |  | ||||||
| 		return false, nil |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	return true, nil |  | ||||||
| } |  | ||||||
| @@ -1,229 +0,0 @@ | |||||||
| package lifecycle |  | ||||||
|  |  | ||||||
| import ( |  | ||||||
| 	"context" |  | ||||||
| 	"crypto/rand" |  | ||||||
| 	"encoding/json" |  | ||||||
| 	"errors" |  | ||||||
| 	"fmt" |  | ||||||
| 	"io" |  | ||||||
| 	"log/slog" |  | ||||||
| 	"mime" |  | ||||||
| 	"net/http" |  | ||||||
| 	"net/url" |  | ||||||
| 	"os" |  | ||||||
| 	"path" |  | ||||||
| 	"path/filepath" |  | ||||||
| 	"runtime" |  | ||||||
| 	"strconv" |  | ||||||
| 	"strings" |  | ||||||
| 	"time" |  | ||||||
|  |  | ||||||
| 	"github.com/ollama/ollama/auth" |  | ||||||
| 	"github.com/ollama/ollama/version" |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| var ( |  | ||||||
| 	UpdateCheckURLBase  = "https://ollama.com/api/update" |  | ||||||
| 	UpdateDownloaded    = false |  | ||||||
| 	UpdateCheckInterval = 60 * 60 * time.Second |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| // TODO - maybe move up to the API package? |  | ||||||
| type UpdateResponse struct { |  | ||||||
| 	UpdateURL     string `json:"url"` |  | ||||||
| 	UpdateVersion string `json:"version"` |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func IsNewReleaseAvailable(ctx context.Context) (bool, UpdateResponse) { |  | ||||||
| 	var updateResp UpdateResponse |  | ||||||
|  |  | ||||||
| 	requestURL, err := url.Parse(UpdateCheckURLBase) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return false, updateResp |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	query := requestURL.Query() |  | ||||||
| 	query.Add("os", runtime.GOOS) |  | ||||||
| 	query.Add("arch", runtime.GOARCH) |  | ||||||
| 	query.Add("version", version.Version) |  | ||||||
| 	query.Add("ts", strconv.FormatInt(time.Now().Unix(), 10)) |  | ||||||
|  |  | ||||||
| 	nonce, err := auth.NewNonce(rand.Reader, 16) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return false, updateResp |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	query.Add("nonce", nonce) |  | ||||||
| 	requestURL.RawQuery = query.Encode() |  | ||||||
|  |  | ||||||
| 	data := []byte(fmt.Sprintf("%s,%s", http.MethodGet, requestURL.RequestURI())) |  | ||||||
| 	signature, err := auth.Sign(ctx, data) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return false, updateResp |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	req, err := http.NewRequestWithContext(ctx, http.MethodGet, requestURL.String(), nil) |  | ||||||
| 	if err != nil { |  | ||||||
| 		slog.Warn(fmt.Sprintf("failed to check for update: %s", err)) |  | ||||||
| 		return false, updateResp |  | ||||||
| 	} |  | ||||||
| 	req.Header.Set("Authorization", signature) |  | ||||||
| 	req.Header.Set("User-Agent", fmt.Sprintf("ollama/%s (%s %s) Go/%s", version.Version, runtime.GOARCH, runtime.GOOS, runtime.Version())) |  | ||||||
|  |  | ||||||
| 	slog.Debug("checking for available update", "requestURL", requestURL) |  | ||||||
| 	resp, err := http.DefaultClient.Do(req) |  | ||||||
| 	if err != nil { |  | ||||||
| 		slog.Warn(fmt.Sprintf("failed to check for update: %s", err)) |  | ||||||
| 		return false, updateResp |  | ||||||
| 	} |  | ||||||
| 	defer resp.Body.Close() |  | ||||||
|  |  | ||||||
| 	if resp.StatusCode == http.StatusNoContent { |  | ||||||
| 		slog.Debug("check update response 204 (current version is up to date)") |  | ||||||
| 		return false, updateResp |  | ||||||
| 	} |  | ||||||
| 	body, err := io.ReadAll(resp.Body) |  | ||||||
| 	if err != nil { |  | ||||||
| 		slog.Warn(fmt.Sprintf("failed to read body response: %s", err)) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	if resp.StatusCode != http.StatusOK { |  | ||||||
| 		slog.Info(fmt.Sprintf("check update error %d - %.96s", resp.StatusCode, string(body))) |  | ||||||
| 		return false, updateResp |  | ||||||
| 	} |  | ||||||
| 	err = json.Unmarshal(body, &updateResp) |  | ||||||
| 	if err != nil { |  | ||||||
| 		slog.Warn(fmt.Sprintf("malformed response checking for update: %s", err)) |  | ||||||
| 		return false, updateResp |  | ||||||
| 	} |  | ||||||
| 	// Extract the version string from the URL in the github release artifact path |  | ||||||
| 	updateResp.UpdateVersion = path.Base(path.Dir(updateResp.UpdateURL)) |  | ||||||
|  |  | ||||||
| 	slog.Info("New update available at " + updateResp.UpdateURL) |  | ||||||
| 	return true, updateResp |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func DownloadNewRelease(ctx context.Context, updateResp UpdateResponse) error { |  | ||||||
| 	// Do a head first to check etag info |  | ||||||
| 	req, err := http.NewRequestWithContext(ctx, http.MethodHead, updateResp.UpdateURL, nil) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	resp, err := http.DefaultClient.Do(req) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return fmt.Errorf("error checking update: %w", err) |  | ||||||
| 	} |  | ||||||
| 	if resp.StatusCode != http.StatusOK { |  | ||||||
| 		return fmt.Errorf("unexpected status attempting to download update %d", resp.StatusCode) |  | ||||||
| 	} |  | ||||||
| 	resp.Body.Close() |  | ||||||
| 	etag := strings.Trim(resp.Header.Get("etag"), "\"") |  | ||||||
| 	if etag == "" { |  | ||||||
| 		slog.Debug("no etag detected, falling back to filename based dedup") |  | ||||||
| 		etag = "_" |  | ||||||
| 	} |  | ||||||
| 	filename := Installer |  | ||||||
| 	_, params, err := mime.ParseMediaType(resp.Header.Get("content-disposition")) |  | ||||||
| 	if err == nil { |  | ||||||
| 		filename = params["filename"] |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	stageFilename := filepath.Join(UpdateStageDir, etag, filename) |  | ||||||
|  |  | ||||||
| 	// Check to see if we already have it downloaded |  | ||||||
| 	_, err = os.Stat(stageFilename) |  | ||||||
| 	if err == nil { |  | ||||||
| 		slog.Info("update already downloaded") |  | ||||||
| 		return nil |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	cleanupOldDownloads() |  | ||||||
|  |  | ||||||
| 	req.Method = http.MethodGet |  | ||||||
| 	resp, err = http.DefaultClient.Do(req) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return fmt.Errorf("error checking update: %w", err) |  | ||||||
| 	} |  | ||||||
| 	defer resp.Body.Close() |  | ||||||
| 	etag = strings.Trim(resp.Header.Get("etag"), "\"") |  | ||||||
| 	if etag == "" { |  | ||||||
| 		slog.Debug("no etag detected, falling back to filename based dedup") // TODO probably can get rid of this redundant log |  | ||||||
| 		etag = "_" |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	stageFilename = filepath.Join(UpdateStageDir, etag, filename) |  | ||||||
|  |  | ||||||
| 	_, err = os.Stat(filepath.Dir(stageFilename)) |  | ||||||
| 	if errors.Is(err, os.ErrNotExist) { |  | ||||||
| 		if err := os.MkdirAll(filepath.Dir(stageFilename), 0o755); err != nil { |  | ||||||
| 			return fmt.Errorf("create ollama dir %s: %v", filepath.Dir(stageFilename), err) |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	payload, err := io.ReadAll(resp.Body) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return fmt.Errorf("failed to read body response: %w", err) |  | ||||||
| 	} |  | ||||||
| 	fp, err := os.OpenFile(stageFilename, os.O_WRONLY|os.O_CREATE|os.O_TRUNC, 0o755) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return fmt.Errorf("write payload %s: %w", stageFilename, err) |  | ||||||
| 	} |  | ||||||
| 	defer fp.Close() |  | ||||||
| 	if n, err := fp.Write(payload); err != nil || n != len(payload) { |  | ||||||
| 		return fmt.Errorf("write payload %s: %d vs %d -- %w", stageFilename, n, len(payload), err) |  | ||||||
| 	} |  | ||||||
| 	slog.Info("new update downloaded " + stageFilename) |  | ||||||
|  |  | ||||||
| 	UpdateDownloaded = true |  | ||||||
| 	return nil |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func cleanupOldDownloads() { |  | ||||||
| 	files, err := os.ReadDir(UpdateStageDir) |  | ||||||
| 	if err != nil && errors.Is(err, os.ErrNotExist) { |  | ||||||
| 		// Expected behavior on first run |  | ||||||
| 		return |  | ||||||
| 	} else if err != nil { |  | ||||||
| 		slog.Warn(fmt.Sprintf("failed to list stage dir: %s", err)) |  | ||||||
| 		return |  | ||||||
| 	} |  | ||||||
| 	for _, file := range files { |  | ||||||
| 		fullname := filepath.Join(UpdateStageDir, file.Name()) |  | ||||||
| 		slog.Debug("cleaning up old download: " + fullname) |  | ||||||
| 		err = os.RemoveAll(fullname) |  | ||||||
| 		if err != nil { |  | ||||||
| 			slog.Warn(fmt.Sprintf("failed to cleanup stale update download %s", err)) |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func StartBackgroundUpdaterChecker(ctx context.Context, cb func(string) error) { |  | ||||||
| 	go func() { |  | ||||||
| 		// Don't blast an update message immediately after startup |  | ||||||
| 		// time.Sleep(30 * time.Second) |  | ||||||
| 		time.Sleep(3 * time.Second) |  | ||||||
|  |  | ||||||
| 		for { |  | ||||||
| 			available, resp := IsNewReleaseAvailable(ctx) |  | ||||||
| 			if available { |  | ||||||
| 				err := DownloadNewRelease(ctx, resp) |  | ||||||
| 				if err != nil { |  | ||||||
| 					slog.Error(fmt.Sprintf("failed to download new release: %s", err)) |  | ||||||
| 				} |  | ||||||
| 				err = cb(resp.UpdateVersion) |  | ||||||
| 				if err != nil { |  | ||||||
| 					slog.Warn(fmt.Sprintf("failed to register update available with tray: %s", err)) |  | ||||||
| 				} |  | ||||||
| 			} |  | ||||||
| 			select { |  | ||||||
| 			case <-ctx.Done(): |  | ||||||
| 				slog.Debug("stopping background update checker") |  | ||||||
| 				return |  | ||||||
| 			default: |  | ||||||
| 				time.Sleep(UpdateCheckInterval) |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 	}() |  | ||||||
| } |  | ||||||
| @@ -1,12 +0,0 @@ | |||||||
| //go:build !windows |  | ||||||
|  |  | ||||||
| package lifecycle |  | ||||||
|  |  | ||||||
| import ( |  | ||||||
| 	"context" |  | ||||||
| 	"errors" |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| func DoUpgrade(cancel context.CancelFunc, done chan int) error { |  | ||||||
| 	return errors.New("not implemented") |  | ||||||
| } |  | ||||||
| @@ -1,78 +0,0 @@ | |||||||
| package lifecycle |  | ||||||
|  |  | ||||||
| import ( |  | ||||||
| 	"context" |  | ||||||
| 	"errors" |  | ||||||
| 	"fmt" |  | ||||||
| 	"log/slog" |  | ||||||
| 	"os" |  | ||||||
| 	"os/exec" |  | ||||||
| 	"path/filepath" |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| func DoUpgrade(cancel context.CancelFunc, done chan int) error { |  | ||||||
| 	files, err := filepath.Glob(filepath.Join(UpdateStageDir, "*", "*.exe")) // TODO generalize for multiplatform |  | ||||||
| 	if err != nil { |  | ||||||
| 		return fmt.Errorf("failed to lookup downloads: %s", err) |  | ||||||
| 	} |  | ||||||
| 	if len(files) == 0 { |  | ||||||
| 		return errors.New("no update downloads found") |  | ||||||
| 	} else if len(files) > 1 { |  | ||||||
| 		// Shouldn't happen |  | ||||||
| 		slog.Warn(fmt.Sprintf("multiple downloads found, using first one %v", files)) |  | ||||||
| 	} |  | ||||||
| 	installerExe := files[0] |  | ||||||
|  |  | ||||||
| 	slog.Info("starting upgrade with " + installerExe) |  | ||||||
| 	slog.Info("upgrade log file " + UpgradeLogFile) |  | ||||||
|  |  | ||||||
| 	// When running in debug mode, we'll be "verbose" and let the installer pop up and prompt |  | ||||||
| 	installArgs := []string{ |  | ||||||
| 		"/CLOSEAPPLICATIONS",                    // Quit the tray app if it's still running |  | ||||||
| 		"/LOG=" + filepath.Base(UpgradeLogFile), // Only relative seems reliable, so set pwd |  | ||||||
| 		"/FORCECLOSEAPPLICATIONS",               // Force close the tray app - might be needed |  | ||||||
| 	} |  | ||||||
| 	// make the upgrade as quiet as possible (no GUI, no prompts) |  | ||||||
| 	installArgs = append(installArgs, |  | ||||||
| 		"/SP", // Skip the "This will install... Do you wish to continue" prompt |  | ||||||
| 		"/SUPPRESSMSGBOXES", |  | ||||||
| 		"/SILENT", |  | ||||||
| 		"/VERYSILENT", |  | ||||||
| 	) |  | ||||||
|  |  | ||||||
| 	// Safeguard in case we have requests in flight that need to drain... |  | ||||||
| 	slog.Info("Waiting for server to shutdown") |  | ||||||
| 	cancel() |  | ||||||
| 	if done != nil { |  | ||||||
| 		<-done |  | ||||||
| 	} else { |  | ||||||
| 		// Shouldn't happen |  | ||||||
| 		slog.Warn("done chan was nil, not actually waiting") |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	slog.Debug(fmt.Sprintf("starting installer: %s %v", installerExe, installArgs)) |  | ||||||
| 	os.Chdir(filepath.Dir(UpgradeLogFile)) //nolint:errcheck |  | ||||||
| 	cmd := exec.Command(installerExe, installArgs...) |  | ||||||
|  |  | ||||||
| 	if err := cmd.Start(); err != nil { |  | ||||||
| 		return fmt.Errorf("unable to start ollama app %w", err) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	if cmd.Process != nil { |  | ||||||
| 		err = cmd.Process.Release() |  | ||||||
| 		if err != nil { |  | ||||||
| 			slog.Error(fmt.Sprintf("failed to release server process: %s", err)) |  | ||||||
| 		} |  | ||||||
| 	} else { |  | ||||||
| 		// TODO - some details about why it didn't start, or is this a pedantic error case? |  | ||||||
| 		return errors.New("installer process did not start") |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	// TODO should we linger for a moment and check to make sure it's actually running by checking the pid? |  | ||||||
|  |  | ||||||
| 	slog.Info("Installer started in background, exiting") |  | ||||||
|  |  | ||||||
| 	os.Exit(0) |  | ||||||
| 	// Not reached |  | ||||||
| 	return nil |  | ||||||
| } |  | ||||||
							
								
								
									
										12
									
								
								app/main.go
									
									
									
									
									
								
							
							
						
						| @@ -1,12 +0,0 @@ | |||||||
| package main |  | ||||||
|  |  | ||||||
| // Compile with the following to get rid of the cmd pop up on windows |  | ||||||
| // go build -ldflags="-H windowsgui" . |  | ||||||
|  |  | ||||||
| import ( |  | ||||||
| 	"github.com/ollama/ollama/app/lifecycle" |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| func main() { |  | ||||||
| 	lifecycle.Run() |  | ||||||
| } |  | ||||||
							
								
								
									
										205
									
								
								app/ollama.iss
									
									
									
									
									
								
							
							
						
						| @@ -1,205 +0,0 @@ | |||||||
| ; Inno Setup Installer for Ollama |  | ||||||
| ; |  | ||||||
| ; To build the installer use the build script invoked from the top of the source tree |  | ||||||
| ;  |  | ||||||
| ; powershell -ExecutionPolicy Bypass -File .\scripts\build_windows.ps |  | ||||||
|  |  | ||||||
|  |  | ||||||
| #define MyAppName "Ollama" |  | ||||||
| #if GetEnv("PKG_VERSION") != "" |  | ||||||
|   #define MyAppVersion GetEnv("PKG_VERSION") |  | ||||||
| #else |  | ||||||
|   #define MyAppVersion "0.0.0" |  | ||||||
| #endif |  | ||||||
| #define MyAppPublisher "Ollama" |  | ||||||
| #define MyAppURL "https://ollama.com/" |  | ||||||
| #define MyAppExeName "ollama app.exe" |  | ||||||
| #define MyIcon ".\assets\app.ico" |  | ||||||
|  |  | ||||||
| [Setup] |  | ||||||
| ; NOTE: The value of AppId uniquely identifies this application. Do not use the same AppId value in installers for other applications. |  | ||||||
| ; (To generate a new GUID, click Tools | Generate GUID inside the IDE.) |  | ||||||
| AppId={{44E83376-CE68-45EB-8FC1-393500EB558C} |  | ||||||
| AppName={#MyAppName} |  | ||||||
| AppVersion={#MyAppVersion} |  | ||||||
| VersionInfoVersion={#MyAppVersion} |  | ||||||
| ;AppVerName={#MyAppName} {#MyAppVersion} |  | ||||||
| AppPublisher={#MyAppPublisher} |  | ||||||
| AppPublisherURL={#MyAppURL} |  | ||||||
| AppSupportURL={#MyAppURL} |  | ||||||
| AppUpdatesURL={#MyAppURL} |  | ||||||
| ArchitecturesAllowed=x64compatible arm64 |  | ||||||
| ArchitecturesInstallIn64BitMode=x64compatible arm64 |  | ||||||
| DefaultDirName={localappdata}\Programs\{#MyAppName} |  | ||||||
| DefaultGroupName={#MyAppName} |  | ||||||
| DisableProgramGroupPage=yes |  | ||||||
| PrivilegesRequired=lowest |  | ||||||
| OutputBaseFilename="OllamaSetup" |  | ||||||
| SetupIconFile={#MyIcon} |  | ||||||
| UninstallDisplayIcon={uninstallexe} |  | ||||||
| Compression=lzma2 |  | ||||||
| SolidCompression=no |  | ||||||
| WizardStyle=modern |  | ||||||
| ChangesEnvironment=yes |  | ||||||
| OutputDir=..\dist\ |  | ||||||
|  |  | ||||||
| ; Disable logging once everything's battle tested |  | ||||||
| ; Filename will be %TEMP%\Setup Log*.txt |  | ||||||
| SetupLogging=yes |  | ||||||
| CloseApplications=yes |  | ||||||
| RestartApplications=no |  | ||||||
| RestartIfNeededByRun=no |  | ||||||
|  |  | ||||||
| ; https://jrsoftware.org/ishelp/index.php?topic=setup_wizardimagefile |  | ||||||
| WizardSmallImageFile=.\assets\setup.bmp |  | ||||||
|  |  | ||||||
| ; TODO verifty actual min windows version... |  | ||||||
| ; OG Win 10 |  | ||||||
| MinVersion=10.0.10240 |  | ||||||
|  |  | ||||||
| ; First release that supports WinRT UI Composition for win32 apps |  | ||||||
| ; MinVersion=10.0.17134 |  | ||||||
| ; First release with XAML Islands - possible UI path forward |  | ||||||
| ; MinVersion=10.0.18362 |  | ||||||
|  |  | ||||||
| ; quiet... |  | ||||||
| DisableDirPage=yes |  | ||||||
| DisableFinishedPage=yes |  | ||||||
| DisableReadyMemo=yes |  | ||||||
| DisableReadyPage=yes |  | ||||||
| DisableStartupPrompt=yes |  | ||||||
| DisableWelcomePage=yes |  | ||||||
|  |  | ||||||
| ; TODO - percentage can't be set less than 100, so how to make it shorter? |  | ||||||
| ; WizardSizePercent=100,80 |  | ||||||
|  |  | ||||||
| #if GetEnv("KEY_CONTAINER") |  | ||||||
| SignTool=MySignTool |  | ||||||
| SignedUninstaller=yes |  | ||||||
| #endif |  | ||||||
|  |  | ||||||
| SetupMutex=OllamaSetupMutex |  | ||||||
|  |  | ||||||
| [Languages] |  | ||||||
| Name: "english"; MessagesFile: "compiler:Default.isl" |  | ||||||
|  |  | ||||||
| [LangOptions] |  | ||||||
| DialogFontSize=12 |  | ||||||
|  |  | ||||||
| [Files] |  | ||||||
| #if DirExists("..\dist\windows-amd64") |  | ||||||
| Source: "..\dist\windows-amd64-app.exe"; DestDir: "{app}"; DestName: "{#MyAppExeName}" ;Check: not IsArm64();  Flags: ignoreversion 64bit |  | ||||||
| Source: "..\dist\windows-amd64\ollama.exe"; DestDir: "{app}"; Check: not IsArm64(); Flags: ignoreversion 64bit |  | ||||||
| Source: "..\dist\windows-amd64\lib\ollama\*"; DestDir: "{app}\lib\ollama\"; Check: not IsArm64(); Flags: ignoreversion 64bit recursesubdirs |  | ||||||
| #endif |  | ||||||
|  |  | ||||||
| #if DirExists("..\dist\windows-arm64") |  | ||||||
| Source: "..\dist\windows-arm64\vc_redist.arm64.exe"; DestDir: "{tmp}"; Check: IsArm64() and vc_redist_needed(); Flags: deleteafterinstall |  | ||||||
| Source: "..\dist\windows-arm64-app.exe"; DestDir: "{app}"; DestName: "{#MyAppExeName}" ;Check: IsArm64();  Flags: ignoreversion 64bit |  | ||||||
| Source: "..\dist\windows-arm64\ollama.exe"; DestDir: "{app}"; Check: IsArm64(); Flags: ignoreversion 64bit |  | ||||||
| Source: "..\dist\windows-arm64\lib\ollama\*"; DestDir: "{app}\lib\ollama\"; Check: IsArm64(); Flags: ignoreversion 64bit recursesubdirs |  | ||||||
| #endif |  | ||||||
|  |  | ||||||
| Source: "..\dist\ollama_welcome.ps1"; DestDir: "{app}"; Flags: ignoreversion |  | ||||||
| Source: ".\assets\app.ico"; DestDir: "{app}"; Flags: ignoreversion |  | ||||||
|  |  | ||||||
| [Icons] |  | ||||||
| Name: "{group}\{#MyAppName}"; Filename: "{app}\{#MyAppExeName}"; IconFilename: "{app}\app.ico" |  | ||||||
| Name: "{userstartup}\{#MyAppName}"; Filename: "{app}\{#MyAppExeName}"; IconFilename: "{app}\app.ico" |  | ||||||
| Name: "{userprograms}\{#MyAppName}"; Filename: "{app}\{#MyAppExeName}"; IconFilename: "{app}\app.ico" |  | ||||||
|  |  | ||||||
| [Run] |  | ||||||
| #if DirExists("..\dist\windows-arm64") |  | ||||||
| Filename: "{tmp}\vc_redist.arm64.exe"; Parameters: "/install /passive /norestart"; Check: IsArm64() and vc_redist_needed(); StatusMsg: "Installing VC++ Redistributables..."; Flags: waituntilterminated |  | ||||||
| #endif |  | ||||||
| Filename: "{cmd}"; Parameters: "/C set PATH={app};%PATH% & ""{app}\{#MyAppExeName}"""; Flags: postinstall nowait runhidden |  | ||||||
|  |  | ||||||
| [UninstallRun] |  | ||||||
| ; Filename: "{cmd}"; Parameters: "/C ""taskkill /im ''{#MyAppExeName}'' /f /t"; Flags: runhidden |  | ||||||
| ; Filename: "{cmd}"; Parameters: "/C ""taskkill /im ollama.exe /f /t"; Flags: runhidden |  | ||||||
| Filename: "taskkill"; Parameters: "/im ""{#MyAppExeName}"" /f /t"; Flags: runhidden |  | ||||||
| Filename: "taskkill"; Parameters: "/im ""ollama.exe"" /f /t"; Flags: runhidden |  | ||||||
| ; HACK!  need to give the server and app enough time to exit |  | ||||||
| ; TODO - convert this to a Pascal code script so it waits until they're no longer running, then completes |  | ||||||
| Filename: "{cmd}"; Parameters: "/c timeout 5"; Flags: runhidden |  | ||||||
|  |  | ||||||
| [UninstallDelete] |  | ||||||
| Type: filesandordirs; Name: "{%TEMP}\ollama*" |  | ||||||
| Type: filesandordirs; Name: "{%LOCALAPPDATA}\Ollama" |  | ||||||
| Type: filesandordirs; Name: "{%LOCALAPPDATA}\Programs\Ollama" |  | ||||||
| Type: filesandordirs; Name: "{%USERPROFILE}\.ollama\models" |  | ||||||
| Type: filesandordirs; Name: "{%USERPROFILE}\.ollama\history" |  | ||||||
| ; NOTE: if the user has a custom OLLAMA_MODELS it will be preserved |  | ||||||
|  |  | ||||||
| [InstallDelete] |  | ||||||
| Type: filesandordirs; Name: "{%TEMP}\ollama*" |  | ||||||
| Type: filesandordirs; Name: "{%LOCALAPPDATA}\Programs\Ollama" |  | ||||||
|  |  | ||||||
| [Messages] |  | ||||||
| WizardReady=Ollama Windows Preview |  | ||||||
| ReadyLabel1=%nLet's get you up and running with your own large language models. |  | ||||||
| SetupAppRunningError=Another Ollama installer is running.%n%nPlease cancel or finish the other installer, then click OK to continue with this install, or Cancel to exit. |  | ||||||
|  |  | ||||||
|  |  | ||||||
| ;FinishedHeadingLabel=Run your first model |  | ||||||
| ;FinishedLabel=%nRun this command in a PowerShell or cmd terminal.%n%n%n    ollama run llama3.2 |  | ||||||
| ;ClickFinish=%n |  | ||||||
|  |  | ||||||
| [Registry] |  | ||||||
| Root: HKCU; Subkey: "Environment"; \ |  | ||||||
|     ValueType: expandsz; ValueName: "Path"; ValueData: "{olddata};{app}"; \ |  | ||||||
|     Check: NeedsAddPath('{app}') |  | ||||||
|  |  | ||||||
| [Code] |  | ||||||
|  |  | ||||||
| function NeedsAddPath(Param: string): boolean; |  | ||||||
| var |  | ||||||
|   OrigPath: string; |  | ||||||
| begin |  | ||||||
|   if not RegQueryStringValue(HKEY_CURRENT_USER, |  | ||||||
|     'Environment', |  | ||||||
|     'Path', OrigPath) |  | ||||||
|   then begin |  | ||||||
|     Result := True; |  | ||||||
|     exit; |  | ||||||
|   end; |  | ||||||
|   { look for the path with leading and trailing semicolon } |  | ||||||
|   { Pos() returns 0 if not found } |  | ||||||
|   Result := Pos(';' + ExpandConstant(Param) + ';', ';' + OrigPath + ';') = 0; |  | ||||||
| end; |  | ||||||
|  |  | ||||||
| { --- VC Runtime libraries discovery code - Only install vc_redist if it isn't already installed ----- } |  | ||||||
| const VCRTL_MIN_V1 = 14; |  | ||||||
| const VCRTL_MIN_V2 = 40; |  | ||||||
| const VCRTL_MIN_V3 = 33807; |  | ||||||
| const VCRTL_MIN_V4 = 0; |  | ||||||
|  |  | ||||||
|  // check if the minimum required vc redist is installed (by looking the registry) |  | ||||||
| function vc_redist_needed (): Boolean; |  | ||||||
| var |  | ||||||
|   sRegKey: string; |  | ||||||
|   v1: Cardinal; |  | ||||||
|   v2: Cardinal; |  | ||||||
|   v3: Cardinal; |  | ||||||
|   v4: Cardinal; |  | ||||||
| begin |  | ||||||
|   sRegKey := 'SOFTWARE\WOW6432Node\Microsoft\VisualStudio\14.0\VC\Runtimes\arm64'; |  | ||||||
|   if (RegQueryDWordValue (HKEY_LOCAL_MACHINE, sRegKey, 'Major', v1)  and |  | ||||||
|       RegQueryDWordValue (HKEY_LOCAL_MACHINE, sRegKey, 'Minor', v2) and |  | ||||||
|       RegQueryDWordValue (HKEY_LOCAL_MACHINE, sRegKey, 'Bld', v3) and |  | ||||||
|       RegQueryDWordValue (HKEY_LOCAL_MACHINE, sRegKey, 'RBld', v4)) then |  | ||||||
|   begin |  | ||||||
|     Log ('VC Redist version: ' + IntToStr (v1) + |  | ||||||
|         '.' + IntToStr (v2) + '.' + IntToStr (v3) + |  | ||||||
|         '.' + IntToStr (v4)); |  | ||||||
|     { Version info was found. Return true if later or equal to our |  | ||||||
|        minimal required version RTL_MIN_Vx } |  | ||||||
|     Result := not ( |  | ||||||
|         (v1 > VCRTL_MIN_V1) or ((v1 = VCRTL_MIN_V1) and |  | ||||||
|          ((v2 > VCRTL_MIN_V2) or ((v2 = VCRTL_MIN_V2) and |  | ||||||
|           ((v3 > VCRTL_MIN_V3) or ((v3 = VCRTL_MIN_V3) and |  | ||||||
|            (v4 >= VCRTL_MIN_V4))))))); |  | ||||||
|   end |  | ||||||
|   else |  | ||||||
|     Result := TRUE; |  | ||||||
| end; |  | ||||||
| @@ -1,29 +0,0 @@ | |||||||
| #include <winver.h> |  | ||||||
|  |  | ||||||
| VS_VERSION_INFO VERSIONINFO |  | ||||||
|  FILEFLAGSMASK 0x3fL |  | ||||||
| #ifdef _DEBUG |  | ||||||
|  FILEFLAGS 0x1L |  | ||||||
| #else |  | ||||||
|  FILEFLAGS 0x0L |  | ||||||
| #endif |  | ||||||
|  FILEOS 0x40004L |  | ||||||
|  FILETYPE 0x1L |  | ||||||
|  FILESUBTYPE 0x0L |  | ||||||
| BEGIN |  | ||||||
|     BLOCK "StringFileInfo" |  | ||||||
|     BEGIN |  | ||||||
|         BLOCK "040904b0" |  | ||||||
|         BEGIN |  | ||||||
|             VALUE "FileDescription", "Ollama" |  | ||||||
|             VALUE "InternalName", "Ollama" |  | ||||||
|             VALUE "OriginalFilename", "ollama app.exe" |  | ||||||
|             VALUE "ProductName", "Ollama" |  | ||||||
|         END |  | ||||||
|     END |  | ||||||
|  |  | ||||||
|     BLOCK "VarFileInfo" |  | ||||||
|     BEGIN |  | ||||||
|         VALUE "Translation", 0x409, 1200 |  | ||||||
|     END |  | ||||||
| END |  | ||||||
| @@ -1,8 +0,0 @@ | |||||||
| # TODO - consider ANSI colors and maybe ASCII art... |  | ||||||
| write-host "" |  | ||||||
| write-host "Welcome to Ollama!" |  | ||||||
| write-host "" |  | ||||||
| write-host "Run your first model:" |  | ||||||
| write-host "" |  | ||||||
| write-host "`tollama run llama3.2" |  | ||||||
| write-host "" |  | ||||||
| @@ -19,7 +19,7 @@ export default function () { | |||||||
|   const [step, setStep] = useState<Step>(Step.WELCOME) |   const [step, setStep] = useState<Step>(Step.WELCOME) | ||||||
|   const [commandCopied, setCommandCopied] = useState<boolean>(false) |   const [commandCopied, setCommandCopied] = useState<boolean>(false) | ||||||
| 
 | 
 | ||||||
|   const command = 'ollama run llama3.2' |   const command = 'ollama run llama2' | ||||||
| 
 | 
 | ||||||
|   return ( |   return ( | ||||||
|     <div className='drag'> |     <div className='drag'> | ||||||
| @@ -162,7 +162,7 @@ app.on('before-quit', () => { | |||||||
|   } |   } | ||||||
| }) | }) | ||||||
| 
 | 
 | ||||||
| const updateURL = `https://ollama.com/api/update?os=${process.platform}&arch=${ | const updateURL = `https://ollama.ai/api/update?os=${process.platform}&arch=${ | ||||||
|   process.arch |   process.arch | ||||||
| }&version=${app.getVersion()}&id=${id()}` | }&version=${app.getVersion()}&id=${id()}` | ||||||
| 
 | 
 | ||||||
| Before Width: | Height: | Size: 17 KiB After Width: | Height: | Size: 17 KiB | 
| @@ -1,97 +0,0 @@ | |||||||
| package store |  | ||||||
|  |  | ||||||
| import ( |  | ||||||
| 	"encoding/json" |  | ||||||
| 	"errors" |  | ||||||
| 	"fmt" |  | ||||||
| 	"log/slog" |  | ||||||
| 	"os" |  | ||||||
| 	"path/filepath" |  | ||||||
| 	"sync" |  | ||||||
|  |  | ||||||
| 	"github.com/google/uuid" |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| type Store struct { |  | ||||||
| 	ID           string `json:"id"` |  | ||||||
| 	FirstTimeRun bool   `json:"first-time-run"` |  | ||||||
| } |  | ||||||
|  |  | ||||||
| var ( |  | ||||||
| 	lock  sync.Mutex |  | ||||||
| 	store Store |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| func GetID() string { |  | ||||||
| 	lock.Lock() |  | ||||||
| 	defer lock.Unlock() |  | ||||||
| 	if store.ID == "" { |  | ||||||
| 		initStore() |  | ||||||
| 	} |  | ||||||
| 	return store.ID |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func GetFirstTimeRun() bool { |  | ||||||
| 	lock.Lock() |  | ||||||
| 	defer lock.Unlock() |  | ||||||
| 	if store.ID == "" { |  | ||||||
| 		initStore() |  | ||||||
| 	} |  | ||||||
| 	return store.FirstTimeRun |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func SetFirstTimeRun(val bool) { |  | ||||||
| 	lock.Lock() |  | ||||||
| 	defer lock.Unlock() |  | ||||||
| 	if store.FirstTimeRun == val { |  | ||||||
| 		return |  | ||||||
| 	} |  | ||||||
| 	store.FirstTimeRun = val |  | ||||||
| 	writeStore(getStorePath()) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // lock must be held |  | ||||||
| func initStore() { |  | ||||||
| 	storeFile, err := os.Open(getStorePath()) |  | ||||||
| 	if err == nil { |  | ||||||
| 		defer storeFile.Close() |  | ||||||
| 		err = json.NewDecoder(storeFile).Decode(&store) |  | ||||||
| 		if err == nil { |  | ||||||
| 			slog.Debug(fmt.Sprintf("loaded existing store %s - ID: %s", getStorePath(), store.ID)) |  | ||||||
| 			return |  | ||||||
| 		} |  | ||||||
| 	} else if !errors.Is(err, os.ErrNotExist) { |  | ||||||
| 		slog.Debug(fmt.Sprintf("unexpected error searching for store: %s", err)) |  | ||||||
| 	} |  | ||||||
| 	slog.Debug("initializing new store") |  | ||||||
| 	store.ID = uuid.New().String() |  | ||||||
| 	writeStore(getStorePath()) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func writeStore(storeFilename string) { |  | ||||||
| 	ollamaDir := filepath.Dir(storeFilename) |  | ||||||
| 	_, err := os.Stat(ollamaDir) |  | ||||||
| 	if errors.Is(err, os.ErrNotExist) { |  | ||||||
| 		if err := os.MkdirAll(ollamaDir, 0o755); err != nil { |  | ||||||
| 			slog.Error(fmt.Sprintf("create ollama dir %s: %v", ollamaDir, err)) |  | ||||||
| 			return |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| 	payload, err := json.Marshal(store) |  | ||||||
| 	if err != nil { |  | ||||||
| 		slog.Error(fmt.Sprintf("failed to marshal store: %s", err)) |  | ||||||
| 		return |  | ||||||
| 	} |  | ||||||
| 	fp, err := os.OpenFile(storeFilename, os.O_WRONLY|os.O_CREATE|os.O_TRUNC, 0o755) |  | ||||||
| 	if err != nil { |  | ||||||
| 		slog.Error(fmt.Sprintf("write store payload %s: %v", storeFilename, err)) |  | ||||||
| 		return |  | ||||||
| 	} |  | ||||||
| 	defer fp.Close() |  | ||||||
| 	if n, err := fp.Write(payload); err != nil || n != len(payload) { |  | ||||||
| 		slog.Error(fmt.Sprintf("write store payload %s: %d vs %d -- %v", storeFilename, n, len(payload), err)) |  | ||||||
| 		return |  | ||||||
| 	} |  | ||||||
| 	slog.Debug("Store contents: " + string(payload)) |  | ||||||
| 	slog.Info(fmt.Sprintf("wrote store: %s", storeFilename)) |  | ||||||
| } |  | ||||||
| @@ -1,13 +0,0 @@ | |||||||
| package store |  | ||||||
|  |  | ||||||
| import ( |  | ||||||
| 	"os" |  | ||||||
| 	"path/filepath" |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| func getStorePath() string { |  | ||||||
| 	// TODO - system wide location? |  | ||||||
|  |  | ||||||
| 	home := os.Getenv("HOME") |  | ||||||
| 	return filepath.Join(home, "Library", "Application Support", "Ollama", "config.json") |  | ||||||
| } |  | ||||||
| @@ -1,16 +0,0 @@ | |||||||
| package store |  | ||||||
|  |  | ||||||
| import ( |  | ||||||
| 	"os" |  | ||||||
| 	"path/filepath" |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| func getStorePath() string { |  | ||||||
| 	if os.Geteuid() == 0 { |  | ||||||
| 		// TODO where should we store this on linux for system-wide operation? |  | ||||||
| 		return "/etc/ollama/config.json" |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	home := os.Getenv("HOME") |  | ||||||
| 	return filepath.Join(home, ".ollama", "config.json") |  | ||||||
| } |  | ||||||
| @@ -1,11 +0,0 @@ | |||||||
| package store |  | ||||||
|  |  | ||||||
| import ( |  | ||||||
| 	"os" |  | ||||||
| 	"path/filepath" |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| func getStorePath() string { |  | ||||||
| 	localAppData := os.Getenv("LOCALAPPDATA") |  | ||||||
| 	return filepath.Join(localAppData, "Ollama", "config.json") |  | ||||||
| } |  | ||||||
| @@ -1,24 +0,0 @@ | |||||||
| package commontray |  | ||||||
|  |  | ||||||
| var ( |  | ||||||
| 	Title   = "Ollama" |  | ||||||
| 	ToolTip = "Ollama" |  | ||||||
|  |  | ||||||
| 	UpdateIconName = "tray_upgrade" |  | ||||||
| 	IconName       = "tray" |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| type Callbacks struct { |  | ||||||
| 	Quit       chan struct{} |  | ||||||
| 	Update     chan struct{} |  | ||||||
| 	DoFirstUse chan struct{} |  | ||||||
| 	ShowLogs   chan struct{} |  | ||||||
| } |  | ||||||
|  |  | ||||||
| type OllamaTray interface { |  | ||||||
| 	GetCallbacks() Callbacks |  | ||||||
| 	Run() |  | ||||||
| 	UpdateAvailable(ver string) error |  | ||||||
| 	DisplayFirstUseNotification() error |  | ||||||
| 	Quit() |  | ||||||
| } |  | ||||||
| @@ -1,28 +0,0 @@ | |||||||
| package tray |  | ||||||
|  |  | ||||||
| import ( |  | ||||||
| 	"fmt" |  | ||||||
| 	"runtime" |  | ||||||
|  |  | ||||||
| 	"github.com/ollama/ollama/app/assets" |  | ||||||
| 	"github.com/ollama/ollama/app/tray/commontray" |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| func NewTray() (commontray.OllamaTray, error) { |  | ||||||
| 	extension := ".png" |  | ||||||
| 	if runtime.GOOS == "windows" { |  | ||||||
| 		extension = ".ico" |  | ||||||
| 	} |  | ||||||
| 	iconName := commontray.UpdateIconName + extension |  | ||||||
| 	updateIcon, err := assets.GetIcon(iconName) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return nil, fmt.Errorf("failed to load icon %s: %w", iconName, err) |  | ||||||
| 	} |  | ||||||
| 	iconName = commontray.IconName + extension |  | ||||||
| 	icon, err := assets.GetIcon(iconName) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return nil, fmt.Errorf("failed to load icon %s: %w", iconName, err) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	return InitPlatformTray(icon, updateIcon) |  | ||||||
| } |  | ||||||
| @@ -1,13 +0,0 @@ | |||||||
| //go:build !windows |  | ||||||
|  |  | ||||||
| package tray |  | ||||||
|  |  | ||||||
| import ( |  | ||||||
| 	"errors" |  | ||||||
|  |  | ||||||
| 	"github.com/ollama/ollama/app/tray/commontray" |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| func InitPlatformTray(icon, updateIcon []byte) (commontray.OllamaTray, error) { |  | ||||||
| 	return nil, errors.New("not implemented") |  | ||||||
| } |  | ||||||
| @@ -1,10 +0,0 @@ | |||||||
| package tray |  | ||||||
|  |  | ||||||
| import ( |  | ||||||
| 	"github.com/ollama/ollama/app/tray/commontray" |  | ||||||
| 	"github.com/ollama/ollama/app/tray/wintray" |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| func InitPlatformTray(icon, updateIcon []byte) (commontray.OllamaTray, error) { |  | ||||||
| 	return wintray.InitTray(icon, updateIcon) |  | ||||||
| } |  | ||||||
| @@ -1,181 +0,0 @@ | |||||||
| //go:build windows |  | ||||||
|  |  | ||||||
| package wintray |  | ||||||
|  |  | ||||||
| import ( |  | ||||||
| 	"fmt" |  | ||||||
| 	"log/slog" |  | ||||||
| 	"sync" |  | ||||||
| 	"unsafe" |  | ||||||
|  |  | ||||||
| 	"golang.org/x/sys/windows" |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| var quitOnce sync.Once |  | ||||||
|  |  | ||||||
| func (t *winTray) Run() { |  | ||||||
| 	nativeLoop() |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func nativeLoop() { |  | ||||||
| 	// Main message pump. |  | ||||||
| 	slog.Debug("starting event handling loop") |  | ||||||
| 	m := &struct { |  | ||||||
| 		WindowHandle windows.Handle |  | ||||||
| 		Message      uint32 |  | ||||||
| 		Wparam       uintptr |  | ||||||
| 		Lparam       uintptr |  | ||||||
| 		Time         uint32 |  | ||||||
| 		Pt           point |  | ||||||
| 		LPrivate     uint32 |  | ||||||
| 	}{} |  | ||||||
| 	for { |  | ||||||
| 		ret, _, err := pGetMessage.Call(uintptr(unsafe.Pointer(m)), 0, 0, 0) |  | ||||||
|  |  | ||||||
| 		// If the function retrieves a message other than WM_QUIT, the return value is nonzero. |  | ||||||
| 		// If the function retrieves the WM_QUIT message, the return value is zero. |  | ||||||
| 		// If there is an error, the return value is -1 |  | ||||||
| 		// https://msdn.microsoft.com/en-us/library/windows/desktop/ms644936(v=vs.85).aspx |  | ||||||
| 		switch int32(ret) { |  | ||||||
| 		case -1: |  | ||||||
| 			slog.Error(fmt.Sprintf("get message failure: %v", err)) |  | ||||||
| 			return |  | ||||||
| 		case 0: |  | ||||||
| 			return |  | ||||||
| 		default: |  | ||||||
| 			pTranslateMessage.Call(uintptr(unsafe.Pointer(m))) //nolint:errcheck |  | ||||||
| 			pDispatchMessage.Call(uintptr(unsafe.Pointer(m)))  //nolint:errcheck |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // WindowProc callback function that processes messages sent to a window. |  | ||||||
| // https://msdn.microsoft.com/en-us/library/windows/desktop/ms633573(v=vs.85).aspx |  | ||||||
| func (t *winTray) wndProc(hWnd windows.Handle, message uint32, wParam, lParam uintptr) (lResult uintptr) { |  | ||||||
| 	const ( |  | ||||||
| 		WM_RBUTTONUP   = 0x0205 |  | ||||||
| 		WM_LBUTTONUP   = 0x0202 |  | ||||||
| 		WM_COMMAND     = 0x0111 |  | ||||||
| 		WM_ENDSESSION  = 0x0016 |  | ||||||
| 		WM_CLOSE       = 0x0010 |  | ||||||
| 		WM_DESTROY     = 0x0002 |  | ||||||
| 		WM_MOUSEMOVE   = 0x0200 |  | ||||||
| 		WM_LBUTTONDOWN = 0x0201 |  | ||||||
| 	) |  | ||||||
| 	switch message { |  | ||||||
| 	case WM_COMMAND: |  | ||||||
| 		menuItemId := int32(wParam) |  | ||||||
| 		// https://docs.microsoft.com/en-us/windows/win32/menurc/wm-command#menus |  | ||||||
| 		switch menuItemId { |  | ||||||
| 		case quitMenuID: |  | ||||||
| 			select { |  | ||||||
| 			case t.callbacks.Quit <- struct{}{}: |  | ||||||
| 			// should not happen but in case not listening |  | ||||||
| 			default: |  | ||||||
| 				slog.Error("no listener on Quit") |  | ||||||
| 			} |  | ||||||
| 		case updateMenuID: |  | ||||||
| 			select { |  | ||||||
| 			case t.callbacks.Update <- struct{}{}: |  | ||||||
| 			// should not happen but in case not listening |  | ||||||
| 			default: |  | ||||||
| 				slog.Error("no listener on Update") |  | ||||||
| 			} |  | ||||||
| 		case diagLogsMenuID: |  | ||||||
| 			select { |  | ||||||
| 			case t.callbacks.ShowLogs <- struct{}{}: |  | ||||||
| 			// should not happen but in case not listening |  | ||||||
| 			default: |  | ||||||
| 				slog.Error("no listener on ShowLogs") |  | ||||||
| 			} |  | ||||||
| 		default: |  | ||||||
| 			slog.Debug(fmt.Sprintf("Unexpected menu item id: %d", menuItemId)) |  | ||||||
| 		} |  | ||||||
| 	case WM_CLOSE: |  | ||||||
| 		boolRet, _, err := pDestroyWindow.Call(uintptr(t.window)) |  | ||||||
| 		if boolRet == 0 { |  | ||||||
| 			slog.Error(fmt.Sprintf("failed to destroy window: %s", err)) |  | ||||||
| 		} |  | ||||||
| 		err = t.wcex.unregister() |  | ||||||
| 		if err != nil { |  | ||||||
| 			slog.Error(fmt.Sprintf("failed to uregister windo %s", err)) |  | ||||||
| 		} |  | ||||||
| 	case WM_DESTROY: |  | ||||||
| 		// same as WM_ENDSESSION, but throws 0 exit code after all |  | ||||||
| 		defer pPostQuitMessage.Call(uintptr(int32(0))) //nolint:errcheck |  | ||||||
| 		fallthrough |  | ||||||
| 	case WM_ENDSESSION: |  | ||||||
| 		t.muNID.Lock() |  | ||||||
| 		if t.nid != nil { |  | ||||||
| 			err := t.nid.delete() |  | ||||||
| 			if err != nil { |  | ||||||
| 				slog.Error(fmt.Sprintf("failed to delete nid: %s", err)) |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 		t.muNID.Unlock() |  | ||||||
| 	case t.wmSystrayMessage: |  | ||||||
| 		switch lParam { |  | ||||||
| 		case WM_MOUSEMOVE, WM_LBUTTONDOWN: |  | ||||||
| 			// Ignore these... |  | ||||||
| 		case WM_RBUTTONUP, WM_LBUTTONUP: |  | ||||||
| 			err := t.showMenu() |  | ||||||
| 			if err != nil { |  | ||||||
| 				slog.Error(fmt.Sprintf("failed to show menu: %s", err)) |  | ||||||
| 			} |  | ||||||
| 		case 0x405: // TODO - how is this magic value derived for the notification left click |  | ||||||
| 			if t.pendingUpdate { |  | ||||||
| 				select { |  | ||||||
| 				case t.callbacks.Update <- struct{}{}: |  | ||||||
| 				// should not happen but in case not listening |  | ||||||
| 				default: |  | ||||||
| 					slog.Error("no listener on Update") |  | ||||||
| 				} |  | ||||||
| 			} else { |  | ||||||
| 				select { |  | ||||||
| 				case t.callbacks.DoFirstUse <- struct{}{}: |  | ||||||
| 				// should not happen but in case not listening |  | ||||||
| 				default: |  | ||||||
| 					slog.Error("no listener on DoFirstUse") |  | ||||||
| 				} |  | ||||||
| 			} |  | ||||||
| 		case 0x404: // Middle click or close notification |  | ||||||
| 			// slog.Debug("doing nothing on close of first time notification") |  | ||||||
| 		default: |  | ||||||
| 			// 0x402 also seems common - what is it? |  | ||||||
| 			slog.Debug(fmt.Sprintf("unmanaged app message, lParm: 0x%x", lParam)) |  | ||||||
| 		} |  | ||||||
| 	case t.wmTaskbarCreated: // on explorer.exe restarts |  | ||||||
| 		t.muNID.Lock() |  | ||||||
| 		err := t.nid.add() |  | ||||||
| 		if err != nil { |  | ||||||
| 			slog.Error(fmt.Sprintf("failed to refresh the taskbar on explorer restart: %s", err)) |  | ||||||
| 		} |  | ||||||
| 		t.muNID.Unlock() |  | ||||||
| 	default: |  | ||||||
| 		// Calls the default window procedure to provide default processing for any window messages that an application does not process. |  | ||||||
| 		// https://msdn.microsoft.com/en-us/library/windows/desktop/ms633572(v=vs.85).aspx |  | ||||||
| 		lResult, _, _ = pDefWindowProc.Call( |  | ||||||
| 			uintptr(hWnd), |  | ||||||
| 			uintptr(message), |  | ||||||
| 			wParam, |  | ||||||
| 			lParam, |  | ||||||
| 		) |  | ||||||
| 	} |  | ||||||
| 	return |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (t *winTray) Quit() { |  | ||||||
| 	quitOnce.Do(quit) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func quit() { |  | ||||||
| 	boolRet, _, err := pPostMessage.Call( |  | ||||||
| 		uintptr(wt.window), |  | ||||||
| 		WM_CLOSE, |  | ||||||
| 		0, |  | ||||||
| 		0, |  | ||||||
| 	) |  | ||||||
| 	if boolRet == 0 { |  | ||||||
| 		slog.Error(fmt.Sprintf("failed to post close message on shutdown %s", err)) |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
| @@ -1,71 +0,0 @@ | |||||||
| //go:build windows |  | ||||||
|  |  | ||||||
| package wintray |  | ||||||
|  |  | ||||||
| import ( |  | ||||||
| 	"fmt" |  | ||||||
| 	"log/slog" |  | ||||||
| 	"unsafe" |  | ||||||
|  |  | ||||||
| 	"golang.org/x/sys/windows" |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| const ( |  | ||||||
| 	updateAvailableMenuID = 1 |  | ||||||
| 	updateMenuID          = updateAvailableMenuID + 1 |  | ||||||
| 	separatorMenuID       = updateMenuID + 1 |  | ||||||
| 	diagLogsMenuID        = separatorMenuID + 1 |  | ||||||
| 	diagSeparatorMenuID   = diagLogsMenuID + 1 |  | ||||||
| 	quitMenuID            = diagSeparatorMenuID + 1 |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| func (t *winTray) initMenus() error { |  | ||||||
| 	if err := t.addOrUpdateMenuItem(diagLogsMenuID, 0, diagLogsMenuTitle, false); err != nil { |  | ||||||
| 		return fmt.Errorf("unable to create menu entries %w\n", err) |  | ||||||
| 	} |  | ||||||
| 	if err := t.addSeparatorMenuItem(diagSeparatorMenuID, 0); err != nil { |  | ||||||
| 		return fmt.Errorf("unable to create menu entries %w", err) |  | ||||||
| 	} |  | ||||||
| 	if err := t.addOrUpdateMenuItem(quitMenuID, 0, quitMenuTitle, false); err != nil { |  | ||||||
| 		return fmt.Errorf("unable to create menu entries %w\n", err) |  | ||||||
| 	} |  | ||||||
| 	return nil |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (t *winTray) UpdateAvailable(ver string) error { |  | ||||||
| 	if !t.updateNotified { |  | ||||||
| 		slog.Debug("updating menu and sending notification for new update") |  | ||||||
| 		if err := t.addOrUpdateMenuItem(updateAvailableMenuID, 0, updateAvailableMenuTitle, true); err != nil { |  | ||||||
| 			return fmt.Errorf("unable to create menu entries %w", err) |  | ||||||
| 		} |  | ||||||
| 		if err := t.addOrUpdateMenuItem(updateMenuID, 0, updateMenutTitle, false); err != nil { |  | ||||||
| 			return fmt.Errorf("unable to create menu entries %w", err) |  | ||||||
| 		} |  | ||||||
| 		if err := t.addSeparatorMenuItem(separatorMenuID, 0); err != nil { |  | ||||||
| 			return fmt.Errorf("unable to create menu entries %w", err) |  | ||||||
| 		} |  | ||||||
| 		iconFilePath, err := iconBytesToFilePath(wt.updateIcon) |  | ||||||
| 		if err != nil { |  | ||||||
| 			return fmt.Errorf("unable to write icon data to temp file: %w", err) |  | ||||||
| 		} |  | ||||||
| 		if err := wt.setIcon(iconFilePath); err != nil { |  | ||||||
| 			return fmt.Errorf("unable to set icon: %w", err) |  | ||||||
| 		} |  | ||||||
| 		t.updateNotified = true |  | ||||||
|  |  | ||||||
| 		t.pendingUpdate = true |  | ||||||
| 		// Now pop up the notification |  | ||||||
| 		t.muNID.Lock() |  | ||||||
| 		defer t.muNID.Unlock() |  | ||||||
| 		copy(t.nid.InfoTitle[:], windows.StringToUTF16(updateTitle)) |  | ||||||
| 		copy(t.nid.Info[:], windows.StringToUTF16(fmt.Sprintf(updateMessage, ver))) |  | ||||||
| 		t.nid.Flags |= NIF_INFO |  | ||||||
| 		t.nid.Timeout = 10 |  | ||||||
| 		t.nid.Size = uint32(unsafe.Sizeof(*wt.nid)) |  | ||||||
| 		err = t.nid.modify() |  | ||||||
| 		if err != nil { |  | ||||||
| 			return err |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| 	return nil |  | ||||||
| } |  | ||||||
| @@ -1,15 +0,0 @@ | |||||||
| //go:build windows |  | ||||||
|  |  | ||||||
| package wintray |  | ||||||
|  |  | ||||||
| const ( |  | ||||||
| 	firstTimeTitle   = "Ollama is running" |  | ||||||
| 	firstTimeMessage = "Click here to get started" |  | ||||||
| 	updateTitle      = "Update available" |  | ||||||
| 	updateMessage    = "Ollama version %s is ready to install" |  | ||||||
|  |  | ||||||
| 	quitMenuTitle            = "Quit Ollama" |  | ||||||
| 	updateAvailableMenuTitle = "An update is available" |  | ||||||
| 	updateMenutTitle         = "Restart to update" |  | ||||||
| 	diagLogsMenuTitle        = "View logs" |  | ||||||
| ) |  | ||||||
| @@ -1,66 +0,0 @@ | |||||||
| //go:build windows |  | ||||||
|  |  | ||||||
| package wintray |  | ||||||
|  |  | ||||||
| import ( |  | ||||||
| 	"unsafe" |  | ||||||
|  |  | ||||||
| 	"golang.org/x/sys/windows" |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| // Contains information that the system needs to display notifications in the notification area. |  | ||||||
| // Used by Shell_NotifyIcon. |  | ||||||
| // https://msdn.microsoft.com/en-us/library/windows/desktop/bb773352(v=vs.85).aspx |  | ||||||
| // https://msdn.microsoft.com/en-us/library/windows/desktop/bb762159 |  | ||||||
| type notifyIconData struct { |  | ||||||
| 	Size                       uint32 |  | ||||||
| 	Wnd                        windows.Handle |  | ||||||
| 	ID, Flags, CallbackMessage uint32 |  | ||||||
| 	Icon                       windows.Handle |  | ||||||
| 	Tip                        [128]uint16 |  | ||||||
| 	State, StateMask           uint32 |  | ||||||
| 	Info                       [256]uint16 |  | ||||||
| 	// Timeout, Version           uint32 |  | ||||||
| 	Timeout uint32 |  | ||||||
|  |  | ||||||
| 	InfoTitle   [64]uint16 |  | ||||||
| 	InfoFlags   uint32 |  | ||||||
| 	GuidItem    windows.GUID |  | ||||||
| 	BalloonIcon windows.Handle |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (nid *notifyIconData) add() error { |  | ||||||
| 	const NIM_ADD = 0x00000000 |  | ||||||
| 	res, _, err := pShellNotifyIcon.Call( |  | ||||||
| 		uintptr(NIM_ADD), |  | ||||||
| 		uintptr(unsafe.Pointer(nid)), |  | ||||||
| 	) |  | ||||||
| 	if res == 0 { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
| 	return nil |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (nid *notifyIconData) modify() error { |  | ||||||
| 	const NIM_MODIFY = 0x00000001 |  | ||||||
| 	res, _, err := pShellNotifyIcon.Call( |  | ||||||
| 		uintptr(NIM_MODIFY), |  | ||||||
| 		uintptr(unsafe.Pointer(nid)), |  | ||||||
| 	) |  | ||||||
| 	if res == 0 { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
| 	return nil |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (nid *notifyIconData) delete() error { |  | ||||||
| 	const NIM_DELETE = 0x00000002 |  | ||||||
| 	res, _, err := pShellNotifyIcon.Call( |  | ||||||
| 		uintptr(NIM_DELETE), |  | ||||||
| 		uintptr(unsafe.Pointer(nid)), |  | ||||||
| 	) |  | ||||||
| 	if res == 0 { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
| 	return nil |  | ||||||
| } |  | ||||||
| @@ -1,488 +0,0 @@ | |||||||
| //go:build windows |  | ||||||
|  |  | ||||||
| package wintray |  | ||||||
|  |  | ||||||
| import ( |  | ||||||
| 	"crypto/md5" |  | ||||||
| 	"encoding/hex" |  | ||||||
| 	"fmt" |  | ||||||
| 	"log/slog" |  | ||||||
| 	"os" |  | ||||||
| 	"path/filepath" |  | ||||||
| 	"sort" |  | ||||||
| 	"sync" |  | ||||||
| 	"syscall" |  | ||||||
| 	"unsafe" |  | ||||||
|  |  | ||||||
| 	"golang.org/x/sys/windows" |  | ||||||
|  |  | ||||||
| 	"github.com/ollama/ollama/app/tray/commontray" |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| // Helpful sources: https://github.com/golang/exp/blob/master/shiny/driver/internal/win32 |  | ||||||
|  |  | ||||||
| // Contains information about loaded resources |  | ||||||
| type winTray struct { |  | ||||||
| 	instance, |  | ||||||
| 	icon, |  | ||||||
| 	cursor, |  | ||||||
| 	window windows.Handle |  | ||||||
|  |  | ||||||
| 	loadedImages   map[string]windows.Handle |  | ||||||
| 	muLoadedImages sync.RWMutex |  | ||||||
|  |  | ||||||
| 	// menus keeps track of the submenus keyed by the menu item ID, plus 0 |  | ||||||
| 	// which corresponds to the main popup menu. |  | ||||||
| 	menus    map[uint32]windows.Handle |  | ||||||
| 	muMenus  sync.RWMutex |  | ||||||
| 	menuOf   map[uint32]windows.Handle |  | ||||||
| 	muMenuOf sync.RWMutex |  | ||||||
| 	// menuItemIcons maintains the bitmap of each menu item (if applies). It's |  | ||||||
| 	// needed to show the icon correctly when showing a previously hidden menu |  | ||||||
| 	// item again. |  | ||||||
| 	// menuItemIcons   map[uint32]windows.Handle |  | ||||||
| 	// muMenuItemIcons sync.RWMutex |  | ||||||
| 	visibleItems   map[uint32][]uint32 |  | ||||||
| 	muVisibleItems sync.RWMutex |  | ||||||
|  |  | ||||||
| 	nid   *notifyIconData |  | ||||||
| 	muNID sync.RWMutex |  | ||||||
| 	wcex  *wndClassEx |  | ||||||
|  |  | ||||||
| 	wmSystrayMessage, |  | ||||||
| 	wmTaskbarCreated uint32 |  | ||||||
|  |  | ||||||
| 	pendingUpdate  bool |  | ||||||
| 	updateNotified bool // Only pop up the notification once - TODO consider daily nag? |  | ||||||
| 	// Callbacks |  | ||||||
| 	callbacks  commontray.Callbacks |  | ||||||
| 	normalIcon []byte |  | ||||||
| 	updateIcon []byte |  | ||||||
| } |  | ||||||
|  |  | ||||||
| var wt winTray |  | ||||||
|  |  | ||||||
| func (t *winTray) GetCallbacks() commontray.Callbacks { |  | ||||||
| 	return t.callbacks |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func InitTray(icon, updateIcon []byte) (*winTray, error) { |  | ||||||
| 	wt.callbacks.Quit = make(chan struct{}) |  | ||||||
| 	wt.callbacks.Update = make(chan struct{}) |  | ||||||
| 	wt.callbacks.ShowLogs = make(chan struct{}) |  | ||||||
| 	wt.callbacks.DoFirstUse = make(chan struct{}) |  | ||||||
| 	wt.normalIcon = icon |  | ||||||
| 	wt.updateIcon = updateIcon |  | ||||||
| 	if err := wt.initInstance(); err != nil { |  | ||||||
| 		return nil, fmt.Errorf("Unable to init instance: %w\n", err) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	if err := wt.createMenu(); err != nil { |  | ||||||
| 		return nil, fmt.Errorf("Unable to create menu: %w\n", err) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	iconFilePath, err := iconBytesToFilePath(wt.normalIcon) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return nil, fmt.Errorf("Unable to write icon data to temp file: %w", err) |  | ||||||
| 	} |  | ||||||
| 	if err := wt.setIcon(iconFilePath); err != nil { |  | ||||||
| 		return nil, fmt.Errorf("Unable to set icon: %w", err) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	return &wt, wt.initMenus() |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (t *winTray) initInstance() error { |  | ||||||
| 	const ( |  | ||||||
| 		className  = "OllamaClass" |  | ||||||
| 		windowName = "" |  | ||||||
| 	) |  | ||||||
|  |  | ||||||
| 	t.wmSystrayMessage = WM_USER + 1 |  | ||||||
| 	t.visibleItems = make(map[uint32][]uint32) |  | ||||||
| 	t.menus = make(map[uint32]windows.Handle) |  | ||||||
| 	t.menuOf = make(map[uint32]windows.Handle) |  | ||||||
|  |  | ||||||
| 	t.loadedImages = make(map[string]windows.Handle) |  | ||||||
|  |  | ||||||
| 	taskbarEventNamePtr, _ := windows.UTF16PtrFromString("TaskbarCreated") |  | ||||||
| 	// https://msdn.microsoft.com/en-us/library/windows/desktop/ms644947 |  | ||||||
| 	res, _, err := pRegisterWindowMessage.Call( |  | ||||||
| 		uintptr(unsafe.Pointer(taskbarEventNamePtr)), |  | ||||||
| 	) |  | ||||||
| 	if res == 0 { // success 0xc000-0xfff |  | ||||||
| 		return fmt.Errorf("failed to register window: %w", err) |  | ||||||
| 	} |  | ||||||
| 	t.wmTaskbarCreated = uint32(res) |  | ||||||
|  |  | ||||||
| 	instanceHandle, _, err := pGetModuleHandle.Call(0) |  | ||||||
| 	if instanceHandle == 0 { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
| 	t.instance = windows.Handle(instanceHandle) |  | ||||||
|  |  | ||||||
| 	// https://msdn.microsoft.com/en-us/library/windows/desktop/ms648072(v=vs.85).aspx |  | ||||||
| 	iconHandle, _, err := pLoadIcon.Call(0, uintptr(IDI_APPLICATION)) |  | ||||||
| 	if iconHandle == 0 { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
| 	t.icon = windows.Handle(iconHandle) |  | ||||||
|  |  | ||||||
| 	// https://msdn.microsoft.com/en-us/library/windows/desktop/ms648391(v=vs.85).aspx |  | ||||||
| 	cursorHandle, _, err := pLoadCursor.Call(0, uintptr(IDC_ARROW)) |  | ||||||
| 	if cursorHandle == 0 { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
| 	t.cursor = windows.Handle(cursorHandle) |  | ||||||
|  |  | ||||||
| 	classNamePtr, err := windows.UTF16PtrFromString(className) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	windowNamePtr, err := windows.UTF16PtrFromString(windowName) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	t.wcex = &wndClassEx{ |  | ||||||
| 		Style:      CS_HREDRAW | CS_VREDRAW, |  | ||||||
| 		WndProc:    windows.NewCallback(t.wndProc), |  | ||||||
| 		Instance:   t.instance, |  | ||||||
| 		Icon:       t.icon, |  | ||||||
| 		Cursor:     t.cursor, |  | ||||||
| 		Background: windows.Handle(6), // (COLOR_WINDOW + 1) |  | ||||||
| 		ClassName:  classNamePtr, |  | ||||||
| 		IconSm:     t.icon, |  | ||||||
| 	} |  | ||||||
| 	if err := t.wcex.register(); err != nil { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	windowHandle, _, err := pCreateWindowEx.Call( |  | ||||||
| 		uintptr(0), |  | ||||||
| 		uintptr(unsafe.Pointer(classNamePtr)), |  | ||||||
| 		uintptr(unsafe.Pointer(windowNamePtr)), |  | ||||||
| 		uintptr(WS_OVERLAPPEDWINDOW), |  | ||||||
| 		uintptr(CW_USEDEFAULT), |  | ||||||
| 		uintptr(CW_USEDEFAULT), |  | ||||||
| 		uintptr(CW_USEDEFAULT), |  | ||||||
| 		uintptr(CW_USEDEFAULT), |  | ||||||
| 		uintptr(0), |  | ||||||
| 		uintptr(0), |  | ||||||
| 		uintptr(t.instance), |  | ||||||
| 		uintptr(0), |  | ||||||
| 	) |  | ||||||
| 	if windowHandle == 0 { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
| 	t.window = windows.Handle(windowHandle) |  | ||||||
|  |  | ||||||
| 	pShowWindow.Call(uintptr(t.window), uintptr(SW_HIDE)) //nolint:errcheck |  | ||||||
|  |  | ||||||
| 	boolRet, _, err := pUpdateWindow.Call(uintptr(t.window)) |  | ||||||
| 	if boolRet == 0 { |  | ||||||
| 		slog.Error(fmt.Sprintf("failed to update window: %s", err)) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	t.muNID.Lock() |  | ||||||
| 	defer t.muNID.Unlock() |  | ||||||
| 	t.nid = ¬ifyIconData{ |  | ||||||
| 		Wnd:             t.window, |  | ||||||
| 		ID:              100, |  | ||||||
| 		Flags:           NIF_MESSAGE, |  | ||||||
| 		CallbackMessage: t.wmSystrayMessage, |  | ||||||
| 	} |  | ||||||
| 	t.nid.Size = uint32(unsafe.Sizeof(*t.nid)) |  | ||||||
|  |  | ||||||
| 	return t.nid.add() |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (t *winTray) createMenu() error { |  | ||||||
| 	menuHandle, _, err := pCreatePopupMenu.Call() |  | ||||||
| 	if menuHandle == 0 { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
| 	t.menus[0] = windows.Handle(menuHandle) |  | ||||||
|  |  | ||||||
| 	// https://msdn.microsoft.com/en-us/library/windows/desktop/ms647575(v=vs.85).aspx |  | ||||||
| 	mi := struct { |  | ||||||
| 		Size, Mask, Style, Max uint32 |  | ||||||
| 		Background             windows.Handle |  | ||||||
| 		ContextHelpID          uint32 |  | ||||||
| 		MenuData               uintptr |  | ||||||
| 	}{ |  | ||||||
| 		Mask: MIM_APPLYTOSUBMENUS, |  | ||||||
| 	} |  | ||||||
| 	mi.Size = uint32(unsafe.Sizeof(mi)) |  | ||||||
|  |  | ||||||
| 	res, _, err := pSetMenuInfo.Call( |  | ||||||
| 		uintptr(t.menus[0]), |  | ||||||
| 		uintptr(unsafe.Pointer(&mi)), |  | ||||||
| 	) |  | ||||||
| 	if res == 0 { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
| 	return nil |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // Contains information about a menu item. |  | ||||||
| // https://msdn.microsoft.com/en-us/library/windows/desktop/ms647578(v=vs.85).aspx |  | ||||||
| type menuItemInfo struct { |  | ||||||
| 	Size, Mask, Type, State     uint32 |  | ||||||
| 	ID                          uint32 |  | ||||||
| 	SubMenu, Checked, Unchecked windows.Handle |  | ||||||
| 	ItemData                    uintptr |  | ||||||
| 	TypeData                    *uint16 |  | ||||||
| 	Cch                         uint32 |  | ||||||
| 	BMPItem                     windows.Handle |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (t *winTray) addOrUpdateMenuItem(menuItemId uint32, parentId uint32, title string, disabled bool) error { |  | ||||||
| 	titlePtr, err := windows.UTF16PtrFromString(title) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	mi := menuItemInfo{ |  | ||||||
| 		Mask:     MIIM_FTYPE | MIIM_STRING | MIIM_ID | MIIM_STATE, |  | ||||||
| 		Type:     MFT_STRING, |  | ||||||
| 		ID:       menuItemId, |  | ||||||
| 		TypeData: titlePtr, |  | ||||||
| 		Cch:      uint32(len(title)), |  | ||||||
| 	} |  | ||||||
| 	mi.Size = uint32(unsafe.Sizeof(mi)) |  | ||||||
| 	if disabled { |  | ||||||
| 		mi.State |= MFS_DISABLED |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	var res uintptr |  | ||||||
| 	t.muMenus.RLock() |  | ||||||
| 	menu := t.menus[parentId] |  | ||||||
| 	t.muMenus.RUnlock() |  | ||||||
| 	if t.getVisibleItemIndex(parentId, menuItemId) != -1 { |  | ||||||
| 		// We set the menu item info based on the menuID |  | ||||||
| 		boolRet, _, err := pSetMenuItemInfo.Call( |  | ||||||
| 			uintptr(menu), |  | ||||||
| 			uintptr(menuItemId), |  | ||||||
| 			0, |  | ||||||
| 			uintptr(unsafe.Pointer(&mi)), |  | ||||||
| 		) |  | ||||||
| 		if boolRet == 0 { |  | ||||||
| 			return fmt.Errorf("failed to set menu item: %w", err) |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	if res == 0 { |  | ||||||
| 		// Menu item does not already exist, create it |  | ||||||
| 		t.muMenus.RLock() |  | ||||||
| 		submenu, exists := t.menus[menuItemId] |  | ||||||
| 		t.muMenus.RUnlock() |  | ||||||
| 		if exists { |  | ||||||
| 			mi.Mask |= MIIM_SUBMENU |  | ||||||
| 			mi.SubMenu = submenu |  | ||||||
| 		} |  | ||||||
| 		t.addToVisibleItems(parentId, menuItemId) |  | ||||||
| 		position := t.getVisibleItemIndex(parentId, menuItemId) |  | ||||||
| 		res, _, err = pInsertMenuItem.Call( |  | ||||||
| 			uintptr(menu), |  | ||||||
| 			uintptr(position), |  | ||||||
| 			1, |  | ||||||
| 			uintptr(unsafe.Pointer(&mi)), |  | ||||||
| 		) |  | ||||||
| 		if res == 0 { |  | ||||||
| 			t.delFromVisibleItems(parentId, menuItemId) |  | ||||||
| 			return err |  | ||||||
| 		} |  | ||||||
| 		t.muMenuOf.Lock() |  | ||||||
| 		t.menuOf[menuItemId] = menu |  | ||||||
| 		t.muMenuOf.Unlock() |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	return nil |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (t *winTray) addSeparatorMenuItem(menuItemId, parentId uint32) error { |  | ||||||
| 	mi := menuItemInfo{ |  | ||||||
| 		Mask: MIIM_FTYPE | MIIM_ID | MIIM_STATE, |  | ||||||
| 		Type: MFT_SEPARATOR, |  | ||||||
| 		ID:   menuItemId, |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	mi.Size = uint32(unsafe.Sizeof(mi)) |  | ||||||
|  |  | ||||||
| 	t.addToVisibleItems(parentId, menuItemId) |  | ||||||
| 	position := t.getVisibleItemIndex(parentId, menuItemId) |  | ||||||
| 	t.muMenus.RLock() |  | ||||||
| 	menu := uintptr(t.menus[parentId]) |  | ||||||
| 	t.muMenus.RUnlock() |  | ||||||
| 	res, _, err := pInsertMenuItem.Call( |  | ||||||
| 		menu, |  | ||||||
| 		uintptr(position), |  | ||||||
| 		1, |  | ||||||
| 		uintptr(unsafe.Pointer(&mi)), |  | ||||||
| 	) |  | ||||||
| 	if res == 0 { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	return nil |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // func (t *winTray) hideMenuItem(menuItemId, parentId uint32) error { |  | ||||||
| // 	const ERROR_SUCCESS syscall.Errno = 0 |  | ||||||
|  |  | ||||||
| // 	t.muMenus.RLock() |  | ||||||
| // 	menu := uintptr(t.menus[parentId]) |  | ||||||
| // 	t.muMenus.RUnlock() |  | ||||||
| // 	res, _, err := pRemoveMenu.Call( |  | ||||||
| // 		menu, |  | ||||||
| // 		uintptr(menuItemId), |  | ||||||
| // 		MF_BYCOMMAND, |  | ||||||
| // 	) |  | ||||||
| // 	if res == 0 && err.(syscall.Errno) != ERROR_SUCCESS { |  | ||||||
| // 		return err |  | ||||||
| // 	} |  | ||||||
| // 	t.delFromVisibleItems(parentId, menuItemId) |  | ||||||
|  |  | ||||||
| // 	return nil |  | ||||||
| // } |  | ||||||
|  |  | ||||||
| func (t *winTray) showMenu() error { |  | ||||||
| 	p := point{} |  | ||||||
| 	boolRet, _, err := pGetCursorPos.Call(uintptr(unsafe.Pointer(&p))) |  | ||||||
| 	if boolRet == 0 { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
| 	boolRet, _, err = pSetForegroundWindow.Call(uintptr(t.window)) |  | ||||||
| 	if boolRet == 0 { |  | ||||||
| 		slog.Warn(fmt.Sprintf("failed to bring menu to foreground: %s", err)) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	boolRet, _, err = pTrackPopupMenu.Call( |  | ||||||
| 		uintptr(t.menus[0]), |  | ||||||
| 		TPM_BOTTOMALIGN|TPM_LEFTALIGN, |  | ||||||
| 		uintptr(p.X), |  | ||||||
| 		uintptr(p.Y), |  | ||||||
| 		0, |  | ||||||
| 		uintptr(t.window), |  | ||||||
| 		0, |  | ||||||
| 	) |  | ||||||
| 	if boolRet == 0 { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	return nil |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (t *winTray) delFromVisibleItems(parent, val uint32) { |  | ||||||
| 	t.muVisibleItems.Lock() |  | ||||||
| 	defer t.muVisibleItems.Unlock() |  | ||||||
| 	visibleItems := t.visibleItems[parent] |  | ||||||
| 	for i, itemval := range visibleItems { |  | ||||||
| 		if val == itemval { |  | ||||||
| 			t.visibleItems[parent] = append(visibleItems[:i], visibleItems[i+1:]...) |  | ||||||
| 			break |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (t *winTray) addToVisibleItems(parent, val uint32) { |  | ||||||
| 	t.muVisibleItems.Lock() |  | ||||||
| 	defer t.muVisibleItems.Unlock() |  | ||||||
| 	if visibleItems, exists := t.visibleItems[parent]; !exists { |  | ||||||
| 		t.visibleItems[parent] = []uint32{val} |  | ||||||
| 	} else { |  | ||||||
| 		newvisible := append(visibleItems, val) |  | ||||||
| 		sort.Slice(newvisible, func(i, j int) bool { return newvisible[i] < newvisible[j] }) |  | ||||||
| 		t.visibleItems[parent] = newvisible |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (t *winTray) getVisibleItemIndex(parent, val uint32) int { |  | ||||||
| 	t.muVisibleItems.RLock() |  | ||||||
| 	defer t.muVisibleItems.RUnlock() |  | ||||||
| 	for i, itemval := range t.visibleItems[parent] { |  | ||||||
| 		if val == itemval { |  | ||||||
| 			return i |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| 	return -1 |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func iconBytesToFilePath(iconBytes []byte) (string, error) { |  | ||||||
| 	bh := md5.Sum(iconBytes) |  | ||||||
| 	dataHash := hex.EncodeToString(bh[:]) |  | ||||||
| 	iconFilePath := filepath.Join(os.TempDir(), "ollama_temp_icon_"+dataHash) |  | ||||||
|  |  | ||||||
| 	if _, err := os.Stat(iconFilePath); os.IsNotExist(err) { |  | ||||||
| 		if err := os.WriteFile(iconFilePath, iconBytes, 0o644); err != nil { |  | ||||||
| 			return "", err |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| 	return iconFilePath, nil |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // Loads an image from file and shows it in tray. |  | ||||||
| // Shell_NotifyIcon: https://msdn.microsoft.com/en-us/library/windows/desktop/bb762159(v=vs.85).aspx |  | ||||||
| func (t *winTray) setIcon(src string) error { |  | ||||||
| 	h, err := t.loadIconFrom(src) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	t.muNID.Lock() |  | ||||||
| 	defer t.muNID.Unlock() |  | ||||||
| 	t.nid.Icon = h |  | ||||||
| 	t.nid.Flags |= NIF_ICON | NIF_TIP |  | ||||||
| 	if toolTipUTF16, err := syscall.UTF16FromString(commontray.ToolTip); err == nil { |  | ||||||
| 		copy(t.nid.Tip[:], toolTipUTF16) |  | ||||||
| 	} else { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
| 	t.nid.Size = uint32(unsafe.Sizeof(*t.nid)) |  | ||||||
|  |  | ||||||
| 	return t.nid.modify() |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // Loads an image from file to be shown in tray or menu item. |  | ||||||
| // LoadImage: https://msdn.microsoft.com/en-us/library/windows/desktop/ms648045(v=vs.85).aspx |  | ||||||
| func (t *winTray) loadIconFrom(src string) (windows.Handle, error) { |  | ||||||
| 	// Save and reuse handles of loaded images |  | ||||||
| 	t.muLoadedImages.RLock() |  | ||||||
| 	h, ok := t.loadedImages[src] |  | ||||||
| 	t.muLoadedImages.RUnlock() |  | ||||||
| 	if !ok { |  | ||||||
| 		srcPtr, err := windows.UTF16PtrFromString(src) |  | ||||||
| 		if err != nil { |  | ||||||
| 			return 0, err |  | ||||||
| 		} |  | ||||||
| 		res, _, err := pLoadImage.Call( |  | ||||||
| 			0, |  | ||||||
| 			uintptr(unsafe.Pointer(srcPtr)), |  | ||||||
| 			IMAGE_ICON, |  | ||||||
| 			0, |  | ||||||
| 			0, |  | ||||||
| 			LR_LOADFROMFILE|LR_DEFAULTSIZE, |  | ||||||
| 		) |  | ||||||
| 		if res == 0 { |  | ||||||
| 			return 0, err |  | ||||||
| 		} |  | ||||||
| 		h = windows.Handle(res) |  | ||||||
| 		t.muLoadedImages.Lock() |  | ||||||
| 		t.loadedImages[src] = h |  | ||||||
| 		t.muLoadedImages.Unlock() |  | ||||||
| 	} |  | ||||||
| 	return h, nil |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (t *winTray) DisplayFirstUseNotification() error { |  | ||||||
| 	t.muNID.Lock() |  | ||||||
| 	defer t.muNID.Unlock() |  | ||||||
| 	copy(t.nid.InfoTitle[:], windows.StringToUTF16(firstTimeTitle)) |  | ||||||
| 	copy(t.nid.Info[:], windows.StringToUTF16(firstTimeMessage)) |  | ||||||
| 	t.nid.Flags |= NIF_INFO |  | ||||||
| 	t.nid.Size = uint32(unsafe.Sizeof(*wt.nid)) |  | ||||||
|  |  | ||||||
| 	return t.nid.modify() |  | ||||||
| } |  | ||||||
| @@ -1,90 +0,0 @@ | |||||||
| //go:build windows |  | ||||||
|  |  | ||||||
| package wintray |  | ||||||
|  |  | ||||||
| import ( |  | ||||||
| 	"runtime" |  | ||||||
|  |  | ||||||
| 	"golang.org/x/sys/windows" |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| var ( |  | ||||||
| 	k32 = windows.NewLazySystemDLL("Kernel32.dll") |  | ||||||
| 	u32 = windows.NewLazySystemDLL("User32.dll") |  | ||||||
| 	s32 = windows.NewLazySystemDLL("Shell32.dll") |  | ||||||
|  |  | ||||||
| 	pCreatePopupMenu       = u32.NewProc("CreatePopupMenu") |  | ||||||
| 	pCreateWindowEx        = u32.NewProc("CreateWindowExW") |  | ||||||
| 	pDefWindowProc         = u32.NewProc("DefWindowProcW") |  | ||||||
| 	pDestroyWindow         = u32.NewProc("DestroyWindow") |  | ||||||
| 	pDispatchMessage       = u32.NewProc("DispatchMessageW") |  | ||||||
| 	pGetCursorPos          = u32.NewProc("GetCursorPos") |  | ||||||
| 	pGetMessage            = u32.NewProc("GetMessageW") |  | ||||||
| 	pGetModuleHandle       = k32.NewProc("GetModuleHandleW") |  | ||||||
| 	pInsertMenuItem        = u32.NewProc("InsertMenuItemW") |  | ||||||
| 	pLoadCursor            = u32.NewProc("LoadCursorW") |  | ||||||
| 	pLoadIcon              = u32.NewProc("LoadIconW") |  | ||||||
| 	pLoadImage             = u32.NewProc("LoadImageW") |  | ||||||
| 	pPostMessage           = u32.NewProc("PostMessageW") |  | ||||||
| 	pPostQuitMessage       = u32.NewProc("PostQuitMessage") |  | ||||||
| 	pRegisterClass         = u32.NewProc("RegisterClassExW") |  | ||||||
| 	pRegisterWindowMessage = u32.NewProc("RegisterWindowMessageW") |  | ||||||
| 	pSetForegroundWindow   = u32.NewProc("SetForegroundWindow") |  | ||||||
| 	pSetMenuInfo           = u32.NewProc("SetMenuInfo") |  | ||||||
| 	pSetMenuItemInfo       = u32.NewProc("SetMenuItemInfoW") |  | ||||||
| 	pShellNotifyIcon       = s32.NewProc("Shell_NotifyIconW") |  | ||||||
| 	pShowWindow            = u32.NewProc("ShowWindow") |  | ||||||
| 	pTrackPopupMenu        = u32.NewProc("TrackPopupMenu") |  | ||||||
| 	pTranslateMessage      = u32.NewProc("TranslateMessage") |  | ||||||
| 	pUnregisterClass       = u32.NewProc("UnregisterClassW") |  | ||||||
| 	pUpdateWindow          = u32.NewProc("UpdateWindow") |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| const ( |  | ||||||
| 	CS_HREDRAW          = 0x0002 |  | ||||||
| 	CS_VREDRAW          = 0x0001 |  | ||||||
| 	CW_USEDEFAULT       = 0x80000000 |  | ||||||
| 	IDC_ARROW           = 32512 // Standard arrow |  | ||||||
| 	IDI_APPLICATION     = 32512 |  | ||||||
| 	IMAGE_ICON          = 1          // Loads an icon |  | ||||||
| 	LR_DEFAULTSIZE      = 0x00000040 // Loads default-size icon for windows(SM_CXICON x SM_CYICON) if cx, cy are set to zero |  | ||||||
| 	LR_LOADFROMFILE     = 0x00000010 // Loads the stand-alone image from the file |  | ||||||
| 	MF_BYCOMMAND        = 0x00000000 |  | ||||||
| 	MFS_DISABLED        = 0x00000003 |  | ||||||
| 	MFT_SEPARATOR       = 0x00000800 |  | ||||||
| 	MFT_STRING          = 0x00000000 |  | ||||||
| 	MIIM_BITMAP         = 0x00000080 |  | ||||||
| 	MIIM_FTYPE          = 0x00000100 |  | ||||||
| 	MIIM_ID             = 0x00000002 |  | ||||||
| 	MIIM_STATE          = 0x00000001 |  | ||||||
| 	MIIM_STRING         = 0x00000040 |  | ||||||
| 	MIIM_SUBMENU        = 0x00000004 |  | ||||||
| 	MIM_APPLYTOSUBMENUS = 0x80000000 |  | ||||||
| 	NIF_ICON            = 0x00000002 |  | ||||||
| 	NIF_TIP             = 0x00000004 |  | ||||||
| 	NIF_INFO            = 0x00000010 |  | ||||||
| 	NIF_MESSAGE         = 0x00000001 |  | ||||||
| 	SW_HIDE             = 0 |  | ||||||
| 	TPM_BOTTOMALIGN     = 0x0020 |  | ||||||
| 	TPM_LEFTALIGN       = 0x0000 |  | ||||||
| 	WM_CLOSE            = 0x0010 |  | ||||||
| 	WM_USER             = 0x0400 |  | ||||||
| 	WS_CAPTION          = 0x00C00000 |  | ||||||
| 	WS_MAXIMIZEBOX      = 0x00010000 |  | ||||||
| 	WS_MINIMIZEBOX      = 0x00020000 |  | ||||||
| 	WS_OVERLAPPED       = 0x00000000 |  | ||||||
| 	WS_OVERLAPPEDWINDOW = WS_OVERLAPPED | WS_CAPTION | WS_SYSMENU | WS_THICKFRAME | WS_MINIMIZEBOX | WS_MAXIMIZEBOX |  | ||||||
| 	WS_SYSMENU          = 0x00080000 |  | ||||||
| 	WS_THICKFRAME       = 0x00040000 |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| // Not sure if this is actually needed on windows |  | ||||||
| func init() { |  | ||||||
| 	runtime.LockOSThread() |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // The POINT structure defines the x- and y- coordinates of a point. |  | ||||||
| // https://msdn.microsoft.com/en-us/library/windows/desktop/dd162805(v=vs.85).aspx |  | ||||||
| type point struct { |  | ||||||
| 	X, Y int32 |  | ||||||
| } |  | ||||||
| @@ -1,45 +0,0 @@ | |||||||
| //go:build windows |  | ||||||
|  |  | ||||||
| package wintray |  | ||||||
|  |  | ||||||
| import ( |  | ||||||
| 	"unsafe" |  | ||||||
|  |  | ||||||
| 	"golang.org/x/sys/windows" |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| // Contains window class information. |  | ||||||
| // It is used with the RegisterClassEx and GetClassInfoEx functions. |  | ||||||
| // https://msdn.microsoft.com/en-us/library/ms633577.aspx |  | ||||||
| type wndClassEx struct { |  | ||||||
| 	Size, Style                        uint32 |  | ||||||
| 	WndProc                            uintptr |  | ||||||
| 	ClsExtra, WndExtra                 int32 |  | ||||||
| 	Instance, Icon, Cursor, Background windows.Handle |  | ||||||
| 	MenuName, ClassName                *uint16 |  | ||||||
| 	IconSm                             windows.Handle |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // Registers a window class for subsequent use in calls to the CreateWindow or CreateWindowEx function. |  | ||||||
| // https://msdn.microsoft.com/en-us/library/ms633587.aspx |  | ||||||
| func (w *wndClassEx) register() error { |  | ||||||
| 	w.Size = uint32(unsafe.Sizeof(*w)) |  | ||||||
| 	res, _, err := pRegisterClass.Call(uintptr(unsafe.Pointer(w))) |  | ||||||
| 	if res == 0 { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
| 	return nil |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // Unregisters a window class, freeing the memory required for the class. |  | ||||||
| // https://msdn.microsoft.com/en-us/library/ms644899.aspx |  | ||||||
| func (w *wndClassEx) unregister() error { |  | ||||||
| 	res, _, err := pUnregisterClass.Call( |  | ||||||
| 		uintptr(unsafe.Pointer(w.ClassName)), |  | ||||||
| 		uintptr(w.Instance), |  | ||||||
| 	) |  | ||||||
| 	if res == 0 { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
| 	return nil |  | ||||||
| } |  | ||||||
							
								
								
									
										92
									
								
								auth/auth.go
									
									
									
									
									
								
							
							
						
						| @@ -1,92 +0,0 @@ | |||||||
| package auth |  | ||||||
|  |  | ||||||
| import ( |  | ||||||
| 	"bytes" |  | ||||||
| 	"context" |  | ||||||
| 	"crypto/rand" |  | ||||||
| 	"encoding/base64" |  | ||||||
| 	"errors" |  | ||||||
| 	"fmt" |  | ||||||
| 	"io" |  | ||||||
| 	"log/slog" |  | ||||||
| 	"os" |  | ||||||
| 	"path/filepath" |  | ||||||
| 	"strings" |  | ||||||
|  |  | ||||||
| 	"golang.org/x/crypto/ssh" |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| const defaultPrivateKey = "id_ed25519" |  | ||||||
|  |  | ||||||
| func keyPath() (string, error) { |  | ||||||
| 	home, err := os.UserHomeDir() |  | ||||||
| 	if err != nil { |  | ||||||
| 		return "", err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	return filepath.Join(home, ".ollama", defaultPrivateKey), nil |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func GetPublicKey() (string, error) { |  | ||||||
| 	keyPath, err := keyPath() |  | ||||||
| 	if err != nil { |  | ||||||
| 		return "", err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	privateKeyFile, err := os.ReadFile(keyPath) |  | ||||||
| 	if err != nil { |  | ||||||
| 		slog.Info(fmt.Sprintf("Failed to load private key: %v", err)) |  | ||||||
| 		return "", err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	privateKey, err := ssh.ParsePrivateKey(privateKeyFile) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return "", err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	publicKey := ssh.MarshalAuthorizedKey(privateKey.PublicKey()) |  | ||||||
|  |  | ||||||
| 	return strings.TrimSpace(string(publicKey)), nil |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func NewNonce(r io.Reader, length int) (string, error) { |  | ||||||
| 	nonce := make([]byte, length) |  | ||||||
| 	if _, err := io.ReadFull(r, nonce); err != nil { |  | ||||||
| 		return "", err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	return base64.RawURLEncoding.EncodeToString(nonce), nil |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func Sign(ctx context.Context, bts []byte) (string, error) { |  | ||||||
| 	keyPath, err := keyPath() |  | ||||||
| 	if err != nil { |  | ||||||
| 		return "", err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	privateKeyFile, err := os.ReadFile(keyPath) |  | ||||||
| 	if err != nil { |  | ||||||
| 		slog.Info(fmt.Sprintf("Failed to load private key: %v", err)) |  | ||||||
| 		return "", err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	privateKey, err := ssh.ParsePrivateKey(privateKeyFile) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return "", err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	// get the pubkey, but remove the type |  | ||||||
| 	publicKey := ssh.MarshalAuthorizedKey(privateKey.PublicKey()) |  | ||||||
| 	parts := bytes.Split(publicKey, []byte(" ")) |  | ||||||
| 	if len(parts) < 2 { |  | ||||||
| 		return "", errors.New("malformed public key") |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	signedData, err := privateKey.Sign(rand.Reader, bts) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return "", err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	// signature is <pubkey>:<signature> |  | ||||||
| 	return fmt.Sprintf("%s:%s", bytes.TrimSpace(parts[1]), base64.StdEncoding.EncodeToString(signedData.Blob)), nil |  | ||||||
| } |  | ||||||
| @@ -1 +0,0 @@ | |||||||
| This is here to make sure the build/ directory exists for the go:embed command |  | ||||||
| @@ -1 +0,0 @@ | |||||||
| This is here to make sure the build/ directory exists for the go:embed command |  | ||||||
| @@ -1,8 +0,0 @@ | |||||||
| package build |  | ||||||
|  |  | ||||||
| import "embed" |  | ||||||
|  |  | ||||||
| // Darwin payloads separated by architecture to avoid duplicate payloads when cross compiling |  | ||||||
|  |  | ||||||
| //go:embed darwin/amd64/* |  | ||||||
| var EmbedFS embed.FS |  | ||||||
| @@ -1,8 +0,0 @@ | |||||||
| package build |  | ||||||
|  |  | ||||||
| import "embed" |  | ||||||
|  |  | ||||||
| // Darwin payloads separated by architecture to avoid duplicate payloads when cross compiling |  | ||||||
|  |  | ||||||
| //go:embed darwin/arm64/* |  | ||||||
| var EmbedFS embed.FS |  | ||||||
| @@ -1,6 +0,0 @@ | |||||||
| package build |  | ||||||
|  |  | ||||||
| import "embed" |  | ||||||
|  |  | ||||||
| //go:embed linux/* |  | ||||||
| var EmbedFS embed.FS |  | ||||||
| @@ -1,8 +0,0 @@ | |||||||
| //go:build !linux && !darwin |  | ||||||
|  |  | ||||||
| package build |  | ||||||
|  |  | ||||||
| import "embed" |  | ||||||
|  |  | ||||||
| // unused on windows |  | ||||||
| var EmbedFS embed.FS |  | ||||||
| @@ -1 +0,0 @@ | |||||||
| This is here to make sure the build/ directory exists for the go:embed command |  | ||||||
| @@ -1 +0,0 @@ | |||||||
| This is here to make sure the build/ directory exists for the go:embed command |  | ||||||