mirror of
https://github.com/signaller-matrix/signaller.git
synced 2024-11-08 19:51:02 +00:00
246 lines
5.9 KiB
Go
246 lines
5.9 KiB
Go
package memory
|
|
|
|
import (
|
|
"fmt"
|
|
"regexp"
|
|
"sort"
|
|
"strings"
|
|
"sync"
|
|
"time"
|
|
|
|
"github.com/signaller-matrix/signaller/internal"
|
|
"github.com/signaller-matrix/signaller/internal/models"
|
|
"github.com/signaller-matrix/signaller/internal/models/common"
|
|
"github.com/signaller-matrix/signaller/internal/models/createroom"
|
|
"github.com/signaller-matrix/signaller/internal/models/rooms"
|
|
mSync "github.com/signaller-matrix/signaller/internal/models/sync"
|
|
"github.com/wangjia184/sortedset"
|
|
)
|
|
|
|
type Backend struct {
|
|
data map[string]internal.User
|
|
rooms map[string]internal.Room
|
|
events *sortedset.SortedSet
|
|
roomAliases map[string]internal.Room
|
|
hostname string
|
|
validateUsernameFunc func(string) error // TODO: create ability to redefine validation func
|
|
mutex sync.RWMutex
|
|
}
|
|
|
|
type Token struct {
|
|
Device string
|
|
}
|
|
|
|
func NewBackend(hostname string) *Backend {
|
|
return &Backend{
|
|
hostname: hostname,
|
|
validateUsernameFunc: defaultValidationUsernameFunc,
|
|
rooms: make(map[string]internal.Room),
|
|
roomAliases: make(map[string]internal.Room),
|
|
events: sortedset.New(),
|
|
data: make(map[string]internal.User)}
|
|
}
|
|
|
|
func (backend *Backend) Register(username, password, device string) (user internal.User, token string, err models.ApiError) {
|
|
backend.mutex.Lock()
|
|
|
|
if backend.validateUsernameFunc != nil {
|
|
err := backend.validateUsernameFunc(username)
|
|
if err != nil {
|
|
return nil, "", models.NewError(models.M_INVALID_USERNAME, err.Error())
|
|
}
|
|
}
|
|
|
|
if _, ok := backend.data[username]; ok {
|
|
backend.mutex.Unlock()
|
|
return nil, "", models.NewError(models.M_USER_IN_USE, "trying to register a user ID which has been taken")
|
|
}
|
|
|
|
user = &User{
|
|
name: username,
|
|
password: password,
|
|
Tokens: make(map[string]Token),
|
|
backend: backend,
|
|
filters: make(map[string]common.Filter)}
|
|
|
|
backend.data[username] = user
|
|
|
|
backend.mutex.Unlock()
|
|
return backend.Login(username, password, device)
|
|
}
|
|
|
|
func (backend *Backend) Login(username, password, device string) (user internal.User, token string, err models.ApiError) {
|
|
backend.mutex.Lock()
|
|
defer backend.mutex.Unlock()
|
|
|
|
user, ok := backend.data[username]
|
|
if !ok {
|
|
return nil, "", models.NewError(models.M_FORBIDDEN, "wrong username")
|
|
}
|
|
|
|
if user.Password() != password {
|
|
return nil, "", models.NewError(models.M_FORBIDDEN, "wrong password")
|
|
}
|
|
|
|
token = internal.RandomString(defaultTokenSize)
|
|
|
|
backend.data[username].(*User).Tokens[token] = Token{Device: device}
|
|
|
|
return user, token, nil
|
|
}
|
|
|
|
func (backend *Backend) Sync(token string, request mSync.SyncRequest) (response *mSync.SyncReply, err models.ApiError) {
|
|
backend.mutex.Lock()
|
|
defer backend.mutex.Unlock()
|
|
|
|
return nil, nil // TODO: implement
|
|
}
|
|
|
|
func (backend *Backend) GetUserByToken(token string) internal.User {
|
|
backend.mutex.RLock()
|
|
defer backend.mutex.RUnlock()
|
|
|
|
for _, user := range backend.data {
|
|
for userToken := range user.(*User).Tokens {
|
|
if userToken == token {
|
|
return user
|
|
}
|
|
}
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (backend *Backend) GetRoomByID(id string) internal.Room {
|
|
backend.mutex.RLock()
|
|
defer backend.mutex.RUnlock()
|
|
|
|
for roomID, room := range backend.rooms {
|
|
if roomID == id {
|
|
return room
|
|
}
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (backend *Backend) GetUserByName(userName string) internal.User {
|
|
backend.mutex.RLock()
|
|
defer backend.mutex.RUnlock()
|
|
|
|
if user, exists := backend.data[userName]; exists {
|
|
return user
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (backend *Backend) PublicRooms(filter string) []internal.Room {
|
|
backend.mutex.RLock()
|
|
defer backend.mutex.RUnlock()
|
|
|
|
var rooms []internal.Room
|
|
|
|
for _, room := range backend.rooms {
|
|
if room.State() == createroom.PublicChat &&
|
|
(strings.Contains(room.Name(), filter) ||
|
|
strings.Contains(room.Topic(), filter) ||
|
|
strings.Contains(room.AliasName(), filter)) {
|
|
rooms = append(rooms, room)
|
|
}
|
|
}
|
|
|
|
sort.Sort(BySize(rooms))
|
|
|
|
return rooms
|
|
}
|
|
|
|
func (backend *Backend) GetRoomByAlias(alias string) internal.Room {
|
|
backend.mutex.RLock()
|
|
defer backend.mutex.RUnlock()
|
|
|
|
alias = internal.StripAlias(backend.hostname, alias)
|
|
|
|
if room, exists := backend.roomAliases[alias]; exists {
|
|
return room
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (backend *Backend) ValidateUsernameFunc() func(string) error {
|
|
backend.mutex.RLock()
|
|
defer backend.mutex.RUnlock()
|
|
|
|
return backend.validateUsernameFunc
|
|
}
|
|
|
|
func defaultValidationUsernameFunc(userName string) error {
|
|
const re = `^\w{5,}$`
|
|
|
|
if !regexp.MustCompile(re).MatchString(userName) {
|
|
return fmt.Errorf("username does not match %s", re)
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (backend *Backend) GetEventByID(id string) rooms.Event {
|
|
backend.mutex.RLock()
|
|
defer backend.mutex.RUnlock()
|
|
|
|
return backend.events.GetByKey(id).Value.(rooms.Event)
|
|
}
|
|
|
|
func (backend *Backend) PutEvent(event rooms.Event) error {
|
|
backend.mutex.Lock()
|
|
defer backend.mutex.Unlock()
|
|
|
|
backend.events.AddOrUpdate(event.EventID, sortedset.SCORE(time.Now().Unix()), event)
|
|
|
|
return nil
|
|
}
|
|
|
|
func (backend *Backend) GetEventsSince(user User, sinceToken string, limit int) []rooms.Event {
|
|
sinceEventNode := backend.events.GetByKey(sinceToken)
|
|
sEvents := backend.events.GetByScoreRange(sinceEventNode.Score(), -1, &sortedset.GetByScoreRangeOptions{
|
|
Limit: limit,
|
|
})
|
|
|
|
events := extractEventsFromNodes(sEvents)
|
|
|
|
var returnEvents []rooms.Event
|
|
for _, event := range events {
|
|
if isEventRelatedToUser(event, user) {
|
|
returnEvents = append(returnEvents, event)
|
|
}
|
|
}
|
|
|
|
return returnEvents
|
|
}
|
|
|
|
func extractEventsFromNodes(nodes []*sortedset.SortedSetNode) []rooms.Event {
|
|
var events []rooms.Event
|
|
for _, e := range nodes {
|
|
events = append(events, e.Value.(rooms.Event))
|
|
}
|
|
|
|
return events
|
|
}
|
|
|
|
func isEventRelatedToUser(event rooms.Event, user User) bool {
|
|
if internal.InArray(event.RoomID, extractRoomIDsFromModel(user.JoinedRooms())) {
|
|
return true
|
|
}
|
|
return false
|
|
}
|
|
|
|
func extractRoomIDsFromModel(rooms []internal.Room) []string {
|
|
var roomIDs []string
|
|
for _, room := range rooms {
|
|
roomIDs = append(roomIDs, room.ID())
|
|
}
|
|
|
|
return roomIDs
|
|
}
|