diff --git a/.dockerignore b/.dockerignore new file mode 100644 index 00000000..f762c3d5 --- /dev/null +++ b/.dockerignore @@ -0,0 +1,12 @@ +.git/ +.github/ +.dockerignore +.gitignore +./docker/Dockerfile +*docker-compose* + +.husky/ +node_modules/ +public/env-config.js +env-config.js +dist/ \ No newline at end of file diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml deleted file mode 100644 index 6d952eca..00000000 --- a/.github/workflows/ci.yml +++ /dev/null @@ -1,35 +0,0 @@ -name: SyncMaster UI CI - -on: - pull_request: - branches: - - '**' - -jobs: - build: - runs-on: ubuntu-latest - - steps: - - name: Checkout repository - uses: actions/checkout@v3 - - - name: Setup Node.js - uses: actions/setup-node@v3 - with: - node-version: '20' - cache: 'yarn' - - - name: Install dependencies - run: yarn - - - name: Run Types check - run: yarn type-check - - - name: Run ESLint - run: yarn lint - - - name: Run Prettier check - run: yarn prettier:check - - - name: Build project - run: yarn build diff --git a/.github/workflows/linters.yml b/.github/workflows/linters.yml new file mode 100644 index 00000000..35325001 --- /dev/null +++ b/.github/workflows/linters.yml @@ -0,0 +1,44 @@ +name: Code analysis + +on: + push: + branches: + - develop + pull_request: + branches-ignore: + - master + workflow_dispatch: + +concurrency: + group: ${{ github.workflow }}-${{ github.head_ref || github.ref }} + cancel-in-progress: true + +jobs: + linters: + runs-on: ubuntu-latest + steps: + - uses: actions/checkout@v2 + + - name: Install modules + run: yarn install --frozen-lockfile + + - name: Cache modules + uses: actions/cache@v4 + with: + path: ./node_modules + key: ${{ runner.os }}-yarn-${{ hashFiles('**/yarn.lock') }} + restore-keys: | + ${{ runner.os }}-yarn-${{ hashFiles('**/yarn.lock') }} + ${{ runner.os }}-yarn- + + - name: Run Type check + run: yarn type-check + + - name: Run ESLint + run: yarn lint + + - name: Run Prettier + run: yarn prettier --check + + - name: Build project + run: yarn build diff --git a/.github/workflows/release.yml b/.github/workflows/release.yml new file mode 100644 index 00000000..41650b34 --- /dev/null +++ b/.github/workflows/release.yml @@ -0,0 +1,71 @@ +name: Release + +on: + push: + branches: + - develop + tags: + - '[0-9]+.[0-9]+.[0-9]+' + workflow_dispatch: + +concurrency: + group: ${{ github.workflow }}-${{ github.head_ref || github.ref }} + cancel-in-progress: true + +jobs: + release: + name: Build & push frontend image to Dockerhub + runs-on: ubuntu-latest + if: github.repository == 'MobileTeleSystems/syncmaster-ui' # prevent running on forks + + steps: + - name: Set up QEMU + uses: docker/setup-qemu-action@v3 + + - name: Set up Docker Buildx + uses: docker/setup-buildx-action@v3 + + - name: Login to Docker Hub + uses: docker/login-action@v3 + with: + username: ${{ secrets.DOCKERHUB_USERNAME }} + password: ${{ secrets.DOCKERHUB_TOKEN }} + + - name: Checkout code + uses: actions/checkout@v4 + + - name: Set tag + id: set_tag + run: | + if [[ "${{ github.ref_type }}" == "branch" && "${{ github.ref_name }}" == "develop" ]]; then + echo "TAG=mtsrus/syncmaster-ui:develop" >> $GITHUB_ENV + elif [[ "${{ github.ref_type }}" == "tag" ]]; then + echo "TAG=mtsrus/syncmaster-ui:latest,mtsrus/syncmaster-ui:${{ github.ref_name }}" >> $GITHUB_ENV + fi + + - name: Build UI image + uses: docker/build-push-action@v6 + with: + tags: ${{ env.TAG }} + context: . + file: ./docker/Dockerfile + target: prod + pull: true + push: true + cache-to: type=inline + cache-from: mtsrus/syncmaster-ui:develop + platforms: | + linux/amd64 + linux/arm64/v8 + provenance: mode=max + + - name: Update DockerHub Description + uses: peter-evans/dockerhub-description@v4 + if: github.ref_type == 'tag' + with: + username: ${{ secrets.DOCKERHUB_USERNAME }} + # this requires token with read+write+delete permissions. read+write is not enough! + password: ${{ secrets.DOCKERHUB_TOKEN }} + repository: mtsrus/syncmaster-ui + short-description: ${{ github.event.repository.description }} + enable-url-completion: true diff --git a/.gitignore b/.gitignore index 38f45faa..736864e2 100644 --- a/.gitignore +++ b/.gitignore @@ -17,4 +17,7 @@ yarn-error.log /**/generated -/**/build \ No newline at end of file +/**/build + +public/env-config.js +env-config.js \ No newline at end of file diff --git a/docker/Dockerfile b/docker/Dockerfile new file mode 100644 index 00000000..08043ee4 --- /dev/null +++ b/docker/Dockerfile @@ -0,0 +1,28 @@ +FROM node:23.1.0 AS build + +WORKDIR /app + +COPY .yarnrc.yml . +COPY .yarn ./.yarn + +COPY package.json yarn.lock ./ +RUN yarn install --immutable + +COPY . . +RUN yarn build + + +FROM nginx:stable AS prod + +WORKDIR /usr/share/nginx/html + +COPY ./docker/nginx.conf /etc/nginx/conf.d/default.conf +COPY --from=build /app/dist . + +COPY ./docker/entrypoint.sh /entrypoint.sh +RUN chmod +x /entrypoint.sh + +EXPOSE 3000 + +ENTRYPOINT ["/entrypoint.sh"] +CMD ["nginx", "-g", "daemon off;"] diff --git a/docker/entrypoint.sh b/docker/entrypoint.sh new file mode 100644 index 00000000..0853fc59 --- /dev/null +++ b/docker/entrypoint.sh @@ -0,0 +1,9 @@ +#!/bin/sh + +cat < /usr/share/nginx/html/env-config.js +window.env = { + API_URL: "${API_URL}", +}; +EOF + +exec "$@" \ No newline at end of file diff --git a/docker/nginx.conf b/docker/nginx.conf new file mode 100644 index 00000000..43455e36 --- /dev/null +++ b/docker/nginx.conf @@ -0,0 +1,16 @@ +server { + listen 3000; + + location / { + root /usr/share/nginx/html; + index index.html index.htm; + try_files $uri $uri/ /index.html; + } + + error_page 500 502 503 504 /50x.html; + + location = /50x.html { + root /usr/share/nginx/html; + } + +} \ No newline at end of file diff --git a/src/app/config/router/instance.tsx b/src/app/config/router/instance.tsx index 389e8274..5a416218 100644 --- a/src/app/config/router/instance.tsx +++ b/src/app/config/router/instance.tsx @@ -1,5 +1,5 @@ import React from 'react'; -import { createBrowserRouter } from 'react-router-dom'; +import { createBrowserRouter, Navigate } from 'react-router-dom'; import { UserDetailPage, UserListPage } from '@pages/user'; import { LoginPage } from '@pages/auth'; import { AuthLayout, ErrorLayout, PrivateLayout } from '@app/layouts'; @@ -40,6 +40,10 @@ export const router = createBrowserRouter([ ), children: [ + { + path: '/', + element: , + }, { path: '/users', element: , diff --git a/src/index.html b/src/index.html index aa4e0767..e3a6686d 100644 --- a/src/index.html +++ b/src/index.html @@ -5,6 +5,7 @@ SyncMaster +
diff --git a/src/shared/config/axios/instance.ts b/src/shared/config/axios/instance.ts index f2e98dd3..df87340d 100644 --- a/src/shared/config/axios/instance.ts +++ b/src/shared/config/axios/instance.ts @@ -3,7 +3,7 @@ import axios from 'axios'; import { requestInterceptor, responseSuccessInterceptor } from './interceptors'; export const axiosInstance = axios.create({ - baseURL: process.env.API_URL, + baseURL: window.env?.API_URL || process.env.API_URL || 'http://localhost:8000/', headers: { Accept: 'application/json', 'Content-Type': 'application/json', diff --git a/src/types/index.d.ts b/src/types/index.d.ts index 2158b266..c172fce3 100644 --- a/src/types/index.d.ts +++ b/src/types/index.d.ts @@ -1,4 +1,10 @@ declare module '*.less' { const content: Record; export default content; +} + +interface Window { + env?: { + API_URL: string; + }; } \ No newline at end of file