mirror of
https://github.com/ergochat/ergo.git
synced 2025-01-10 12:12:37 +01:00
201 lines
4.3 KiB
Go
201 lines
4.3 KiB
Go
// Copyright (c) 2016-2017 Daniel Oaks <daniel@danieloaks.net>
|
|
// released under the MIT license
|
|
|
|
// viewing and modifying accounts, registered channels, dlines, rehashing, etc
|
|
|
|
package irc
|
|
|
|
import (
|
|
"encoding/json"
|
|
"net/http"
|
|
"strconv"
|
|
"time"
|
|
|
|
"fmt"
|
|
|
|
"github.com/gorilla/mux"
|
|
"github.com/tidwall/buntdb"
|
|
)
|
|
|
|
const restErr = "{\"error\":\"An unknown error occurred\"}"
|
|
|
|
// ircServer is used to keep a link to the current running server since this is the best
|
|
// way to do it, given how HTTP handlers dispatch and work.
|
|
var ircServer *Server
|
|
|
|
type restInfoResp struct {
|
|
ServerName string `json:"server-name"`
|
|
NetworkName string `json:"network-name"`
|
|
|
|
Version string `json:"version"`
|
|
}
|
|
|
|
type restStatusResp struct {
|
|
Clients int `json:"clients"`
|
|
Opers int `json:"opers"`
|
|
Channels int `json:"channels"`
|
|
}
|
|
|
|
type restXLinesResp struct {
|
|
DLines map[string]IPBanInfo `json:"dlines"`
|
|
KLines map[string]IPBanInfo `json:"klines"`
|
|
}
|
|
|
|
type restAcct struct {
|
|
Name string `json:"name"`
|
|
RegisteredAt time.Time `json:"registered-at"`
|
|
Clients int `json:"clients"`
|
|
}
|
|
|
|
type restAccountsResp struct {
|
|
Verified map[string]restAcct `json:"verified"`
|
|
}
|
|
|
|
type restRehashResp struct {
|
|
Successful bool `json:"successful"`
|
|
Error string `json:"error"`
|
|
Time time.Time `json:"time"`
|
|
}
|
|
|
|
func restInfo(w http.ResponseWriter, r *http.Request) {
|
|
rs := restInfoResp{
|
|
Version: SemVer,
|
|
ServerName: ircServer.name,
|
|
NetworkName: ircServer.networkName,
|
|
}
|
|
b, err := json.Marshal(rs)
|
|
if err != nil {
|
|
fmt.Fprintln(w, restErr)
|
|
} else {
|
|
fmt.Fprintln(w, string(b))
|
|
}
|
|
}
|
|
|
|
func restStatus(w http.ResponseWriter, r *http.Request) {
|
|
rs := restStatusResp{
|
|
Clients: ircServer.clients.Count(),
|
|
Opers: len(ircServer.operators),
|
|
Channels: ircServer.channels.Len(),
|
|
}
|
|
b, err := json.Marshal(rs)
|
|
if err != nil {
|
|
fmt.Fprintln(w, restErr)
|
|
} else {
|
|
fmt.Fprintln(w, string(b))
|
|
}
|
|
}
|
|
|
|
func restGetXLines(w http.ResponseWriter, r *http.Request) {
|
|
rs := restXLinesResp{
|
|
DLines: ircServer.dlines.AllBans(),
|
|
KLines: ircServer.klines.AllBans(),
|
|
}
|
|
b, err := json.Marshal(rs)
|
|
if err != nil {
|
|
fmt.Fprintln(w, restErr)
|
|
} else {
|
|
fmt.Fprintln(w, string(b))
|
|
}
|
|
}
|
|
|
|
func restGetAccounts(w http.ResponseWriter, r *http.Request) {
|
|
rs := restAccountsResp{
|
|
Verified: make(map[string]restAcct),
|
|
}
|
|
|
|
// get accounts
|
|
err := ircServer.store.View(func(tx *buntdb.Tx) error {
|
|
tx.AscendKeys("account.exists *", func(key, value string) bool {
|
|
key = key[len("account.exists "):]
|
|
_, err := tx.Get(fmt.Sprintf(keyAccountVerified, key))
|
|
verified := err == nil
|
|
fmt.Println(fmt.Sprintf(keyAccountVerified, key))
|
|
|
|
// get other details
|
|
name, _ := tx.Get(fmt.Sprintf(keyAccountName, key))
|
|
regTimeStr, _ := tx.Get(fmt.Sprintf(keyAccountRegTime, key))
|
|
regTimeInt, _ := strconv.ParseInt(regTimeStr, 10, 64)
|
|
regTime := time.Unix(regTimeInt, 0)
|
|
|
|
var clients int
|
|
acct := ircServer.accounts[key]
|
|
if acct != nil {
|
|
clients = len(acct.Clients)
|
|
}
|
|
|
|
if verified {
|
|
rs.Verified[key] = restAcct{
|
|
Name: name,
|
|
RegisteredAt: regTime,
|
|
Clients: clients,
|
|
}
|
|
} else {
|
|
//TODO(dan): Add to unverified list
|
|
}
|
|
|
|
return true // true to continue I guess?
|
|
})
|
|
|
|
return nil
|
|
})
|
|
|
|
b, err := json.Marshal(rs)
|
|
if err != nil {
|
|
fmt.Fprintln(w, restErr)
|
|
} else {
|
|
fmt.Fprintln(w, string(b))
|
|
}
|
|
}
|
|
|
|
func restRehash(w http.ResponseWriter, r *http.Request) {
|
|
err := ircServer.rehash()
|
|
|
|
rs := restRehashResp{
|
|
Successful: err == nil,
|
|
Time: time.Now(),
|
|
}
|
|
if err != nil {
|
|
rs.Error = err.Error()
|
|
}
|
|
|
|
b, err := json.Marshal(rs)
|
|
if err != nil {
|
|
fmt.Fprintln(w, restErr)
|
|
} else {
|
|
fmt.Fprintln(w, string(b))
|
|
}
|
|
}
|
|
|
|
func StartRestAPI(s *Server, listenAddr string) (*http.Server, error) {
|
|
// so handlers can ref it later
|
|
ircServer = s
|
|
|
|
// start router
|
|
r := mux.NewRouter()
|
|
|
|
// GET methods
|
|
rg := r.Methods("GET").Subrouter()
|
|
rg.HandleFunc("/info", restInfo)
|
|
rg.HandleFunc("/status", restStatus)
|
|
rg.HandleFunc("/xlines", restGetXLines)
|
|
rg.HandleFunc("/accounts", restGetAccounts)
|
|
|
|
// PUT methods
|
|
rp := r.Methods("POST").Subrouter()
|
|
rp.HandleFunc("/rehash", restRehash)
|
|
|
|
// start api
|
|
httpserver := http.Server{
|
|
Addr: listenAddr,
|
|
Handler: r,
|
|
}
|
|
|
|
go func() {
|
|
if err := httpserver.ListenAndServe(); err != nil {
|
|
s.logger.Error("listeners", fmt.Sprintf("Rest API listenAndServe error: %s", err))
|
|
}
|
|
}()
|
|
|
|
return &httpserver, nil
|
|
}
|