diff options
| author | nsfisis <nsfisis@gmail.com> | 2024-08-17 21:11:07 +0900 |
|---|---|---|
| committer | nsfisis <nsfisis@gmail.com> | 2024-08-17 21:11:07 +0900 |
| commit | 01d3120fd7129f573d88f7aa7c227b3ef93fe368 (patch) | |
| tree | eea4f5ac0f025c7bdbff71d2ee83b4dce99355ef | |
| parent | f926ef682de637b717d3b0cc0eaee43c59e83c95 (diff) | |
| parent | b923a9d6534820d33f42bc65c47ae22889bde922 (diff) | |
| download | phperkaigi-2025-albatross-01d3120fd7129f573d88f7aa7c227b3ef93fe368.tar.gz phperkaigi-2025-albatross-01d3120fd7129f573d88f7aa7c227b3ef93fe368.tar.zst phperkaigi-2025-albatross-01d3120fd7129f573d88f7aa7c227b3ef93fe368.zip | |
Merge branch 'feat/icon'
| -rw-r--r-- | backend/account/icon.go | 85 | ||||
| -rw-r--r-- | backend/admin/handler.go | 26 | ||||
| -rw-r--r-- | backend/admin/templates/users.html | 3 | ||||
| -rw-r--r-- | backend/auth/auth.go | 13 | ||||
| -rw-r--r-- | backend/auth/fortee/fortee.go | 46 | ||||
| -rw-r--r-- | backend/db/query.sql.go | 16 | ||||
| -rw-r--r-- | backend/fortee/fortee.go | 62 | ||||
| -rw-r--r-- | backend/fortee/generated.go (renamed from backend/auth/fortee/generated.go) | 117 | ||||
| -rw-r--r-- | backend/gen/oapi-codegen.fortee.yaml | 2 | ||||
| -rw-r--r-- | backend/main.go | 9 | ||||
| -rw-r--r-- | backend/query.sql | 5 | ||||
| -rw-r--r-- | compose.local.yaml | 3 | ||||
| -rw-r--r-- | compose.prod.yaml | 4 | ||||
| -rw-r--r-- | frontend/app/components/GolfPlayApps/GolfPlayAppGaming.tsx | 26 | ||||
| -rw-r--r-- | frontend/app/components/GolfWatchApps/GolfWatchAppGaming.tsx | 38 | ||||
| -rw-r--r-- | frontend/app/routes/dashboard.tsx | 11 | ||||
| -rw-r--r-- | nginx.conf | 4 | ||||
| -rw-r--r-- | openapi/fortee.yaml | 33 |
18 files changed, 442 insertions, 61 deletions
diff --git a/backend/account/icon.go b/backend/account/icon.go new file mode 100644 index 0000000..40632c6 --- /dev/null +++ b/backend/account/icon.go @@ -0,0 +1,85 @@ +package account + +import ( + "context" + "fmt" + "io" + "net/http" + "net/url" + "os" + "path" + "path/filepath" + "time" + + "github.com/nsfisis/iosdc-japan-2024-albatross/backend/db" + "github.com/nsfisis/iosdc-japan-2024-albatross/backend/fortee" +) + +func FetchIcon( + ctx context.Context, + q *db.Queries, + userID int, +) error { + ctx, cancel := context.WithTimeout(ctx, 15*time.Second) + defer cancel() + + // Fetch user. + user, err := q.GetUserByID(ctx, int32(userID)) + if err != nil { + return fmt.Errorf("failed to fetch user icon (uid=%d): %w", userID, err) + } + // Fetch user icon URL. + avatarURL, err := fortee.GetUserAvatarURL(ctx, user.Username) + if err != nil { + return fmt.Errorf("failed to fetch user icon (uid=%d): %w", userID, err) + } + // Download user icon file. + filePath := fmt.Sprintf("/files/%s/icon%s", url.PathEscape(user.Username), path.Ext(avatarURL)) + if err := downloadFile(ctx, fortee.Endpoint+avatarURL, "/data"+filePath); err != nil { + return fmt.Errorf("failed to fetch user icon (uid=%d): %w", userID, err) + } + // Save user icon path. + if err := q.UpdateUserIconPath(ctx, db.UpdateUserIconPathParams{ + UserID: int32(userID), + IconPath: &filePath, + }); err != nil { + return fmt.Errorf("failed to fetch user icon (uid=%d): %w", userID, err) + } + return nil +} + +func downloadFile(ctx context.Context, url string, filePath string) error { + req, err := http.NewRequestWithContext(ctx, "GET", url, nil) + if err != nil { + return fmt.Errorf("failed to download file (%s): %w", url, err) + } + + client := &http.Client{} + res, err := client.Do(req) + if err != nil { + return fmt.Errorf("failed to download file (%s): %w", url, err) + } + defer res.Body.Close() + + if res.StatusCode != http.StatusOK { + return fmt.Errorf("failed to download file (%s): status %d", url, res.StatusCode) + } + + fileDir := filepath.Dir(filePath) + if err := os.MkdirAll(fileDir, 0755); err != nil { + return fmt.Errorf("failed to create directory (%s): %w", fileDir, err) + } + + file, err := os.Create(filePath) + if err != nil { + return fmt.Errorf("failed to open file (%s): %w", filePath, err) + } + defer file.Close() + + _, err = io.Copy(file, res.Body) + if err != nil { + return fmt.Errorf("failed to save file (%s): %w", filePath, err) + } + + return nil +} diff --git a/backend/admin/handler.go b/backend/admin/handler.go index 41eacd4..1b60de5 100644 --- a/backend/admin/handler.go +++ b/backend/admin/handler.go @@ -1,7 +1,9 @@ package admin import ( + "context" "errors" + "log" "net/http" "strconv" "time" @@ -10,6 +12,7 @@ import ( "github.com/jackc/pgx/v5/pgtype" "github.com/labstack/echo/v4" + "github.com/nsfisis/iosdc-japan-2024-albatross/backend/account" "github.com/nsfisis/iosdc-japan-2024-albatross/backend/auth" "github.com/nsfisis/iosdc-japan-2024-albatross/backend/db" ) @@ -58,6 +61,7 @@ func (h *Handler) RegisterHandlers(g *echo.Group) { g.GET("/dashboard", h.getDashboard) g.GET("/users", h.getUsers) g.GET("/users/:userID", h.getUserEdit) + g.POST("/users/:userID/fetch-icon", h.postUserFetchIcon) g.GET("/games", h.getGames) g.GET("/games/:gameID", h.getGameEdit) g.POST("/games/:gameID", h.postGameEdit) @@ -116,6 +120,28 @@ func (h *Handler) getUserEdit(c echo.Context) error { }) } +func (h *Handler) postUserFetchIcon(c echo.Context) error { + userID, err := strconv.Atoi(c.Param("userID")) + if err != nil { + return echo.NewHTTPError(http.StatusBadRequest, "Invalid user id") + } + row, err := h.q.GetUserByID(c.Request().Context(), int32(userID)) + if err != nil { + if errors.Is(err, pgx.ErrNoRows) { + return echo.NewHTTPError(http.StatusNotFound) + } + return echo.NewHTTPError(http.StatusInternalServerError, err.Error()) + } + go func() { + err := account.FetchIcon(context.Background(), h.q, int(row.UserID)) + if err != nil { + log.Printf("%v", err) + // The failure is intentionally ignored. Retry manually if needed. + } + }() + return c.Redirect(http.StatusSeeOther, "/iosdc-japan/2024/code-battle/admin/users") +} + func (h *Handler) getGames(c echo.Context) error { rows, err := h.q.ListGames(c.Request().Context()) if err != nil { diff --git a/backend/admin/templates/users.html b/backend/admin/templates/users.html index 3543457..3dc03c0 100644 --- a/backend/admin/templates/users.html +++ b/backend/admin/templates/users.html @@ -11,6 +11,9 @@ <a href="/iosdc-japan/2024/code-battle/admin/users/{{ .UserID }}"> {{ .DisplayName }} (id={{ .UserID }} username={{ .Username }}){{ if .IsAdmin }} <em>admin</em>{{ end }} </a> + <form method="post" action="/iosdc-japan/2024/code-battle/admin/users/{{ .UserID }}/fetch-icon"> + <button type="submit">Fetch Icon</button> + </form> </li> {{ end }} </ul> diff --git a/backend/auth/auth.go b/backend/auth/auth.go index 0e55d8d..2266c50 100644 --- a/backend/auth/auth.go +++ b/backend/auth/auth.go @@ -3,13 +3,15 @@ package auth import ( "context" "errors" + "log" "time" "github.com/jackc/pgx/v5" "golang.org/x/crypto/bcrypt" - "github.com/nsfisis/iosdc-japan-2024-albatross/backend/auth/fortee" + "github.com/nsfisis/iosdc-japan-2024-albatross/backend/account" "github.com/nsfisis/iosdc-japan-2024-albatross/backend/db" + "github.com/nsfisis/iosdc-japan-2024-albatross/backend/fortee" ) var ( @@ -98,6 +100,13 @@ func signup( }); err != nil { return 0, err } + go func() { + err := account.FetchIcon(context.Background(), queries, int(userID)) + if err != nil { + log.Printf("%v", err) + // The failure is intentionally ignored. Retry manually if needed. + } + }() return int(userID), nil } @@ -119,7 +128,7 @@ func verifyForteeAccount(ctx context.Context, username string, password string) ctx, cancel := context.WithTimeout(ctx, forteeAPITimeout) defer cancel() - canonicalizedUsername, err := fortee.LoginFortee(ctx, username, password) + canonicalizedUsername, err := fortee.Login(ctx, username, password) if errors.Is(err, context.DeadlineExceeded) { return "", ErrForteeLoginTimeout } diff --git a/backend/auth/fortee/fortee.go b/backend/auth/fortee/fortee.go deleted file mode 100644 index 25ca9c5..0000000 --- a/backend/auth/fortee/fortee.go +++ /dev/null @@ -1,46 +0,0 @@ -package fortee - -import ( - "context" - "errors" - "net/http" -) - -const ( - apiEndpoint = "https://fortee.jp" -) - -var ( - ErrLoginFailed = errors.New("fortee login failed") -) - -func LoginFortee(ctx context.Context, username string, password string) (string, error) { - client, err := NewClientWithResponses(apiEndpoint, WithRequestEditorFn(addAcceptHeader)) - if err != nil { - return "", err - } - res, err := client.PostLoginWithFormdataBodyWithResponse(ctx, PostLoginFormdataRequestBody{ - Username: username, - Password: password, - }) - if err != nil { - return "", err - } - if res.StatusCode() != http.StatusOK { - return "", ErrLoginFailed - } - resOk := res.JSON200 - if !resOk.LoggedIn { - return "", ErrLoginFailed - } - if resOk.User == nil { - return "", ErrLoginFailed - } - return resOk.User.Username, nil -} - -// fortee API denies requests without Accept header. -func addAcceptHeader(_ context.Context, req *http.Request) error { - req.Header.Set("Accept", "application/json") - return nil -} diff --git a/backend/db/query.sql.go b/backend/db/query.sql.go index 02e660a..35b7edd 100644 --- a/backend/db/query.sql.go +++ b/backend/db/query.sql.go @@ -584,3 +584,19 @@ func (q *Queries) UpdateGameState(ctx context.Context, arg UpdateGameStateParams _, err := q.db.Exec(ctx, updateGameState, arg.GameID, arg.State) return err } + +const updateUserIconPath = `-- name: UpdateUserIconPath :exec +UPDATE users +SET icon_path = $2 +WHERE user_id = $1 +` + +type UpdateUserIconPathParams struct { + UserID int32 + IconPath *string +} + +func (q *Queries) UpdateUserIconPath(ctx context.Context, arg UpdateUserIconPathParams) error { + _, err := q.db.Exec(ctx, updateUserIconPath, arg.UserID, arg.IconPath) + return err +} diff --git a/backend/fortee/fortee.go b/backend/fortee/fortee.go new file mode 100644 index 0000000..cfe4eff --- /dev/null +++ b/backend/fortee/fortee.go @@ -0,0 +1,62 @@ +package fortee + +import ( + "context" + "errors" + "net/http" +) + +const ( + Endpoint = "https://fortee.jp" +) + +var ( + ErrLoginFailed = errors.New("fortee login failed") + ErrUserNotFound = errors.New("fortee user not found") +) + +func Login(ctx context.Context, username string, password string) (string, error) { + client, err := NewClientWithResponses(Endpoint, WithRequestEditorFn(addAcceptHeader)) + if err != nil { + return "", err + } + res, err := client.PostLoginWithFormdataBodyWithResponse(ctx, PostLoginFormdataRequestBody{ + Username: username, + Password: password, + }) + if err != nil { + return "", err + } + if res.StatusCode() != http.StatusOK { + return "", ErrLoginFailed + } + resOk := res.JSON200 + if !resOk.LoggedIn { + return "", ErrLoginFailed + } + if resOk.User == nil { + return "", ErrLoginFailed + } + return resOk.User.Username, nil +} + +func GetUserAvatarURL(ctx context.Context, username string) (string, error) { + client, err := NewClientWithResponses(Endpoint, WithRequestEditorFn(addAcceptHeader)) + if err != nil { + return "", err + } + res, err := client.GetUserWithResponse(ctx, username) + if err != nil { + return "", err + } + if res.StatusCode() != http.StatusOK { + return "", ErrUserNotFound + } + return res.JSON200.AvatarURL, nil +} + +// fortee API denies requests without Accept header. +func addAcceptHeader(_ context.Context, req *http.Request) error { + req.Header.Set("Accept", "application/json") + return nil +} diff --git a/backend/auth/fortee/generated.go b/backend/fortee/generated.go index 53529f9..5291449 100644 --- a/backend/auth/fortee/generated.go +++ b/backend/fortee/generated.go @@ -101,6 +101,9 @@ type ClientInterface interface { PostLoginWithBody(ctx context.Context, contentType string, body io.Reader, reqEditors ...RequestEditorFn) (*http.Response, error) PostLoginWithFormdataBody(ctx context.Context, body PostLoginFormdataRequestBody, reqEditors ...RequestEditorFn) (*http.Response, error) + + // GetUser request + GetUser(ctx context.Context, username string, reqEditors ...RequestEditorFn) (*http.Response, error) } func (c *Client) PostLoginWithBody(ctx context.Context, contentType string, body io.Reader, reqEditors ...RequestEditorFn) (*http.Response, error) { @@ -127,6 +130,18 @@ func (c *Client) PostLoginWithFormdataBody(ctx context.Context, body PostLoginFo return c.Client.Do(req) } +func (c *Client) GetUser(ctx context.Context, username string, reqEditors ...RequestEditorFn) (*http.Response, error) { + req, err := NewGetUserRequest(c.Server, username) + if err != nil { + return nil, err + } + req = req.WithContext(ctx) + if err := c.applyEditors(ctx, req, reqEditors); err != nil { + return nil, err + } + return c.Client.Do(req) +} + // NewPostLoginRequestWithFormdataBody calls the generic PostLogin builder with application/x-www-form-urlencoded body func NewPostLoginRequestWithFormdataBody(server string, body PostLoginFormdataRequestBody) (*http.Request, error) { var bodyReader io.Reader @@ -167,6 +182,40 @@ func NewPostLoginRequestWithBody(server string, contentType string, body io.Read return req, nil } +// NewGetUserRequest generates requests for GetUser +func NewGetUserRequest(server string, username string) (*http.Request, error) { + var err error + + var pathParam0 string + + pathParam0, err = runtime.StyleParamWithLocation("simple", false, "username", runtime.ParamLocationPath, username) + if err != nil { + return nil, err + } + + serverURL, err := url.Parse(server) + if err != nil { + return nil, err + } + + operationPath := fmt.Sprintf("/api/user/view/%s", pathParam0) + if operationPath[0] == '/' { + operationPath = "." + operationPath + } + + queryURL, err := serverURL.Parse(operationPath) + if err != nil { + return nil, err + } + + req, err := http.NewRequest("GET", queryURL.String(), nil) + if err != nil { + return nil, err + } + + return req, nil +} + func (c *Client) applyEditors(ctx context.Context, req *http.Request, additionalEditors []RequestEditorFn) error { for _, r := range c.RequestEditors { if err := r(ctx, req); err != nil { @@ -214,6 +263,9 @@ type ClientWithResponsesInterface interface { PostLoginWithBodyWithResponse(ctx context.Context, contentType string, body io.Reader, reqEditors ...RequestEditorFn) (*PostLoginResponse, error) PostLoginWithFormdataBodyWithResponse(ctx context.Context, body PostLoginFormdataRequestBody, reqEditors ...RequestEditorFn) (*PostLoginResponse, error) + + // GetUserWithResponse request + GetUserWithResponse(ctx context.Context, username string, reqEditors ...RequestEditorFn) (*GetUserResponse, error) } type PostLoginResponse struct { @@ -243,6 +295,32 @@ func (r PostLoginResponse) StatusCode() int { return 0 } +type GetUserResponse struct { + Body []byte + HTTPResponse *http.Response + JSON200 *struct { + AvatarURL string `json:"avatar_url"` + Username string `json:"username"` + UUID string `json:"uuid"` + } +} + +// Status returns HTTPResponse.Status +func (r GetUserResponse) Status() string { + if r.HTTPResponse != nil { + return r.HTTPResponse.Status + } + return http.StatusText(0) +} + +// StatusCode returns HTTPResponse.StatusCode +func (r GetUserResponse) StatusCode() int { + if r.HTTPResponse != nil { + return r.HTTPResponse.StatusCode + } + return 0 +} + // PostLoginWithBodyWithResponse request with arbitrary body returning *PostLoginResponse func (c *ClientWithResponses) PostLoginWithBodyWithResponse(ctx context.Context, contentType string, body io.Reader, reqEditors ...RequestEditorFn) (*PostLoginResponse, error) { rsp, err := c.PostLoginWithBody(ctx, contentType, body, reqEditors...) @@ -260,6 +338,15 @@ func (c *ClientWithResponses) PostLoginWithFormdataBodyWithResponse(ctx context. return ParsePostLoginResponse(rsp) } +// GetUserWithResponse request returning *GetUserResponse +func (c *ClientWithResponses) GetUserWithResponse(ctx context.Context, username string, reqEditors ...RequestEditorFn) (*GetUserResponse, error) { + rsp, err := c.GetUser(ctx, username, reqEditors...) + if err != nil { + return nil, err + } + return ParseGetUserResponse(rsp) +} + // ParsePostLoginResponse parses an HTTP response from a PostLoginWithResponse call func ParsePostLoginResponse(rsp *http.Response) (*PostLoginResponse, error) { bodyBytes, err := io.ReadAll(rsp.Body) @@ -290,3 +377,33 @@ func ParsePostLoginResponse(rsp *http.Response) (*PostLoginResponse, error) { return response, nil } + +// ParseGetUserResponse parses an HTTP response from a GetUserWithResponse call +func ParseGetUserResponse(rsp *http.Response) (*GetUserResponse, error) { + bodyBytes, err := io.ReadAll(rsp.Body) + defer func() { _ = rsp.Body.Close() }() + if err != nil { + return nil, err + } + + response := &GetUserResponse{ + Body: bodyBytes, + HTTPResponse: rsp, + } + + switch { + case strings.Contains(rsp.Header.Get("Content-Type"), "json") && rsp.StatusCode == 200: + var dest struct { + AvatarURL string `json:"avatar_url"` + Username string `json:"username"` + UUID string `json:"uuid"` + } + if err := json.Unmarshal(bodyBytes, &dest); err != nil { + return nil, err + } + response.JSON200 = &dest + + } + + return response, nil +} diff --git a/backend/gen/oapi-codegen.fortee.yaml b/backend/gen/oapi-codegen.fortee.yaml index 3bd5819..1799757 100644 --- a/backend/gen/oapi-codegen.fortee.yaml +++ b/backend/gen/oapi-codegen.fortee.yaml @@ -2,7 +2,7 @@ package: fortee generate: models: true client: true -output: ../auth/fortee/generated.go +output: ../fortee/generated.go output-options: skip-prune: true nullable-type: true diff --git a/backend/main.go b/backend/main.go index 3296957..890e666 100644 --- a/backend/main.go +++ b/backend/main.go @@ -96,6 +96,15 @@ func main() { return c.Redirect(http.StatusPermanentRedirect, "http://localhost:5173/iosdc-japan/2024/code-battle/logout") }) + // For local dev: This is never used in production because the reverse + // proxy directly handles /files. + filesGroup := e.Group("/iosdc-japan/2024/code-battle/files") + filesGroup.Use(middleware.StaticWithConfig(middleware.StaticConfig{ + Root: "/", + Filesystem: http.Dir("/data/files"), + IgnoreBase: true, + })) + go gameHubs.Run() go func() { diff --git a/backend/query.sql b/backend/query.sql index 6b0ecdd..dc5f384 100644 --- a/backend/query.sql +++ b/backend/query.sql @@ -13,6 +13,11 @@ INSERT INTO users (username, display_name, is_admin) VALUES ($1, $1, false) RETURNING user_id; +-- name: UpdateUserIconPath :exec +UPDATE users +SET icon_path = $2 +WHERE user_id = $1; + -- name: ListUsers :many SELECT * FROM users ORDER BY users.user_id; diff --git a/compose.local.yaml b/compose.local.yaml index cfcb41e..bf7c0c6 100644 --- a/compose.local.yaml +++ b/compose.local.yaml @@ -4,6 +4,8 @@ services: context: ./backend ports: - '127.0.0.1:8002:80' + volumes: + - files-data:/data/files:rw depends_on: db: condition: service_healthy @@ -67,3 +69,4 @@ services: volumes: db-data: + files-data: diff --git a/compose.prod.yaml b/compose.prod.yaml index 07ff19e..a3b840c 100644 --- a/compose.prod.yaml +++ b/compose.prod.yaml @@ -5,6 +5,7 @@ services: - '127.0.0.1:8002:80' volumes: - ./nginx.conf:/etc/nginx/nginx.conf:ro + - files-data:/var/www/files:ro depends_on: - api-server - app-server @@ -15,6 +16,8 @@ services: context: ./backend expose: - 80 + volumes: + - files-data:/data/files:rw depends_on: db: condition: service_healthy @@ -77,3 +80,4 @@ services: volumes: db-data: + files-data: diff --git a/frontend/app/components/GolfPlayApps/GolfPlayAppGaming.tsx b/frontend/app/components/GolfPlayApps/GolfPlayAppGaming.tsx index 4730583..a6c4550 100644 --- a/frontend/app/components/GolfPlayApps/GolfPlayAppGaming.tsx +++ b/frontend/app/components/GolfPlayApps/GolfPlayAppGaming.tsx @@ -55,12 +55,26 @@ export default function GolfPlayAppGaming({ <div className="text-gray-100">{gameDisplayName}</div> <div className="text-2xl">{leftTime}</div> </div> - <div className="font-bold text-end"> - <Link to={"/dashboard"} className="text-gray-100"> - {playerInfo.displayName} - </Link> - <div className="text-2xl">{playerInfo.score}</div> - </div> + <Link to={"/dashboard"}> + <div className="flex gap-4 my-auto font-bold"> + <div className="text-6xl">{playerInfo.score}</div> + <div className="text-end"> + <div className="text-gray-100">Player 1</div> + <div className="text-2xl">{playerInfo.displayName}</div> + </div> + {playerInfo.iconPath && ( + <img + src={ + process.env.NODE_ENV === "development" + ? `http://localhost:8002/iosdc-japan/2024/code-battle${playerInfo.iconPath}` + : `/iosdc-japan/2024/code-battle${playerInfo.iconPath}` + } + alt={`${playerInfo.displayName} のアイコン`} + className="w-12 h-12 rounded-full my-auto border-4 border-white" + /> + )} + </div> + </Link> </div> <div className="grow grid grid-cols-3 divide-x divide-gray-300"> <div className="p-4"> diff --git a/frontend/app/components/GolfWatchApps/GolfWatchAppGaming.tsx b/frontend/app/components/GolfWatchApps/GolfWatchAppGaming.tsx index c4c3a53..63c232b 100644 --- a/frontend/app/components/GolfWatchApps/GolfWatchAppGaming.tsx +++ b/frontend/app/components/GolfWatchApps/GolfWatchAppGaming.tsx @@ -47,9 +47,22 @@ export default function GolfWatchAppGaming({ <div className="min-h-screen bg-gray-100 flex flex-col"> <div className="text-white bg-iosdc-japan grid grid-cols-3 px-4 py-2"> <div className="font-bold flex justify-between my-auto"> - <div> - <div className="text-gray-100">Player 1</div> - <div className="text-2xl">{playerInfoA.displayName}</div> + <div className="flex gap-4"> + {playerInfoA.iconPath && ( + <img + src={ + process.env.NODE_ENV === "development" + ? `http://localhost:8002/iosdc-japan/2024/code-battle${playerInfoA.iconPath}` + : `/iosdc-japan/2024/code-battle${playerInfoA.iconPath}` + } + alt={`${playerInfoA.displayName} のアイコン`} + className="w-12 h-12 rounded-full my-auto border-4 border-white" + /> + )} + <div> + <div className="text-gray-100">Player 1</div> + <div className="text-2xl">{playerInfoA.displayName}</div> + </div> </div> <div className="text-6xl">{playerInfoA.score}</div> </div> @@ -59,9 +72,22 @@ export default function GolfWatchAppGaming({ </div> <div className="font-bold flex justify-between my-auto"> <div className="text-6xl">{playerInfoB.score}</div> - <div> - <div className="text-gray-100">Player 2</div> - <div className="text-2xl">{playerInfoB.displayName}</div> + <div className="flex gap-4 text-end"> + <div> + <div className="text-gray-100">Player 2</div> + <div className="text-2xl">{playerInfoB.displayName}</div> + </div> + {playerInfoB.iconPath && ( + <img + src={ + process.env.NODE_ENV === "development" + ? `http://localhost:8002/iosdc-japan/2024/code-battle${playerInfoB.iconPath}` + : `/iosdc-japan/2024/code-battle${playerInfoB.iconPath}` + } + alt={`${playerInfoB.displayName} のアイコン`} + className="w-12 h-12 rounded-full my-auto border-4 border-white" + /> + )} </div> </div> </div> diff --git a/frontend/app/routes/dashboard.tsx b/frontend/app/routes/dashboard.tsx index 99c64f2..614b3ae 100644 --- a/frontend/app/routes/dashboard.tsx +++ b/frontend/app/routes/dashboard.tsx @@ -31,6 +31,17 @@ export default function Dashboard() { return ( <div className="p-6 bg-gray-100 min-h-screen flex flex-col items-center"> + {user.icon_path && ( + <img + src={ + process.env.NODE_ENV === "development" + ? `http://localhost:8002/iosdc-japan/2024/code-battle${user.icon_path}` + : `/iosdc-japan/2024/code-battle${user.icon_path}` + } + alt={`${user.display_name} のアイコン`} + className="w-24 h-24 rounded-full mb-4" + /> + )} <h1 className="text-2xl font-bold mb-4"> <span className="text-gray-800">{user.display_name}</span> <span className="text-gray-500 ml-2">@{user.username}</span> @@ -12,6 +12,10 @@ http { server { listen 80; + location /iosdc-japan/2024/code-battle/files/ { + alias /var/www/files/; + } + location /iosdc-japan/2024/code-battle/api/ { proxy_pass http://api-server; proxy_set_header Host $host; diff --git a/openapi/fortee.yaml b/openapi/fortee.yaml index 707a29b..9237954 100644 --- a/openapi/fortee.yaml +++ b/openapi/fortee.yaml @@ -44,3 +44,36 @@ paths: - username required: - loggedIn + /api/user/view/{username}: + get: + operationId: getUser + summary: Get a user + parameters: + - in: path + name: username + schema: + type: string + required: true + responses: + '200': + description: User found + content: + application/json: + schema: + type: object + properties: + uuid: + type: string + example: "11111111-1111-1111-1111-111111111111" + username: + type: string + example: "john" + avatar_url: + type: string + example: "/files/_user/11111111-1111-1111-1111-111111111111.jpg" + required: + - uuid + - username + - avatar_url + '404': + description: User not found |
