cmd.go 23 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018
  1. package cmd
  2. import (
  3. "bytes"
  4. "context"
  5. "crypto/ed25519"
  6. "crypto/rand"
  7. "crypto/sha256"
  8. "encoding/pem"
  9. "errors"
  10. "fmt"
  11. "io"
  12. "log"
  13. "net"
  14. "net/http"
  15. "os"
  16. "os/exec"
  17. "os/signal"
  18. "path/filepath"
  19. "runtime"
  20. "strings"
  21. "syscall"
  22. "time"
  23. "github.com/olekukonko/tablewriter"
  24. "github.com/spf13/cobra"
  25. "golang.org/x/crypto/ssh"
  26. "golang.org/x/term"
  27. "github.com/jmorganca/ollama/api"
  28. "github.com/jmorganca/ollama/format"
  29. "github.com/jmorganca/ollama/parser"
  30. "github.com/jmorganca/ollama/progress"
  31. "github.com/jmorganca/ollama/readline"
  32. "github.com/jmorganca/ollama/server"
  33. "github.com/jmorganca/ollama/version"
  34. )
  35. func CreateHandler(cmd *cobra.Command, args []string) error {
  36. filename, _ := cmd.Flags().GetString("file")
  37. filename, err := filepath.Abs(filename)
  38. if err != nil {
  39. return err
  40. }
  41. client, err := api.ClientFromEnvironment()
  42. if err != nil {
  43. return err
  44. }
  45. p := progress.NewProgress(os.Stderr)
  46. defer p.Stop()
  47. bars := make(map[string]*progress.Bar)
  48. status := "transferring context"
  49. spinner := progress.NewSpinner(status)
  50. p.Add(status, spinner)
  51. modelfile, err := os.ReadFile(filename)
  52. if err != nil {
  53. return err
  54. }
  55. commands, err := parser.Parse(bytes.NewReader(modelfile))
  56. if err != nil {
  57. return err
  58. }
  59. home, err := os.UserHomeDir()
  60. if err != nil {
  61. return err
  62. }
  63. for _, c := range commands {
  64. switch c.Name {
  65. case "model", "adapter":
  66. path := c.Args
  67. if path == "~" {
  68. path = home
  69. } else if strings.HasPrefix(path, "~/") {
  70. path = filepath.Join(home, path[2:])
  71. }
  72. bin, err := os.Open(path)
  73. if errors.Is(err, os.ErrNotExist) && c.Name == "model" {
  74. continue
  75. } else if err != nil {
  76. return err
  77. }
  78. defer bin.Close()
  79. hash := sha256.New()
  80. if _, err := io.Copy(hash, bin); err != nil {
  81. return err
  82. }
  83. bin.Seek(0, io.SeekStart)
  84. digest := fmt.Sprintf("sha256:%x", hash.Sum(nil))
  85. if err = client.CreateBlob(cmd.Context(), digest, bin); err != nil {
  86. return err
  87. }
  88. modelfile = bytes.ReplaceAll(modelfile, []byte(c.Args), []byte("@"+digest))
  89. }
  90. }
  91. fn := func(resp api.ProgressResponse) error {
  92. if resp.Digest != "" {
  93. spinner.Stop()
  94. bar, ok := bars[resp.Digest]
  95. if !ok {
  96. bar = progress.NewBar(resp.Status, resp.Total, resp.Completed)
  97. bars[resp.Digest] = bar
  98. p.Add(resp.Digest, bar)
  99. }
  100. bar.Set(resp.Completed)
  101. } else if status != resp.Status {
  102. spinner.Stop()
  103. status = resp.Status
  104. spinner = progress.NewSpinner(status)
  105. p.Add(status, spinner)
  106. }
  107. return nil
  108. }
  109. request := api.CreateRequest{Name: args[0], Path: filename, Modelfile: string(modelfile)}
  110. if err := client.Create(context.Background(), &request, fn); err != nil {
  111. return err
  112. }
  113. if spinner != nil {
  114. spinner.Stop()
  115. }
  116. return nil
  117. }
  118. func RunHandler(cmd *cobra.Command, args []string) error {
  119. client, err := api.ClientFromEnvironment()
  120. if err != nil {
  121. return err
  122. }
  123. name := args[0]
  124. // check if the model exists on the server
  125. _, err = client.Show(context.Background(), &api.ShowRequest{Name: name})
  126. var statusError api.StatusError
  127. switch {
  128. case errors.As(err, &statusError) && statusError.StatusCode == http.StatusNotFound:
  129. if err := PullHandler(cmd, args); err != nil {
  130. return err
  131. }
  132. case err != nil:
  133. return err
  134. }
  135. return RunGenerate(cmd, args)
  136. }
  137. func PushHandler(cmd *cobra.Command, args []string) error {
  138. client, err := api.ClientFromEnvironment()
  139. if err != nil {
  140. return err
  141. }
  142. insecure, err := cmd.Flags().GetBool("insecure")
  143. if err != nil {
  144. return err
  145. }
  146. p := progress.NewProgress(os.Stderr)
  147. defer p.Stop()
  148. bars := make(map[string]*progress.Bar)
  149. var status string
  150. var spinner *progress.Spinner
  151. fn := func(resp api.ProgressResponse) error {
  152. if resp.Digest != "" {
  153. if spinner != nil {
  154. spinner.Stop()
  155. spinner = nil
  156. }
  157. bar, ok := bars[resp.Digest]
  158. if !ok {
  159. bar = progress.NewBar(resp.Status, resp.Total, resp.Completed)
  160. bars[resp.Digest] = bar
  161. p.Add(resp.Digest, bar)
  162. }
  163. bar.Set(resp.Completed)
  164. } else if status != resp.Status {
  165. if spinner != nil {
  166. spinner.Stop()
  167. spinner = nil
  168. }
  169. status = resp.Status
  170. spinner = progress.NewSpinner(status)
  171. p.Add(status, spinner)
  172. }
  173. return nil
  174. }
  175. request := api.PushRequest{Name: args[0], Insecure: insecure}
  176. if err := client.Push(context.Background(), &request, fn); err != nil {
  177. return err
  178. }
  179. if spinner != nil {
  180. spinner.Stop()
  181. }
  182. return nil
  183. }
  184. func ListHandler(cmd *cobra.Command, args []string) error {
  185. client, err := api.ClientFromEnvironment()
  186. if err != nil {
  187. return err
  188. }
  189. models, err := client.List(context.Background())
  190. if err != nil {
  191. return err
  192. }
  193. var data [][]string
  194. for _, m := range models.Models {
  195. if len(args) == 0 || strings.HasPrefix(m.Name, args[0]) {
  196. data = append(data, []string{m.Name, m.Digest[:12], format.HumanBytes(m.Size), format.HumanTime(m.ModifiedAt, "Never")})
  197. }
  198. }
  199. table := tablewriter.NewWriter(os.Stdout)
  200. table.SetHeader([]string{"NAME", "ID", "SIZE", "MODIFIED"})
  201. table.SetHeaderAlignment(tablewriter.ALIGN_LEFT)
  202. table.SetAlignment(tablewriter.ALIGN_LEFT)
  203. table.SetHeaderLine(false)
  204. table.SetBorder(false)
  205. table.SetNoWhiteSpace(true)
  206. table.SetTablePadding("\t")
  207. table.AppendBulk(data)
  208. table.Render()
  209. return nil
  210. }
  211. func DeleteHandler(cmd *cobra.Command, args []string) error {
  212. client, err := api.ClientFromEnvironment()
  213. if err != nil {
  214. return err
  215. }
  216. for _, name := range args {
  217. req := api.DeleteRequest{Name: name}
  218. if err := client.Delete(context.Background(), &req); err != nil {
  219. return err
  220. }
  221. fmt.Printf("deleted '%s'\n", name)
  222. }
  223. return nil
  224. }
  225. func ShowHandler(cmd *cobra.Command, args []string) error {
  226. client, err := api.ClientFromEnvironment()
  227. if err != nil {
  228. return err
  229. }
  230. if len(args) != 1 {
  231. return errors.New("missing model name")
  232. }
  233. license, errLicense := cmd.Flags().GetBool("license")
  234. modelfile, errModelfile := cmd.Flags().GetBool("modelfile")
  235. parameters, errParams := cmd.Flags().GetBool("parameters")
  236. system, errSystem := cmd.Flags().GetBool("system")
  237. template, errTemplate := cmd.Flags().GetBool("template")
  238. for _, boolErr := range []error{errLicense, errModelfile, errParams, errSystem, errTemplate} {
  239. if boolErr != nil {
  240. return errors.New("error retrieving flags")
  241. }
  242. }
  243. flagsSet := 0
  244. showType := ""
  245. if license {
  246. flagsSet++
  247. showType = "license"
  248. }
  249. if modelfile {
  250. flagsSet++
  251. showType = "modelfile"
  252. }
  253. if parameters {
  254. flagsSet++
  255. showType = "parameters"
  256. }
  257. if system {
  258. flagsSet++
  259. showType = "system"
  260. }
  261. if template {
  262. flagsSet++
  263. showType = "template"
  264. }
  265. if flagsSet > 1 {
  266. return errors.New("only one of '--license', '--modelfile', '--parameters', '--system', or '--template' can be specified")
  267. } else if flagsSet == 0 {
  268. return errors.New("one of '--license', '--modelfile', '--parameters', '--system', or '--template' must be specified")
  269. }
  270. req := api.ShowRequest{Name: args[0]}
  271. resp, err := client.Show(context.Background(), &req)
  272. if err != nil {
  273. return err
  274. }
  275. switch showType {
  276. case "license":
  277. fmt.Println(resp.License)
  278. case "modelfile":
  279. fmt.Println(resp.Modelfile)
  280. case "parameters":
  281. fmt.Println(resp.Parameters)
  282. case "system":
  283. fmt.Println(resp.System)
  284. case "template":
  285. fmt.Println(resp.Template)
  286. }
  287. return nil
  288. }
  289. func CopyHandler(cmd *cobra.Command, args []string) error {
  290. client, err := api.ClientFromEnvironment()
  291. if err != nil {
  292. return err
  293. }
  294. req := api.CopyRequest{Source: args[0], Destination: args[1]}
  295. if err := client.Copy(context.Background(), &req); err != nil {
  296. return err
  297. }
  298. fmt.Printf("copied '%s' to '%s'\n", args[0], args[1])
  299. return nil
  300. }
  301. func PullHandler(cmd *cobra.Command, args []string) error {
  302. insecure, err := cmd.Flags().GetBool("insecure")
  303. if err != nil {
  304. return err
  305. }
  306. client, err := api.ClientFromEnvironment()
  307. if err != nil {
  308. return err
  309. }
  310. p := progress.NewProgress(os.Stderr)
  311. defer p.Stop()
  312. bars := make(map[string]*progress.Bar)
  313. var status string
  314. var spinner *progress.Spinner
  315. fn := func(resp api.ProgressResponse) error {
  316. if resp.Digest != "" {
  317. if spinner != nil {
  318. spinner.Stop()
  319. spinner = nil
  320. }
  321. bar, ok := bars[resp.Digest]
  322. if !ok {
  323. bar = progress.NewBar(resp.Status, resp.Total, resp.Completed)
  324. bars[resp.Digest] = bar
  325. p.Add(resp.Digest, bar)
  326. }
  327. bar.Set(resp.Completed)
  328. } else if status != resp.Status {
  329. if spinner != nil {
  330. spinner.Stop()
  331. spinner = nil
  332. }
  333. status = resp.Status
  334. spinner = progress.NewSpinner(status)
  335. p.Add(status, spinner)
  336. }
  337. return nil
  338. }
  339. request := api.PullRequest{Name: args[0], Insecure: insecure}
  340. if err := client.Pull(context.Background(), &request, fn); err != nil {
  341. return err
  342. }
  343. if spinner != nil {
  344. spinner.Stop()
  345. }
  346. return nil
  347. }
  348. func RunGenerate(cmd *cobra.Command, args []string) error {
  349. format, err := cmd.Flags().GetString("format")
  350. if err != nil {
  351. return err
  352. }
  353. prompts := args[1:]
  354. // prepend stdin to the prompt if provided
  355. if !term.IsTerminal(int(os.Stdin.Fd())) {
  356. in, err := io.ReadAll(os.Stdin)
  357. if err != nil {
  358. return err
  359. }
  360. prompts = append([]string{string(in)}, prompts...)
  361. }
  362. // output is being piped
  363. if !term.IsTerminal(int(os.Stdout.Fd())) {
  364. return generate(cmd, args[0], strings.Join(prompts, " "), false, format)
  365. }
  366. wordWrap := os.Getenv("TERM") == "xterm-256color"
  367. nowrap, err := cmd.Flags().GetBool("nowordwrap")
  368. if err != nil {
  369. return err
  370. }
  371. if nowrap {
  372. wordWrap = false
  373. }
  374. // prompts are provided via stdin or args so don't enter interactive mode
  375. if len(prompts) > 0 {
  376. return generate(cmd, args[0], strings.Join(prompts, " "), wordWrap, format)
  377. }
  378. return generateInteractive(cmd, args[0], wordWrap, format)
  379. }
  380. type generateContextKey string
  381. func generate(cmd *cobra.Command, model, prompt string, wordWrap bool, format string) error {
  382. client, err := api.ClientFromEnvironment()
  383. if err != nil {
  384. return err
  385. }
  386. var latest api.GenerateResponse
  387. generateContext, ok := cmd.Context().Value(generateContextKey("context")).([]int)
  388. if !ok {
  389. generateContext = []int{}
  390. }
  391. termWidth, _, err := term.GetSize(int(os.Stdout.Fd()))
  392. if err != nil {
  393. wordWrap = false
  394. }
  395. cancelCtx, cancel := context.WithCancel(context.Background())
  396. defer cancel()
  397. sigChan := make(chan os.Signal, 1)
  398. signal.Notify(sigChan, syscall.SIGINT)
  399. var abort bool
  400. go func() {
  401. <-sigChan
  402. cancel()
  403. abort = true
  404. }()
  405. var currentLineLength int
  406. var wordBuffer string
  407. request := api.GenerateRequest{Model: model, Prompt: prompt, Context: generateContext, Format: format}
  408. fn := func(response api.GenerateResponse) error {
  409. latest = response
  410. if wordWrap {
  411. for _, ch := range response.Response {
  412. if currentLineLength+1 > termWidth-5 {
  413. // backtrack the length of the last word and clear to the end of the line
  414. fmt.Printf("\x1b[%dD\x1b[K\n", len(wordBuffer))
  415. fmt.Printf("%s%c", wordBuffer, ch)
  416. currentLineLength = len(wordBuffer) + 1
  417. } else {
  418. fmt.Print(string(ch))
  419. currentLineLength += 1
  420. switch ch {
  421. case ' ':
  422. wordBuffer = ""
  423. case '\n':
  424. currentLineLength = 0
  425. default:
  426. wordBuffer += string(ch)
  427. }
  428. }
  429. }
  430. } else {
  431. fmt.Print(response.Response)
  432. }
  433. return nil
  434. }
  435. if err := client.Generate(cancelCtx, &request, fn); err != nil {
  436. if strings.Contains(err.Error(), "context canceled") && abort {
  437. return nil
  438. }
  439. return err
  440. }
  441. if prompt != "" {
  442. fmt.Println()
  443. fmt.Println()
  444. }
  445. if !latest.Done {
  446. if abort {
  447. return nil
  448. }
  449. return errors.New("unexpected end of response")
  450. }
  451. verbose, err := cmd.Flags().GetBool("verbose")
  452. if err != nil {
  453. return err
  454. }
  455. if verbose {
  456. latest.Summary()
  457. }
  458. ctx := cmd.Context()
  459. ctx = context.WithValue(ctx, generateContextKey("context"), latest.Context)
  460. cmd.SetContext(ctx)
  461. return nil
  462. }
  463. func generateInteractive(cmd *cobra.Command, model string, wordWrap bool, format string) error {
  464. // load the model
  465. if err := generate(cmd, model, "", false, ""); err != nil {
  466. return err
  467. }
  468. usage := func() {
  469. fmt.Fprintln(os.Stderr, "Available Commands:")
  470. fmt.Fprintln(os.Stderr, " /set Set session variables")
  471. fmt.Fprintln(os.Stderr, " /show Show model information")
  472. fmt.Fprintln(os.Stderr, " /bye Exit")
  473. fmt.Fprintln(os.Stderr, " /?, /help Help for a command")
  474. fmt.Fprintln(os.Stderr, "")
  475. fmt.Fprintln(os.Stderr, "Use \"\"\" to begin a multi-line message.")
  476. fmt.Fprintln(os.Stderr, "")
  477. }
  478. usageSet := func() {
  479. fmt.Fprintln(os.Stderr, "Available Commands:")
  480. fmt.Fprintln(os.Stderr, " /set history Enable history")
  481. fmt.Fprintln(os.Stderr, " /set nohistory Disable history")
  482. fmt.Fprintln(os.Stderr, " /set wordwrap Enable wordwrap")
  483. fmt.Fprintln(os.Stderr, " /set nowordwrap Disable wordwrap")
  484. fmt.Fprintln(os.Stderr, " /set format json Enable JSON mode")
  485. fmt.Fprintln(os.Stderr, " /set noformat Disable formatting")
  486. fmt.Fprintln(os.Stderr, " /set verbose Show LLM stats")
  487. fmt.Fprintln(os.Stderr, " /set quiet Disable LLM stats")
  488. fmt.Fprintln(os.Stderr, "")
  489. }
  490. usageShow := func() {
  491. fmt.Fprintln(os.Stderr, "Available Commands:")
  492. fmt.Fprintln(os.Stderr, " /show license Show model license")
  493. fmt.Fprintln(os.Stderr, " /show modelfile Show Modelfile for this model")
  494. fmt.Fprintln(os.Stderr, " /show parameters Show parameters for this model")
  495. fmt.Fprintln(os.Stderr, " /show system Show system prompt")
  496. fmt.Fprintln(os.Stderr, " /show template Show prompt template")
  497. fmt.Fprintln(os.Stderr, "")
  498. }
  499. prompt := readline.Prompt{
  500. Prompt: ">>> ",
  501. AltPrompt: "... ",
  502. Placeholder: "Send a message (/? for help)",
  503. AltPlaceholder: `Use """ to end multi-line input`,
  504. }
  505. scanner, err := readline.New(prompt)
  506. if err != nil {
  507. return err
  508. }
  509. fmt.Print(readline.StartBracketedPaste)
  510. defer fmt.Printf(readline.EndBracketedPaste)
  511. var multiLineBuffer string
  512. for {
  513. line, err := scanner.Readline()
  514. switch {
  515. case errors.Is(err, io.EOF):
  516. fmt.Println()
  517. return nil
  518. case errors.Is(err, readline.ErrInterrupt):
  519. if line == "" {
  520. fmt.Println("\nUse Ctrl-D or /bye to exit.")
  521. }
  522. continue
  523. case err != nil:
  524. return err
  525. }
  526. line = strings.TrimSpace(line)
  527. switch {
  528. case scanner.Prompt.UseAlt:
  529. if strings.HasSuffix(line, `"""`) {
  530. scanner.Prompt.UseAlt = false
  531. multiLineBuffer += strings.TrimSuffix(line, `"""`)
  532. line = multiLineBuffer
  533. multiLineBuffer = ""
  534. } else {
  535. multiLineBuffer += line + " "
  536. continue
  537. }
  538. case strings.HasPrefix(line, `"""`):
  539. scanner.Prompt.UseAlt = true
  540. multiLineBuffer = strings.TrimPrefix(line, `"""`) + " "
  541. continue
  542. case strings.HasPrefix(line, "/list"):
  543. args := strings.Fields(line)
  544. if err := ListHandler(cmd, args[1:]); err != nil {
  545. return err
  546. }
  547. case strings.HasPrefix(line, "/set"):
  548. args := strings.Fields(line)
  549. if len(args) > 1 {
  550. switch args[1] {
  551. case "history":
  552. scanner.HistoryEnable()
  553. case "nohistory":
  554. scanner.HistoryDisable()
  555. case "wordwrap":
  556. wordWrap = true
  557. fmt.Println("Set 'wordwrap' mode.")
  558. case "nowordwrap":
  559. wordWrap = false
  560. fmt.Println("Set 'nowordwrap' mode.")
  561. case "verbose":
  562. cmd.Flags().Set("verbose", "true")
  563. fmt.Println("Set 'verbose' mode.")
  564. case "quiet":
  565. cmd.Flags().Set("verbose", "false")
  566. fmt.Println("Set 'quiet' mode.")
  567. case "format":
  568. if len(args) < 3 || args[2] != "json" {
  569. fmt.Println("Invalid or missing format. For 'json' mode use '/set format json'")
  570. } else {
  571. format = args[2]
  572. fmt.Printf("Set format to '%s' mode.\n", args[2])
  573. }
  574. case "noformat":
  575. format = ""
  576. fmt.Println("Disabled format.")
  577. default:
  578. fmt.Printf("Unknown command '/set %s'. Type /? for help\n", args[1])
  579. }
  580. } else {
  581. usageSet()
  582. }
  583. case strings.HasPrefix(line, "/show"):
  584. args := strings.Fields(line)
  585. if len(args) > 1 {
  586. client, err := api.ClientFromEnvironment()
  587. if err != nil {
  588. fmt.Println("error: couldn't connect to ollama server")
  589. return err
  590. }
  591. resp, err := client.Show(cmd.Context(), &api.ShowRequest{Name: model})
  592. if err != nil {
  593. fmt.Println("error: couldn't get model")
  594. return err
  595. }
  596. switch args[1] {
  597. case "license":
  598. if resp.License == "" {
  599. fmt.Print("No license was specified for this model.\n\n")
  600. } else {
  601. fmt.Println(resp.License)
  602. }
  603. case "modelfile":
  604. fmt.Println(resp.Modelfile)
  605. case "parameters":
  606. if resp.Parameters == "" {
  607. fmt.Print("No parameters were specified for this model.\n\n")
  608. } else {
  609. fmt.Println(resp.Parameters)
  610. }
  611. case "system":
  612. if resp.System == "" {
  613. fmt.Print("No system prompt was specified for this model.\n\n")
  614. } else {
  615. fmt.Println(resp.System)
  616. }
  617. case "template":
  618. if resp.Template == "" {
  619. fmt.Print("No prompt template was specified for this model.\n\n")
  620. } else {
  621. fmt.Println(resp.Template)
  622. }
  623. default:
  624. fmt.Printf("Unknown command '/show %s'. Type /? for help\n", args[1])
  625. }
  626. } else {
  627. usageShow()
  628. }
  629. case strings.HasPrefix(line, "/help"), strings.HasPrefix(line, "/?"):
  630. args := strings.Fields(line)
  631. if len(args) > 1 {
  632. switch args[1] {
  633. case "set", "/set":
  634. usageSet()
  635. case "show", "/show":
  636. usageShow()
  637. }
  638. } else {
  639. usage()
  640. }
  641. case line == "/exit", line == "/bye":
  642. return nil
  643. case strings.HasPrefix(line, "/"):
  644. args := strings.Fields(line)
  645. fmt.Printf("Unknown command '%s'. Type /? for help\n", args[0])
  646. }
  647. if len(line) > 0 && line[0] != '/' {
  648. if err := generate(cmd, model, line, wordWrap, format); err != nil {
  649. return err
  650. }
  651. }
  652. }
  653. }
  654. func RunServer(cmd *cobra.Command, _ []string) error {
  655. host, port, err := net.SplitHostPort(os.Getenv("OLLAMA_HOST"))
  656. if err != nil {
  657. host, port = "127.0.0.1", "11434"
  658. if ip := net.ParseIP(strings.Trim(os.Getenv("OLLAMA_HOST"), "[]")); ip != nil {
  659. host = ip.String()
  660. }
  661. }
  662. if err := initializeKeypair(); err != nil {
  663. return err
  664. }
  665. ln, err := net.Listen("tcp", net.JoinHostPort(host, port))
  666. if err != nil {
  667. return err
  668. }
  669. var origins []string
  670. if o := os.Getenv("OLLAMA_ORIGINS"); o != "" {
  671. origins = strings.Split(o, ",")
  672. }
  673. return server.Serve(ln, origins)
  674. }
  675. func initializeKeypair() error {
  676. home, err := os.UserHomeDir()
  677. if err != nil {
  678. return err
  679. }
  680. privKeyPath := filepath.Join(home, ".ollama", "id_ed25519")
  681. pubKeyPath := filepath.Join(home, ".ollama", "id_ed25519.pub")
  682. _, err = os.Stat(privKeyPath)
  683. if os.IsNotExist(err) {
  684. fmt.Printf("Couldn't find '%s'. Generating new private key.\n", privKeyPath)
  685. _, privKey, err := ed25519.GenerateKey(rand.Reader)
  686. if err != nil {
  687. return err
  688. }
  689. privKeyBytes, err := format.OpenSSHPrivateKey(privKey, "")
  690. if err != nil {
  691. return err
  692. }
  693. err = os.MkdirAll(filepath.Dir(privKeyPath), 0o755)
  694. if err != nil {
  695. return fmt.Errorf("could not create directory %w", err)
  696. }
  697. err = os.WriteFile(privKeyPath, pem.EncodeToMemory(privKeyBytes), 0o600)
  698. if err != nil {
  699. return err
  700. }
  701. sshPrivateKey, err := ssh.NewSignerFromKey(privKey)
  702. if err != nil {
  703. return err
  704. }
  705. pubKeyData := ssh.MarshalAuthorizedKey(sshPrivateKey.PublicKey())
  706. err = os.WriteFile(pubKeyPath, pubKeyData, 0o644)
  707. if err != nil {
  708. return err
  709. }
  710. fmt.Printf("Your new public key is: \n\n%s\n", string(pubKeyData))
  711. }
  712. return nil
  713. }
  714. func startMacApp(client *api.Client) error {
  715. exe, err := os.Executable()
  716. if err != nil {
  717. return err
  718. }
  719. link, err := os.Readlink(exe)
  720. if err != nil {
  721. return err
  722. }
  723. if !strings.Contains(link, "Ollama.app") {
  724. return fmt.Errorf("could not find ollama app")
  725. }
  726. path := strings.Split(link, "Ollama.app")
  727. if err := exec.Command("/usr/bin/open", "-a", path[0]+"Ollama.app").Run(); err != nil {
  728. return err
  729. }
  730. // wait for the server to start
  731. timeout := time.After(5 * time.Second)
  732. tick := time.Tick(500 * time.Millisecond)
  733. for {
  734. select {
  735. case <-timeout:
  736. return errors.New("timed out waiting for server to start")
  737. case <-tick:
  738. if err := client.Heartbeat(context.Background()); err == nil {
  739. return nil // server has started
  740. }
  741. }
  742. }
  743. }
  744. func checkServerHeartbeat(_ *cobra.Command, _ []string) error {
  745. client, err := api.ClientFromEnvironment()
  746. if err != nil {
  747. return err
  748. }
  749. if err := client.Heartbeat(context.Background()); err != nil {
  750. if !strings.Contains(err.Error(), "connection refused") {
  751. return err
  752. }
  753. if runtime.GOOS == "darwin" {
  754. if err := startMacApp(client); err != nil {
  755. return fmt.Errorf("could not connect to ollama app, is it running?")
  756. }
  757. } else {
  758. return fmt.Errorf("could not connect to ollama server, run 'ollama serve' to start it")
  759. }
  760. }
  761. return nil
  762. }
  763. func NewCLI() *cobra.Command {
  764. log.SetFlags(log.LstdFlags | log.Lshortfile)
  765. rootCmd := &cobra.Command{
  766. Use: "ollama",
  767. Short: "Large language model runner",
  768. SilenceUsage: true,
  769. SilenceErrors: true,
  770. CompletionOptions: cobra.CompletionOptions{
  771. DisableDefaultCmd: true,
  772. },
  773. Version: version.Version,
  774. }
  775. cobra.EnableCommandSorting = false
  776. createCmd := &cobra.Command{
  777. Use: "create MODEL",
  778. Short: "Create a model from a Modelfile",
  779. Args: cobra.ExactArgs(1),
  780. PreRunE: checkServerHeartbeat,
  781. RunE: CreateHandler,
  782. }
  783. createCmd.Flags().StringP("file", "f", "Modelfile", "Name of the Modelfile (default \"Modelfile\")")
  784. showCmd := &cobra.Command{
  785. Use: "show MODEL",
  786. Short: "Show information for a model",
  787. Args: cobra.ExactArgs(1),
  788. PreRunE: checkServerHeartbeat,
  789. RunE: ShowHandler,
  790. }
  791. showCmd.Flags().Bool("license", false, "Show license of a model")
  792. showCmd.Flags().Bool("modelfile", false, "Show Modelfile of a model")
  793. showCmd.Flags().Bool("parameters", false, "Show parameters of a model")
  794. showCmd.Flags().Bool("template", false, "Show template of a model")
  795. showCmd.Flags().Bool("system", false, "Show system prompt of a model")
  796. runCmd := &cobra.Command{
  797. Use: "run MODEL [PROMPT]",
  798. Short: "Run a model",
  799. Args: cobra.MinimumNArgs(1),
  800. PreRunE: checkServerHeartbeat,
  801. RunE: RunHandler,
  802. }
  803. runCmd.Flags().Bool("verbose", false, "Show timings for response")
  804. runCmd.Flags().Bool("insecure", false, "Use an insecure registry")
  805. runCmd.Flags().Bool("nowordwrap", false, "Don't wrap words to the next line automatically")
  806. runCmd.Flags().String("format", "", "Response format (e.g. json)")
  807. serveCmd := &cobra.Command{
  808. Use: "serve",
  809. Aliases: []string{"start"},
  810. Short: "Start ollama",
  811. Args: cobra.ExactArgs(0),
  812. RunE: RunServer,
  813. }
  814. pullCmd := &cobra.Command{
  815. Use: "pull MODEL",
  816. Short: "Pull a model from a registry",
  817. Args: cobra.ExactArgs(1),
  818. PreRunE: checkServerHeartbeat,
  819. RunE: PullHandler,
  820. }
  821. pullCmd.Flags().Bool("insecure", false, "Use an insecure registry")
  822. pushCmd := &cobra.Command{
  823. Use: "push MODEL",
  824. Short: "Push a model to a registry",
  825. Args: cobra.ExactArgs(1),
  826. PreRunE: checkServerHeartbeat,
  827. RunE: PushHandler,
  828. }
  829. pushCmd.Flags().Bool("insecure", false, "Use an insecure registry")
  830. listCmd := &cobra.Command{
  831. Use: "list",
  832. Aliases: []string{"ls"},
  833. Short: "List models",
  834. PreRunE: checkServerHeartbeat,
  835. RunE: ListHandler,
  836. }
  837. copyCmd := &cobra.Command{
  838. Use: "cp SOURCE TARGET",
  839. Short: "Copy a model",
  840. Args: cobra.ExactArgs(2),
  841. PreRunE: checkServerHeartbeat,
  842. RunE: CopyHandler,
  843. }
  844. deleteCmd := &cobra.Command{
  845. Use: "rm MODEL [MODEL...]",
  846. Short: "Remove a model",
  847. Args: cobra.MinimumNArgs(1),
  848. PreRunE: checkServerHeartbeat,
  849. RunE: DeleteHandler,
  850. }
  851. rootCmd.AddCommand(
  852. serveCmd,
  853. createCmd,
  854. showCmd,
  855. runCmd,
  856. pullCmd,
  857. pushCmd,
  858. listCmd,
  859. copyCmd,
  860. deleteCmd,
  861. )
  862. return rootCmd
  863. }