7 Commits

Author SHA1 Message Date
7c067839af feat: 添加 gitea 构建脚本
All checks were successful
Build and Push Docker Image / build-and-push (push) Successful in 11m50s
2025-07-15 23:14:05 +08:00
22fd87a07a fix: 调整邮箱正则 2025-07-15 23:13:27 +08:00
1d825c97f3 feat: 调整过期时间 2025-06-19 22:45:10 +08:00
044cd3082e feat: 代理全部类型账户的 ProfileKey 2025-06-19 01:12:19 +08:00
e7c4dc58b7 feat: 添加公钥缓存功能 2025-06-05 02:25:03 +08:00
2722e85a6a fix: 修复证书问题 2025-06-01 20:51:06 +08:00
dcec80c184 fix: 修复 docker 无法启动 2025-06-01 19:30:42 +08:00
5 changed files with 116 additions and 42 deletions

View File

@@ -29,41 +29,26 @@ jobs:
cache-dependency-path: frontend cache-dependency-path: frontend
- name: Build Frontend - name: Build Frontend
run: | run: |
mkdir build
make assets make assets
rm -rf /host/${{ gitea.workspace }} && mkdir -p /host/${{ gitea.workspace }}
cp -a . /host/${{ gitea.workspace }}/
- name: Build Yggdrasil Server - name: Build Yggdrasil Server
uses: crazy-max/ghaction-xgo@v2 uses: crazy-max/ghaction-xgo@v2
with: with:
xgo_version: latest xgo_version: latest
go_version: 1.24 go_version: 1.24
dest: "${{ gitea.workspace }}/build" dest: build
prefix: yggdrasil prefix: yggdrasil
targets: windows/amd64,linux/amd64,linux/arm64,darwin/amd64,darwin/arm64 targets: linux/amd64,linux/arm64
v: true v: true
x: false x: false
race: false race: false
ldflags: -s -w -buildid= ldflags: -s -w -buildid=
tags: nomsgpack sqlite mysql tags: nomsgpack sqlite mysql
trimpath: true trimpath: true
- name: Store Back Binaries
- name: Create ZIP archive
run: | run: |
cp -rv ./config_example.ini ./assets ./build/ || exit 1 cp -a /host/${{ gitea.workspace }}/build/. build
pushd build || exit 1
ls -1 yggdrasil-* | while read LINE; do
PREFIX="${LINE%.*}"
SUFFIX="$(echo "$LINE" | grep -osE '\.\w+' || printf '')"
cp -v "$LINE" "yggdrasil$SUFFIX"
FILE="../$PREFIX.zip"
zip -9rv "$FILE" "yggdrasil$SUFFIX" *.ini assets
DGST="$FILE.dgst"
openssl dgst -md5 "$FILE" | sed 's/([^)]*)//g' >>"$DGST"
openssl dgst -sha1 "$FILE" | sed 's/([^)]*)//g' >>"$DGST"
openssl dgst -sha256 "$FILE" | sed 's/([^)]*)//g' >>"$DGST"
openssl dgst -sha512 "$FILE" | sed 's/([^)]*)//g' >>"$DGST"
done
popd || exit 1
- name: Set up QEMU - name: Set up QEMU
uses: docker/setup-qemu-action@v2 uses: docker/setup-qemu-action@v2
- name: Set up Docker Buildx - name: Set up Docker Buildx

View File

@@ -1,8 +1,10 @@
FROM alpine:latest FROM debian:12-slim
LABEL maintainer="Gardel <sunxinao@hotmail.com>" LABEL maintainer="Gardel <sunxinao@hotmail.com>"
LABEL "Description"="Go Yggdrasil Server" LABEL "Description"="Go Yggdrasil Server"
RUN apt-get update && apt-get install -y ca-certificates
RUN update-ca-certificates
ARG TARGETOS ARG TARGETOS
ARG TARGETARCH ARG TARGETARCH
RUN mkdir -p /app RUN mkdir -p /app

View File

@@ -64,6 +64,7 @@ type HomeRouter interface {
type homeRouterImpl struct { type homeRouterImpl struct {
serverMeta ServerMeta serverMeta ServerMeta
myPubKey KeyPair myPubKey KeyPair
cachedPubKey *PublicKeys
} }
func NewHomeRouter(meta *ServerMeta) HomeRouter { func NewHomeRouter(meta *ServerMeta) HomeRouter {
@@ -81,6 +82,10 @@ func (h *homeRouterImpl) Home(c *gin.Context) {
} }
func (h *homeRouterImpl) PublicKeys(c *gin.Context) { func (h *homeRouterImpl) PublicKeys(c *gin.Context) {
if h.cachedPubKey != nil {
c.JSON(http.StatusOK, h.cachedPubKey)
return
}
publicKeys := PublicKeys{} publicKeys := PublicKeys{}
err := util.GetObject("https://api.minecraftservices.com/publickeys", &publicKeys) err := util.GetObject("https://api.minecraftservices.com/publickeys", &publicKeys)
if err != nil { if err != nil {
@@ -90,4 +95,5 @@ func (h *homeRouterImpl) PublicKeys(c *gin.Context) {
publicKeys.ProfilePropertyKeys = append(publicKeys.ProfilePropertyKeys, h.myPubKey) publicKeys.ProfilePropertyKeys = append(publicKeys.ProfilePropertyKeys, h.myPubKey)
publicKeys.PlayerCertificateKeys = append(publicKeys.PlayerCertificateKeys, h.myPubKey) publicKeys.PlayerCertificateKeys = append(publicKeys.PlayerCertificateKeys, h.myPubKey)
c.JSON(http.StatusOK, publicKeys) c.JSON(http.StatusOK, publicKeys)
h.cachedPubKey = &publicKeys
} }

View File

@@ -118,7 +118,7 @@ func (u *userServiceImpl) Register(username, password, profileName, ip string) (
} else if _, err := mojangUsernameToUUID(profileName); err == nil { } else if _, err := mojangUsernameToUUID(profileName); err == nil {
return nil, util.NewForbiddenOperationError("profileName duplicate") return nil, util.NewForbiddenOperationError("profileName duplicate")
} }
matched, err := regexp.MatchString("^(\\w){3,}(\\.\\w+)*@(\\w){2,}((\\.\\w+)+)$", username) matched, err := regexp.MatchString("^\\w+@(\\w){2,}((\\.\\w+)+)$", username)
if err != nil { if err != nil {
return nil, err return nil, err
} }
@@ -394,12 +394,24 @@ func (u *userServiceImpl) QueryProfile(profileId uuid.UUID, unsigned bool, textu
func (u *userServiceImpl) ProfileKey(accessToken string) (resp *ProfileKeyResponse, err error) { func (u *userServiceImpl) ProfileKey(accessToken string) (resp *ProfileKeyResponse, err error) {
token, ok := u.tokenService.GetToken(accessToken) token, ok := u.tokenService.GetToken(accessToken)
var profileId uuid.UUID
if ok && token.GetAvailableLevel() == model.Valid { if ok && token.GetAvailableLevel() == model.Valid {
profileId = token.SelectedProfile.Id
} else {
id, _, err := util.ParseOfficialToken(accessToken)
if err != nil {
return nil, err
}
profileId, err = util.ToUUID(id)
if err != nil {
return nil, err
}
}
resp = new(ProfileKeyResponse) resp = new(ProfileKeyResponse)
now := time.Now().UTC() now := time.Now().UTC()
resp.RefreshedAfter = now resp.RefreshedAfter = now
resp.ExpiresAt = now.Add(10 * time.Minute) resp.ExpiresAt = now.Add(90 * 24 * time.Hour)
keyPair, err := u.getProfileKey(token.SelectedProfile.Id) keyPair, err := u.getProfileKey(profileId)
if err != nil { if err != nil {
return nil, err return nil, err
} }
@@ -411,12 +423,6 @@ func (u *userServiceImpl) ProfileKey(accessToken string) (resp *ProfileKeyRespon
} }
resp.PublicKeySignature = sign resp.PublicKeySignature = sign
resp.PublicKeySignatureV2 = sign resp.PublicKeySignatureV2 = sign
} else {
err = util.PostForString("https://api.minecraftservices.com/player/certificates", accessToken, []byte(""), resp)
if err != nil {
return nil, err
}
}
return resp, nil return resp, nil
} }

75
util/token_utils.go Normal file
View File

@@ -0,0 +1,75 @@
/*
* Copyright (C) 2025. Gardel <sunxinao@hotmail.com> and contributors
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU Affero General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU Affero General Public License for more details.
*
* You should have received a copy of the GNU Affero General Public License
* along with this program. If not, see <https://www.gnu.org/licenses/>.
*/
package util
import (
"encoding/base64"
"encoding/json"
"errors"
"strings"
)
type officialTokenPayload struct {
Xuid string `json:"xuid"`
Agg string `json:"agg"`
Sub string `json:"sub"`
Auth string `json:"auth"`
Ns string `json:"ns"`
Roles []interface{} `json:"roles"`
Iss string `json:"iss"`
Flags []string `json:"flags"`
Profiles struct {
Mc string `json:"mc"`
} `json:"profiles"`
Platform string `json:"platform"`
Pfd []struct {
Type string `json:"type"`
Id string `json:"id"`
Name string `json:"name"`
} `json:"pfd"`
Nbf int `json:"nbf"`
Exp int `json:"exp"`
Iat int `json:"iat"`
}
func ParseOfficialToken(token string) (id, name string, err error) {
firstDot := strings.IndexRune(token, '.')
if firstDot == -1 {
return id, name, errors.New("invalid token")
}
secondDot := 1 + firstDot + strings.IndexRune(token[firstDot+1:], '.')
if secondDot == -1 {
return id, name, errors.New("invalid token")
}
jsonBase64 := token[firstDot+1 : secondDot]
jsonDecoded, err := base64.RawURLEncoding.DecodeString(jsonBase64)
if err != nil {
return id, name, err
}
payload := officialTokenPayload{}
err = json.Unmarshal(jsonDecoded, &payload)
if err != nil {
return id, name, err
}
if payload.Pfd == nil || len(payload.Pfd) == 0 {
return id, name, errors.New("invalid token")
}
id = payload.Pfd[0].Id
name = payload.Pfd[0].Name
return id, name, nil
}