[hw-3] loms service

This commit is contained in:
Никита Шубин
2025-06-20 10:11:59 +00:00
parent c8e056bc99
commit b88dfe6db5
73 changed files with 8837 additions and 52 deletions

119
loms/internal/app/app.go Normal file
View File

@@ -0,0 +1,119 @@
package app
import (
"context"
"fmt"
"net"
"net/http"
"os"
"time"
"github.com/grpc-ecosystem/grpc-gateway/v2/runtime"
"github.com/rs/zerolog"
"github.com/rs/zerolog/log"
"google.golang.org/grpc"
"google.golang.org/grpc/credentials/insecure"
"google.golang.org/grpc/reflection"
"route256/loms/internal/app/server"
ordersRepository "route256/loms/internal/domain/repository/orders"
stocksRepository "route256/loms/internal/domain/repository/stocks"
"route256/loms/internal/domain/service"
"route256/loms/internal/infra/config"
mw "route256/loms/internal/infra/grpc/middleware"
pb "route256/pkg/api/loms/v1"
)
type App struct {
config *config.Config
controller *server.Server
}
func NewApp(configPath string) (*App, error) {
c, err := config.LoadConfig(configPath)
if err != nil {
return nil, fmt.Errorf("unable to load config: %w", err)
}
log.Logger = log.Output(zerolog.ConsoleWriter{Out: os.Stderr})
zerolog.SetGlobalLevel(zerolog.InfoLevel)
if c.Service.LogLevel != "" {
level, logErr := zerolog.ParseLevel(c.Service.LogLevel)
if logErr != nil {
return nil, fmt.Errorf("unknown log level `%s` provided: %w", c.Service.LogLevel, logErr)
}
zerolog.SetGlobalLevel(level)
}
log.WithLevel(zerolog.GlobalLevel()).Msgf("using logging level=`%s`", zerolog.GlobalLevel().String())
stockRepo, err := stocksRepository.NewInMemoryRepository(100)
if err != nil {
return nil, fmt.Errorf("stocksRepository.NewInMemoryRepository: %w", err)
}
orderRepo := ordersRepository.NewInMemoryRepository(100)
service := service.NewLomsService(orderRepo, stockRepo)
controller := server.NewServer(service)
app := &App{
config: c,
controller: controller,
}
return app, nil
}
func (app *App) ListenAndServe() error {
grpcAddr := fmt.Sprintf("%s:%s", app.config.Service.Host, app.config.Service.GRPCPort)
l, err := net.Listen("tcp", grpcAddr)
if err != nil {
return err
}
grpcServer := grpc.NewServer(
grpc.ChainUnaryInterceptor(
mw.Logging,
mw.Validate,
),
)
reflection.Register(grpcServer)
pb.RegisterLOMSServer(grpcServer, app.controller)
go func() {
if err = grpcServer.Serve(l); err != nil {
log.Fatal().Msgf("failed to serve: %v", err)
}
}()
log.Info().Msgf("Serving grpc loms at grpc://%s", l.Addr())
// Setup HTTP gateway
conn, err := grpc.NewClient(
grpcAddr,
grpc.WithTransportCredentials(insecure.NewCredentials()),
)
if err != nil {
return fmt.Errorf("grpc.NewClient: %w", err)
}
gwmux := runtime.NewServeMux()
if err = pb.RegisterLOMSHandler(context.Background(), gwmux, conn); err != nil {
return fmt.Errorf("pb.RegisterLOMSHandler: %w", err)
}
gwServer := &http.Server{
Addr: fmt.Sprintf("%s:%s", app.config.Service.Host, app.config.Service.HTTPPort),
Handler: gwmux,
ReadTimeout: 10 * time.Second,
}
log.Info().Msgf("Serving http loms at http://%s", gwServer.Addr)
return gwServer.ListenAndServe()
}

View File

@@ -0,0 +1,148 @@
package server
import (
"context"
"fmt"
"github.com/pkg/errors"
"google.golang.org/grpc/codes"
"google.golang.org/grpc/status"
"google.golang.org/protobuf/types/known/emptypb"
"route256/loms/internal/domain/entity"
"route256/loms/internal/domain/model"
pb "route256/pkg/api/loms/v1"
)
var _ pb.LOMSServer = (*Server)(nil)
type LomsService interface {
OrderCreate(ctx context.Context, orderReq *pb.OrderCreateRequest) (entity.ID, error)
OrderInfo(ctx context.Context, orderID entity.ID) (*entity.Order, error)
OrderPay(ctx context.Context, orderID entity.ID) error
OrderCancel(ctx context.Context, orderID entity.ID) error
StocksInfo(ctx context.Context, sku entity.Sku) (uint32, error)
}
type Server struct {
pb.UnimplementedLOMSServer
service LomsService
}
func NewServer(lomsService LomsService) *Server {
return &Server{
service: lomsService,
}
}
func mapOrderStatus(pbStatus pb.OrderStatus) (string, error) {
switch pbStatus {
case pb.OrderStatus_ORDER_STATUS_AWAITING_PAYMENT:
return "awaiting payment", nil
case pb.OrderStatus_ORDER_STATUS_CANCELLED:
return "cancelled", nil
case pb.OrderStatus_ORDER_STATUS_FAILED:
return "failed", nil
case pb.OrderStatus_ORDER_STATUS_NEW:
return "new", nil
case pb.OrderStatus_ORDER_STATUS_PAYED:
return "payed", nil
default:
return "", fmt.Errorf("unexpected OrderStatus: %v", pbStatus)
}
}
func (s *Server) OrderCreate(ctx context.Context, req *pb.OrderCreateRequest) (*pb.OrderCreateResponse, error) {
id, err := s.service.OrderCreate(ctx, req)
switch {
case errors.Is(err, model.ErrInvalidInput):
return nil, status.Error(codes.InvalidArgument, err.Error())
case errors.Is(err, model.ErrNotEnoughStocks):
return nil, status.Error(codes.FailedPrecondition, err.Error())
case err != nil:
return nil, status.Error(codes.Internal, err.Error())
}
return &pb.OrderCreateResponse{OrderId: int64(id)}, nil
}
func (s *Server) OrderInfo(ctx context.Context, req *pb.OrderInfoRequest) (*pb.OrderInfoResponse, error) {
ord, err := s.service.OrderInfo(ctx, entity.ID(req.OrderId))
switch {
case errors.Is(err, model.ErrInvalidInput):
return nil, status.Error(codes.InvalidArgument, err.Error())
case errors.Is(err, model.ErrOrderNotFound):
return nil, status.Error(codes.NotFound, err.Error())
case err != nil:
return nil, status.Error(codes.Internal, err.Error())
}
items := make([]*pb.OrderItem, len(ord.Items))
for i, item := range ord.Items {
items[i] = &pb.OrderItem{
Sku: int64(item.ID),
Count: item.Count,
}
}
orderStatus, err := mapOrderStatus(pb.OrderStatus(pb.OrderStatus_value[ord.Status]))
if err != nil {
return nil, err
}
resp := &pb.OrderInfoResponse{
Status: orderStatus,
UserId: int64(ord.UserID),
Items: items,
}
return resp, nil
}
func (s *Server) OrderPay(ctx context.Context, req *pb.OrderPayRequest) (*emptypb.Empty, error) {
err := s.service.OrderPay(ctx, entity.ID(req.OrderId))
switch {
case errors.Is(err, model.ErrInvalidInput):
return nil, status.Error(codes.InvalidArgument, err.Error())
case errors.Is(err, model.ErrOrderNotFound):
return nil, status.Error(codes.NotFound, err.Error())
case errors.Is(err, model.ErrOrderInvalidStatus):
return nil, status.Error(codes.FailedPrecondition, err.Error())
case err != nil:
return nil, status.Error(codes.Internal, err.Error())
}
return &emptypb.Empty{}, nil
}
func (s *Server) OrderCancel(ctx context.Context, req *pb.OrderCancelRequest) (*emptypb.Empty, error) {
err := s.service.OrderCancel(ctx, entity.ID(req.OrderId))
switch {
case errors.Is(err, model.ErrInvalidInput):
return nil, status.Error(codes.InvalidArgument, err.Error())
case errors.Is(err, model.ErrOrderNotFound):
return nil, status.Error(codes.NotFound, err.Error())
case errors.Is(err, model.ErrOrderInvalidStatus):
return nil, status.Error(codes.FailedPrecondition, err.Error())
case err != nil:
return nil, status.Error(codes.Internal, err.Error())
}
return &emptypb.Empty{}, nil
}
func (s *Server) StocksInfo(ctx context.Context, req *pb.StocksInfoRequest) (*pb.StocksInfoResponse, error) {
count, err := s.service.StocksInfo(ctx, entity.Sku(req.Sku))
switch {
case errors.Is(err, model.ErrInvalidInput):
return nil, status.Error(codes.InvalidArgument, err.Error())
case errors.Is(err, model.ErrOrderNotFound), errors.Is(err, model.ErrUnknownStock):
return nil, status.Error(codes.NotFound, err.Error())
case err != nil:
return nil, status.Error(codes.Internal, err.Error())
}
return &pb.StocksInfoResponse{Count: count}, nil
}