Compare commits

..

No commits in common. "master" and "v0.0.1" have entirely different histories.

15 changed files with 119 additions and 170 deletions

View File

@ -5,8 +5,8 @@ flags = -ldflags "-w -s" -v
all: def all: def
@date +"%Y-%m-%d %H:%M:%S" @date +"%Y-%m-%d %H:%M:%S"
go build ${flags} -o ${binarys} src/main.go src/option.go ${version} ${binpack} && ./${binarys} forever restart &>/dev/null go build ${flags} -o ${binarys} src/main.go ${version} ${binpack} && ./${binarys} forever restart &>/dev/null
def: def:
@[ -f ${version} ] || echo "package main">${version} @[ -f ${version} ] || echo "package main">${version}
@[ -f ${binpack} ] || echo "package main">${binpack} @[ -f ${binpack} ] || echo "package main">${binpack}

View File

@ -1,26 +1,19 @@
# ollama-story # ContextOS
基于 ContextOS 开发的 ollama 部署工具与应用界面 ContextOS 通过集群化、模块化、自动化的方式,只用一个 20M 大小的程序文件,就可以在各种设备上,一键启动完整的云计算服务与云研发环境
## 安装 ContextOS ## 源码安装
首先安装 ContextOS 云操作系统。 ### 克隆编译
```sh ```sh
https://shylinux.com git clone https://shylinux.com/x/ContextOS
cd ContextOS; source etc/miss.sh
``` ```
## 安装 ollama-story ### 启动服务
打开 ContextOS 的应用商店,下载 `20250208-ollama-story`
```sh ```sh
http://localhost:9020/c/store#2025-ContextOS ish_miss_serve
``` ```
## 使用 ollama-story ### 访问网页
打开 `web.chat.ollama.chat` 的应用。
```sh ```sh
http://localhost:9020/s/20250208-ollama-story/c/web.chat.ollama.chat open http://localhost:9020
``` ```
## 查看 ollama-story
打开 `web.chat.ollama.chat` 的源码。
```sh
http://localhost:9020/s/20250208-ollama-story/c/vimer?path=src/&file=client/chat.go
```

7
go.mod
View File

@ -3,9 +3,6 @@ module shylinux.com/x/ollama-story
go 1.13 go 1.13
require ( require (
shylinux.com/x/ice v1.5.73 shylinux.com/x/ice v1.5.59
shylinux.com/x/icebergs v1.9.75 shylinux.com/x/icebergs v1.9.59
shylinux.com/x/toolkits v1.0.19
) )
require github.com/gomarkdown/markdown v0.0.0-20250207164621-7a1f277a159e // indirect

14
go.sum
View File

@ -1,15 +1,13 @@
github.com/gomarkdown/markdown v0.0.0-20250207164621-7a1f277a159e h1:ESHlT0RVZphh4JGBz49I5R6nTdC8Qyc08vU25GQHzzQ=
github.com/gomarkdown/markdown v0.0.0-20250207164621-7a1f277a159e/go.mod h1:JDGcbDT52eL4fju3sZ4TeHGsQwhG9nbDV21aMyhwPoA=
shylinux.com/x/go-git/v5 v5.6.7 h1:WD5QSco7m3QooPCgdvQ6/GyGIFPun8C+hex5N41LYlk= shylinux.com/x/go-git/v5 v5.6.7 h1:WD5QSco7m3QooPCgdvQ6/GyGIFPun8C+hex5N41LYlk=
shylinux.com/x/go-git/v5 v5.6.7/go.mod h1:Qb0lA+uIrofZg8NQerhYcJHgGWixFqvS6p3aJ/L5Nlk= shylinux.com/x/go-git/v5 v5.6.7/go.mod h1:Qb0lA+uIrofZg8NQerhYcJHgGWixFqvS6p3aJ/L5Nlk=
shylinux.com/x/go-qrcode v0.0.3 h1:RMo+Vidbgq3HatLBj7DDXcTbTLFUwzis5K7TqBkD38U= shylinux.com/x/go-qrcode v0.0.3 h1:RMo+Vidbgq3HatLBj7DDXcTbTLFUwzis5K7TqBkD38U=
shylinux.com/x/go-qrcode v0.0.3/go.mod h1:KAbtU+KwiiABMZ/CJ0zh9PI2AX82Uf9rRYcQ4ODm4po= shylinux.com/x/go-qrcode v0.0.3/go.mod h1:KAbtU+KwiiABMZ/CJ0zh9PI2AX82Uf9rRYcQ4ODm4po=
shylinux.com/x/ice v1.5.73 h1:wp+KeQ14hLOdxxSou/Y0zHi48Jm4nLx1rFOwGz8ix04= shylinux.com/x/ice v1.5.59 h1:9TugxrDR5rlH+fMnm1L3sLtxoZXY5j0sR4SRLQyqcgg=
shylinux.com/x/ice v1.5.73/go.mod h1:uGI73gYkk+FiJt5qmy/aYhoKRJ4wobcJk3WrPGh8Waw= shylinux.com/x/ice v1.5.59/go.mod h1:/jZUXZGsBpTdHy/c8JkpWtVPGg6uPHOghlUULzg4R7o=
shylinux.com/x/icebergs v1.9.75 h1:9zAm0ppwUWl0As/vqfQG/QR/vwXa5zmfdIGE/gmvErQ= shylinux.com/x/icebergs v1.9.59 h1:fNEnFxyTJB2chG4zUeNpgfH8gXAiwUJOJvYFPJGDsek=
shylinux.com/x/icebergs v1.9.75/go.mod h1:3Bdp3tjzw+hUKJF+kR8pfsrbjAf72DVZmCaE8/MPFtk= shylinux.com/x/icebergs v1.9.59/go.mod h1:d8sN77l5UZA+h8/swZ9OzqRCruoSYHWYjg5qKWnQI2s=
shylinux.com/x/toolkits v0.7.10/go.mod h1:CHDJarGlDkg60kVsvMLYL/a5hAnRLEOShiEsMOuEp0Q= shylinux.com/x/toolkits v0.7.10/go.mod h1:CHDJarGlDkg60kVsvMLYL/a5hAnRLEOShiEsMOuEp0Q=
shylinux.com/x/toolkits v1.0.19 h1:Nrx0xYRc5ph1WS66EZ1hJUCe+2FdSWQ4QP6tBlguikQ= shylinux.com/x/toolkits v1.0.16 h1:7Oh454uAyBLfflIFEQooLNzml4pqXIReiaxEVA/YXaU=
shylinux.com/x/toolkits v1.0.19/go.mod h1:CHDJarGlDkg60kVsvMLYL/a5hAnRLEOShiEsMOuEp0Q= shylinux.com/x/toolkits v1.0.16/go.mod h1:CHDJarGlDkg60kVsvMLYL/a5hAnRLEOShiEsMOuEp0Q=
shylinux.com/x/websocket v0.0.4 h1:AJpwblePoOpiE6C8NrvgNYpKTotXMLrDDX2chTvx44Q= shylinux.com/x/websocket v0.0.4 h1:AJpwblePoOpiE6C8NrvgNYpKTotXMLrDDX2chTvx44Q=
shylinux.com/x/websocket v0.0.4/go.mod h1:3UGWkjTu3ie5NAZen7J+uLPBrO7DFeKloj6Jxo13Oiw= shylinux.com/x/websocket v0.0.4/go.mod h1:3UGWkjTu3ie5NAZen7J+uLPBrO7DFeKloj6Jxo13Oiw=

View File

@ -1,6 +1,5 @@
$content { padding:10px 20px; } $output>div.message { padding:20px; overflow:auto; }
$content>div.request { background-color:#94ec69; color:black; padding:10px; float:right; } $output>div.message>div.request { background-color:#94ec69; padding:10px; float:right; }
$content>div.response { clear:both; } $output>div.message>div.request>span { float:right; }
$content>div.response p { margin:5px 0; } $output>div.message>div.response { white-space:break-spaces; padding:0px; clear:both; }
$display { background-color:var(--plugin-bg-color); height:120px; } $output>div.request { padding:20px; }
$display textarea { margin:10px; width:calc(100% - 20px); }

View File

@ -9,18 +9,19 @@ import (
) )
type chat struct { type chat struct {
client client list string `name:"list name auto" help:"大模型对话"`
request string `name:"request model* role* content* which*"`
list string `name:"list list" help:"大模型对话" icon:"src/main.png"`
} }
func (s chat) Request(m *ice.Message, arg ...string) { func (s chat) Request(m *ice.Message, arg ...string) {
m.Cmdy(web.SPIDE, OLLAMA, web.SPIDE_STREAM, http.MethodPost, "/api/chat", web.SPIDE_DATA, kit.Format(kit.Dict( m.Optionv(web.SPIDE_STREAM, func(text string) { web.PushNoticeGrow(m.Message, text) })
m.OptionSimple("model"), "stream", true, "messages", kit.List(kit.Dict(m.OptionSimple("role", "content"))), m.Cmdy(web.SPIDE, ice.DEV, web.SPIDE_STREAM, http.MethodPost, "http://localhost:11434/api/chat",
))) web.SPIDE_DATA, kit.Format(kit.Dict("model", "deepseek-r1", "stream", true,
"messages", kit.List(kit.Dict("role", "user", "content", arg[0])),
)),
)
} }
func (s chat) List(m *ice.Message, arg ...string) { func (s chat) List(m *ice.Message, arg ...string) {
m.Cmdy(s.client).Display("").DisplayCSS("") m.Display("").DisplayCSS("")
} }
func init() { ice.ChatModCmd(chat{}) } func init() { ice.CodeModCmd(chat{}) }

View File

@ -1,40 +1,28 @@
Volcanos(chat.ONIMPORT, { Volcanos(chat.ONIMPORT, {
_init: function(can, msg) { can.require(["https://unpkg.com/showdown/dist/showdown.min.js"]) _init: function(can, msg) {
can.ui = can.onappend.layout(can), can.ui.responseList = [], can.onmotion.toggle(can, can.ui.display, true) can.ui = can.page.Append(can, can._output, [
msg.Table(function(value) { can.onimport.item(can, { {view: "message"},
name: value.NAME, nick: value.NAME+" ("+value.SIZE+")", {view: "request", list: [
}, function(event, item, show, target) { {type: "textarea", onkeydown: function(event) {
can.onimport.tabsCache(can, item, target, function() { can.onimport.content(can, target) }) if (event.key == "Enter") {
can.onappend._status(can, msg), can.onimport.layout(can) can.onaction.request(event, can, event.target.value), event.target.value = "", can.onkeymap.prevent(event)
}) }) }
}, }},
content: function(can, target) {
can.page.Append(can, can.ui.content, [
{view: chat.RESPONSE, list: [
{text: can.user.trans(can, "Please Ask Anything!", "有什么问题尽管问吧!")},
]}, ]},
]) ])
can.page.Append(can, can.ui.display, [ can.ui.response = can.page.Append(can, can.ui.message, [{view: "response", list: [{text: "有什么问题尽管问吧!"}]}])._target
{type: html.TEXTAREA, placeholder: can.user.trans(can, "Please Input Your Question!", "请输入的您的问题!"), onkeydown: function(event) {
if (event.key == code.ENTER) {
can.onaction.request(event, can, event.target.value), event.target.value = "", can.onkeymap.prevent(event)
}
}},
])
}, },
_grow: function(can, msg, which, text) { _grow: function(can, msg, text) { var data = JSON.parse(text)
var target = can.ui.responseList[parseInt(which)-1], data = JSON.parse(text) can.page.Append(can, can.ui.response, [{text: data.message.content}]), can.ui.message.scrollBy(0, 1000)
if (window.showdown) { target._text = (target._text||"")+data.message.content
var converter = new showdown.Converter(); target.innerHTML = converter.makeHtml(target._text)
} else {
can.page.Append(can, target, [{text: data.message.content}])
} can.ui.content.scrollBy(0, 1000)
}, },
layout: function(can) {
can.page.style(can, can.ui.message, html.MAX_HEIGHT, can.ConfHeight()-140)
}
}) })
Volcanos(chat.ONACTION, { Volcanos(chat.ONACTION, {
request: function(event, can, text) { can.page.Append(can, can.ui.content, [{view: chat.REQUEST, list: [{text: text}]}]) request: function(event, can, text) {
var response = can.page.Append(can, can.ui.content, [{view: chat.RESPONSE}])._target; can.ui.responseList.push(response) can.page.Append(can, can.ui.message, [{view: "request", list: [{text: text}]}])
can.request(event, {model: can.db.value.name, role: "user", content: text, which: ""+can.ui.responseList.length}) can.ui.response = can.page.Append(can, can.ui.message, [{view: "response"}])._target
can.runAction(event, chat.REQUEST, [], function(msg) {}) can.runAction(event, "request", [text], function(msg) {})
}, },
}) })

View File

@ -3,32 +3,21 @@ package client
import ( import (
"shylinux.com/x/ice" "shylinux.com/x/ice"
"shylinux.com/x/icebergs/base/cli" "shylinux.com/x/icebergs/base/cli"
"shylinux.com/x/icebergs/base/mdb"
"shylinux.com/x/icebergs/base/nfs"
"shylinux.com/x/icebergs/base/web"
)
const (
OLLAMA = "ollama"
) )
type client struct { type client struct {
list string `name:"list NAME auto" help:"大模型"` list string `name:"list NAME auto" help:"大模型"`
} }
func (s client) Init(m *ice.Message, arg ...string) {
m.Cmd(web.SPIDE, mdb.CREATE, "http://localhost:11434/api/chat", OLLAMA, nfs.SRC_MAIN_PNG)
}
func (s client) List(m *ice.Message, arg ...string) { func (s client) List(m *ice.Message, arg ...string) {
if len(arg) == 0 { if len(arg) == 0 {
m.SplitIndex(s.cmdx(m, ice.LIST)) m.SplitIndex(s.cmdx(m, "list"))
} else { } else if len(arg) == 1 {
m.Echo(s.cmdx(m, ice.SHOW, arg[0])) m.Echo(s.cmdx(m, "show", arg[0]))
} }
} }
func init() { ice.ChatModCmd(client{}) }
func (s client) cmdx(m *ice.Message, arg ...string) string { func (s client) cmdx(m *ice.Message, arg ...string) string {
return m.Cmdx(cli.SYSTEM, OLLAMA, arg) return m.Cmdx(cli.SYSTEM, "ollama", arg)
} }
func init() { ice.Cmd("web.chat.ollama.client", client{}) }

View File

@ -1,9 +1,7 @@
package main package main
import ( import "shylinux.com/x/ice"
"shylinux.com/x/ice" import _ "shylinux.com/x/ollama-story/src/server"
_ "shylinux.com/x/ollama-story/src/client" import _ "shylinux.com/x/ollama-story/src/client"
_ "shylinux.com/x/ollama-story/src/server"
)
func main() { print(ice.Run()) } func main() { print(ice.Run()) }

Binary file not shown.

Before

Width:  |  Height:  |  Size: 7.3 KiB

View File

@ -1,60 +1,53 @@
title "ollama-story" title "ContextOS"
refer `
官网 https://ollama.com/ chapter "实践"
源码 https://github.com/ollama/ollama label `
模型 https://ollama.com/library/deepseek-r1 code wiki chat team mall
应用 https://feizhuke.com/ linux nginx docker redis mysql
bash git golang vim tmux
` `
chapter "应用" chapter "理论"
refer ` label `集群化 模块化 自动化`
源码 https://github.com/shylinux/ollama-story
帖子 https://bbs.shylinux.com/forum.php?mod=viewthread&tid=6&extra=page%3D1
showdown https://igithub.com/showdownjs/showdown
nodejs https://nodejs.org/en
`
field web.chat.ollama.chat
chapter "常用模型" chapter "项目"
refer ` label `
硅基流动 https://siliconflow.cn/ matrix release program
深度求索 https://www.deepseek.com/ intshell icebergs volcanos
字节豆包 https://www.doubao.com/ contexts toolkits learning
腾讯元宝 https://yuanbao.tencent.com/
通义千问 https://tongyi.aliyun.com/
文心一言 https://yiyan.baidu.com/
月之暗面 https://kimi.moonshot.cn/
` `
chapter "常用工具" section "火山架"
section "AnythingLLM" field "趋势图" web.code.git.trends args `volcanos`
refer ` field "架构图" web.code.git.spides args `volcanos`
官网 https://anythingllm.com/
文档 https://docs.anythingllm.com/ section "冰山架"
源码 https://github.com/Mintplex-Labs/anything-llm field "趋势图" web.code.git.trends args `icebergs`
field "架构图" web.code.git.spides args `icebergs`
label `
ctx mdb web aaa
lex yac ssh gdb
tcp nfs cli log
` `
section "Cherry Studio" chain `
refer ` web
官网 https://cherry-ai.com/ code
文档 https://docs.cherry-ai.com/ wiki
源码 https://github.com/CherryHQ/cherry-studio chart
chat
macos
oauth
location
team
mall
` `
section "OBS Studio" label `
refer ` websocket webview qrcode xterm ssh git
官网 https://obsproject.com/ vim bash tmux alpha input lex yac
文档 https://obsproject.com/kb/quick-start-guide node java coder github chrome
源码 https://github.com/obsproject/obs-studio wx lark wework
` ` compact true
section "community"
refer ` section "神农架"
https://x.com/ field "趋势图" web.code.git.trends args `intshell`
https://discord.com/ field "架构图" web.code.git.spides args `intshell`
https://telegram.org/
https://www.facebook.com/
https://www.youtube.com/
https://mail.google.com/
https://www.bilibili.com/
https://www.xiaohongshu.com/
https://www.douyin.com/
https://weibo.com/
`

View File

@ -1,6 +0,0 @@
package main
import (
_ "shylinux.com/x/ice/devops"
_ "shylinux.com/x/icebergs/misc/md"
)

View File

@ -4,11 +4,12 @@ import "shylinux.com/x/ice"
type server struct { type server struct {
ice.Hash ice.Hash
list string `name:"list hash auto" help:"大模型"`
list string `name:"list hash auto" help:"server"`
} }
func (s server) List(m *ice.Message, arg ...string) { func (s server) List(m *ice.Message, arg ...string) {
s.Hash.List(m, arg...) s.Hash.List(m, arg...)
} }
func init() { ice.ChatModCmd(server{}) } func init() { ice.Cmd("web.chat.ollama.server", server{}) }

6
src/server/server.shy Normal file
View File

@ -0,0 +1,6 @@
title "ollama"
refer `
官网 https://ollama.com/
源码 https://github.com/ollama/ollama
简介 https://blog.csdn.net/qq_40999403/article/details/139320266
`

View File

@ -7,14 +7,6 @@
"time": "2025-02-08 09:20:30.381" "time": "2025-02-08 09:20:30.381"
} }
}, },
"937681425ca043f9908761fbcdf3c813": {
"meta": {
"icon": "src/main.png",
"index": "web.chat.ollama.chat",
"name": "chat",
"time": "2025-02-08 15:27:08.564"
}
},
"adf694a4d378bcc855218c464fecb7d9": { "adf694a4d378bcc855218c464fecb7d9": {
"meta": { "meta": {
"icon": "usr/icons/Messages.png", "icon": "usr/icons/Messages.png",