Compare commits

...

10 commits

Author SHA1 Message Date
246fdc86b5 Merge branch 'dev-ars' into dev
All checks were successful
Build and Deploy Go App / build (push) Successful in 6m26s
Build and Deploy Go App / deploy (push) Successful in 26s
2025-11-27 07:11:30 +03:00
658d666fec feat: query for update usertitle 2025-11-27 07:08:06 +03:00
f2589e05e8 fix: now 409 on try to add existing usertitle 2025-11-27 07:06:18 +03:00
e98d2c6509
cicd: build auth using actions
All checks were successful
Build and Deploy Go App / build (push) Successful in 6m39s
Build and Deploy Go App / deploy (push) Successful in 27s
2025-11-27 06:35:43 +03:00
4c74315291
Merge branch 'front' into auth 2025-11-27 06:31:42 +03:00
4c643d80bb
feat: added title page 2025-11-27 06:29:36 +03:00
68294dd13c
fix: oapi shitty generation 2025-11-27 06:11:55 +03:00
a225d1fb60
feat: signup return username 2025-11-25 04:13:52 +03:00
e64e770783
feat: use SetCookie for access and refresh tokens 2025-11-23 03:32:58 +03:00
bbe57e07d5
feat: initial auth service support 2025-11-15 02:53:25 +03:00
26 changed files with 343 additions and 130 deletions

View file

@ -20,9 +20,9 @@ jobs:
go-version: '^1.25' go-version: '^1.25'
check-latest: false check-latest: false
cache-dependency-path: | cache-dependency-path: |
modules/backend/go.sum go.sum
- name: Build Go app - name: Build backend
run: | run: |
cd modules/backend cd modules/backend
go mod tidy go mod tidy
@ -35,6 +35,19 @@ jobs:
name: nyanimedb-backend.tar.gz name: nyanimedb-backend.tar.gz
path: modules/backend/nyanimedb-backend.tar.gz path: modules/backend/nyanimedb-backend.tar.gz
- name: Build auth
run: |
cd modules/auth
go mod tidy
go build -o auth .
tar -czvf nyanimedb-auth.tar.gz auth
- name: Upload built auth to artifactory
uses: actions/upload-artifact@v3
with:
name: nyanimedb-auth.tar.gz
path: modules/auth/nyanimedb-auth.tar.gz
# Build frontend # Build frontend
- uses: actions/setup-node@v5 - uses: actions/setup-node@v5
with: with:
@ -76,6 +89,14 @@ jobs:
push: true push: true
tags: meowgit.nekoea.red/nihonium/nyanimedb-backend:latest tags: meowgit.nekoea.red/nihonium/nyanimedb-backend:latest
- name: Build and push auth image
uses: docker/build-push-action@v6
with:
context: .
file: Dockerfiles/Dockerfile_auth
push: true
tags: meowgit.nekoea.red/nihonium/nyanimedb-auth:latest
- name: Build and push frontend image - name: Build and push frontend image
uses: docker/build-push-action@v6 uses: docker/build-push-action@v6
with: with:

View file

@ -86,6 +86,7 @@ paths:
description: Unknown server error description: Unknown server error
'/titles/{title_id}': '/titles/{title_id}':
get: get:
operationId: getTitle
summary: Get title description summary: Get title description
parameters: parameters:
- name: title_id - name: title_id
@ -116,6 +117,7 @@ paths:
description: Unknown server error description: Unknown server error
'/users/{user_id}': '/users/{user_id}':
get: get:
operationId: getUsersId
summary: Get user info summary: Get user info
parameters: parameters:
- name: user_id - name: user_id

View file

@ -1,5 +1,6 @@
get: get:
summary: Get title description summary: Get title description
operationId: getTitle
parameters: parameters:
- in: path - in: path
name: title_id name: title_id

View file

@ -128,7 +128,6 @@ post:
application/json: application/json:
schema: schema:
$ref: '../schemas/UserTitleMini.yaml' $ref: '../schemas/UserTitleMini.yaml'
'400': '400':
description: Invalid request body (missing fields, invalid types, etc.) description: Invalid request body (missing fields, invalid types, etc.)
'401': '401':
@ -180,7 +179,6 @@ patch:
application/json: application/json:
schema: schema:
$ref: '../schemas/UserTitleMini.yaml' $ref: '../schemas/UserTitleMini.yaml'
'400': '400':
description: Invalid request body (missing fields, invalid types, etc.) description: Invalid request body (missing fields, invalid types, etc.)
'401': '401':

View file

@ -1,5 +1,6 @@
get: get:
summary: Get user info summary: Get user info
operationId: getUsersId
parameters: parameters:
- in: path - in: path
name: user_id name: user_id

View file

@ -60,4 +60,3 @@ properties:
additionalProperties: additionalProperties:
type: number type: number
format: double format: double
additionalProperties: true

View file

@ -20,4 +20,4 @@ properties:
format: int64 format: int64
ctime: ctime:
type: string type: string
format: date-time format: date-time

View file

@ -1,7 +1,6 @@
package handlers package handlers
import ( import (
"context"
"encoding/json" "encoding/json"
"fmt" "fmt"
oapi "nyanimedb/api" oapi "nyanimedb/api"
@ -17,11 +16,11 @@ func NewServer(db *sqlc.Queries) Server {
return Server{db: db} return Server{db: db}
} }
func sql2StorageType(s *sqlc.StorageTypeT) (*oapi.ImageStorageType, error) { func sql2StorageType(s *sqlc.StorageTypeT) (*oapi.StorageType, error) {
if s == nil { if s == nil {
return nil, nil return nil, nil
} }
var t oapi.ImageStorageType var t oapi.StorageType
switch *s { switch *s {
case sqlc.StorageTypeTLocal: case sqlc.StorageTypeTLocal:
t = oapi.Local t = oapi.Local
@ -33,7 +32,7 @@ func sql2StorageType(s *sqlc.StorageTypeT) (*oapi.ImageStorageType, error) {
return &t, nil return &t, nil
} }
func (s Server) mapTitle(ctx context.Context, title sqlc.GetTitleByIDRow) (oapi.Title, error) { func (s Server) mapTitle(title sqlc.GetTitleByIDRow) (oapi.Title, error) {
oapi_title := oapi.Title{ oapi_title := oapi.Title{
EpisodesAired: title.EpisodesAired, EpisodesAired: title.EpisodesAired,

View file

@ -144,7 +144,7 @@ func (s Server) GetTitlesTitleId(ctx context.Context, request oapi.GetTitlesTitl
return oapi.GetTitlesTitleId500Response{}, nil return oapi.GetTitlesTitleId500Response{}, nil
} }
oapi_title, err = s.mapTitle(ctx, sqlc_title) oapi_title, err = s.mapTitle(sqlc_title)
if err != nil { if err != nil {
log.Errorf("%v", err) log.Errorf("%v", err)
return oapi.GetTitlesTitleId500Response{}, nil return oapi.GetTitlesTitleId500Response{}, nil
@ -238,7 +238,7 @@ func (s Server) GetTitles(ctx context.Context, request oapi.GetTitlesRequestObje
// _title.TitleStorageType = string(s) // _title.TitleStorageType = string(s)
// } // }
t, err := s.mapTitle(ctx, _title) t, err := s.mapTitle(_title)
if err != nil { if err != nil {
log.Errorf("%v", err) log.Errorf("%v", err)
return oapi.GetTitles500Response{}, nil return oapi.GetTitles500Response{}, nil

View file

@ -2,6 +2,7 @@ package handlers
import ( import (
"context" "context"
"errors"
"fmt" "fmt"
oapi "nyanimedb/api" oapi "nyanimedb/api"
sqlc "nyanimedb/sql" sqlc "nyanimedb/sql"
@ -9,24 +10,12 @@ import (
"time" "time"
"github.com/jackc/pgx/v5" "github.com/jackc/pgx/v5"
"github.com/jackc/pgx/v5/pgconn"
"github.com/jackc/pgx/v5/pgtype" "github.com/jackc/pgx/v5/pgtype"
"github.com/oapi-codegen/runtime/types" "github.com/oapi-codegen/runtime/types"
log "github.com/sirupsen/logrus" log "github.com/sirupsen/logrus"
) )
// type Server struct {
// db *sqlc.Queries
// }
// func NewServer(db *sqlc.Queries) Server {
// return Server{db: db}
// }
// func parseInt64(s string) (int32, error) {
// i, err := strconv.ParseInt(s, 10, 64)
// return int32(i), err
// }
func mapUser(u sqlc.GetUserByIDRow) (oapi.User, error) { func mapUser(u sqlc.GetUserByIDRow) (oapi.User, error) {
i := oapi.Image{ i := oapi.Image{
Id: u.AvatarID, Id: u.AvatarID,
@ -202,7 +191,7 @@ func (s Server) mapUsertitle(ctx context.Context, t sqlc.SearchUserTitlesRow) (o
// StudioImagePath: title.StudioImagePath, // StudioImagePath: title.StudioImagePath,
} }
oapi_title, err := s.mapTitle(ctx, _title) oapi_title, err := s.mapTitle(_title)
if err != nil { if err != nil {
return oapi_usertitle, fmt.Errorf("mapUsertitle: %v", err) return oapi_usertitle, fmt.Errorf("mapUsertitle: %v", err)
} }
@ -368,19 +357,26 @@ func (s Server) AddUserTitle(ctx context.Context, request oapi.AddUserTitleReque
} }
params := sqlc.InsertUserTitleParams{ params := sqlc.InsertUserTitleParams{
UserID: request.UserId, UserID: request.UserId,
TitleID: request.Body.TitleId, TitleID: request.Body.TitleId,
Status: *status, Status: *status,
Rate: request.Body.Rate, Rate: request.Body.Rate,
ReviewID: request.Body.ReviewId,
} }
user_title, err := s.db.InsertUserTitle(ctx, params) user_title, err := s.db.InsertUserTitle(ctx, params)
if err != nil { if err != nil {
log.Errorf("%v", err) var pgErr *pgconn.PgError
return oapi.AddUserTitle500Response{}, nil if errors.As(err, &pgErr) {
// fmt.Println(pgErr.Message) // => syntax error at end of input
// fmt.Println(pgErr.Code) // => 42601
if pgErr.Code == "23505" { //duplicate key value
return oapi.AddUserTitle409Response{}, nil
}
} else {
log.Errorf("%v", err)
return oapi.AddUserTitle500Response{}, nil
}
} }
oapi_status, err := sql2usertitlestatus(user_title.Status) oapi_status, err := sql2usertitlestatus(user_title.Status)
if err != nil { if err != nil {
log.Errorf("%v", err) log.Errorf("%v", err)
@ -406,3 +402,13 @@ func (s Server) AddUserTitle(ctx context.Context, request oapi.AddUserTitleReque
return oapi.AddUserTitle200JSONResponse(oapi_usertitle), nil return oapi.AddUserTitle200JSONResponse(oapi_usertitle), nil
} }
// DeleteUserTitle implements oapi.StrictServerInterface.
func (s Server) DeleteUserTitle(ctx context.Context, request oapi.DeleteUserTitleRequestObject) (oapi.DeleteUserTitleResponseObject, error) {
panic("unimplemented")
}
// UpdateUserTitle implements oapi.StrictServerInterface.
func (s Server) UpdateUserTitle(ctx context.Context, request oapi.UpdateUserTitleRequestObject) (oapi.UpdateUserTitleResponseObject, error) {
panic("unimplemented")
}

View file

@ -461,21 +461,13 @@ VALUES (
) )
RETURNING user_id, title_id, status, rate, review_id, ctime; RETURNING user_id, title_id, status, rate, review_id, ctime;
-- -- name: UpdateUserTitle :one -- name: UpdateUserTitle :one
-- UPDATE usertitles -- Fails with sql.ErrNoRows if (user_id, title_id) not found
-- SET UPDATE usertitles
-- status = COALESCE(sqlc.narg('status'), status), SET
-- rate = COALESCE(sqlc.narg('rate'), rate), status = COALESCE(sqlc.narg('status')::usertitle_status_t, status),
-- review_id = COALESCE(sqlc.narg('review_id'), review_id) rate = COALESCE(sqlc.narg('rate')::int, rate)
-- WHERE user_id = $1 AND title_id = $2 WHERE
-- RETURNING *; user_id = sqlc.arg('user_id')
AND title_id = sqlc.arg('title_id')
-- -- name: DeleteUserTitle :exec RETURNING *;
-- DELETE FROM usertitles
-- WHERE user_id = $1 AND ($2::int IS NULL OR title_id = $2);
-- -- name: ListTags :many
-- SELECT tag_id, tag_names
-- FROM tags
-- ORDER BY tag_id
-- LIMIT $1 OFFSET $2;

View file

@ -2,6 +2,7 @@ import React from "react";
import { BrowserRouter as Router, Routes, Route } from "react-router-dom"; import { BrowserRouter as Router, Routes, Route } from "react-router-dom";
import UsersIdPage from "./pages/UsersIdPage/UsersIdPage"; import UsersIdPage from "./pages/UsersIdPage/UsersIdPage";
import TitlesPage from "./pages/TitlesPage/TitlesPage"; import TitlesPage from "./pages/TitlesPage/TitlesPage";
import TitlePage from "./pages/TitlePage/TitlePage";
import { LoginPage } from "./pages/LoginPage/LoginPage"; import { LoginPage } from "./pages/LoginPage/LoginPage";
import { Header } from "./components/Header/Header"; import { Header } from "./components/Header/Header";
@ -24,7 +25,9 @@ const App: React.FC = () => {
/> />
<Route path="/users/:id" element={<UsersIdPage />} /> <Route path="/users/:id" element={<UsersIdPage />} />
<Route path="/titles" element={<TitlesPage />} /> <Route path="/titles" element={<TitlesPage />} />
<Route path="/titles/:id" element={<TitlePage />} />
</Routes> </Routes>
</Router> </Router>
); );

View file

@ -20,7 +20,7 @@ export type OpenAPIConfig = {
}; };
export const OpenAPI: OpenAPIConfig = { export const OpenAPI: OpenAPIConfig = {
BASE: '/api/v1', BASE: 'http://10.1.0.65:8081/api/v1',
VERSION: '1.0.0', VERSION: '1.0.0',
WITH_CREDENTIALS: false, WITH_CREDENTIALS: false,
CREDENTIALS: 'include', CREDENTIALS: 'include',

View file

@ -12,6 +12,7 @@ export type { CursorObj } from './models/CursorObj';
export type { Image } from './models/Image'; export type { Image } from './models/Image';
export type { ReleaseSeason } from './models/ReleaseSeason'; export type { ReleaseSeason } from './models/ReleaseSeason';
export type { Review } from './models/Review'; export type { Review } from './models/Review';
export type { StorageType } from './models/StorageType';
export type { Studio } from './models/Studio'; export type { Studio } from './models/Studio';
export type { Tag } from './models/Tag'; export type { Tag } from './models/Tag';
export type { Tags } from './models/Tags'; export type { Tags } from './models/Tags';
@ -21,6 +22,7 @@ export type { TitleSort } from './models/TitleSort';
export type { TitleStatus } from './models/TitleStatus'; export type { TitleStatus } from './models/TitleStatus';
export type { User } from './models/User'; export type { User } from './models/User';
export type { UserTitle } from './models/UserTitle'; export type { UserTitle } from './models/UserTitle';
export type { UserTitleMini } from './models/UserTitleMini';
export type { UserTitleStatus } from './models/UserTitleStatus'; export type { UserTitleStatus } from './models/UserTitleStatus';
export { DefaultService } from './services/DefaultService'; export { DefaultService } from './services/DefaultService';

View file

@ -2,12 +2,10 @@
/* istanbul ignore file */ /* istanbul ignore file */
/* tslint:disable */ /* tslint:disable */
/* eslint-disable */ /* eslint-disable */
import type { StorageType } from './StorageType';
export type Image = { export type Image = {
id?: number; id?: number;
/** storage_type?: StorageType;
* Image storage type
*/
storage_type?: 's3' | 'local';
image_path?: string; image_path?: string;
}; };

View file

@ -0,0 +1,8 @@
/* generated using openapi-typescript-codegen -- do not edit */
/* istanbul ignore file */
/* tslint:disable */
/* eslint-disable */
/**
* Image storage type
*/
export type StorageType = 's3' | 'local';

View file

@ -2,4 +2,30 @@
/* istanbul ignore file */ /* istanbul ignore file */
/* tslint:disable */ /* tslint:disable */
/* eslint-disable */ /* eslint-disable */
export type Title = Record<string, any>; import type { Image } from './Image';
import type { ReleaseSeason } from './ReleaseSeason';
import type { Studio } from './Studio';
import type { Tags } from './Tags';
import type { TitleStatus } from './TitleStatus';
export type Title = {
/**
* Unique title ID (primary key)
*/
id: number;
/**
* Localized titles. Key = language (ISO 639-1), value = list of names
*/
title_names: Record<string, Array<string>>;
studio?: Studio;
tags: Tags;
poster?: Image;
title_status?: TitleStatus;
rating?: number;
rating_count?: number;
release_year?: number;
release_season?: ReleaseSeason;
episodes_aired?: number;
episodes_all?: number;
episodes_len?: Record<string, number>;
};

View file

@ -0,0 +1,14 @@
/* generated using openapi-typescript-codegen -- do not edit */
/* istanbul ignore file */
/* tslint:disable */
/* eslint-disable */
import type { UserTitleStatus } from './UserTitleStatus';
export type UserTitleMini = {
user_id: number;
title_id: number;
status: UserTitleStatus;
rate?: number;
review_id?: number;
ctime?: string;
};

View file

@ -9,6 +9,7 @@ import type { TitleSort } from '../models/TitleSort';
import type { TitleStatus } from '../models/TitleStatus'; import type { TitleStatus } from '../models/TitleStatus';
import type { User } from '../models/User'; import type { User } from '../models/User';
import type { UserTitle } from '../models/UserTitle'; import type { UserTitle } from '../models/UserTitle';
import type { UserTitleMini } from '../models/UserTitleMini';
import type { UserTitleStatus } from '../models/UserTitleStatus'; import type { UserTitleStatus } from '../models/UserTitleStatus';
import type { CancelablePromise } from '../core/CancelablePromise'; import type { CancelablePromise } from '../core/CancelablePromise';
import { OpenAPI } from '../core/OpenAPI'; import { OpenAPI } from '../core/OpenAPI';
@ -78,7 +79,7 @@ export class DefaultService {
* @returns Title Title description * @returns Title Title description
* @throws ApiError * @throws ApiError
*/ */
public static getTitles1( public static getTitle(
titleId: number, titleId: number,
fields: string = 'all', fields: string = 'all',
): CancelablePromise<Title> { ): CancelablePromise<Title> {
@ -105,7 +106,7 @@ export class DefaultService {
* @returns User User info * @returns User User info
* @throws ApiError * @throws ApiError
*/ */
public static getUsers( public static getUsersId(
userId: string, userId: string,
fields: string = 'all', fields: string = 'all',
): CancelablePromise<User> { ): CancelablePromise<User> {
@ -248,22 +249,17 @@ export class DefaultService {
* User adding title to list af watched, status required * User adding title to list af watched, status required
* @param userId ID of the user to assign the title to * @param userId ID of the user to assign the title to
* @param requestBody * @param requestBody
* @returns any Title successfully added to user * @returns UserTitleMini Title successfully added to user
* @throws ApiError * @throws ApiError
*/ */
public static addUserTitle( public static addUserTitle(
userId: number, userId: number,
requestBody: UserTitle, requestBody: {
): CancelablePromise<{
data?: {
user_id: number;
title_id: number; title_id: number;
status: UserTitleStatus; status: UserTitleStatus;
rate?: number; rate?: number;
review_id?: number; },
ctime?: string; ): CancelablePromise<UserTitleMini> {
};
}> {
return __request(OpenAPI, { return __request(OpenAPI, {
method: 'POST', method: 'POST',
url: '/users/{user_id}/titles', url: '/users/{user_id}/titles',
@ -282,4 +278,37 @@ export class DefaultService {
}, },
}); });
} }
/**
* Update a usertitle
* User updating title list of watched
* @param userId ID of the user to assign the title to
* @param requestBody
* @returns UserTitleMini Title successfully updated
* @throws ApiError
*/
public static updateUserTitle(
userId: number,
requestBody: {
title_id: number;
status?: UserTitleStatus;
rate?: number;
},
): CancelablePromise<UserTitleMini> {
return __request(OpenAPI, {
method: 'PATCH',
url: '/users/{user_id}/titles',
path: {
'user_id': userId,
},
body: requestBody,
mediaType: 'application/json',
errors: {
400: `Invalid request body (missing fields, invalid types, etc.)`,
401: `Unauthorized — missing or invalid auth token`,
403: `Forbidden — user not allowed to update title`,
404: `User or Title not found`,
500: `Internal server error`,
},
});
}
} }

View file

@ -20,7 +20,7 @@ export type OpenAPIConfig = {
}; };
export const OpenAPI: OpenAPIConfig = { export const OpenAPI: OpenAPIConfig = {
BASE: '/auth', BASE: 'http://10.1.0.65:8081/auth',
VERSION: '1.0.0', VERSION: '1.0.0',
WITH_CREDENTIALS: false, WITH_CREDENTIALS: false,
CREDENTIALS: 'include', CREDENTIALS: 'include',

View file

@ -1,64 +1,140 @@
// import React, { useEffect, useState } from "react"; import { useEffect, useState } from "react";
// import { useParams } from "react-router-dom"; import { useParams } from "react-router-dom";
// import { DefaultService } from "../../api/services/DefaultService"; import { DefaultService } from "../../api/services/DefaultService";
// import type { User } from "../../api/models/User"; import type { Title, UserTitleStatus } from "../../api";
// import styles from "./UserPage.module.css"; import {
ClockIcon,
CheckCircleIcon,
PlayCircleIcon,
XCircleIcon,
} from "@heroicons/react/24/solid";
// const UserPage: React.FC = () => { const STATUS_BUTTONS: { status: UserTitleStatus; icon: React.ReactNode; label: string }[] = [
// const { id } = useParams<{ id: string }>(); { status: "planned", icon: <ClockIcon className="w-6 h-6" />, label: "Planned" },
// const [user, setUser] = useState<User | null>(null); { status: "finished", icon: <CheckCircleIcon className="w-6 h-6" />, label: "Finished" },
// const [loading, setLoading] = useState(true); { status: "in-progress", icon: <PlayCircleIcon className="w-6 h-6" />, label: "In Progress" },
// const [error, setError] = useState<string | null>(null); { status: "dropped", icon: <XCircleIcon className="w-6 h-6" />, label: "Dropped" },
];
// useEffect(() => { export default function TitlePage() {
// if (!id) return; const params = useParams();
const titleId = Number(params.id);
// const getTitleInfo = async () => { const [title, setTitle] = useState<Title | null>(null);
// try { const [loading, setLoading] = useState(true);
// const userInfo = await DefaultService.getTitle(id, "all"); const [error, setError] = useState<string | null>(null);
// setUser(userInfo);
// } catch (err) {
// console.error(err);
// setError("Failed to fetch user info.");
// } finally {
// setLoading(false);
// }
// };
// getTitleInfo();
// }, [id]);
// if (loading) return <div className={styles.loader}>Loading...</div>; const [userStatus, setUserStatus] = useState<UserTitleStatus | null>(null);
// if (error) return <div className={styles.error}>{error}</div>; const [updatingStatus, setUpdatingStatus] = useState(false);
// if (!user) return <div className={styles.error}>User not found.</div>;
// return ( useEffect(() => {
// <div className={styles.container}> const fetchTitle = async () => {
// <div className={styles.card}> setLoading(true);
// <div className={styles.avatar}> try {
// {user.avatar_id ? ( const data = await DefaultService.getTitle(titleId, "all");
// <img setTitle(data);
// src={`/images/${user.avatar_id}.png`} setError(null);
// alt="User Avatar" } catch (err: any) {
// className={styles.avatarImg} console.error(err);
// /> setError(err?.message || "Failed to fetch title");
// ) : ( } finally {
// <div className={styles.avatarPlaceholder}> setLoading(false);
// {user.disp_name?.[0] || "U"} }
// </div> };
// )} fetchTitle();
// </div> }, [titleId]);
// <div className={styles.info}> const handleStatusClick = async (status: UserTitleStatus) => {
// <h1 className={styles.name}>{user.disp_name || user.nickname}</h1> if (updatingStatus || userStatus === status) return;
// <p className={styles.nickname}>@{user.nickname}</p>
// {user.user_desc && <p className={styles.desc}>{user.user_desc}</p>}
// <p className={styles.created}>
// Joined: {new Date(user.creation_date).toLocaleDateString()}
// </p>
// </div>
// </div>
// </div>
// );
// };
// export default UserPage; const userId = Number(localStorage.getItem("userId"));
if (!userId) {
alert("You must be logged in to set status.");
return;
}
setUpdatingStatus(true);
try {
await DefaultService.addUserTitle(userId, {
title_id: titleId,
status,
});
setUserStatus(status);
} catch (err: any) {
console.error(err);
alert(err?.message || "Failed to set status");
} finally {
setUpdatingStatus(false);
}
};
const getTagsString = () =>
title?.tags?.map(tag => tag.en).filter(Boolean).join(", ");
if (loading) return <div className="mt-20 font-medium text-black">Loading title...</div>;
if (error) return <div className="mt-20 text-red-600 font-medium">{error}</div>;
if (!title) return null;
return (
<div className="w-full min-h-screen bg-gray-50 p-6 flex justify-center">
<div className="flex flex-col md:flex-row bg-white shadow-lg rounded-xl max-w-4xl w-full p-6 gap-6">
{/* Постер */}
<div className="flex flex-col items-center">
<img
src={title.poster?.image_path || "/default-poster.png"}
alt={title.title_names?.en?.[0] || "Title poster"}
className="w-48 h-72 object-cover rounded-lg mb-4"
/>
{/* Статус кнопки с иконками */}
<div className="flex gap-2 mt-2 flex-wrap justify-center">
{STATUS_BUTTONS.map(btn => (
<button
key={btn.status}
onClick={() => handleStatusClick(btn.status)}
disabled={updatingStatus}
className={`p-2 rounded-lg transition flex items-center justify-center ${
userStatus === btn.status
? "bg-blue-600 text-white"
: "bg-gray-200 text-gray-700 hover:bg-gray-300"
}`}
title={btn.label}
>
{btn.icon}
</button>
))}
</div>
</div>
{/* Информация о тайтле */}
<div className="flex-1 flex flex-col">
<h1 className="text-3xl font-bold mb-2">
{title.title_names?.en?.[0] || "Untitled"}
</h1>
{title.studio && <p className="text-gray-700 mb-1">Studio: {title.studio.name}</p>}
{title.title_status && <p className="text-gray-700 mb-1">Status: {title.title_status}</p>}
{title.rating !== undefined && (
<p className="text-gray-700 mb-1">
Rating: {title.rating} ({title.rating_count} votes)
</p>
)}
{title.release_year && (
<p className="text-gray-700 mb-1">
Released: {title.release_year} {title.release_season || ""}
</p>
)}
{title.episodes_aired !== undefined && (
<p className="text-gray-700 mb-1">
Episodes: {title.episodes_aired}/{title.episodes_all}
</p>
)}
{title.tags && title.tags.length > 0 && (
<p className="text-gray-700 mb-1">
Tags: {getTagsString()}
</p>
)}
</div>
</div>
</div>
);
}

View file

@ -15,7 +15,7 @@ const UserPage: React.FC = () => {
const getUserInfo = async () => { const getUserInfo = async () => {
try { try {
const userInfo = await DefaultService.getUsers(id, "all"); // <-- use dynamic id const userInfo = await DefaultService.getUsersId(id, "all"); // <-- use dynamic id
setUser(userInfo); setUser(userInfo);
} catch (err) { } catch (err) {
console.error(err); console.error(err);

View file

@ -41,7 +41,7 @@ export default function UsersIdPage({ userId }: UsersIdPageProps) {
if (!id) return; if (!id) return;
setLoadingUser(true); setLoadingUser(true);
try { try {
const result = await DefaultService.getUsers(id, "all"); const result = await DefaultService.getUsersId(id, "all");
setUser(result); setUser(result);
setErrorUser(null); setErrorUser(null);
} catch (err: any) { } catch (err: any) {

View file

@ -179,6 +179,6 @@ END;
$$ LANGUAGE plpgsql; $$ LANGUAGE plpgsql;
CREATE TRIGGER set_ctime_on_update CREATE TRIGGER set_ctime_on_update
AFTER UPDATE ON usertitles BEFORE UPDATE ON usertitles
FOR EACH ROW FOR EACH ROW
EXECUTE FUNCTION set_ctime(); EXECUTE FUNCTION set_ctime();

View file

@ -925,3 +925,41 @@ func (q *Queries) UpdateUser(ctx context.Context, arg UpdateUserParams) (UpdateU
) )
return i, err return i, err
} }
const updateUserTitle = `-- name: UpdateUserTitle :one
UPDATE usertitles
SET
status = COALESCE($1::usertitle_status_t, status),
rate = COALESCE($2::int, rate)
WHERE
user_id = $3
AND title_id = $4
RETURNING user_id, title_id, status, rate, review_id, ctime
`
type UpdateUserTitleParams struct {
Status NullUsertitleStatusT `json:"status"`
Rate *int32 `json:"rate"`
UserID int64 `json:"user_id"`
TitleID int64 `json:"title_id"`
}
// Fails with sql.ErrNoRows if (user_id, title_id) not found
func (q *Queries) UpdateUserTitle(ctx context.Context, arg UpdateUserTitleParams) (Usertitle, error) {
row := q.db.QueryRow(ctx, updateUserTitle,
arg.Status,
arg.Rate,
arg.UserID,
arg.TitleID,
)
var i Usertitle
err := row.Scan(
&i.UserID,
&i.TitleID,
&i.Status,
&i.Rate,
&i.ReviewID,
&i.Ctime,
)
return i, err
}