// Copyright (c) 2020 Shivaram Lingamneni // released under the MIT license package irc import ( "bufio" "fmt" "os" "strconv" "time" "github.com/ergochat/ergo/irc/history" "github.com/ergochat/ergo/irc/modes" "github.com/ergochat/ergo/irc/utils" ) type CanDelete uint const ( canDeleteAny CanDelete = iota // User is allowed to delete any message (for a given channel/PM) canDeleteSelf // User is allowed to delete their own messages (ditto) canDeleteNone // User is not allowed to delete any message (ditto) ) const ( histservHelp = `HistServ provides commands related to history.` ) func histservEnabled(config *Config) bool { return config.History.Enabled } func historyComplianceEnabled(config *Config) bool { return config.History.Enabled && config.History.Persistent.Enabled && config.History.Retention.EnableAccountIndexing } var ( histservCommands = map[string]*serviceCommand{ "forget": { handler: histservForgetHandler, help: `Syntax: $bFORGET $b FORGET deletes all history messages sent by an account.`, helpShort: `$bFORGET$b deletes all history messages sent by an account.`, capabs: []string{"history"}, enabled: histservEnabled, minParams: 1, maxParams: 1, }, "delete": { handler: histservDeleteHandler, help: `Syntax: $bDELETE $b DELETE deletes an individual message by its msgid. The target is the channel name. The msgid is the ID as can be found in the tags of that message.`, helpShort: `$bDELETE$b deletes an individual message by its target and msgid.`, enabled: histservEnabled, minParams: 2, maxParams: 2, }, "export": { handler: histservExportHandler, help: `Syntax: $bEXPORT $b EXPORT exports all messages sent by an account as JSON. This can be used at the request of the account holder.`, helpShort: `$bEXPORT$b exports all messages sent by an account as JSON.`, enabled: historyComplianceEnabled, capabs: []string{"history"}, minParams: 1, maxParams: 1, }, "play": { handler: histservPlayHandler, help: `Syntax: $bPLAY [limit]$b PLAY plays back history messages, rendering them into direct messages from HistServ. 'target' is a channel name or nickname to query, and 'limit' is a message count or a time duration. Note that message playback may be incomplete or degraded, relative to direct playback from /HISTORY or CHATHISTORY.`, helpShort: `$bPLAY$b plays back history messages.`, enabled: histservEnabled, minParams: 1, maxParams: 2, }, } ) func histservForgetHandler(service *ircService, server *Server, client *Client, command string, params []string, rb *ResponseBuffer) { accountName := server.accounts.AccountToAccountName(params[0]) if accountName == "" { service.Notice(rb, client.t("Could not look up account name, proceeding anyway")) accountName = params[0] } server.ForgetHistory(accountName) service.Notice(rb, fmt.Sprintf(client.t("Enqueued account %s for message deletion"), accountName)) } // Returns: // // 1. `canDeleteAny` if the client allowed to delete other users' messages from the target, ie.: // - the client is a channel operator, or // - the client is an operator with "history" capability // // 2. `canDeleteSelf` if the client is allowed to delete their own messages from the target // 3. `canDeleteNone` otherwise func deletionPolicy(server *Server, client *Client, target string) CanDelete { isOper := client.HasRoleCapabs("history") if isOper { return canDeleteAny } else { if server.Config().History.Retention.AllowIndividualDelete { channel := server.channels.Get(target) if channel != nil && channel.ClientIsAtLeast(client, modes.Operator) { return canDeleteAny } else { return canDeleteSelf } } else { return canDeleteNone } } } func histservDeleteHandler(service *ircService, server *Server, client *Client, command string, params []string, rb *ResponseBuffer) { target, msgid := params[0], params[1] // Fix #1881 2 params are required canDelete := deletionPolicy(server, client, target) accountName := "*" if canDelete == canDeleteNone { service.Notice(rb, client.t("Insufficient privileges")) return } else if canDelete == canDeleteSelf { accountName = client.AccountName() if accountName == "*" { service.Notice(rb, client.t("Insufficient privileges")) return } } err := server.DeleteMessage(target, msgid, accountName) if err == nil { service.Notice(rb, client.t("Successfully deleted message")) } else { isOper := client.HasRoleCapabs("history") if isOper { service.Notice(rb, fmt.Sprintf(client.t("Error deleting message: %v"), err)) } else { service.Notice(rb, client.t("Could not delete message")) } } } func histservExportHandler(service *ircService, server *Server, client *Client, command string, params []string, rb *ResponseBuffer) { cfAccount, err := CasefoldName(params[0]) if err != nil { service.Notice(rb, client.t("Invalid account name")) return } config := server.Config() // don't include the account name in the filename because of escaping concerns filename := fmt.Sprintf("%s-%s.json", utils.GenerateSecretToken(), time.Now().UTC().Format(IRCv3TimestampFormat)) pathname := config.getOutputPath(filename) outfile, err := os.Create(pathname) if err != nil { service.Notice(rb, fmt.Sprintf(client.t("Error opening export file: %v"), err)) } else { service.Notice(rb, fmt.Sprintf(client.t("Started exporting data for account %[1]s to file %[2]s"), cfAccount, filename)) } go histservExportAndNotify(service, server, cfAccount, outfile, filename, client.Nick()) } func histservExportAndNotify(service *ircService, server *Server, cfAccount string, outfile *os.File, filename, alertNick string) { defer server.HandlePanic(nil) defer outfile.Close() writer := bufio.NewWriter(outfile) defer writer.Flush() server.historyDB.Export(cfAccount, writer) client := server.clients.Get(alertNick) if client != nil && client.HasRoleCapabs("history") { client.Send(nil, service.prefix, "NOTICE", client.Nick(), fmt.Sprintf(client.t("Data export for %[1]s completed and written to %[2]s"), cfAccount, filename)) } } func histservPlayHandler(service *ircService, server *Server, client *Client, command string, params []string, rb *ResponseBuffer) { items, _, err := easySelectHistory(server, client, params) if err != nil { service.Notice(rb, client.t("Could not retrieve history")) return } playMessage := func(timestamp time.Time, nick, message string) { service.Notice(rb, fmt.Sprintf("%s <%s> %s", timestamp.Format("15:04:05"), NUHToNick(nick), message)) } for _, item := range items { // TODO: support a few more of these, maybe JOIN/PART/QUIT if item.Type != history.Privmsg && item.Type != history.Notice { continue } if len(item.Message.Split) == 0 { playMessage(item.Message.Time, item.Nick, item.Message.Message) } else { for _, pair := range item.Message.Split { playMessage(item.Message.Time, item.Nick, pair.Message) } } } service.Notice(rb, client.t("End of history playback")) } // handles parameter parsing and history queries for /HISTORY and /HISTSERV PLAY func easySelectHistory(server *Server, client *Client, params []string) (items []history.Item, channel *Channel, err error) { channel, sequence, err := server.GetHistorySequence(nil, client, params[0]) if sequence == nil || err != nil { return nil, nil, errNoSuchChannel } var duration time.Duration maxChathistoryLimit := server.Config().History.ChathistoryMax limit := 100 if maxChathistoryLimit < limit { limit = maxChathistoryLimit } if len(params) > 1 { providedLimit, err := strconv.Atoi(params[1]) if err == nil && providedLimit != 0 { limit = providedLimit if maxChathistoryLimit < limit { limit = maxChathistoryLimit } } else if err != nil { duration, err = time.ParseDuration(params[1]) if err == nil { limit = maxChathistoryLimit } } } if duration == 0 { items, err = sequence.Between(history.Selector{}, history.Selector{}, limit) } else { now := time.Now().UTC() start := history.Selector{Time: now} end := history.Selector{Time: now.Add(-duration)} items, err = sequence.Between(start, end, limit) } return }