From 98abbd524b68064c368d935360bd16a4eefd06f8 Mon Sep 17 00:00:00 2001 From: p0t4t0sandwich Date: Sat, 20 Apr 2024 06:17:20 -0600 Subject: [PATCH] Added protobuf support to MCStatus and simplified the response behaviour (to be offloaded to the actual site) --- Makefile | 3 + modules/game_server_status/service.go | 4 +- modules/game_server_status/types.go | 12 +- modules/mcstatus/handler.go | 32 ++- modules/mcstatus/service.go | 18 +- modules/mcstatus/types.go | 236 +++++++-------- modules/proto/mcstatuspb/mcstatus.pb.go | 365 ++++++++++++++++++++++++ proto/mcstatus.proto | 27 ++ public/api/v1/openapi.json | 244 ++++++---------- responses/problem.go | 3 +- responses/response.go | 4 +- 11 files changed, 647 insertions(+), 301 deletions(-) create mode 100644 modules/proto/mcstatuspb/mcstatus.pb.go create mode 100644 proto/mcstatus.proto diff --git a/Makefile b/Makefile index d19d52e..f2dc4a2 100644 --- a/Makefile +++ b/Makefile @@ -4,3 +4,6 @@ gen: protoc -I=./proto --go_out=./modules/proto ./proto/problem.proto sed -i 's/json:"\(.*\),omitempty"/json:"\1,omitempty" xml:"\1,omitempty"/g' ./modules/proto/problempb/problem.pb.go + + protoc -I=./proto --go_out=./modules/proto ./proto/mcstatus.proto + sed -i 's/json:"\(.*\),omitempty"/json:"\1" xml:"\1"/g' ./modules/proto/mcstatuspb/mcstatus.pb.go diff --git a/modules/game_server_status/service.go b/modules/game_server_status/service.go index 2f88c7b..b2e3b15 100644 --- a/modules/game_server_status/service.go +++ b/modules/game_server_status/service.go @@ -92,11 +92,11 @@ func (s *service) QueryGameServer(game string, host string, port int) (*GameServ for _, v := range MinecraftList { if v == game { isBedrock := game != "minecraft" - response, err := mcstatus.NewService().GetServerStatus(host, port, isBedrock, true) + response, err := mcstatus.NewService().GetServerStatus(host, port, isBedrock, true, port) if err != nil { return nil, err } - return (*mcstatusResponse)(response).Normalize(), nil + return (*mcServerStatus)(response).Normalize(), nil } } for _, v := range GameQList { diff --git a/modules/game_server_status/types.go b/modules/game_server_status/types.go index d621674..8f1c22e 100644 --- a/modules/game_server_status/types.go +++ b/modules/game_server_status/types.go @@ -61,22 +61,22 @@ type GameQResponse struct { } // Type alias -type mcstatusResponse mcstatus.MCStatusResponse +type mcServerStatus mcstatus.MCServerStatus // Normalize - Normalize Minecraft response -func (mc *mcstatusResponse) Normalize() *GameServerStatus { +func (mc *mcServerStatus) Normalize() *GameServerStatus { players := make([]Player, len(mc.Players)) for i, v := range mc.Players { - players[i] = Player{Name: v.Name, ID: v.UUID.String()} + players[i] = Player{Name: v.Name, ID: v.Uuid} } return &GameServerStatus{ Host: mc.Host, - Port: mc.Port, + Port: int(mc.Port), Name: mc.Name, MapName: mc.Map, - MaxPlayers: mc.MaxPlayers, - NumPlayers: mc.NumPlayers, + MaxPlayers: int(mc.MaxPlayers), + NumPlayers: int(mc.NumPlayers), Players: players, QueryType: QueryTypeMinecraft, Raw: mc.Raw, diff --git a/modules/mcstatus/handler.go b/modules/mcstatus/handler.go index 76ef842..89fb637 100644 --- a/modules/mcstatus/handler.go +++ b/modules/mcstatus/handler.go @@ -4,6 +4,7 @@ import ( "image/png" "net/http" "strconv" + "strings" "github.com/NeuralNexusDev/neuralnexus-api/responses" ) @@ -16,14 +17,14 @@ func ApplyRoutes(mux *http.ServeMux) *http.ServeMux { return mux } -// Route that returns the server status +// GetServerStatus - Route that returns the server status func GetServerStatus(s MCStatusService) http.HandlerFunc { return func(w http.ResponseWriter, r *http.Request) { host := r.PathValue("host") isBedrock := r.URL.Query().Get("bedrock") == "true" queryEnabled := r.URL.Query().Get("query") == "true" raw := r.URL.Query().Get("raw") == "true" - port, err := strconv.Atoi(r.URL.Query().Get("port")) + port, err := strconv.Atoi(host[strings.LastIndex(host, ":")+1:]) if err != nil { if isBedrock { port = 19132 @@ -31,38 +32,39 @@ func GetServerStatus(s MCStatusService) http.HandlerFunc { port = 25565 } } + queryPort, err := strconv.Atoi(r.URL.Query().Get("query_port")) + if err != nil { + queryPort = port + } - status, err := s.GetServerStatus(host, port, isBedrock, queryEnabled) + status, err := s.GetServerStatus(host, port, isBedrock, queryEnabled, queryPort) if err != nil { - responses.SendAndEncodeInternalServerError(w, r, err.Error()) + responses.SendAndEncodeNotFound(w, r, err.Error()) return } if !raw { status.Raw = nil } - responses.SendAndEncodeStruct(w, r, http.StatusOK, status) } } -// Route that returns the server icon as a PNG (base64 encoded string didn't work for some reason) +// GetIcon - Route that returns the server icon as a PNG func GetIcon(s MCStatusService) http.HandlerFunc { return func(w http.ResponseWriter, r *http.Request) { host := r.PathValue("host") isBedrock := r.URL.Query().Get("bedrock") == "true" - queryEnabled := r.URL.Query().Get("query") == "true" - port, err := strconv.Atoi(r.URL.Query().Get("port")) + if isBedrock { + responses.SendAndEncodeBadRequest(w, r, "Bedrock servers do not have icons.") + } + port, err := strconv.Atoi(host[strings.LastIndex(host, ":")+1:]) if err != nil { - if isBedrock { - port = 19132 - } else { - port = 25565 - } + port = 25565 } - status, err := s.GetServerStatus(host, port, isBedrock, queryEnabled) + status, err := s.GetJavaServerStatus(host, port, false, 0) if err != nil { - responses.SendAndEncodeInternalServerError(w, r, err.Error()) + responses.SendAndEncodeNotFound(w, r, err.Error()) return } diff --git a/modules/mcstatus/service.go b/modules/mcstatus/service.go index 4791b0a..aa6425c 100644 --- a/modules/mcstatus/service.go +++ b/modules/mcstatus/service.go @@ -11,9 +11,9 @@ import ( // MCStatusService - Minecraft Status service type MCStatusService interface { - GetJavaServerStatus(host string, port int, queryEnabled bool) (*MCStatusResponse, error) - GetBedrockServerStatus(host string, port int) (*MCStatusResponse, error) - GetServerStatus(host string, port int, isBedrock bool, queryEnabled bool) (*MCStatusResponse, error) + GetJavaServerStatus(host string, port int, queryEnabled bool, queryPort int) (*MCServerStatus, error) + GetBedrockServerStatus(host string, port int) (*MCServerStatus, error) + GetServerStatus(host string, port int, isBedrock bool, queryEnabled bool, queryPort int) (*MCServerStatus, error) } // service - Minecraft Status service implementation @@ -25,14 +25,14 @@ func NewService() MCStatusService { } // GetJavaServerStatus - Get Java server status -func (s *service) GetJavaServerStatus(host string, port int, queryEnabled bool) (*MCStatusResponse, error) { +func (s *service) GetJavaServerStatus(host string, port int, queryEnabled bool, queryPort int) (*MCServerStatus, error) { pinger := minequery.NewPinger( minequery.WithTimeout(5*time.Second), minequery.WithProtocolVersion16(minequery.Ping16ProtocolVersion162), minequery.WithProtocolVersion17(minequery.Ping17ProtocolVersion119), ) - var status *MCStatusResponse = nil + var status *MCServerStatus = nil s17, err := pinger.Ping17(host, port) if err == nil { status = GetPing17Status(s17) @@ -62,14 +62,14 @@ func (s *service) GetJavaServerStatus(host string, port int, queryEnabled bool) } if status != nil { status.Host = host - status.Port = port + status.Port = int32(port) return status, nil } return nil, errors.New("failed to get java server status") } // GetBedrockServerStatus - Get Bedrock server status -func (s *service) GetBedrockServerStatus(host string, port int) (*MCStatusResponse, error) { +func (s *service) GetBedrockServerStatus(host string, port int) (*MCServerStatus, error) { connect := host + ":" + fmt.Sprint(port) status, err := bedrockping.Query(connect, 5*time.Second, 150*time.Millisecond) if err != nil { @@ -79,9 +79,9 @@ func (s *service) GetBedrockServerStatus(host string, port int) (*MCStatusRespon } // GetServerStatus - Get server status -func (s *service) GetServerStatus(host string, port int, isBedrock bool, queryEnabled bool) (*MCStatusResponse, error) { +func (s *service) GetServerStatus(host string, port int, isBedrock bool, queryEnabled bool, queryPort int) (*MCServerStatus, error) { if isBedrock { return s.GetBedrockServerStatus(host, port) } - return s.GetJavaServerStatus(host, port, queryEnabled) + return s.GetJavaServerStatus(host, port, queryEnabled, queryPort) } diff --git a/modules/mcstatus/types.go b/modules/mcstatus/types.go index f96783f..3a50edc 100644 --- a/modules/mcstatus/types.go +++ b/modules/mcstatus/types.go @@ -9,34 +9,19 @@ import ( "regexp" "strings" + "github.com/NeuralNexusDev/neuralnexus-api/modules/proto/mcstatuspb" "github.com/ZeroErrors/go-bedrockping" "github.com/dreamscached/minequery/v2" - "github.com/google/uuid" ) -// MCStatusResponse - Minecraft Status response -type MCStatusResponse struct { - Host string `json:"host" xml:"host"` - Port int `json:"port" xml:"port"` - Name string `json:"name" xml:"name"` - MOTD string `json:"motd" xml:"motd"` - Map string `json:"map" xml:"map"` - MaxPlayers int `json:"max_players" xml:"max_players"` - NumPlayers int `json:"num_players" xml:"num_players"` - Players []Player `json:"players" xml:"players"` - Version string `json:"version" xml:"version"` - Favicon string `json:"favicon" xml:"favicon"` +// MCServerStatus - Minecraft Status response +type MCServerStatus struct { + mcstatuspb.ServerStatus ServerType ServerType `json:"server_type" xml:"server_type"` Raw interface{} `json:"raw,omitempty" xml:"raw,omitempty"` Icon image.Image `json:"-" xml:"-"` } -// Player - Player info -type Player struct { - Name string `json:"name" xml:"name"` - UUID uuid.UUID `json:"uuid" xml:"uuid"` -} - // ServerType - Server type enum type ServerType string @@ -46,6 +31,28 @@ const ( ServerTypeBedrock ServerType = "bedrock" ) +// NewServerStatus - Create a new server status +func NewServerStatus(host string, port int, name string, motd string, mapName string, maxPlayers int, numPlayers int, players []*mcstatuspb.Player, version string, favicon string, serverType ServerType, raw interface{}, icon image.Image) *MCServerStatus { + return &MCServerStatus{ + ServerStatus: mcstatuspb.ServerStatus{ + Host: host, + Port: int32(port), + Name: name, + Motd: motd, + Map: mapName, + MaxPlayers: int32(maxPlayers), + NumPlayers: int32(numPlayers), + Players: players, + Version: version, + Favicon: favicon, + ServerType: mcstatuspb.ServerType(mcstatuspb.ServerType_value[strings.ToUpper(string(serverType))]), + }, + ServerType: serverType, + Raw: raw, + Icon: icon, + } +} + // MOTDToName - Convert MOTD to name func MOTDToName(motd string) string { name := strings.ReplaceAll(motd, "\n", " ") @@ -56,6 +63,9 @@ func MOTDToName(motd string) string { // ImgToBase64 - Convert image.Image to base64 string func ImgToBase64(i image.Image) string { + if i == nil { + return "" + } var buff bytes.Buffer png.Encode(&buff, i) var encodedString string = base64.StdEncoding.EncodeToString(buff.Bytes()) @@ -81,106 +91,121 @@ func LoadImgFromFile(path string) (image.Image, error) { } // GetPing17Status - Get the Java server status -func GetPing17Status(s *minequery.Status17) *MCStatusResponse { +func GetPing17Status(s *minequery.Status17) *MCServerStatus { icon := s.Icon s.Icon = nil // TODO: Parse the motd and keep data motd := s.Description.String() - var playerList []Player = []Player{} + playerList := []*mcstatuspb.Player{} for _, player := range s.SamplePlayers { - playerList = append(playerList, Player{ + playerList = append(playerList, &mcstatuspb.Player{ Name: player.Nickname, - UUID: player.UUID, + Uuid: player.UUID.String(), }) } - return &MCStatusResponse{ - Name: MOTDToName(motd), - MOTD: strings.ReplaceAll(motd, "\n", "\\n"), - MaxPlayers: s.MaxPlayers, - NumPlayers: s.OnlinePlayers, - Players: playerList, - Favicon: ImgToBase64(icon), - ServerType: ServerTypeJava, - Icon: icon, - Raw: s, - } + return NewServerStatus( + "", 0, + MOTDToName(motd), + strings.ReplaceAll(motd, "\n", "\\n"), + "", + s.MaxPlayers, + s.OnlinePlayers, + playerList, + s.VersionName, + ImgToBase64(icon), + ServerTypeJava, + s, + icon, + ) } // GetPing16Status - Get the Java server status -func GetPing16Status(s *minequery.Status16) *MCStatusResponse { +func GetPing16Status(s *minequery.Status16) *MCServerStatus { motd := s.MOTD - return &MCStatusResponse{ - Name: MOTDToName(motd), - MOTD: strings.ReplaceAll(motd, "\n", "\\n"), - MaxPlayers: s.MaxPlayers, - NumPlayers: s.OnlinePlayers, - Players: []Player{}, - Favicon: LegacyIcon, - ServerType: ServerTypeJava, - Icon: ImgLegacyIcon, - Raw: s, - } + return NewServerStatus( + "", 0, + MOTDToName(motd), + strings.ReplaceAll(motd, "\n", "\\n"), + "", + s.MaxPlayers, + s.OnlinePlayers, + []*mcstatuspb.Player{}, + "1.6", + "", + ServerTypeJava, + s, + nil, + ) } // GetPing14Status - Get the Java server status -func GetPing14Status(s *minequery.Status14) *MCStatusResponse { +func GetPing14Status(s *minequery.Status14) *MCServerStatus { motd := s.MOTD - return &MCStatusResponse{ - Name: MOTDToName(motd), - MOTD: strings.ReplaceAll(motd, "\n", "\\n"), - MaxPlayers: s.MaxPlayers, - NumPlayers: s.OnlinePlayers, - Players: []Player{}, - Favicon: LegacyIcon, - ServerType: ServerTypeJava, - Icon: ImgLegacyIcon, - Raw: s, - } + return NewServerStatus( + "", 0, + MOTDToName(motd), + strings.ReplaceAll(motd, "\n", "\\n"), + "", + s.MaxPlayers, + s.OnlinePlayers, + []*mcstatuspb.Player{}, + "1.4-1.5", + "", + ServerTypeJava, + s, + nil, + ) } // GetBeta18Status - Get the Java server status -func GetBeta18Status(s *minequery.StatusBeta18) *MCStatusResponse { +func GetBeta18Status(s *minequery.StatusBeta18) *MCServerStatus { motd := s.MOTD - return &MCStatusResponse{ - Name: MOTDToName(motd), - MOTD: strings.ReplaceAll(motd, "\n", "\\n"), - MaxPlayers: s.MaxPlayers, - NumPlayers: s.OnlinePlayers, - Players: []Player{}, - Favicon: LegacyIcon, - ServerType: ServerTypeJava, - Icon: ImgLegacyIcon, - Raw: s, - } + return NewServerStatus( + "", 0, + MOTDToName(motd), + strings.ReplaceAll(motd, "\n", "\\n"), + "", + s.MaxPlayers, + s.OnlinePlayers, + []*mcstatuspb.Player{}, + "b1.8-1.3", + "", + ServerTypeJava, + s, + nil, + ) } // GetQueryStatus - Get the Java server status -func GetQueryStatus(s *minequery.FullQueryStatus) *MCStatusResponse { +func GetQueryStatus(s *minequery.FullQueryStatus) *MCServerStatus { motd := s.MOTD - var playerList []Player = []Player{} + playerList := []*mcstatuspb.Player{} for _, player := range s.SamplePlayers { - playerList = append(playerList, Player{ + playerList = append(playerList, &mcstatuspb.Player{ Name: player, }) } - return &MCStatusResponse{ - Name: MOTDToName(motd), - MOTD: strings.ReplaceAll(motd, "\n", "\\n"), - MaxPlayers: s.MaxPlayers, - NumPlayers: s.OnlinePlayers, - Players: playerList, - Favicon: LegacyIcon, - ServerType: ServerTypeJava, - Icon: ImgLegacyIcon, - Raw: s, - } + return NewServerStatus( + "", 0, + MOTDToName(motd), + strings.ReplaceAll(motd, "\n", "\\n"), + "", + s.MaxPlayers, + s.OnlinePlayers, + playerList, + s.Version, + "", + ServerTypeJava, + s, + nil, + ) } // GetBedrockStatus - Get the Bedrock server status -func GetBedrockStatus(s bedrockping.Response) *MCStatusResponse { +func GetBedrockStatus(s bedrockping.Response) *MCServerStatus { motd := s.ServerName if len(s.Extra) > 1 { motd += "\\n" + s.Extra[1] @@ -191,31 +216,18 @@ func GetBedrockStatus(s bedrockping.Response) *MCStatusResponse { mapName = s.Extra[2] } - return &MCStatusResponse{ - Name: MOTDToName(motd), - MOTD: strings.ReplaceAll(motd, "\n", "\\n"), - Map: mapName, - MaxPlayers: s.MaxPlayers, - NumPlayers: s.PlayerCount, - Players: []Player{}, - Favicon: BedrockIcon, - ServerType: ServerTypeBedrock, - Icon: ImgBedrockIcon, - Raw: s, - } + return NewServerStatus( + "", 0, + MOTDToName(motd), + strings.ReplaceAll(motd, "\n", "\\n"), + mapName, + s.MaxPlayers, + s.PlayerCount, + []*mcstatuspb.Player{}, + s.MCPEVersion, + "", + ServerTypeJava, + s, + nil, + ) } - -// DefaultIcon - Default icon -const DefaultIcon = "" - -// LegacyIcon - Legacy icon -const LegacyIcon = "" - -// BedrockIcon - Bedrock icon -const BedrockIcon = "" - -var ( - ImgDefaultIcon, _ = LoadImgFromFile("public/mcstatus/icons/default.png") - ImgLegacyIcon, _ = LoadImgFromFile("public/mcstatus/icons/legacy.png") - ImgBedrockIcon, _ = LoadImgFromFile("public/mcstatus/icons/bedrock.png") -) diff --git a/modules/proto/mcstatuspb/mcstatus.pb.go b/modules/proto/mcstatuspb/mcstatus.pb.go new file mode 100644 index 0000000..6b718d7 --- /dev/null +++ b/modules/proto/mcstatuspb/mcstatus.pb.go @@ -0,0 +1,365 @@ +// Code generated by protoc-gen-go. DO NOT EDIT. +// versions: +// protoc-gen-go v1.33.0 +// protoc v5.26.1 +// source: mcstatus.proto + +package mcstatuspb + +import ( + protoreflect "google.golang.org/protobuf/reflect/protoreflect" + protoimpl "google.golang.org/protobuf/runtime/protoimpl" + reflect "reflect" + sync "sync" +) + +const ( + // Verify that this generated code is sufficiently up-to-date. + _ = protoimpl.EnforceVersion(20 - protoimpl.MinVersion) + // Verify that runtime/protoimpl is sufficiently up-to-date. + _ = protoimpl.EnforceVersion(protoimpl.MaxVersion - 20) +) + +type ServerType int32 + +const ( + ServerType_JAVA ServerType = 0 + ServerType_BEDROCK ServerType = 1 +) + +// Enum value maps for ServerType. +var ( + ServerType_name = map[int32]string{ + 0: "JAVA", + 1: "BEDROCK", + } + ServerType_value = map[string]int32{ + "JAVA": 0, + "BEDROCK": 1, + } +) + +func (x ServerType) Enum() *ServerType { + p := new(ServerType) + *p = x + return p +} + +func (x ServerType) String() string { + return protoimpl.X.EnumStringOf(x.Descriptor(), protoreflect.EnumNumber(x)) +} + +func (ServerType) Descriptor() protoreflect.EnumDescriptor { + return file_mcstatus_proto_enumTypes[0].Descriptor() +} + +func (ServerType) Type() protoreflect.EnumType { + return &file_mcstatus_proto_enumTypes[0] +} + +func (x ServerType) Number() protoreflect.EnumNumber { + return protoreflect.EnumNumber(x) +} + +// Deprecated: Use ServerType.Descriptor instead. +func (ServerType) EnumDescriptor() ([]byte, []int) { + return file_mcstatus_proto_rawDescGZIP(), []int{0} +} + +type ServerStatus struct { + state protoimpl.MessageState + sizeCache protoimpl.SizeCache + unknownFields protoimpl.UnknownFields + + Host string `protobuf:"bytes,1,opt,name=host,proto3" json:"host" xml:"host"` + Port int32 `protobuf:"varint,2,opt,name=port,proto3" json:"port" xml:"port"` + Name string `protobuf:"bytes,3,opt,name=name,proto3" json:"name" xml:"name"` + Motd string `protobuf:"bytes,4,opt,name=motd,proto3" json:"motd" xml:"motd"` + Map string `protobuf:"bytes,5,opt,name=map,proto3" json:"map" xml:"map"` + MaxPlayers int32 `protobuf:"varint,6,opt,name=max_players,json=maxPlayers,proto3" json:"max_players" xml:"max_players"` + NumPlayers int32 `protobuf:"varint,7,opt,name=num_players,json=numPlayers,proto3" json:"num_players" xml:"num_players"` + Players []*Player `protobuf:"bytes,8,rep,name=players,proto3" json:"players" xml:"players"` + Version string `protobuf:"bytes,9,opt,name=version,proto3" json:"version" xml:"version"` + Favicon string `protobuf:"bytes,10,opt,name=favicon,proto3" json:"favicon" xml:"favicon"` + ServerType ServerType `protobuf:"varint,11,opt,name=server_type,json=serverType,proto3,enum=mcstatuspb.ServerType" json:"server_type" xml:"server_type"` +} + +func (x *ServerStatus) Reset() { + *x = ServerStatus{} + if protoimpl.UnsafeEnabled { + mi := &file_mcstatus_proto_msgTypes[0] + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + ms.StoreMessageInfo(mi) + } +} + +func (x *ServerStatus) String() string { + return protoimpl.X.MessageStringOf(x) +} + +func (*ServerStatus) ProtoMessage() {} + +func (x *ServerStatus) ProtoReflect() protoreflect.Message { + mi := &file_mcstatus_proto_msgTypes[0] + if protoimpl.UnsafeEnabled && x != nil { + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + if ms.LoadMessageInfo() == nil { + ms.StoreMessageInfo(mi) + } + return ms + } + return mi.MessageOf(x) +} + +// Deprecated: Use ServerStatus.ProtoReflect.Descriptor instead. +func (*ServerStatus) Descriptor() ([]byte, []int) { + return file_mcstatus_proto_rawDescGZIP(), []int{0} +} + +func (x *ServerStatus) GetHost() string { + if x != nil { + return x.Host + } + return "" +} + +func (x *ServerStatus) GetPort() int32 { + if x != nil { + return x.Port + } + return 0 +} + +func (x *ServerStatus) GetName() string { + if x != nil { + return x.Name + } + return "" +} + +func (x *ServerStatus) GetMotd() string { + if x != nil { + return x.Motd + } + return "" +} + +func (x *ServerStatus) GetMap() string { + if x != nil { + return x.Map + } + return "" +} + +func (x *ServerStatus) GetMaxPlayers() int32 { + if x != nil { + return x.MaxPlayers + } + return 0 +} + +func (x *ServerStatus) GetNumPlayers() int32 { + if x != nil { + return x.NumPlayers + } + return 0 +} + +func (x *ServerStatus) GetPlayers() []*Player { + if x != nil { + return x.Players + } + return nil +} + +func (x *ServerStatus) GetVersion() string { + if x != nil { + return x.Version + } + return "" +} + +func (x *ServerStatus) GetFavicon() string { + if x != nil { + return x.Favicon + } + return "" +} + +func (x *ServerStatus) GetServerType() ServerType { + if x != nil { + return x.ServerType + } + return ServerType_JAVA +} + +type Player struct { + state protoimpl.MessageState + sizeCache protoimpl.SizeCache + unknownFields protoimpl.UnknownFields + + Name string `protobuf:"bytes,1,opt,name=name,proto3" json:"name" xml:"name"` + Uuid string `protobuf:"bytes,2,opt,name=uuid,proto3" json:"uuid" xml:"uuid"` +} + +func (x *Player) Reset() { + *x = Player{} + if protoimpl.UnsafeEnabled { + mi := &file_mcstatus_proto_msgTypes[1] + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + ms.StoreMessageInfo(mi) + } +} + +func (x *Player) String() string { + return protoimpl.X.MessageStringOf(x) +} + +func (*Player) ProtoMessage() {} + +func (x *Player) ProtoReflect() protoreflect.Message { + mi := &file_mcstatus_proto_msgTypes[1] + if protoimpl.UnsafeEnabled && x != nil { + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + if ms.LoadMessageInfo() == nil { + ms.StoreMessageInfo(mi) + } + return ms + } + return mi.MessageOf(x) +} + +// Deprecated: Use Player.ProtoReflect.Descriptor instead. +func (*Player) Descriptor() ([]byte, []int) { + return file_mcstatus_proto_rawDescGZIP(), []int{1} +} + +func (x *Player) GetName() string { + if x != nil { + return x.Name + } + return "" +} + +func (x *Player) GetUuid() string { + if x != nil { + return x.Uuid + } + return "" +} + +var File_mcstatus_proto protoreflect.FileDescriptor + +var file_mcstatus_proto_rawDesc = []byte{ + 0x0a, 0x0e, 0x6d, 0x63, 0x73, 0x74, 0x61, 0x74, 0x75, 0x73, 0x2e, 0x70, 0x72, 0x6f, 0x74, 0x6f, + 0x12, 0x0a, 0x6d, 0x63, 0x73, 0x74, 0x61, 0x74, 0x75, 0x73, 0x70, 0x62, 0x22, 0xcd, 0x02, 0x0a, + 0x0c, 0x53, 0x65, 0x72, 0x76, 0x65, 0x72, 0x53, 0x74, 0x61, 0x74, 0x75, 0x73, 0x12, 0x12, 0x0a, + 0x04, 0x68, 0x6f, 0x73, 0x74, 0x18, 0x01, 0x20, 0x01, 0x28, 0x09, 0x52, 0x04, 0x68, 0x6f, 0x73, + 0x74, 0x12, 0x12, 0x0a, 0x04, 0x70, 0x6f, 0x72, 0x74, 0x18, 0x02, 0x20, 0x01, 0x28, 0x05, 0x52, + 0x04, 0x70, 0x6f, 0x72, 0x74, 0x12, 0x12, 0x0a, 0x04, 0x6e, 0x61, 0x6d, 0x65, 0x18, 0x03, 0x20, + 0x01, 0x28, 0x09, 0x52, 0x04, 0x6e, 0x61, 0x6d, 0x65, 0x12, 0x12, 0x0a, 0x04, 0x6d, 0x6f, 0x74, + 0x64, 0x18, 0x04, 0x20, 0x01, 0x28, 0x09, 0x52, 0x04, 0x6d, 0x6f, 0x74, 0x64, 0x12, 0x10, 0x0a, + 0x03, 0x6d, 0x61, 0x70, 0x18, 0x05, 0x20, 0x01, 0x28, 0x09, 0x52, 0x03, 0x6d, 0x61, 0x70, 0x12, + 0x1f, 0x0a, 0x0b, 0x6d, 0x61, 0x78, 0x5f, 0x70, 0x6c, 0x61, 0x79, 0x65, 0x72, 0x73, 0x18, 0x06, + 0x20, 0x01, 0x28, 0x05, 0x52, 0x0a, 0x6d, 0x61, 0x78, 0x50, 0x6c, 0x61, 0x79, 0x65, 0x72, 0x73, + 0x12, 0x1f, 0x0a, 0x0b, 0x6e, 0x75, 0x6d, 0x5f, 0x70, 0x6c, 0x61, 0x79, 0x65, 0x72, 0x73, 0x18, + 0x07, 0x20, 0x01, 0x28, 0x05, 0x52, 0x0a, 0x6e, 0x75, 0x6d, 0x50, 0x6c, 0x61, 0x79, 0x65, 0x72, + 0x73, 0x12, 0x2c, 0x0a, 0x07, 0x70, 0x6c, 0x61, 0x79, 0x65, 0x72, 0x73, 0x18, 0x08, 0x20, 0x03, + 0x28, 0x0b, 0x32, 0x12, 0x2e, 0x6d, 0x63, 0x73, 0x74, 0x61, 0x74, 0x75, 0x73, 0x70, 0x62, 0x2e, + 0x50, 0x6c, 0x61, 0x79, 0x65, 0x72, 0x52, 0x07, 0x70, 0x6c, 0x61, 0x79, 0x65, 0x72, 0x73, 0x12, + 0x18, 0x0a, 0x07, 0x76, 0x65, 0x72, 0x73, 0x69, 0x6f, 0x6e, 0x18, 0x09, 0x20, 0x01, 0x28, 0x09, + 0x52, 0x07, 0x76, 0x65, 0x72, 0x73, 0x69, 0x6f, 0x6e, 0x12, 0x18, 0x0a, 0x07, 0x66, 0x61, 0x76, + 0x69, 0x63, 0x6f, 0x6e, 0x18, 0x0a, 0x20, 0x01, 0x28, 0x09, 0x52, 0x07, 0x66, 0x61, 0x76, 0x69, + 0x63, 0x6f, 0x6e, 0x12, 0x37, 0x0a, 0x0b, 0x73, 0x65, 0x72, 0x76, 0x65, 0x72, 0x5f, 0x74, 0x79, + 0x70, 0x65, 0x18, 0x0b, 0x20, 0x01, 0x28, 0x0e, 0x32, 0x16, 0x2e, 0x6d, 0x63, 0x73, 0x74, 0x61, + 0x74, 0x75, 0x73, 0x70, 0x62, 0x2e, 0x53, 0x65, 0x72, 0x76, 0x65, 0x72, 0x54, 0x79, 0x70, 0x65, + 0x52, 0x0a, 0x73, 0x65, 0x72, 0x76, 0x65, 0x72, 0x54, 0x79, 0x70, 0x65, 0x22, 0x30, 0x0a, 0x06, + 0x50, 0x6c, 0x61, 0x79, 0x65, 0x72, 0x12, 0x12, 0x0a, 0x04, 0x6e, 0x61, 0x6d, 0x65, 0x18, 0x01, + 0x20, 0x01, 0x28, 0x09, 0x52, 0x04, 0x6e, 0x61, 0x6d, 0x65, 0x12, 0x12, 0x0a, 0x04, 0x75, 0x75, + 0x69, 0x64, 0x18, 0x02, 0x20, 0x01, 0x28, 0x09, 0x52, 0x04, 0x75, 0x75, 0x69, 0x64, 0x2a, 0x23, + 0x0a, 0x0a, 0x53, 0x65, 0x72, 0x76, 0x65, 0x72, 0x54, 0x79, 0x70, 0x65, 0x12, 0x08, 0x0a, 0x04, + 0x4a, 0x41, 0x56, 0x41, 0x10, 0x00, 0x12, 0x0b, 0x0a, 0x07, 0x42, 0x45, 0x44, 0x52, 0x4f, 0x43, + 0x4b, 0x10, 0x01, 0x42, 0x0e, 0x5a, 0x0c, 0x2e, 0x2f, 0x6d, 0x63, 0x73, 0x74, 0x61, 0x74, 0x75, + 0x73, 0x70, 0x62, 0x62, 0x06, 0x70, 0x72, 0x6f, 0x74, 0x6f, 0x33, +} + +var ( + file_mcstatus_proto_rawDescOnce sync.Once + file_mcstatus_proto_rawDescData = file_mcstatus_proto_rawDesc +) + +func file_mcstatus_proto_rawDescGZIP() []byte { + file_mcstatus_proto_rawDescOnce.Do(func() { + file_mcstatus_proto_rawDescData = protoimpl.X.CompressGZIP(file_mcstatus_proto_rawDescData) + }) + return file_mcstatus_proto_rawDescData +} + +var file_mcstatus_proto_enumTypes = make([]protoimpl.EnumInfo, 1) +var file_mcstatus_proto_msgTypes = make([]protoimpl.MessageInfo, 2) +var file_mcstatus_proto_goTypes = []interface{}{ + (ServerType)(0), // 0: mcstatuspb.ServerType + (*ServerStatus)(nil), // 1: mcstatuspb.ServerStatus + (*Player)(nil), // 2: mcstatuspb.Player +} +var file_mcstatus_proto_depIdxs = []int32{ + 2, // 0: mcstatuspb.ServerStatus.players:type_name -> mcstatuspb.Player + 0, // 1: mcstatuspb.ServerStatus.server_type:type_name -> mcstatuspb.ServerType + 2, // [2:2] is the sub-list for method output_type + 2, // [2:2] is the sub-list for method input_type + 2, // [2:2] is the sub-list for extension type_name + 2, // [2:2] is the sub-list for extension extendee + 0, // [0:2] is the sub-list for field type_name +} + +func init() { file_mcstatus_proto_init() } +func file_mcstatus_proto_init() { + if File_mcstatus_proto != nil { + return + } + if !protoimpl.UnsafeEnabled { + file_mcstatus_proto_msgTypes[0].Exporter = func(v interface{}, i int) interface{} { + switch v := v.(*ServerStatus); i { + case 0: + return &v.state + case 1: + return &v.sizeCache + case 2: + return &v.unknownFields + default: + return nil + } + } + file_mcstatus_proto_msgTypes[1].Exporter = func(v interface{}, i int) interface{} { + switch v := v.(*Player); i { + case 0: + return &v.state + case 1: + return &v.sizeCache + case 2: + return &v.unknownFields + default: + return nil + } + } + } + type x struct{} + out := protoimpl.TypeBuilder{ + File: protoimpl.DescBuilder{ + GoPackagePath: reflect.TypeOf(x{}).PkgPath(), + RawDescriptor: file_mcstatus_proto_rawDesc, + NumEnums: 1, + NumMessages: 2, + NumExtensions: 0, + NumServices: 0, + }, + GoTypes: file_mcstatus_proto_goTypes, + DependencyIndexes: file_mcstatus_proto_depIdxs, + EnumInfos: file_mcstatus_proto_enumTypes, + MessageInfos: file_mcstatus_proto_msgTypes, + }.Build() + File_mcstatus_proto = out.File + file_mcstatus_proto_rawDesc = nil + file_mcstatus_proto_goTypes = nil + file_mcstatus_proto_depIdxs = nil +} diff --git a/proto/mcstatus.proto b/proto/mcstatus.proto new file mode 100644 index 0000000..96b1348 --- /dev/null +++ b/proto/mcstatus.proto @@ -0,0 +1,27 @@ +syntax = "proto3"; +package mcstatuspb; +option go_package = "./mcstatuspb"; + +message ServerStatus { + string host = 1; + int32 port = 2; + string name = 3; + string motd = 4; + string map = 5; + int32 max_players = 6; + int32 num_players = 7; + repeated Player players = 8; + string version = 9; + string favicon = 10; + ServerType server_type = 11; +} + +message Player { + string name = 1; + string uuid = 2; +} + +enum ServerType { + JAVA = 0; + BEDROCK = 1; +} diff --git a/public/api/v1/openapi.json b/public/api/v1/openapi.json index 835c9a5..c3e2ea7 100644 --- a/public/api/v1/openapi.json +++ b/public/api/v1/openapi.json @@ -268,38 +268,9 @@ "get": { "summary": "Get Minecraft server status", "description": "Get Minecraft server status", - "requestBody": { - "content": { - "application/json": { - "schema": { - "type": "object", - "required": [ - "address" - ], - "properties": { - "address": { - "type": "string" - }, - "port": { - "type": "integer" - }, - "query_port": { - "type": "integer" - }, - "is_bedrock": { - "type": "boolean" - }, - "query_enabled": { - "type": "boolean" - } - } - } - } - } - }, "parameters": [ { - "name": "address", + "name": "host", "in": "path", "description": "Minecraft server address", "required": true, @@ -308,36 +279,36 @@ } }, { - "name": "port", + "name": "bedrock", "in": "query", - "description": "Minecraft server port", + "description": "Is the server bedrock edition", "required": false, "schema": { - "type": "integer" + "type": "boolean" } }, { - "name": "query_port", + "name": "query", "in": "query", - "description": "Minecraft server query port", + "description": "Is query enabled on the server", "required": false, "schema": { - "type": "integer" + "type": "boolean" } }, { - "name": "is_bedrock", + "name": "query_port", "in": "query", - "description": "Is the server bedrock edition", + "description": "Minecraft server query port", "required": false, "schema": { - "type": "boolean" + "type": "integer" } }, { - "name": "query_enabled", + "name": "raw", "in": "query", - "description": "Is the server query enabled", + "description": "Return raw server status", "required": false, "schema": { "type": "boolean" @@ -350,39 +321,17 @@ "content": { "application/json": { "schema": { - "type": "object", - "required": [ - "name", - "map", - "maxplayers", - "players", - "connect", - "version", - "favicon" - ], - "properties": { - "name": { - "type": "string" - }, - "map": { - "type": "string" - }, - "maxplayers": { - "type": "integer" - }, - "players": { - "type": "integer" - }, - "connect": { - "type": "string" - }, - "version": { - "type": "string" - }, - "favicon": { - "type": "string" - } - } + "$ref": "#/components/schemas/MCServerStatus" + } + }, + "application/xml": { + "schema": { + "$ref": "#/components/schemas/MCServerStatus" + } + }, + "application/x-protobuf": { + "schema": { + "$ref": "#/components/schemas/MCServerStatus" } } } @@ -425,16 +374,6 @@ } } } - }, - "500": { - "description": "Internal Server Error", - "content": { - "text/plain": { - "schema": { - "type": "string" - } - } - } } } } @@ -443,29 +382,6 @@ "get": { "summary": "Get Minecraft server icon", "description": "Get Minecraft server icon", - "requestBody": { - "content": { - "application/json": { - "schema": { - "type": "object", - "required": [ - "address" - ], - "properties": { - "address": { - "type": "string" - }, - "port": { - "type": "integer" - }, - "is_bedrock": { - "type": "boolean" - } - } - } - } - } - }, "parameters": [ { "name": "address", @@ -475,24 +391,6 @@ "schema": { "type": "string" } - }, - { - "name": "port", - "in": "query", - "description": "Minecraft server port", - "required": false, - "schema": { - "type": "integer" - } - }, - { - "name": "is_bedrock", - "in": "query", - "description": "Is the server bedrock edition", - "required": false, - "schema": { - "type": "boolean" - } } ], "responses": { @@ -507,37 +405,8 @@ } } }, - "204": { - "description": "No Content, bedrock server detected, so a png of a bedrock block is returned", - "content": { - "image/png": { - "schema": { - "type": "string", - "format": "binary" - } - } - } - }, "404": { - "description": "Not Found, returns generic offline server icon", - "content": { - "image/png": { - "schema": { - "type": "string", - "format": "binary" - } - } - } - }, - "500": { - "description": "Internal Server Error", - "content": { - "text/plain": { - "schema": { - "type": "string" - } - } - } + "$ref": "#/components/responses/404NotFound" } } } @@ -618,6 +487,71 @@ "type": "string" } } + }, + "MCServerStatus": { + "type": "object", + "required": [ + "host", + "port", + "name", + "motd", + "map", + "max_players", + "num_players", + "version", + "favicon", + "server_type" + ], + "properties": { + "host": { + "type": "string" + }, + "port": { + "type": "integer" + }, + "name": { + "type": "string" + }, + "motd": { + "type": "string" + }, + "map": { + "type": "string" + }, + "max_players": { + "type": "integer" + }, + "num_players": { + "type": "integer" + }, + "players": { + "type": "array", + "items": { + "type": "object", + "properties": { + "name": { + "type": "string" + }, + "uuid": { + "type": "string" + } + } + } + }, + "version": { + "type": "string" + }, + "favicon": { + "type": "string" + }, + "server_type": { + "type": "string", + "enum": [ + "java", + "bedrock" + ] + } + } } }, "parameters": { diff --git a/responses/problem.go b/responses/problem.go index 78435e0..989cc77 100644 --- a/responses/problem.go +++ b/responses/problem.go @@ -42,7 +42,8 @@ func (problem *problem) SendAndEncodeProblem(w http.ResponseWriter, r *http.Requ case "application/xml": content += "xml" structBytes, _ = xml.Marshal(problem) - default: + } + if structBytes == nil { content += "json" structBytes, _ = json.Marshal(problem) } diff --git a/responses/response.go b/responses/response.go index 0b1e1fe..a3dcff3 100644 --- a/responses/response.go +++ b/responses/response.go @@ -23,10 +23,12 @@ func SendAndEncodeStruct[T any](w http.ResponseWriter, r *http.Request, statusCo case "application/xml": content += "xml" structBytes, _ = xml.Marshal(data) - default: + } + if structBytes == nil { content += "json" structBytes, _ = json.Marshal(data) } + w.Header().Set("Content-Type", content) w.WriteHeader(statusCode) w.Write(structBytes)