ADD New Pre-Commit hooks and Formatting code

This commit is contained in:
Kentai Radiquum 2024-04-23 18:30:35 +05:00
parent 5c9c3e67fa
commit 9e75a0783c
Signed by: Radiquum
GPG key ID: 858E8EE696525EED
26 changed files with 4163 additions and 105 deletions

3
.gitignore vendored
View file

@ -42,3 +42,6 @@ yarn-error.log*
## typescript ## typescript
*.tsbuildinfo *.tsbuildinfo
next-env.d.ts next-env.d.ts
# traefik
traefik/traefik

View file

@ -1,11 +1,20 @@
# See https://pre-commit.com for more information # See https://pre-commit.com for more information
# See https://pre-commit.com/hooks.html for more hooks # See https://pre-commit.com/hooks.html for more hooks
repos: repos:
- repo: https://github.com/pre-commit/mirrors-mypy
rev: v1.9.0
hooks:
- id: mypy
args: [--no-strict-optional, --ignore-missing-imports]
additional_dependencies: [tokenize-rt==3.2.0, types-requests]
files: ^(backend/)
- repo: https://github.com/psf/black - repo: https://github.com/psf/black
rev: 24.4.0 rev: 24.4.0
hooks: hooks:
- id: black - id: black
args: [--safe] args: [--safe]
files: ^(backend/)
- repo: https://github.com/pre-commit/pre-commit-hooks - repo: https://github.com/pre-commit/pre-commit-hooks
rev: v4.5.0 rev: v4.5.0
@ -22,15 +31,37 @@ repos:
hooks: hooks:
- id: flake8 - id: flake8
language_version: python3 language_version: python3
args: [--ignore=E501]
files: ^(backend/)
- repo: https://github.com/asottile/reorder_python_imports - repo: https://github.com/asottile/reorder_python_imports
rev: v3.12.0 rev: v3.12.0
hooks: hooks:
- id: reorder-python-imports - id: reorder-python-imports
args: [--py39-plus] args: [--py39-plus]
files: ^(backend/)
- repo: https://github.com/asottile/pyupgrade - repo: https://github.com/asottile/pyupgrade
rev: v3.15.2 rev: v3.15.2
hooks: hooks:
- id: pyupgrade - id: pyupgrade
args: [--py39-plus] args: [--py39-plus]
files: ^(backend/)
- repo: local
hooks:
- id: next-lint
name: next-lint
entry: bash -c 'cd ./frontend && npm run lint'
language: system
files: ^(frontend/)
stages:
- manual
- repo: https://github.com/pre-commit/mirrors-prettier
rev: v3.1.0 # Use the sha or tag you want to point at
hooks:
- id: prettier
additional_dependencies:
- prettier@3.1.0
files: ^(frontend/)

View file

@ -17,8 +17,8 @@ Please note that AniX is an unofficial project and is not affiliated with the de
### Prerequisites ### Prerequisites
* Python 3.6 or later ([https://www.python.org/downloads/](https://www.python.org/downloads/)) - Python 3.6 or later ([https://www.python.org/downloads/](https://www.python.org/downloads/))
* Node.js and npm ([https://nodejs.org/en](https://nodejs.org/en)) - Node.js and npm ([https://nodejs.org/en](https://nodejs.org/en))
### Setting Up the Backend ### Setting Up the Backend
@ -38,8 +38,6 @@ Please note that AniX is an unofficial project and is not affiliated with the de
pip install -r ./requirements.txt pip install -r ./requirements.txt
``` ```
5. (Optional) Create a `.env` file in the project root directory to store sensitive information like API keys.
### Setting Up the Frontend ### Setting Up the Frontend
1. Navigate to the `frontend` directory. 1. Navigate to the `frontend` directory.
@ -54,26 +52,33 @@ Please note that AniX is an unofficial project and is not affiliated with the de
1. Start the backend server: 1. Start the backend server:
```bash ```bash
cd .. # Navigate back to the project root directory cd ./backend
source ./venv/bin/activate
uvicorn main:app --reload uvicorn main:app --reload
``` ```
2. Start the frontend development server: 2. Start the frontend development server:
```bash ```bash
cd frontend cd ./frontend
npm run dev npm run dev
``` ```
3. Start local reverse proxy server like traefik 3. Start local traefik reverse proxy
```to be added soon``` 1. download a traefik binary from [github](https://github.com/traefik/traefik/releases/tag/v2.11.2)
2. place it in traefik directory and make it executable
3. add the following line to the hosts file `127.0.0.1 anix.test.local`
4. run `sudo ./traefik --configFile ./traefik.yml`
This will start the development server for both the backend and frontend. You can access the AniX web client in your browser at ```no url for now```. This will start the development server for both the backend and frontend. You can access the AniX web client in your browser at `anix.test.local` via http only.
<!-- ## Development ## Development
The code for both the backend and frontend is well-commented and should be easy to understand and modify. Feel free to make changes and experiment with the project. --> 0. Install global pre-commit `pip install pre-commit`
1. Install pre-commit hooks `pre-commit install #inside repository folder`
Feel free to make changes and experiment with the project.
## Deployment ## Deployment

View file

@ -31,11 +31,13 @@ TAGS = [
] ]
app = FastAPI( app = FastAPI(
openapi_tags=TAGS, openapi_tags=TAGS,
title="AniX API", title="AniX API",
description="unofficial API proxy for Anixart android application.", description="unofficial API proxy for Anixart android application.",
openapi_url="/api/openapi.json", openapi_url="/api/openapi.json",
docs_url="/api/docs", redoc_url=None) docs_url="/api/docs",
redoc_url=None,
)
app.include_router(profile.router, prefix="/api/profile", tags=["Profile"]) app.include_router(profile.router, prefix="/api/profile", tags=["Profile"])
app.include_router(auth.router, prefix="/api/auth", tags=["Profile"]) app.include_router(auth.router, prefix="/api/auth", tags=["Profile"])

View file

@ -1,10 +1,22 @@
from typing import TypedDict
from typing import Union from typing import Union
import requests import requests
from fastapi import Request from fastapi import Request
class Endpoints(TypedDict):
release: dict[str, str]
profile: str
filter: str
auth: str
user: dict[str, str]
search: str
statistic: dict[str, str]
API_URL = "https://api.anixart.tv" API_URL = "https://api.anixart.tv"
ENDPOINTS = { ENDPOINTS: Endpoints = {
"release": { "release": {
"info": f"{API_URL}/release", "info": f"{API_URL}/release",
"episode": f"{API_URL}/episode", "episode": f"{API_URL}/episode",
@ -30,10 +42,9 @@ USER_AGENT = "AnixartApp/8.2.1-23121216 (Android 11; SDK 30; arm64-v8a;)"
async def apiRequest( async def apiRequest(
# noqa: E501
request: Request = None, request: Request = None,
endpoint: str = "", endpoint: Union[str, Endpoints] = "",
path: str = "", path: Union[str, int] = "",
query: str = "", query: str = "",
data: Union[None, str, dict] = None, data: Union[None, str, dict] = None,
): ):

View file

@ -11,9 +11,7 @@ async def GetReleaseById(request: Request, release_id: str):
return await apiRequest(request, ENDPOINTS["release"]["info"], release_id) return await apiRequest(request, ENDPOINTS["release"]["info"], release_id)
@router.get( @router.get("/{release_id}/voiceover", summary="Get release voiceover info")
"/{release_id}/voiceover", summary="Get release voiceover info"
)
async def GetReleaseVoiceover(request: Request, release_id: str): async def GetReleaseVoiceover(request: Request, release_id: str):
return await apiRequest(request, ENDPOINTS["release"]["episode"], release_id) return await apiRequest(request, ENDPOINTS["release"]["episode"], release_id)

View file

@ -1,14 +1,17 @@
from typing import Union
from fastapi import APIRouter from fastapi import APIRouter
from fastapi import Request from fastapi import Request
from modules.proxy import apiRequest from modules.proxy import apiRequest
from modules.proxy import ENDPOINTS from modules.proxy import ENDPOINTS
from typing import Union
router = APIRouter() router = APIRouter()
@router.get("/{user_id}", summary="Get user profile by user ID") @router.get("/{user_id}", summary="Get user profile by user ID")
async def getUserById(request: Request, user_id: str, short: bool = False, token: Union[None, str] = None): async def getUserById(
request: Request, user_id: str, short: bool = False, token: Union[None, str] = None
):
query = "" query = ""
if token: if token:
query = f"?token={token}" query = f"?token={token}"
@ -23,5 +26,5 @@ async def getUserById(request: Request, user_id: str, short: bool = False, token
"login": res["profile"]["login"], "login": res["profile"]["login"],
"avatar": res["profile"]["avatar"], "avatar": res["profile"]["avatar"],
}, },
"is_my_profile": res["is_my_profile"] "is_my_profile": res["is_my_profile"],
} }

3
frontend/.eslintrc.json Normal file
View file

@ -0,0 +1,3 @@
{
"extends": "next/core-web-vitals"
}

View file

@ -1,11 +1,11 @@
export const API_URL = "http://anix.test.local/api"; export const API_URL = "/api";
export const endpoints = { export const endpoints = {
index: { index: {
last: `${API_URL}/index/last`, last: `${API_URL}/index/last`,
ongoing: `${API_URL}/index/ongoing`, ongoing: `${API_URL}/index/ongoing`,
announce: `${API_URL}/index/announce`, announce: `${API_URL}/index/announce`,
finished: `${API_URL}/index/finished`, finished: `${API_URL}/index/finished`,
}, },
search: `${API_URL}/search` search: `${API_URL}/search`,
}; };

View file

@ -3,7 +3,8 @@ import { ReleaseCard } from "@/app/components/ReleaseCard/ReleaseCard";
export const CardList = (props) => { export const CardList = (props) => {
return props.data.map((item) => { return props.data.map((item) => {
return ( return (
<ReleaseCard key={item.id} <ReleaseCard
key={item.id}
id={item.id} id={item.id}
title={item.title_ru} title={item.title_ru}
poster={item.image} poster={item.image}

View file

@ -20,37 +20,40 @@ export const ColorPicker = (props) => {
const [mode, setMode] = useState(ui("mode")); const [mode, setMode] = useState(ui("mode"));
return ( return (
<dialog className="active left round bottom small" style={{blockSize: "unset"}}> <dialog
<h5>Выбор темы</h5> className="active left round bottom small"
<div className="grid center-align"> style={{ blockSize: "unset" }}
{colors.map((item) => { >
return ( <h5>Выбор темы</h5>
<button <div className="grid center-align">
key={item.color} {colors.map((item) => {
className={`circle border small ${item.color} s2`} return (
onClick={() => props.theme(item.hex)} <button
></button> key={item.color}
); className={`circle border small ${item.color} s2`}
})} onClick={() => props.theme(item.hex)}
</div> ></button>
<div className="medium-divider"></div> );
<nav> })}
<button </div>
className={`circle small transparent`} <div className="medium-divider"></div>
onClick={() => { <nav>
props.mode(); <button
setMode(ui("mode")); className={`circle small transparent`}
}} onClick={() => {
> props.mode();
{mode == "light" ? <i>dark_mode</i> : <i>light_mode</i>} setMode(ui("mode"));
</button> }}
<button >
className={`circle small transparent `} {mode == "light" ? <i>dark_mode</i> : <i>light_mode</i>}
onClick={() => props.setColorPicker(!props.colorPicker)} </button>
> <button
<i>close</i> className={`circle small transparent `}
</button> onClick={() => props.setColorPicker(!props.colorPicker)}
</nav> >
</dialog> <i>close</i>
</button>
</nav>
</dialog>
); );
}; };

View file

@ -1,9 +1,9 @@
.radiquum-pink{ .radiquum-pink {
background-color: #ffc8ff !important; background-color: #ffc8ff !important;
} }
.fuxigen-blue{ .fuxigen-blue {
background-color: #0087c7 !important; background-color: #0087c7 !important;
}
.anixart-red {
background-color: #e54040 !important;
} }
.anixart-red{
background-color: #e54040 !important;
}

View file

@ -4,12 +4,10 @@ import Link from "next/link";
export const LogInNeeded = (props) => { export const LogInNeeded = (props) => {
return ( return (
<div className="absolute padding error center middle round"> <div className="absolute padding error center middle round">
<i className="extra">no_accounts</i> <i className="extra">no_accounts</i>
<h5>Требуется авторизация</h5> <h5>Требуется авторизация</h5>
<p> <p>Для доступа к этой вкладке требуется авторизация в аккаунте anixart</p>
Для доступа к этой вкладке требуется авторизация в аккаунте anixart </div>
</p>
</div>
); );
}; };

View file

@ -60,7 +60,6 @@ export const NavigationRail = (props) => {
> >
<i>palette</i> <i>palette</i>
</button> </button>
</nav> </nav>
); );
}; };

View file

@ -5,11 +5,18 @@ import Link from "next/link";
export const ReleaseCard = (props) => { export const ReleaseCard = (props) => {
return ( return (
<Link href={`/release/${props.id}`} className="s3"> <Link href={`/release/${props.id}`} className="s3">
<article className="no-padding round fill" style={{width: 284, height: 508}}> <article
className="no-padding round fill"
style={{ width: 284, height: 508 }}
>
<img className="responsive large top-round" src={props.poster} /> <img className="responsive large top-round" src={props.poster} />
<div className="padding"> <div className="padding">
<h6>{`${props.title.substring(0, 36)}${[...props.title].length > 36 ? "..." : ""}`}</h6> <h6>{`${props.title.substring(0, 36)}${
<p>{`${props.description}${[...props.description].length > 160 ? "..." : ""}`}</p> [...props.title].length > 36 ? "..." : ""
}`}</h6>
<p>{`${props.description}${
[...props.description].length > 160 ? "..." : ""
}`}</p>
</div> </div>
</article> </article>
</Link> </Link>

View file

@ -17,8 +17,10 @@ body {
} }
} */ } */
body, nav.left, main{ body,
transition: background .2s; nav.left,
main {
transition: background 0.2s;
transform-origin: left; transform-origin: left;
min-height: 100dvh; min-height: 100dvh;
} }

View file

@ -1,5 +1,5 @@
import "./globals.css"; import "./globals.css";
import {App} from "@/app/App" import { App } from "@/app/App";
export const metadata = { export const metadata = {
title: "AniX", title: "AniX",

View file

@ -1,3 +1,3 @@
export default () => { export default () => {
return (<p>login page</p>) return <p>login page</p>;
} };

View file

@ -23,7 +23,7 @@ export default function Home() {
return params.toString(); return params.toString();
}, },
[searchParams] [searchParams],
); );
// set list on initial page load // set list on initial page load

View file

@ -32,7 +32,7 @@ export default function Search() {
return params.toString(); return params.toString();
}, },
[searchParams] [searchParams],
); );
async function fetchData(query, page = 0) { async function fetchData(query, page = 0) {

View file

@ -3,5 +3,6 @@
"paths": { "paths": {
"@/*": ["./*"] "@/*": ["./*"]
} }
} },
"exclude": ["node_modules", "**/.next/**", "**/_next/**", "**/dist/**"]
} }

File diff suppressed because it is too large Load diff

View file

@ -1,5 +1,6 @@
{ {
"name": "frontend", "name": "frontend",
"type": "module",
"version": "0.1.0", "version": "0.1.0",
"private": true, "private": true,
"scripts": { "scripts": {
@ -17,6 +18,7 @@
"zustand": "^4.5.2" "zustand": "^4.5.2"
}, },
"devDependencies": { "devDependencies": {
"eslint-config-next": "14.2.2",
"postcss": "^8" "postcss": "^8"
} }
} }

View file

@ -1,11 +0,0 @@
/** @type {import('tailwindcss').Config} */
module.exports = {
content: [
"./pages/**/*.{js,ts,jsx,tsx,mdx}",
"./components/**/*.{js,ts,jsx,tsx,mdx}",
"./app/**/*.{js,ts,jsx,tsx,mdx}",
],
theme: {
},
plugins: [],
};

View file

@ -0,0 +1,25 @@
http:
routers:
dashboard:
rule: Host(`traefik.test.local`) && (PathPrefix(`/api`) || PathPrefix(`/dashboard`))
service: api@internal
anix-api:
entryPoints:
- web
rule: Host(`anix.test.local`) && PathPrefix(`/api`)
service: anix-api-svc
anix-front:
entryPoints:
- web
rule: Host(`anix.test.local`)
service: anix-front-svc
services:
anix-api-svc:
loadBalancer:
servers:
- url: "http://127.0.0.1:8000"
anix-front-svc:
loadBalancer:
servers:
- url: "http://127.0.0.1:3000"

12
traefik/traefik.yml Normal file
View file

@ -0,0 +1,12 @@
entryPoints:
web:
address: ":80"
api:
insecure: false
dashboard: true
providers:
file:
filename: "./traefik-dynamic.yml"
watch: true