mirror of
https://github.com/elyby/chrly.git
synced 2025-01-25 21:12:09 +05:30
220 lines
5.7 KiB
Go
220 lines
5.7 KiB
Go
package worker
|
|
|
|
import (
|
|
"encoding/json"
|
|
"strconv"
|
|
|
|
"github.com/assembla/cony"
|
|
"github.com/mono83/slf/wd"
|
|
"github.com/streadway/amqp"
|
|
|
|
"elyby/minecraft-skinsystem/db"
|
|
"elyby/minecraft-skinsystem/interfaces"
|
|
"elyby/minecraft-skinsystem/model"
|
|
)
|
|
|
|
type Services struct {
|
|
AmqpClient *cony.Client
|
|
SkinsRepo interfaces.SkinsRepository
|
|
AccountsAPI interfaces.AccountsAPI
|
|
Logger wd.Watchdog
|
|
}
|
|
|
|
type UsernameChanged struct {
|
|
AccountId int `json:"accountId"`
|
|
OldUsername string `json:"oldUsername"`
|
|
NewUsername string `json:"newUsername"`
|
|
}
|
|
|
|
type SkinChanged struct {
|
|
AccountId int `json:"userId"`
|
|
Uuid string `json:"uuid"`
|
|
SkinId int `json:"skinId"`
|
|
Hash string `json:"hash"`
|
|
Is1_8 bool `json:"is1_8"`
|
|
IsSlim bool `json:"isSlim"`
|
|
Url string `json:"url"`
|
|
MojangTextures string `json:"mojangTextures"`
|
|
MojangSignature string `json:"mojangSignature"`
|
|
}
|
|
|
|
const exchangeName string = "events"
|
|
const queueName string = "skinsystem-accounts-events"
|
|
|
|
func (service *Services) Run() error {
|
|
clientErrs, consumerErrs, deliveryChannel := setupClient(service.AmqpClient)
|
|
shouldReturnError := true
|
|
|
|
for service.AmqpClient.Loop() {
|
|
select {
|
|
case msg := <-deliveryChannel:
|
|
shouldReturnError = false
|
|
service.HandleDelivery(&msg)
|
|
case err := <-consumerErrs:
|
|
if shouldReturnError {
|
|
return err
|
|
}
|
|
|
|
service.Logger.Error("Consume error: :err", wd.ErrParam(err))
|
|
case err := <-clientErrs:
|
|
if shouldReturnError {
|
|
return err
|
|
}
|
|
|
|
service.Logger.Error("Client error: :err", wd.ErrParam(err))
|
|
}
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (service *Services) HandleDelivery(delivery *amqp.Delivery) {
|
|
service.Logger.Debug("Incoming message with routing key " + delivery.RoutingKey)
|
|
var result bool = true
|
|
switch delivery.RoutingKey {
|
|
case "accounts.username-changed":
|
|
var event *UsernameChanged
|
|
json.Unmarshal(delivery.Body, &event)
|
|
result = service.HandleChangeUsername(event)
|
|
case "accounts.skin-changed":
|
|
var event *SkinChanged
|
|
json.Unmarshal(delivery.Body, &event)
|
|
result = service.HandleSkinChanged(event)
|
|
default:
|
|
service.Logger.Info("Unknown delivery with routing key " + delivery.RoutingKey)
|
|
delivery.Ack(false)
|
|
return
|
|
}
|
|
|
|
if result {
|
|
delivery.Ack(false)
|
|
} else {
|
|
delivery.Reject(true)
|
|
}
|
|
}
|
|
|
|
func (service *Services) HandleChangeUsername(event *UsernameChanged) bool {
|
|
service.Logger.IncCounter("worker.change_username", 1)
|
|
if event.OldUsername == "" {
|
|
service.Logger.IncCounter("worker.change_username_empty_old_username", 1)
|
|
record := &model.Skin{
|
|
UserId: event.AccountId,
|
|
Username: event.NewUsername,
|
|
}
|
|
|
|
service.SkinsRepo.Save(record)
|
|
|
|
return true
|
|
}
|
|
|
|
record, err := service.SkinsRepo.FindByUserId(event.AccountId)
|
|
if err != nil {
|
|
service.Logger.Info("Cannot find user id :accountId. Trying to search.", wd.IntParam("accountId", event.AccountId))
|
|
if _, isSkinNotFound := err.(*db.SkinNotFoundError); !isSkinNotFound {
|
|
service.Logger.Error("Unknown error when requesting a skin from the repository: :err", wd.ErrParam(err))
|
|
}
|
|
|
|
service.Logger.IncCounter("worker.change_username_id_not_found", 1)
|
|
record = &model.Skin{
|
|
UserId: event.AccountId,
|
|
}
|
|
}
|
|
|
|
record.Username = event.NewUsername
|
|
service.SkinsRepo.Save(record)
|
|
|
|
return true
|
|
}
|
|
|
|
// TODO: возможно стоит добавить проверку на совпадение id аккаунтов
|
|
func (service *Services) HandleSkinChanged(event *SkinChanged) bool {
|
|
service.Logger.IncCounter("worker.skin_changed", 1)
|
|
var record *model.Skin
|
|
record, err := service.SkinsRepo.FindByUserId(event.AccountId)
|
|
if err != nil {
|
|
if _, isSkinNotFound := err.(*db.SkinNotFoundError); !isSkinNotFound {
|
|
service.Logger.Error("Unknown error when requesting a skin from the repository: :err", wd.ErrParam(err))
|
|
}
|
|
|
|
service.Logger.IncCounter("worker.skin_changed_id_not_found", 1)
|
|
service.Logger.Info("Cannot find user id :accountId. Trying to search.", wd.IntParam("accountId", event.AccountId))
|
|
response, err := service.AccountsAPI.AccountInfo("id", strconv.Itoa(event.AccountId))
|
|
if err != nil {
|
|
service.Logger.IncCounter("worker.skin_changed_id_not_restored", 1)
|
|
service.Logger.Error(
|
|
"Cannot restore user info for :accountId: :err",
|
|
wd.IntParam("accountId", event.AccountId),
|
|
wd.ErrParam(err),
|
|
)
|
|
|
|
return true
|
|
}
|
|
|
|
service.Logger.IncCounter("worker.skin_changed_id_restored", 1)
|
|
service.Logger.Info("User info successfully restored.")
|
|
|
|
record = &model.Skin{
|
|
UserId: response.Id,
|
|
Username: response.Username,
|
|
}
|
|
}
|
|
|
|
record.Uuid = event.Uuid
|
|
record.SkinId = event.SkinId
|
|
record.Hash = event.Hash
|
|
record.Is1_8 = event.Is1_8
|
|
record.IsSlim = event.IsSlim
|
|
record.Url = event.Url
|
|
record.MojangTextures = event.MojangTextures
|
|
record.MojangSignature = event.MojangSignature
|
|
|
|
service.SkinsRepo.Save(record)
|
|
|
|
return true
|
|
}
|
|
|
|
func setupClient(client *cony.Client) (<-chan error, <-chan error, <-chan amqp.Delivery ) {
|
|
exchange := cony.Exchange{
|
|
Name: exchangeName,
|
|
Kind: "topic",
|
|
Durable: true,
|
|
AutoDelete: false,
|
|
}
|
|
|
|
queue := &cony.Queue{
|
|
Name: queueName,
|
|
Durable: true,
|
|
AutoDelete: false,
|
|
Exclusive: false,
|
|
}
|
|
|
|
usernameEventBinding := cony.Binding{
|
|
Exchange: exchange,
|
|
Queue: queue,
|
|
Key: "accounts.username-changed",
|
|
}
|
|
|
|
skinEventBinding := cony.Binding{
|
|
Exchange: exchange,
|
|
Queue: queue,
|
|
Key: "accounts.skin-changed",
|
|
}
|
|
|
|
declarations := []cony.Declaration{
|
|
cony.DeclareExchange(exchange),
|
|
cony.DeclareQueue(queue),
|
|
cony.DeclareBinding(usernameEventBinding),
|
|
cony.DeclareBinding(skinEventBinding),
|
|
}
|
|
|
|
client.Declare(declarations)
|
|
|
|
consumer := cony.NewConsumer(queue,
|
|
cony.Qos(10),
|
|
cony.AutoTag(),
|
|
)
|
|
client.Consume(consumer)
|
|
|
|
return client.Errors(), consumer.Errors(), consumer.Deliveries()
|
|
}
|