windows: Support alt install paths, fit and finish (#6967)

* windows: Support alt install paths

Advanced users are leveraging innosetup's /DIR switch to target
an alternate location, but we get confused by things not existing in the LocalAppData dir.
This also hardens the server path lookup code for a future attempt to unify with a ./bin prefix

* Fit and finish improvements for windows app

Document alternate install location instructions for binaries and model.
Pop up progress UI for upgrades (automatic, with cancel button).
Expose non-default port in menu to disambiguate mutiple instances.
Set minimum Windows version to 10 22H2
This commit is contained in:
Daniel Hiltgen 2024-10-30 09:24:31 -07:00 committed by GitHub
parent db1842b9e1
commit 91dfbb1bba
No known key found for this signature in database
GPG key ID: B5690EEEBB952194
7 changed files with 59 additions and 19 deletions

View file

@ -11,10 +11,12 @@ import (
"github.com/ollama/ollama/app/store" "github.com/ollama/ollama/app/store"
"github.com/ollama/ollama/app/tray" "github.com/ollama/ollama/app/tray"
"github.com/ollama/ollama/envconfig"
) )
func Run() { func Run() {
InitLogging() InitLogging()
slog.Info("app config", "env", envconfig.Values())
ctx, cancel := context.WithCancel(context.Background()) ctx, cancel := context.WithCancel(context.Background())
var done chan int var done chan int

View file

@ -36,8 +36,13 @@ func init() {
ServerLogFile = filepath.Join(AppDataDir, "server.log") ServerLogFile = filepath.Join(AppDataDir, "server.log")
UpgradeLogFile = filepath.Join(AppDataDir, "upgrade.log") UpgradeLogFile = filepath.Join(AppDataDir, "upgrade.log")
// Executables are stored in APPDATA exe, err := os.Executable()
AppDir = filepath.Join(localAppData, "Programs", "Ollama") if err != nil {
slog.Warn("error discovering executable directory", "error", err)
AppDir = filepath.Join(localAppData, "Programs", "Ollama")
} else {
AppDir = filepath.Dir(exe)
}
// Make sure we have PATH set correctly for any spawned children // Make sure we have PATH set correctly for any spawned children
paths := strings.Split(os.Getenv("PATH"), ";") paths := strings.Split(os.Getenv("PATH"), ";")
@ -64,7 +69,7 @@ func init() {
} }
// Make sure our logging dir exists // Make sure our logging dir exists
_, err := os.Stat(AppDataDir) _, err = os.Stat(AppDataDir)
if errors.Is(err, os.ErrNotExist) { if errors.Is(err, os.ErrNotExist) {
if err := os.MkdirAll(AppDataDir, 0o755); err != nil { if err := os.MkdirAll(AppDataDir, 0o755); err != nil {
slog.Error(fmt.Sprintf("create ollama dir %s: %v", AppDataDir, err)) slog.Error(fmt.Sprintf("create ollama dir %s: %v", AppDataDir, err))

View file

@ -18,11 +18,17 @@ func getCLIFullPath(command string) string {
var cmdPath string var cmdPath string
appExe, err := os.Executable() appExe, err := os.Executable()
if err == nil { if err == nil {
// Check both the same location as the tray app, as well as ./bin
cmdPath = filepath.Join(filepath.Dir(appExe), command) cmdPath = filepath.Join(filepath.Dir(appExe), command)
_, err := os.Stat(cmdPath) _, err := os.Stat(cmdPath)
if err == nil { if err == nil {
return cmdPath return cmdPath
} }
cmdPath = filepath.Join(filepath.Dir(appExe), "bin", command)
_, err = os.Stat(cmdPath)
if err == nil {
return cmdPath
}
} }
cmdPath, err = exec.LookPath(command) cmdPath, err = exec.LookPath(command)
if err == nil { if err == nil {

View file

@ -26,19 +26,15 @@ func DoUpgrade(cancel context.CancelFunc, done chan int) error {
slog.Info("starting upgrade with " + installerExe) slog.Info("starting upgrade with " + installerExe)
slog.Info("upgrade log file " + UpgradeLogFile) slog.Info("upgrade log file " + UpgradeLogFile)
// When running in debug mode, we'll be "verbose" and let the installer pop up and prompt // make the upgrade show progress, but non interactive
installArgs := []string{ installArgs := []string{
"/CLOSEAPPLICATIONS", // Quit the tray app if it's still running "/CLOSEAPPLICATIONS", // Quit the tray app if it's still running
"/LOG=" + filepath.Base(UpgradeLogFile), // Only relative seems reliable, so set pwd "/LOG=" + filepath.Base(UpgradeLogFile), // Only relative seems reliable, so set pwd
"/FORCECLOSEAPPLICATIONS", // Force close the tray app - might be needed "/FORCECLOSEAPPLICATIONS", // Force close the tray app - might be needed
} "/SP", // Skip the "This will install... Do you wish to continue" prompt
// make the upgrade as quiet as possible (no GUI, no prompts) "/NOCANCEL", // Disable the ability to cancel upgrade mid-flight to avoid partially installed upgrades
installArgs = append(installArgs,
"/SP", // Skip the "This will install... Do you wish to continue" prompt
"/SUPPRESSMSGBOXES",
"/SILENT", "/SILENT",
"/VERYSILENT", }
)
// Safeguard in case we have requests in flight that need to drain... // Safeguard in case we have requests in flight that need to drain...
slog.Info("Waiting for server to shutdown") slog.Info("Waiting for server to shutdown")

View file

@ -53,8 +53,8 @@ RestartIfNeededByRun=no
; https://jrsoftware.org/ishelp/index.php?topic=setup_wizardimagefile ; https://jrsoftware.org/ishelp/index.php?topic=setup_wizardimagefile
WizardSmallImageFile=.\assets\setup.bmp WizardSmallImageFile=.\assets\setup.bmp
; TODO verifty actual min windows version... ; Ollama requires Windows 10 22H2 or newer for proper unicode rendering
; OG Win 10 ; TODO: consider setting this to 10.0.19045
MinVersion=10.0.10240 MinVersion=10.0.10240
; First release that supports WinRT UI Composition for win32 apps ; First release that supports WinRT UI Composition for win32 apps

View file

@ -11,12 +11,13 @@ import (
) )
const ( const (
updateAvailableMenuID = 1 _ = iota
updateMenuID = updateAvailableMenuID + 1 updateAvailableMenuID
separatorMenuID = updateMenuID + 1 updateMenuID
diagLogsMenuID = separatorMenuID + 1 separatorMenuID
diagSeparatorMenuID = diagLogsMenuID + 1 diagLogsMenuID
quitMenuID = diagSeparatorMenuID + 1 diagSeparatorMenuID
quitMenuID
) )
func (t *winTray) initMenus() error { func (t *winTray) initMenus() error {

View file

@ -25,6 +25,32 @@ Logs will often be helpful in diagnosing the problem (see
Ollama uses unicode characters for progress indication, which may render as unknown squares in some older terminal fonts in Windows 10. If you see this, try changing your terminal font settings. Ollama uses unicode characters for progress indication, which may render as unknown squares in some older terminal fonts in Windows 10. If you see this, try changing your terminal font settings.
## Filesystem Requirements
The Ollama install does not require Administrator, and installs in your home directory by default. You'll need at least 4GB of space for the binary install. Once you've installed Ollama, you'll need additional space for storing the Large Language models, which can be tens to hundreds of GB in size. If your home directory doesn't have enough space, you can change where the binaries are installed, and where the models are stored.
### Changing Install Location
To install the Ollama application in a location different than your home directory, start the installer with the following flag
```powershell
OllamaSetup.exe /DIR="d:\some\location"
```
### Changing Model Location
To change where Ollama stores the downloaded models instead of using your home directory, set the environment variable `OLLAMA_MODELS` in your user account.
1. Start the Settings (Windows 11) or Control Panel (Windows 10) application and search for _environment variables_.
2. Click on _Edit environment variables for your account_.
3. Edit or create a new variable for your user account for `OLLAMA_MODELS` where you want the models stored
4. Click OK/Apply to save.
If Ollama is already running, Quit the tray application and relaunch it from the Start menu, or a new terminal started after you saved the environment variables.
## API Access ## API Access
Here's a quick example showing API access from `powershell` Here's a quick example showing API access from `powershell`
@ -52,6 +78,10 @@ the explorer window by hitting `<cmd>+R` and type in:
The Ollama Windows installer registers an Uninstaller application. Under `Add or remove programs` in Windows Settings, you can uninstall Ollama. The Ollama Windows installer registers an Uninstaller application. Under `Add or remove programs` in Windows Settings, you can uninstall Ollama.
> [!NOTE]
> If you have [changed the OLLAMA_MODELS location](#changing-model-location), the installer will not remove your downloaded models
## Standalone CLI ## Standalone CLI
The easiest way to install Ollama on Windows is to use the `OllamaSetup.exe` The easiest way to install Ollama on Windows is to use the `OllamaSetup.exe`