diff --git a/README.md b/README.md index e4cf757..222468a 100644 --- a/README.md +++ b/README.md @@ -1,7 +1,7 @@ # milla -Milla is an IRC bot that sends things over to an LLM when you ask it questions, prints the answer with syntax-hilighting.
-Currently Supported Models: +Milla is an IRC bot that sends things over to an LLM when you ask it questions and prints the answer with optional syntax-hilighting.
+Currently Supported: - Ollama - Openai @@ -9,38 +9,185 @@ Currently Supported Models: ![milla](./milla.png) -### Config - -config: - -```toml -ircServer = "irc.terminaldweller.com" -ircPort = 6697 -ircNick = "mybot" -ircSaslUser = "mybot" -ircSaslPass = "mypass" -ircChannels = ["#mychannel1", "#mychannel2"] -ollamaEndpoint = "" -temp = 0.2 -ollamaSystem = "" -requestTimeout = 10 -millaReconnectDelay = 60 -enableSasl = true -model = "llama2-uncensored" -chromaStyle = "rose-pine-moon" -chromaFormatter = "terminal256" -provider = "ollama" # ollama, chatgpt, gemini -apikey = "key" -topP = 0.9 -topK = 20 +milla accepts one cli arg which tells it where to look for the config file:
+ +```$ milla -help +Usage of ./milla: + -config string + path to the config file (default "./config.toml") +``` + +## Config + +An exhaustive example is in `config-example.toml`. + +#### ircServer + +The address for the IRC server to connect to. + +#### ircNick + +The nick the bot should use. + +#### enableSasl + +Whether to use SASL for authentication. + +#### ircSaslUser + +The SASL username. + +#### ircSaslPass + +The SASL password for SASL plain authentication. + +#### ollamaEndpoint + +The address for the Ollama chat endpoint. + +#### model + +The name of the model to use. + +#### chromaStyle + +The style to use for syntax highlighting done by [chroma](https://github.com/alecthomas/chroma). This is basically what's called a "theme". + +#### chromaFormatter + +The formatter to use. This tells chroma how to generate the color in the output. The supported options are: + +- `noop` for no syntax highlighting +- `terminal` for 8-color terminals +- `terminal8` for 8-color terminals +- `terminal16` for 16-color terminals +- `terminal256` for 256-color terminals +- `terminal16m` for treucolor terminals +- `html` for HTML output + +#### provider + +Which LLM provider to use. The supported options are: + +- [ollama](https://github.com/ollama/ollama) +- chatgpt +- gemini + +#### apikey + +The apikey to use for the LLM provider. + +#### ollamaSystem + +The system message to use for ollama. + +#### clientCertPath + +The path to the client certificate to use for SASL external authentication. + +#### serverPass + +The password to use for the IRC server the bot is trying to connect to if the server has a password. + +#### bind + +Which address to bind to for the IRC server. + +#### temp + +The temperature to config the model with. + +#### requestTimeout + +The timeout for requests made to the LLM provider. + +#### millaReconnectDelay + +How much to wait before reconnecting to the IRC server. + +#### ircPort + +Which port to connect to for the IRC server. + +#### keepAlive + +#### memoryLimit + +How many conversations to keep in memory for a model. + +#### pingDelay + +Ping delay for the IRC server. + +#### pingTimeout + +Ping timeout for the IRC server. + +#### topP + +#### topK + +#### skipTLSVerify + +Skip verifying the IRC server's TLS certificate. This only makes sense if you are trying to connect to an IRC server with a self-signed certificate. + +#### useTLS + +Whether to use TLS to connect to the IRC server. This option is provided to support usage on overlay networks such as Tor, i2p and [yggdrassil](https://github.com/yggdrasil-network/yggdrasil-go). + +#### disableSTSFallback + +#### allowFlood + +Disable [girc](https://github.com/lrstanley/girc)'s built-in flood protection. + +#### debug + +Whether to enable debug logging. The logs are written to stdout. + +#### out + +Whether to write raw messages to stdout. + +#### admins + +List of adimns for the bot. Only admins can use commands. + +``` +admins = ["admin1", "admin2"] ``` -### Deploy +#### ircChannels + +List of channels for the bot to join when it connects to the server. + +``` +ircChannels = ["#channel1", "#channel2"] +``` + +## Commands + +#### help + +Prints the help message. + +#### get -You can use the provided compose file:
+Get the value of a config option. Use the same name as the config file but capitalized. + +#### getall + +Get the value of all config options. + +#### set + +Set a config option on the fly. Use the same name as the config file but capitalized. + +## Deploy + +An example docker compose file is provided in the repo under `docker-compose.yaml`. ```yaml -version: "3.9" services: milla: image: milla @@ -50,6 +197,7 @@ services: resources: limits: memory: 64M + user: ${UID}:${GID} logging: driver: "json-file" options: @@ -60,13 +208,21 @@ services: command: ["--config", "/opt/milla/config.toml"] volumes: - ./config.toml:/opt/milla/config.toml + - /etc/ssl/certs:/etc/ssl/certs:ro cap_drop: - ALL - dns: - - 9.9.9.9 environment: - SERVER_DEPLOYMENT_TYPE=deployment - entrypoint: ["/milla/milla"] networks: millanet: ``` + +## Thanks + +- [girc](https://github.com/lrstanley/girc) +- [chroma](https://github.com/alecthomas/chroma) +- [ollama](https://github.com/ollama/ollama) + +## Similar Projects + +- [soulshack](https://github.com/pkdindustries/soulshack) diff --git a/config-example.toml b/config-example.toml new file mode 100644 index 0000000..9a5a56e --- /dev/null +++ b/config-example.toml @@ -0,0 +1,33 @@ +ircServer = "irc.libera.caht" +ircNick = "milla" +ircSaslUser = "" +ircSaslPass = "" +ollamaEndpoint = "http://127.0.0.1:11434/api/chat" +model = "ollama2-uncensored" +chromaStyle = "monokai" +chromaFormatter = "terminal16m" +provider = "ollama" +apikey = "xxx" +ollamaSystem = "" +clientCertPath = "" +serverPass = "" +bind = "" +temp = 0.5 +requestTimeout = 10 +millaReconnectDelay = 60 +ircPort = 6669 +keepAlive = 20 +memoryLimit = 20 +pingDelay = 20 +pingTimeout = 600 +topP = 0.9 +topK = 0 +enableSasl = false +skipTLSVerify = false +useTLS = true +disableSTSFallback = true +allowFlood = false +debug = false +out = false +admins = ["nick1", "nick2"] +ircChannels = ["##milla1", "##milla2"] diff --git a/docker-compose.yaml b/docker-compose.yaml index 69d0ba5..6e64496 100644 --- a/docker-compose.yaml +++ b/docker-compose.yaml @@ -1,4 +1,3 @@ -version: "3.9" services: milla: image: milla @@ -8,6 +7,7 @@ services: resources: limits: memory: 64M + user: ${UID}:${GID} logging: driver: "json-file" options: @@ -15,14 +15,11 @@ services: networks: - millanet restart: unless-stopped - restart: no command: ["--config", "/opt/milla/config.toml"] volumes: - ./config.toml:/opt/milla/config.toml - /etc/ssl/certs:/etc/ssl/certs:ro cap_drop: - ALL - environment: - - SERVER_DEPLOYMENT_TYPE=deployment networks: millanet: diff --git a/main.go b/main.go index 7555e00..b470738 100644 --- a/main.go +++ b/main.go @@ -13,7 +13,6 @@ import ( "net/url" "os" "reflect" - "regexp" "strconv" "strings" "time" @@ -138,26 +137,26 @@ func returnGeminiResponse(resp *genai.GenerateContentResponse) string { return result } -func extractLast256ColorEscapeCode(str string) (string, error) { - pattern256F := `\033\[38;5;(\d+)m` - // pattern256B := `\033\[48;5;(\d+)m` - // pattern16mF := `\033\[38;2;(\d+);(\d+);(\d+)m` - // pattern16mB := `\033\[48;2;(\d+);(\d+);(\d+)m` +// func extractLast256ColorEscapeCode(str string) (string, error) { +// pattern256F := `\033\[38;5;(\d+)m` +// // pattern256B := `\033\[48;5;(\d+)m` +// // pattern16mF := `\033\[38;2;(\d+);(\d+);(\d+)m` +// // pattern16mB := `\033\[48;2;(\d+);(\d+);(\d+)m` - r, err := regexp.Compile(pattern256F) - if err != nil { - return "", fmt.Errorf("failed to compile regular expression: %w", err) - } +// r, err := regexp.Compile(pattern256F) +// if err != nil { +// return "", fmt.Errorf("failed to compile regular expression: %w", err) +// } - matches := r.FindAllStringSubmatch(str, -1) - if len(matches) == 0 { - return "", nil - } +// matches := r.FindAllStringSubmatch(str, -1) +// if len(matches) == 0 { +// return "", nil +// } - lastMatch := matches[len(matches)-1] +// lastMatch := matches[len(matches)-1] - return lastMatch[1], nil -} +// return lastMatch[1], nil +// } func chunker(inputString string, chromaFormatter string) []string { chunks := strings.Split(inputString, "\n") @@ -677,9 +676,9 @@ func runIRC(appConfig TomlConfig, ircChan chan *girc.Client) { } } - if appConfig.EnableSasl && appConfig.ClientCertPath != "" { - // TODO - add client cert support - } + // if appConfig.EnableSasl && appConfig.ClientCertPath != "" { + // // TODO - add client cert support + // } irc.Handlers.AddBg(girc.CONNECTED, func(c *girc.Client, e girc.Event) { for _, channel := range appConfig.IrcChannels {