mirror of
https://github.com/diamondburned/cchat-mock.git
synced 2024-11-14 01:42:45 +00:00
141 lines
3 KiB
Go
141 lines
3 KiB
Go
// Package mock contains a mock cchat backend.
|
|
package mock
|
|
|
|
import (
|
|
"encoding/json"
|
|
"errors"
|
|
"strconv"
|
|
|
|
"github.com/diamondburned/cchat"
|
|
"github.com/diamondburned/cchat/services"
|
|
)
|
|
|
|
func init() {
|
|
services.RegisterService(&Service{})
|
|
}
|
|
|
|
// ErrInvalidSession is returned if SessionRestore is given a bad session.
|
|
var ErrInvalidSession = errors.New("invalid session")
|
|
|
|
type Service struct{}
|
|
|
|
var (
|
|
_ cchat.Service = (*Service)(nil)
|
|
_ cchat.Configurator = (*Service)(nil)
|
|
_ cchat.SessionRestorer = (*Service)(nil)
|
|
)
|
|
|
|
func (s Service) Name() string {
|
|
return "Mock"
|
|
}
|
|
|
|
func (s Service) RestoreSession(storage map[string]string) (cchat.Session, error) {
|
|
username, ok := storage["username"]
|
|
if !ok {
|
|
return nil, ErrInvalidSession
|
|
}
|
|
|
|
return newSession(username), nil
|
|
}
|
|
|
|
func (s Service) Authenticate() cchat.Authenticator {
|
|
return Authenticator{}
|
|
}
|
|
|
|
type Authenticator struct{}
|
|
|
|
var _ cchat.Authenticator = (*Authenticator)(nil)
|
|
|
|
func (Authenticator) AuthenticateForm() []cchat.AuthenticateEntry {
|
|
return []cchat.AuthenticateEntry{
|
|
{
|
|
Name: "Username",
|
|
},
|
|
{
|
|
Name: "Password (ignored)",
|
|
Secret: true,
|
|
},
|
|
{
|
|
Name: "Paragraph (ignored)",
|
|
Multiline: true,
|
|
},
|
|
}
|
|
}
|
|
|
|
func (Authenticator) Authenticate(form []string) (cchat.Session, error) {
|
|
return newSession(form[0]), nil
|
|
}
|
|
|
|
var (
|
|
// channel.go @ emulateAustralianInternet
|
|
internetCanFail = true
|
|
// 500ms ~ 3s
|
|
internetMinLatency = 500
|
|
internetMaxLatency = 3000
|
|
)
|
|
|
|
func (s Service) Configuration() (map[string]string, error) {
|
|
return map[string]string{
|
|
"internet.canFail": strconv.FormatBool(internetCanFail),
|
|
"internet.minLatency": strconv.Itoa(internetMinLatency),
|
|
"internet.maxLatency": strconv.Itoa(internetMaxLatency),
|
|
}, nil
|
|
}
|
|
|
|
func (s Service) SetConfiguration(config map[string]string) error {
|
|
for _, err := range []error{
|
|
// shit code, would not recommend. It's only an ok-ish idea here because
|
|
// unmarshalConfig() returns ErrInvalidConfigAtField.
|
|
unmarshalConfig(config, "internet.canFail", &internetCanFail),
|
|
unmarshalConfig(config, "internet.minLatency", &internetMinLatency),
|
|
unmarshalConfig(config, "internet.maxLatency", &internetMaxLatency),
|
|
} {
|
|
if err != nil {
|
|
return err
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func unmarshalConfig(config map[string]string, key string, value interface{}) error {
|
|
if err := json.Unmarshal([]byte(config[key]), value); err != nil {
|
|
return &cchat.ErrInvalidConfigAtField{
|
|
Key: key,
|
|
Err: err,
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
type Session struct {
|
|
username string
|
|
servers []cchat.Server
|
|
lastid uint32 // used for generation
|
|
}
|
|
|
|
var (
|
|
_ cchat.Session = (*Session)(nil)
|
|
_ cchat.SessionSaver = (*Session)(nil)
|
|
)
|
|
|
|
func newSession(username string) *Session {
|
|
ses := &Session{username: username}
|
|
ses.servers = GenerateServers(ses)
|
|
return ses
|
|
}
|
|
|
|
func (s *Session) Name() (string, error) {
|
|
return s.username, nil
|
|
}
|
|
|
|
func (s *Session) Servers(container cchat.ServersContainer) error {
|
|
container.SetServers(s.servers)
|
|
return nil
|
|
}
|
|
|
|
func (s *Session) Save() (map[string]string, error) {
|
|
return map[string]string{
|
|
"username": s.username,
|
|
}, nil
|
|
}
|