mirror of
https://github.com/Radiquum/AniX.git
synced 2025-04-06 16:24:40 +00:00
ADD New Pre-Commit hooks and Formatting code
This commit is contained in:
parent
5c9c3e67fa
commit
9e75a0783c
26 changed files with 4163 additions and 105 deletions
3
.gitignore
vendored
3
.gitignore
vendored
|
@ -42,3 +42,6 @@ yarn-error.log*
|
||||||
## typescript
|
## typescript
|
||||||
*.tsbuildinfo
|
*.tsbuildinfo
|
||||||
next-env.d.ts
|
next-env.d.ts
|
||||||
|
|
||||||
|
# traefik
|
||||||
|
traefik/traefik
|
||||||
|
|
|
@ -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/)
|
27
README.md
27
README.md
|
@ -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
|
||||||
|
|
||||||
|
|
|
@ -35,7 +35,9 @@ app = FastAPI(
|
||||||
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"])
|
||||||
|
|
|
@ -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,
|
||||||
):
|
):
|
||||||
|
|
|
@ -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)
|
||||||
|
|
||||||
|
|
|
@ -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
3
frontend/.eslintrc.json
Normal file
|
@ -0,0 +1,3 @@
|
||||||
|
{
|
||||||
|
"extends": "next/core-web-vitals"
|
||||||
|
}
|
|
@ -1,4 +1,4 @@
|
||||||
export const API_URL = "http://anix.test.local/api";
|
export const API_URL = "/api";
|
||||||
|
|
||||||
export const endpoints = {
|
export const endpoints = {
|
||||||
index: {
|
index: {
|
||||||
|
@ -7,5 +7,5 @@ export const endpoints = {
|
||||||
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`,
|
||||||
};
|
};
|
|
@ -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}
|
||||||
|
|
|
@ -20,7 +20,10 @@ 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
|
||||||
|
className="active left round bottom small"
|
||||||
|
style={{ blockSize: "unset" }}
|
||||||
|
>
|
||||||
<h5>Выбор темы</h5>
|
<h5>Выбор темы</h5>
|
||||||
<div className="grid center-align">
|
<div className="grid center-align">
|
||||||
{colors.map((item) => {
|
{colors.map((item) => {
|
||||||
|
|
|
@ -7,9 +7,7 @@ export const LogInNeeded = (props) => {
|
||||||
<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
|
|
||||||
</p>
|
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
|
@ -60,7 +60,6 @@ export const NavigationRail = (props) => {
|
||||||
>
|
>
|
||||||
<i>palette</i>
|
<i>palette</i>
|
||||||
</button>
|
</button>
|
||||||
|
|
||||||
</nav>
|
</nav>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
|
@ -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>
|
||||||
|
|
|
@ -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;
|
||||||
}
|
}
|
||||||
|
|
|
@ -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",
|
||||||
|
|
|
@ -1,3 +1,3 @@
|
||||||
export default () => {
|
export default () => {
|
||||||
return (<p>login page</p>)
|
return <p>login page</p>;
|
||||||
}
|
};
|
||||||
|
|
|
@ -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
|
||||||
|
|
|
@ -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) {
|
||||||
|
|
|
@ -3,5 +3,6 @@
|
||||||
"paths": {
|
"paths": {
|
||||||
"@/*": ["./*"]
|
"@/*": ["./*"]
|
||||||
}
|
}
|
||||||
}
|
},
|
||||||
|
"exclude": ["node_modules", "**/.next/**", "**/_next/**", "**/dist/**"]
|
||||||
}
|
}
|
||||||
|
|
3963
frontend/package-lock.json
generated
3963
frontend/package-lock.json
generated
File diff suppressed because it is too large
Load diff
|
@ -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"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -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: [],
|
|
||||||
};
|
|
25
traefik/traefik-dynamic.yml
Normal file
25
traefik/traefik-dynamic.yml
Normal 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
12
traefik/traefik.yml
Normal file
|
@ -0,0 +1,12 @@
|
||||||
|
entryPoints:
|
||||||
|
web:
|
||||||
|
address: ":80"
|
||||||
|
|
||||||
|
api:
|
||||||
|
insecure: false
|
||||||
|
dashboard: true
|
||||||
|
|
||||||
|
providers:
|
||||||
|
file:
|
||||||
|
filename: "./traefik-dynamic.yml"
|
||||||
|
watch: true
|
Loading…
Add table
Reference in a new issue