package server import ( "bytes" "context" "crypto/rand" "crypto/sha256" "encoding/base64" "encoding/hex" "encoding/json" "fmt" "io" "log" "net/http" "os" "path" "strings" "time" "golang.org/x/crypto/ssh" "github.com/jmorganca/ollama/api" ) type AuthRedirect struct { Realm string Service string Scope string } type SignatureData struct { Method string Path string Data []byte } func generateNonce(length int) (string, error) { nonce := make([]byte, length) _, err := rand.Read(nonce) if err != nil { return "", err } return base64.RawURLEncoding.EncodeToString(nonce), nil } func (r AuthRedirect) URL() (string, error) { nonce, err := generateNonce(16) if err != nil { return "", err } scopes := []string{} for _, s := range strings.Split(r.Scope, " ") { scopes = append(scopes, fmt.Sprintf("scope=%s", s)) } scopeStr := strings.Join(scopes, "&") return fmt.Sprintf("%s?service=%s&%s&ts=%d&nonce=%s", r.Realm, r.Service, scopeStr, time.Now().Unix(), nonce), nil } func getAuthToken(ctx context.Context, redirData AuthRedirect, regOpts *RegistryOptions) (string, error) { url, err := redirData.URL() if err != nil { return "", err } home, err := os.UserHomeDir() if err != nil { return "", err } keyPath := path.Join(home, ".ollama", "id_ed25519") rawKey, err := os.ReadFile(keyPath) if err != nil { log.Printf("Failed to load private key: %v", err) return "", err } s := SignatureData{ Method: "GET", Path: url, Data: nil, } if !strings.HasPrefix(s.Path, "http") { if regOpts.Insecure { s.Path = "http://" + url } else { s.Path = "https://" + url } } sig, err := s.Sign(rawKey) if err != nil { return "", err } headers := make(http.Header) headers.Set("Authorization", sig) resp, err := makeRequest(ctx, "GET", url, headers, nil, regOpts) if err != nil { log.Printf("couldn't get token: %q", err) } defer resp.Body.Close() if resp.StatusCode != http.StatusOK { body, _ := io.ReadAll(resp.Body) return "", fmt.Errorf("on pull registry responded with code %d: %s", resp.StatusCode, body) } respBody, err := io.ReadAll(resp.Body) if err != nil { return "", err } var tok api.TokenResponse if err := json.Unmarshal(respBody, &tok); err != nil { return "", err } return tok.Token, nil } // Bytes returns a byte slice of the data to sign for the request func (s SignatureData) Bytes() []byte { // We first derive the content hash of the request body using: // base64(hex(sha256(request body))) hash := sha256.Sum256(s.Data) hashHex := make([]byte, hex.EncodedLen(len(hash))) hex.Encode(hashHex, hash[:]) contentHash := base64.StdEncoding.EncodeToString(hashHex) // We then put the entire request together in a serialize string using: // ",," // e.g. "GET,http://localhost,OTdkZjM1O..." return []byte(strings.Join([]string{s.Method, s.Path, contentHash}, ",")) } // SignData takes a SignatureData object and signs it with a raw private key func (s SignatureData) Sign(rawKey []byte) (string, error) { privateKey, err := ssh.ParseRawPrivateKey(rawKey) if err != nil { return "", err } signer, err := ssh.NewSignerFromKey(privateKey) if err != nil { return "", err } // get the pubkey, but remove the type pubKey := ssh.MarshalAuthorizedKey(signer.PublicKey()) parts := bytes.Split(pubKey, []byte(" ")) if len(parts) < 2 { return "", fmt.Errorf("malformed public key") } signedData, err := signer.Sign(nil, s.Bytes()) if err != nil { return "", err } // signature is : sig := fmt.Sprintf("%s:%s", bytes.TrimSpace(parts[1]), base64.StdEncoding.EncodeToString(signedData.Blob)) return sig, nil }