Compare commits

...

4 Commits

Author SHA1 Message Date
8f8182039d v0.2.0 2026-01-28 16:31:02 +03:00
05dadc3de3 logger enchancment 2026-01-28 13:40:18 +03:00
37397ba90f added sleep on fetching updates 2026-01-26 14:36:42 +03:00
c503b68814 photo keyboard 2026-01-22 20:08:23 +03:00
9 changed files with 148 additions and 62 deletions

View File

@@ -5,6 +5,7 @@ import (
"encoding/json" "encoding/json"
"fmt" "fmt"
"io" "io"
"log"
"net/http" "net/http"
"os" "os"
"sort" "sort"
@@ -104,15 +105,28 @@ func NewBot(settings *BotSettings) *Bot {
if settings.Debug { if settings.Debug {
level = DEBUG level = DEBUG
} }
bot.logger = CreateLogger().Level(level)
bot.logger = CreateLogger().Level(level).PrintTraceback(true)
bot.logger.AddWriter(bot.logger.CreateStdoutWriter())
if settings.WriteToFile { if settings.WriteToFile {
bot.logger = bot.logger.OpenFile(fmt.Sprintf("%s/main.log", strings.TrimRight(settings.LoggerBasePath, "/"))) path := fmt.Sprintf("%s/main.log", strings.TrimRight(settings.LoggerBasePath, "/"))
fileWriter, err := bot.logger.CreateFileWriter(path)
if err != nil {
bot.logger.Fatal(err)
}
bot.logger.AddWriter(fileWriter)
} }
bot.logger = bot.logger.PrintTraceback(true)
if settings.UseRequestLogger { if settings.UseRequestLogger {
bot.requestLogger = CreateLogger().Level(level).Prefix("REQUESTS") bot.requestLogger = CreateLogger().Level(level).Prefix("REQUESTS")
bot.requestLogger.AddWriter(bot.requestLogger.CreateStdoutWriter())
if settings.WriteToFile { if settings.WriteToFile {
bot.requestLogger = bot.requestLogger.OpenFile(fmt.Sprintf("%s/requests.log", strings.TrimRight(settings.LoggerBasePath, "/"))) path := fmt.Sprintf("%s/requests.log", strings.TrimRight(settings.LoggerBasePath, "/"))
fileWriter, err := bot.requestLogger.CreateFileWriter(path)
if err != nil {
bot.logger.Fatal(err)
}
bot.requestLogger.AddWriter(fileWriter)
} }
} }
@@ -120,11 +134,11 @@ func NewBot(settings *BotSettings) *Bot {
} }
func (b *Bot) Close() { func (b *Bot) Close() {
err := b.logger.f.Close() for _, writer := range b.logger.writers {
if err != nil { err := writer.Close()
fmt.Println(err) if err != nil {
} else { log.Println(err)
fmt.Println("log closed") }
} }
} }
@@ -132,11 +146,11 @@ func (b *Bot) InitDatabaseContext(ctx *DatabaseContext) *Bot {
b.dbContext = ctx b.dbContext = ctx
return b return b
} }
func (b *Bot) AddDatabaseLogger(writer func(db *DatabaseContext) LoggerWriter) *Bot { func (b *Bot) AddDatabaseLogger(writer func(db *DatabaseContext) *LoggerWriter) *Bot {
w := []LoggerWriter{writer(b.dbContext)} w := writer(b.dbContext)
b.logger.AddWriters(w) b.logger.AddWriter(w)
if b.requestLogger != nil { if b.requestLogger != nil {
b.requestLogger.AddWriters(w) b.requestLogger.AddWriter(w)
} }
return b return b
} }
@@ -177,7 +191,7 @@ func (b *Bot) Debug(debug bool) *Bot {
func (b *Bot) AddPlugins(plugin ...*Plugin) *Bot { func (b *Bot) AddPlugins(plugin ...*Plugin) *Bot {
b.plugins = append(b.plugins, plugin...) b.plugins = append(b.plugins, plugin...)
for _, p := range plugin { for _, p := range plugin {
b.logger.Debug(fmt.Sprintf("plugins with name \"%s\" was registered", p.Name)) b.logger.Debug(fmt.Sprintf("plugins with name \"%s\" registered", p.Name))
} }
return b return b
} }
@@ -194,7 +208,7 @@ func (b *Bot) AddMiddleware(middleware ...*Middleware) *Bot {
b.middlewares = append(b.middlewares, middleware...) b.middlewares = append(b.middlewares, middleware...)
for _, m := range middleware { for _, m := range middleware {
b.logger.Debug(fmt.Sprintf("middleware with name \"%s\" was registered", m.Name)) b.logger.Debug(fmt.Sprintf("middleware with name \"%s\" registered", m.Name))
} }
return b return b
} }
@@ -218,6 +232,7 @@ func (b *Bot) Run() {
if err != nil { if err != nil {
b.logger.Error(err) b.logger.Error(err)
} }
time.Sleep(time.Millisecond * 10)
} }
}() }()
@@ -370,6 +385,31 @@ func (ctx *MsgContext) EditCallbackf(format string, keyboard *InlineKeyboard, ar
return ctx.EditCallback(fmt.Sprintf(format, args...), keyboard) return ctx.EditCallback(fmt.Sprintf(format, args...), keyboard)
} }
func (ctx *MsgContext) editPhotoText(messageId int, text string, kb *InlineKeyboard) *AnswerMessage {
params := &EditMessageCaptionP{
ChatID: ctx.Msg.Chat.ID,
MessageID: messageId,
Caption: text,
ParseMode: ParseMD,
}
if kb != nil {
params.ReplyMarkup = kb.Get()
}
msg, err := ctx.Bot.EditMessageCaption(params)
if err != nil {
ctx.Bot.logger.Error(err)
}
return &AnswerMessage{
MessageID: msg.MessageID, ctx: ctx, Text: text, IsMedia: true,
}
}
func (m *AnswerMessage) EditCaption(text string) *AnswerMessage {
return m.ctx.editPhotoText(m.MessageID, text, nil)
}
func (m *AnswerMessage) EditCaptionKeyboard(text string, kb *InlineKeyboard) *AnswerMessage {
return m.ctx.editPhotoText(m.MessageID, text, kb)
}
func (ctx *MsgContext) answer(text string, keyboard *InlineKeyboard) *AnswerMessage { func (ctx *MsgContext) answer(text string, keyboard *InlineKeyboard) *AnswerMessage {
params := &SendMessageP{ params := &SendMessageP{
ChatID: ctx.Msg.Chat.ID, ChatID: ctx.Msg.Chat.ID,
@@ -399,20 +439,30 @@ func (ctx *MsgContext) Keyboard(text string, kb *InlineKeyboard) *AnswerMessage
return ctx.answer(text, kb) return ctx.answer(text, kb)
} }
func (ctx *MsgContext) AnswerPhoto(photoId string, text string) *AnswerMessage { func (ctx *MsgContext) answerPhoto(photoId, text string, kb *InlineKeyboard) *AnswerMessage {
_, err := ctx.Bot.SendPhoto(&SendPhotoP{ params := &SendPhotoP{
ChatID: ctx.Msg.Chat.ID, ChatID: ctx.Msg.Chat.ID,
Caption: text, Caption: text,
Photo: photoId, Photo: photoId,
ParseMode: ParseMD, ParseMode: ParseMD,
}) }
if kb != nil {
params.ReplyMarkup = kb.Get()
}
msg, err := ctx.Bot.SendPhoto(params)
if err != nil { if err != nil {
ctx.Bot.logger.Error(err) ctx.Bot.logger.Error(err)
} }
return &AnswerMessage{ return &AnswerMessage{
MessageID: ctx.Msg.MessageID, ctx: ctx, IsMedia: true, Text: text, MessageID: msg.MessageID, ctx: ctx, Text: text, IsMedia: true,
} }
} }
func (ctx *MsgContext) AnswerPhoto(photoId, text string) *AnswerMessage {
return ctx.answerPhoto(photoId, text, nil)
}
func (ctx *MsgContext) AnswerPhotoKeyboard(photoId, text string, kb *InlineKeyboard) *AnswerMessage {
return ctx.answerPhoto(photoId, text, kb)
}
func (ctx *MsgContext) delete(messageId int) { func (ctx *MsgContext) delete(messageId int) {
_, err := ctx.Bot.DeleteMessage(&DeleteMessageP{ _, err := ctx.Bot.DeleteMessage(&DeleteMessageP{

View File

@@ -2,6 +2,7 @@ package laniakea
import ( import (
"fmt" "fmt"
"io"
"os" "os"
"path/filepath" "path/filepath"
"runtime" "runtime"
@@ -12,16 +13,30 @@ import (
"github.com/fatih/color" "github.com/fatih/color"
) )
type LoggerWriter func(level LogLevel, prefix, traceback string, message []any) type LoggerWriter struct {
writer io.Writer
writeFn LoggerWriterFn
logger *Logger
}
func (w *LoggerWriter) SetFn(fn LoggerWriterFn) {
w.writeFn = fn
}
func (w *LoggerWriter) Write(p []byte) (n int, err error) {
return w.writer.Write(p)
}
func (w *LoggerWriter) Close() error {
return w.writer.(io.Closer).Close()
}
type LoggerWriterFn func(level LogLevel, prefix, traceback string, message []any) error
type Logger struct { type Logger struct {
prefix string prefix string
level LogLevel level LogLevel
printTraceback bool printTraceback bool
printTime bool printTime bool
writers []LoggerWriter writers []*LoggerWriter
f *os.File
} }
type LogLevel struct { type LogLevel struct {
@@ -60,16 +75,34 @@ func CreateLogger() *Logger {
} }
} }
func (l *Logger) OpenFile(name string) *Logger { func (l *Logger) CreateFileWriter(path string) (*LoggerWriter, error) {
err := os.MkdirAll(filepath.Dir(name), os.ModePerm) err := os.MkdirAll(filepath.Dir(path), os.ModePerm)
if err != nil { if err != nil {
l.Fatal(err) return nil, err
} }
l.f, err = os.OpenFile(name, os.O_APPEND|os.O_CREATE|os.O_WRONLY, 0644) file, err := os.OpenFile(path, os.O_APPEND|os.O_CREATE|os.O_WRONLY, 0644)
if err != nil { if err != nil {
l.Fatal(err) return nil, err
} }
return l
writer := &LoggerWriter{
writer: file, logger: l,
}
writer.writeFn = func(level LogLevel, prefix, traceback string, message []any) error {
_, err = writer.Write([]byte(writer.logger.buildString(level, message) + "\n"))
return err
}
return writer, nil
}
func (l *Logger) CreateStdoutWriter() *LoggerWriter {
writer := &LoggerWriter{
writer: os.Stdout, logger: l,
}
writer.writeFn = func(level LogLevel, prefix, traceback string, message []any) error {
_, err := color.New(level.c).Fprint(writer.writer, writer.logger.buildString(level, message))
return err
}
return writer
} }
func (l *Logger) Prefix(prefix string) *Logger { func (l *Logger) Prefix(prefix string) *Logger {
@@ -88,11 +121,18 @@ func (l *Logger) PrintTime(b bool) *Logger {
l.printTime = b l.printTime = b
return l return l
} }
func (l *Logger) AddWriters(writers []LoggerWriter) *Logger { func (l *Logger) AddWriters(writers ...*LoggerWriter) *Logger {
l.writers = append(l.writers, writers...) l.writers = append(l.writers, writers...)
return l return l
} }
func (l *Logger) AddWriter(writer *LoggerWriter) *Logger {
l.writers = append(l.writers, writer)
return l
}
func (l *Logger) Infof(format string, m ...any) {
l.print(INFO, []any{fmt.Sprintf(format, m...)})
}
func (l *Logger) Info(m ...any) { func (l *Logger) Info(m ...any) {
l.print(INFO, m) l.print(INFO, m)
} }
@@ -209,19 +249,11 @@ func (l *Logger) print(level LogLevel, m []any) {
if l.level.n < level.n { if l.level.n < level.n {
return return
} }
_, err := color.New(level.c).Println(l.buildString(level, m))
if err != nil {
l.Fatal(err)
return
}
for _, writer := range l.writers { for _, writer := range l.writers {
writer(level, l.prefix, l.formatFullTraceback(l.getFullTraceback(4)), m) err := writer.writeFn(level, l.prefix, l.formatFullTraceback(l.getFullTraceback(0)), m)
} if err != nil {
l.Error(err)
if l.f != nil {
if _, err := l.f.Write([]byte(l.buildString(level, m) + "\n")); err != nil {
l.Fatal(err)
} }
} }
} }

View File

@@ -87,19 +87,20 @@ func (b *Bot) SendMessage(params *SendMessageP) (*Message, error) {
} }
type SendPhotoP struct { type SendPhotoP struct {
BusinessConnectionID string `json:"business_connection_id,omitempty"` BusinessConnectionID string `json:"business_connection_id,omitempty"`
ChatID int `json:"chat_id"` ChatID int `json:"chat_id"`
MessageThreadID int `json:"message_thread_id,omitempty"` MessageThreadID int `json:"message_thread_id,omitempty"`
ParseMode ParseMode `json:"parse_mode,omitempty"` ParseMode ParseMode `json:"parse_mode,omitempty"`
Photo string `json:"photo"` Photo string `json:"photo"`
Caption string `json:"caption,omitempty"` Caption string `json:"caption,omitempty"`
CaptionEntities []*MessageEntity `json:"caption_entities,omitempty"` CaptionEntities []*MessageEntity `json:"caption_entities,omitempty"`
ShowCaptionAboveMedia bool `json:"show_caption_above_media"` ShowCaptionAboveMedia bool `json:"show_caption_above_media"`
HasSpoiler bool `json:"has_spoiler"` HasSpoiler bool `json:"has_spoiler"`
DisableNotifications bool `json:"disable_notifications,omitempty"` DisableNotifications bool `json:"disable_notifications,omitempty"`
ProtectContent bool `json:"protect_content,omitempty"` ProtectContent bool `json:"protect_content,omitempty"`
AllowPaidBroadcast bool `json:"allow_paid_broadcast,omitempty"` AllowPaidBroadcast bool `json:"allow_paid_broadcast,omitempty"`
MessageEffectID string `json:"message_effect_id,omitempty"` MessageEffectID string `json:"message_effect_id,omitempty"`
ReplyMarkup InlineKeyboardMarkup `json:"reply_markup,omitempty"`
} }
func (b *Bot) SendPhoto(params *SendPhotoP) (*Message, error) { func (b *Bot) SendPhoto(params *SendPhotoP) (*Message, error) {
@@ -133,12 +134,13 @@ func (b *Bot) EditMessageText(params *EditMessageTextP) (*Message, error) {
} }
type EditMessageCaptionP struct { type EditMessageCaptionP struct {
BusinessConnectionID string `json:"business_connection_id,omitempty"` BusinessConnectionID string `json:"business_connection_id,omitempty"`
ChatID int `json:"chat_id,omitempty"` ChatID int `json:"chat_id,omitempty"`
MessageID int `json:"message_id,omitempty"` MessageID int `json:"message_id,omitempty"`
InlineMessageID string `json:"inline_message_id,omitempty"` InlineMessageID string `json:"inline_message_id,omitempty"`
Caption string `json:"caption"` Caption string `json:"caption"`
ParseMode ParseMode `json:"parse_mode,omitempty"` ParseMode ParseMode `json:"parse_mode,omitempty"`
ReplyMarkup InlineKeyboardMarkup `json:"reply_markup,omitempty"`
} }
func (b *Bot) EditMessageCaption(params *EditMessageCaptionP) (*Message, error) { func (b *Bot) EditMessageCaption(params *EditMessageCaptionP) (*Message, error) {

View File

@@ -1,5 +1,7 @@
package laniakea package laniakea
import "log"
type CommandExecutor func(ctx *MsgContext, dbContext *DatabaseContext) type CommandExecutor func(ctx *MsgContext, dbContext *DatabaseContext)
type PluginBuilder struct { type PluginBuilder struct {
@@ -44,8 +46,8 @@ func (p *PluginBuilder) UpdateListener(listener CommandExecutor) *PluginBuilder
} }
func (p *PluginBuilder) Build() *Plugin { func (p *PluginBuilder) Build() *Plugin {
if len(p.commands) == 0 { if len(p.commands) == 0 && len(p.payloads) == 0 {
return nil log.Println("no command or payloads")
} }
plugin := &Plugin{ plugin := &Plugin{
Name: p.name, Name: p.name,