Everything is validated now
This commit is contained in:
parent
23ba777e5a
commit
9170f8f7aa
@ -1,6 +1,7 @@
|
|||||||
image: node:latest
|
image: node:latest
|
||||||
|
|
||||||
stages:
|
stages:
|
||||||
|
- setup
|
||||||
- test
|
- test
|
||||||
- build
|
- build
|
||||||
- package
|
- package
|
||||||
@ -23,18 +24,33 @@ before_script:
|
|||||||
- yarn install --cache-folder .yarn --frozen-lockfile
|
- yarn install --cache-folder .yarn --frozen-lockfile
|
||||||
- cd ..
|
- cd ..
|
||||||
|
|
||||||
test_backend:
|
test_build_dto:
|
||||||
|
stage: setup
|
||||||
cache:
|
cache:
|
||||||
<<: *global_cache
|
<<: *global_cache
|
||||||
policy: pull-push
|
policy: pull-push
|
||||||
|
before_script: []
|
||||||
|
script:
|
||||||
|
- cd dto
|
||||||
|
- yarn install --frozen-lockfile
|
||||||
|
- yarn lint
|
||||||
|
- yarn build
|
||||||
|
- cd ..
|
||||||
|
- yarn install --cache-folder .yarn --frozen-lockfile
|
||||||
|
- yarn add ./dto
|
||||||
|
- cd frontend
|
||||||
|
- yarn install --cache-folder .yarn --frozen-lockfile
|
||||||
|
- yarn add ../dto
|
||||||
|
|
||||||
|
|
||||||
|
test_backend:
|
||||||
|
needs: ["test_build_dto"]
|
||||||
stage: test
|
stage: test
|
||||||
script:
|
script:
|
||||||
- yarn lint
|
- yarn lint
|
||||||
|
|
||||||
test_frontend:
|
test_frontend:
|
||||||
cache:
|
needs: ["test_build_dto"]
|
||||||
<<: *global_cache
|
|
||||||
policy: pull-push
|
|
||||||
stage: test
|
stage: test
|
||||||
script:
|
script:
|
||||||
- cd frontend
|
- cd frontend
|
||||||
|
402
dto/.gitignore
vendored
Normal file
402
dto/.gitignore
vendored
Normal file
@ -0,0 +1,402 @@
|
|||||||
|
# Created by .ignore support plugin (hsz.mobi)
|
||||||
|
### JetBrains template
|
||||||
|
# Covers JetBrains IDEs: IntelliJ, RubyMine, PhpStorm, AppCode, PyCharm, CLion, Android Studio and Webstorm
|
||||||
|
# Reference: https://intellij-support.jetbrains.com/hc/en-us/articles/206544839
|
||||||
|
|
||||||
|
# User-specific stuff:
|
||||||
|
.idea/**/workspace.xml
|
||||||
|
.idea/**/tasks.xml
|
||||||
|
.idea/dictionaries
|
||||||
|
|
||||||
|
# Sensitive or high-churn files:
|
||||||
|
.idea/**/dataSources/
|
||||||
|
.idea/**/dataSources.ids
|
||||||
|
.idea/**/dataSources.xml
|
||||||
|
.idea/**/dataSources.local.xml
|
||||||
|
.idea/**/sqlDataSources.xml
|
||||||
|
.idea/**/dynamic.xml
|
||||||
|
.idea/**/uiDesigner.xml
|
||||||
|
|
||||||
|
# Gradle:
|
||||||
|
.idea/**/gradle.xml
|
||||||
|
.idea/**/libraries
|
||||||
|
|
||||||
|
# CMake
|
||||||
|
cmake-build-debug/
|
||||||
|
|
||||||
|
# Mongo Explorer plugin:
|
||||||
|
.idea/**/mongoSettings.xml
|
||||||
|
|
||||||
|
## File-based project format:
|
||||||
|
*.iws
|
||||||
|
|
||||||
|
## Plugin-specific files:
|
||||||
|
|
||||||
|
# IntelliJ
|
||||||
|
out/
|
||||||
|
|
||||||
|
# mpeltonen/sbt-idea plugin
|
||||||
|
.idea_modules/
|
||||||
|
|
||||||
|
# JIRA plugin
|
||||||
|
atlassian-ide-plugin.xml
|
||||||
|
|
||||||
|
# Cursive Clojure plugin
|
||||||
|
.idea/replstate.xml
|
||||||
|
|
||||||
|
# Crashlytics plugin (for Android Studio and IntelliJ)
|
||||||
|
com_crashlytics_export_strings.xml
|
||||||
|
crashlytics.properties
|
||||||
|
crashlytics-build.properties
|
||||||
|
fabric.properties
|
||||||
|
### VisualStudio template
|
||||||
|
## Ignore Visual Studio temporary files, build results, and
|
||||||
|
## files generated by popular Visual Studio add-ons.
|
||||||
|
##
|
||||||
|
## Get latest from https://github.com/github/gitignore/blob/master/VisualStudio.gitignore
|
||||||
|
|
||||||
|
# User-specific files
|
||||||
|
*.suo
|
||||||
|
*.user
|
||||||
|
*.userosscache
|
||||||
|
*.sln.docstates
|
||||||
|
|
||||||
|
# User-specific files (MonoDevelop/Xamarin Studio)
|
||||||
|
*.userprefs
|
||||||
|
|
||||||
|
# Build results
|
||||||
|
[Dd]ebug/
|
||||||
|
[Dd]ebugPublic/
|
||||||
|
[Rr]elease/
|
||||||
|
[Rr]eleases/
|
||||||
|
x64/
|
||||||
|
x86/
|
||||||
|
bld/
|
||||||
|
[Bb]in/
|
||||||
|
[Oo]bj/
|
||||||
|
[Ll]og/
|
||||||
|
|
||||||
|
# Visual Studio 2015 cache/options directory
|
||||||
|
.vs/
|
||||||
|
# Uncomment if you have tasks that create the project's static files in wwwroot
|
||||||
|
#wwwroot/
|
||||||
|
|
||||||
|
# MSTest test Results
|
||||||
|
[Tt]est[Rr]esult*/
|
||||||
|
[Bb]uild[Ll]og.*
|
||||||
|
|
||||||
|
# NUNIT
|
||||||
|
*.VisualState.xml
|
||||||
|
TestResult.xml
|
||||||
|
|
||||||
|
# Build Results of an ATL Project
|
||||||
|
[Dd]ebugPS/
|
||||||
|
[Rr]eleasePS/
|
||||||
|
dlldata.c
|
||||||
|
|
||||||
|
# Benchmark Results
|
||||||
|
BenchmarkDotNet.Artifacts/
|
||||||
|
|
||||||
|
# .NET Core
|
||||||
|
project.lock.json
|
||||||
|
project.fragment.lock.json
|
||||||
|
artifacts/
|
||||||
|
**/Properties/launchSettings.json
|
||||||
|
|
||||||
|
*_i.c
|
||||||
|
*_p.c
|
||||||
|
*_i.h
|
||||||
|
*.ilk
|
||||||
|
*.meta
|
||||||
|
*.obj
|
||||||
|
*.pch
|
||||||
|
*.pdb
|
||||||
|
*.pgc
|
||||||
|
*.pgd
|
||||||
|
*.rsp
|
||||||
|
*.sbr
|
||||||
|
*.tlb
|
||||||
|
*.tli
|
||||||
|
*.tlh
|
||||||
|
*.tmp
|
||||||
|
*.tmp_proj
|
||||||
|
*.log
|
||||||
|
*.vspscc
|
||||||
|
*.vssscc
|
||||||
|
.builds
|
||||||
|
*.pidb
|
||||||
|
*.svclog
|
||||||
|
*.scc
|
||||||
|
|
||||||
|
# Chutzpah Test files
|
||||||
|
_Chutzpah*
|
||||||
|
|
||||||
|
# Visual C++ cache files
|
||||||
|
ipch/
|
||||||
|
*.aps
|
||||||
|
*.ncb
|
||||||
|
*.opendb
|
||||||
|
*.opensdf
|
||||||
|
*.sdf
|
||||||
|
*.cachefile
|
||||||
|
*.VC.db
|
||||||
|
*.VC.VC.opendb
|
||||||
|
|
||||||
|
# Visual Studio profiler
|
||||||
|
*.psess
|
||||||
|
*.vsp
|
||||||
|
*.vspx
|
||||||
|
*.sap
|
||||||
|
|
||||||
|
# Visual Studio Trace Files
|
||||||
|
*.e2e
|
||||||
|
|
||||||
|
# TFS 2012 Local Workspace
|
||||||
|
$tf/
|
||||||
|
|
||||||
|
# Guidance Automation Toolkit
|
||||||
|
*.gpState
|
||||||
|
|
||||||
|
# ReSharper is a .NET coding add-in
|
||||||
|
_ReSharper*/
|
||||||
|
*.[Rr]e[Ss]harper
|
||||||
|
*.DotSettings.user
|
||||||
|
|
||||||
|
# JustCode is a .NET coding add-in
|
||||||
|
.JustCode
|
||||||
|
|
||||||
|
# TeamCity is a build add-in
|
||||||
|
_TeamCity*
|
||||||
|
|
||||||
|
# DotCover is a Code Coverage Tool
|
||||||
|
*.dotCover
|
||||||
|
|
||||||
|
# AxoCover is a Code Coverage Tool
|
||||||
|
.axoCover/*
|
||||||
|
!.axoCover/settings.json
|
||||||
|
|
||||||
|
# Visual Studio code coverage results
|
||||||
|
*.coverage
|
||||||
|
*.coveragexml
|
||||||
|
|
||||||
|
# NCrunch
|
||||||
|
_NCrunch_*
|
||||||
|
.*crunch*.local.xml
|
||||||
|
nCrunchTemp_*
|
||||||
|
|
||||||
|
# MightyMoose
|
||||||
|
*.mm.*
|
||||||
|
AutoTest.Net/
|
||||||
|
|
||||||
|
# Web workbench (sass)
|
||||||
|
.sass-cache/
|
||||||
|
|
||||||
|
# Installshield output folder
|
||||||
|
[Ee]xpress/
|
||||||
|
|
||||||
|
# DocProject is a documentation generator add-in
|
||||||
|
DocProject/buildhelp/
|
||||||
|
DocProject/Help/*.HxT
|
||||||
|
DocProject/Help/*.HxC
|
||||||
|
DocProject/Help/*.hhc
|
||||||
|
DocProject/Help/*.hhk
|
||||||
|
DocProject/Help/*.hhp
|
||||||
|
DocProject/Help/Html2
|
||||||
|
DocProject/Help/html
|
||||||
|
|
||||||
|
# Click-Once directory
|
||||||
|
publish/
|
||||||
|
|
||||||
|
# Publish Web Output
|
||||||
|
*.[Pp]ublish.xml
|
||||||
|
*.azurePubxml
|
||||||
|
# Note: Comment the next line if you want to checkin your web deploy settings,
|
||||||
|
# but database connection strings (with potential passwords) will be unencrypted
|
||||||
|
*.pubxml
|
||||||
|
*.publishproj
|
||||||
|
|
||||||
|
# Microsoft Azure Web App publish settings. Comment the next line if you want to
|
||||||
|
# checkin your Azure Web App publish settings, but sensitive information contained
|
||||||
|
# in these scripts will be unencrypted
|
||||||
|
PublishScripts/
|
||||||
|
|
||||||
|
# NuGet Packages
|
||||||
|
*.nupkg
|
||||||
|
# The packages folder can be ignored because of Package Restore
|
||||||
|
**/[Pp]ackages/*
|
||||||
|
# except build/, which is used as an MSBuild target.
|
||||||
|
!**/[Pp]ackages/build/
|
||||||
|
# Uncomment if necessary however generally it will be regenerated when needed
|
||||||
|
#!**/[Pp]ackages/repositories.config
|
||||||
|
# NuGet v3's project.json files produces more ignorable files
|
||||||
|
*.nuget.props
|
||||||
|
*.nuget.targets
|
||||||
|
|
||||||
|
# Microsoft Azure Build Output
|
||||||
|
csx/
|
||||||
|
*.build.csdef
|
||||||
|
|
||||||
|
# Microsoft Azure Emulator
|
||||||
|
ecf/
|
||||||
|
rcf/
|
||||||
|
|
||||||
|
# Windows Store app package directories and files
|
||||||
|
AppPackages/
|
||||||
|
BundleArtifacts/
|
||||||
|
Package.StoreAssociation.xml
|
||||||
|
_pkginfo.txt
|
||||||
|
*.appx
|
||||||
|
|
||||||
|
# Visual Studio cache files
|
||||||
|
# files ending in .cache can be ignored
|
||||||
|
*.[Cc]ache
|
||||||
|
# but keep track of directories ending in .cache
|
||||||
|
!*.[Cc]ache/
|
||||||
|
|
||||||
|
# Others
|
||||||
|
ClientBin/
|
||||||
|
~$*
|
||||||
|
*~
|
||||||
|
*.dbmdl
|
||||||
|
*.dbproj.schemaview
|
||||||
|
*.jfm
|
||||||
|
*.pfx
|
||||||
|
*.publishsettings
|
||||||
|
orleans.codegen.cs
|
||||||
|
|
||||||
|
# Since there are multiple workflows, uncomment next line to ignore bower_components
|
||||||
|
# (https://github.com/github/gitignore/pull/1529#issuecomment-104372622)
|
||||||
|
#bower_components/
|
||||||
|
|
||||||
|
# RIA/Silverlight projects
|
||||||
|
Generated_Code/
|
||||||
|
|
||||||
|
# Backup & report files from converting an old project file
|
||||||
|
# to a newer Visual Studio version. Backup files are not needed,
|
||||||
|
# because we have git ;-)
|
||||||
|
_UpgradeReport_Files/
|
||||||
|
Backup*/
|
||||||
|
UpgradeLog*.XML
|
||||||
|
UpgradeLog*.htm
|
||||||
|
|
||||||
|
# SQL Server files
|
||||||
|
*.mdf
|
||||||
|
*.ldf
|
||||||
|
*.ndf
|
||||||
|
|
||||||
|
# Business Intelligence projects
|
||||||
|
*.rdl.data
|
||||||
|
*.bim.layout
|
||||||
|
*.bim_*.settings
|
||||||
|
|
||||||
|
# Microsoft Fakes
|
||||||
|
FakesAssemblies/
|
||||||
|
|
||||||
|
# GhostDoc plugin setting file
|
||||||
|
*.GhostDoc.xml
|
||||||
|
|
||||||
|
# Node.js Tools for Visual Studio
|
||||||
|
.ntvs_analysis.dat
|
||||||
|
node_modules/
|
||||||
|
|
||||||
|
# Typescript v1 declaration files
|
||||||
|
typings/
|
||||||
|
|
||||||
|
# Visual Studio 6 build log
|
||||||
|
*.plg
|
||||||
|
|
||||||
|
# Visual Studio 6 workspace options file
|
||||||
|
*.opt
|
||||||
|
|
||||||
|
# Visual Studio 6 auto-generated workspace file (contains which files were open etc.)
|
||||||
|
*.vbw
|
||||||
|
|
||||||
|
# Visual Studio LightSwitch build output
|
||||||
|
**/*.HTMLClient/GeneratedArtifacts
|
||||||
|
**/*.DesktopClient/GeneratedArtifacts
|
||||||
|
**/*.DesktopClient/ModelManifest.xml
|
||||||
|
**/*.Server/GeneratedArtifacts
|
||||||
|
**/*.Server/ModelManifest.xml
|
||||||
|
_Pvt_Extensions
|
||||||
|
|
||||||
|
# Paket dependency manager
|
||||||
|
.paket/paket.exe
|
||||||
|
paket-files/
|
||||||
|
|
||||||
|
# FAKE - F# Make
|
||||||
|
.fake/
|
||||||
|
|
||||||
|
# JetBrains Rider
|
||||||
|
.idea/
|
||||||
|
*.sln.iml
|
||||||
|
|
||||||
|
# IDE - VSCode
|
||||||
|
.vscode/*
|
||||||
|
!.vscode/settings.json
|
||||||
|
!.vscode/tasks.json
|
||||||
|
!.vscode/launch.json
|
||||||
|
!.vscode/extensions.json
|
||||||
|
|
||||||
|
# CodeRush
|
||||||
|
.cr/
|
||||||
|
|
||||||
|
# Python Tools for Visual Studio (PTVS)
|
||||||
|
__pycache__/
|
||||||
|
*.pyc
|
||||||
|
|
||||||
|
# Cake - Uncomment if you are using it
|
||||||
|
# tools/**
|
||||||
|
# !tools/packages.config
|
||||||
|
|
||||||
|
# Tabs Studio
|
||||||
|
*.tss
|
||||||
|
|
||||||
|
# Telerik's JustMock configuration file
|
||||||
|
*.jmconfig
|
||||||
|
|
||||||
|
# BizTalk build output
|
||||||
|
*.btp.cs
|
||||||
|
*.btm.cs
|
||||||
|
*.odx.cs
|
||||||
|
*.xsd.cs
|
||||||
|
|
||||||
|
# OpenCover UI analysis results
|
||||||
|
OpenCover/
|
||||||
|
coverage/
|
||||||
|
|
||||||
|
### macOS template
|
||||||
|
# General
|
||||||
|
.DS_Store
|
||||||
|
.AppleDouble
|
||||||
|
.LSOverride
|
||||||
|
|
||||||
|
# Icon must end with two \r
|
||||||
|
Icon
|
||||||
|
|
||||||
|
# Thumbnails
|
||||||
|
._*
|
||||||
|
|
||||||
|
# Files that might appear in the root of a volume
|
||||||
|
.DocumentRevisions-V100
|
||||||
|
.fseventsd
|
||||||
|
.Spotlight-V100
|
||||||
|
.TemporaryItems
|
||||||
|
.Trashes
|
||||||
|
.VolumeIcon.icns
|
||||||
|
.com.apple.timemachine.donotpresent
|
||||||
|
|
||||||
|
# Directories potentially created on remote AFP share
|
||||||
|
.AppleDB
|
||||||
|
.AppleDesktop
|
||||||
|
Network Trash Folder
|
||||||
|
Temporary Items
|
||||||
|
.apdisk
|
||||||
|
|
||||||
|
=======
|
||||||
|
# Local
|
||||||
|
.env
|
||||||
|
dist
|
||||||
|
lib
|
||||||
|
|
||||||
|
files
|
||||||
|
sqlite.db
|
@ -2,13 +2,18 @@
|
|||||||
"name": "dto",
|
"name": "dto",
|
||||||
"private": true,
|
"private": true,
|
||||||
"version": "1.0.0",
|
"version": "1.0.0",
|
||||||
"main": "src/index.ts",
|
"main": "lib/index.js",
|
||||||
|
"types": "lib/index.d.ts",
|
||||||
|
"files": [
|
||||||
|
"/lib"
|
||||||
|
],
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"class-transformer": "^0.5.1",
|
"class-transformer": "^0.5.1",
|
||||||
"class-validator": "^0.13.2"
|
"class-validator": "^0.13.2"
|
||||||
},
|
},
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"lint": "eslint \"src/**/*.ts\" && tsc --no-emit",
|
"lint": "eslint \"src/**/*.ts\" && tsc --noEmit",
|
||||||
"lint-fix": "eslint \"src/**/*.ts\" --fix"
|
"lint-fix": "eslint \"src/**/*.ts\" --fix",
|
||||||
|
"build": "tsc"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,2 +1,3 @@
|
|||||||
export * as Requests from './requests';
|
export * as Requests from './requests';
|
||||||
export * as Responses from './responses';
|
export * as Responses from './responses';
|
||||||
|
export { validateSync, validateAsync, validateAsyncInline } from './utils';
|
||||||
|
@ -1,14 +1,36 @@
|
|||||||
import { BaseRequest } from './base';
|
import { BaseRequest } from './base';
|
||||||
import { IsEmail, IsNotEmpty } from 'class-validator';
|
import {
|
||||||
|
IsBoolean,
|
||||||
|
IsEmail,
|
||||||
|
IsNotEmpty,
|
||||||
|
IsOptional,
|
||||||
|
IsString
|
||||||
|
} from 'class-validator';
|
||||||
|
|
||||||
export class AuthSignUpRequest extends BaseRequest {
|
export class SignUpRequest extends BaseRequest {
|
||||||
@IsEmail()
|
@IsEmail()
|
||||||
username: string;
|
username: string;
|
||||||
|
|
||||||
@IsNotEmpty()
|
@IsNotEmpty()
|
||||||
|
@IsString()
|
||||||
password: string;
|
password: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
export class AuthLoginRequest extends AuthSignUpRequest {
|
export class LoginRequest extends SignUpRequest {
|
||||||
|
@IsOptional()
|
||||||
|
@IsNotEmpty()
|
||||||
|
@IsString()
|
||||||
otp?: string;
|
otp?: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export class TfaComplete extends BaseRequest {
|
||||||
|
@IsNotEmpty()
|
||||||
|
@IsString()
|
||||||
|
code: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
export class TfaSetup extends BaseRequest {
|
||||||
|
@IsNotEmpty()
|
||||||
|
@IsBoolean()
|
||||||
|
mail: boolean;
|
||||||
|
}
|
||||||
|
@ -1,7 +1,5 @@
|
|||||||
import { BaseRequest } from './base';
|
import { BaseRequest } from './base';
|
||||||
import { IsInt, IsNotEmpty, Min } from 'class-validator';
|
import { IsInt, IsNotEmpty, IsString, Min } from 'class-validator';
|
||||||
|
|
||||||
export type CreateFileRequest = CreateFolderRequest;
|
|
||||||
|
|
||||||
export class CreateFolderRequest extends BaseRequest {
|
export class CreateFolderRequest extends BaseRequest {
|
||||||
@IsInt()
|
@IsInt()
|
||||||
@ -9,6 +7,7 @@ export class CreateFolderRequest extends BaseRequest {
|
|||||||
parent: number;
|
parent: number;
|
||||||
|
|
||||||
@IsNotEmpty()
|
@IsNotEmpty()
|
||||||
|
@IsString()
|
||||||
name: string;
|
name: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -17,3 +16,5 @@ export class DeleteRequest extends BaseRequest {
|
|||||||
@Min(1)
|
@Min(1)
|
||||||
node: number;
|
node: number;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export class CreateFileRequest extends CreateFolderRequest {}
|
||||||
|
@ -1,20 +1,27 @@
|
|||||||
import { SuccessResponse } from './base';
|
import { SuccessResponse } from './base';
|
||||||
import { IsBase32, IsJWT, IsNotEmpty } from 'class-validator';
|
import { IsBase32, IsJWT, IsNotEmpty } from 'class-validator';
|
||||||
|
import { ValidateConstructor } from '../utils';
|
||||||
|
|
||||||
export type TfaRequiredResponse = SuccessResponse;
|
@ValidateConstructor
|
||||||
export type RemoveTfaResponse = SuccessResponse;
|
|
||||||
export type RequestEmailTfaResponse = SuccessResponse;
|
|
||||||
export type TfaCompletedResponse = SuccessResponse;
|
|
||||||
export type SignupResponse = SuccessResponse;
|
|
||||||
export type RefreshResponse = LoginResponse;
|
|
||||||
|
|
||||||
export class LoginResponse extends SuccessResponse {
|
export class LoginResponse extends SuccessResponse {
|
||||||
|
constructor(jwt: string) {
|
||||||
|
super();
|
||||||
|
this.jwt = jwt;
|
||||||
|
}
|
||||||
|
|
||||||
@IsNotEmpty()
|
@IsNotEmpty()
|
||||||
@IsJWT()
|
@IsJWT()
|
||||||
jwt: string;
|
jwt: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ValidateConstructor
|
||||||
export class RequestTotpTfaResponse extends SuccessResponse {
|
export class RequestTotpTfaResponse extends SuccessResponse {
|
||||||
|
constructor(qrCode: string, secret: string) {
|
||||||
|
super();
|
||||||
|
this.qrCode = qrCode;
|
||||||
|
this.secret = secret;
|
||||||
|
}
|
||||||
|
|
||||||
@IsNotEmpty()
|
@IsNotEmpty()
|
||||||
qrCode: string;
|
qrCode: string;
|
||||||
|
|
||||||
@ -22,3 +29,10 @@ export class RequestTotpTfaResponse extends SuccessResponse {
|
|||||||
@IsBase32()
|
@IsBase32()
|
||||||
secret: string;
|
secret: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export class TfaRequiredResponse extends SuccessResponse {}
|
||||||
|
export class RemoveTfaResponse extends SuccessResponse {}
|
||||||
|
export class RequestEmailTfaResponse extends SuccessResponse {}
|
||||||
|
export class TfaCompletedResponse extends SuccessResponse {}
|
||||||
|
export class SignupResponse extends SuccessResponse {}
|
||||||
|
export class RefreshResponse extends LoginResponse {}
|
||||||
|
@ -1,17 +1,39 @@
|
|||||||
import { SuccessResponse } from './base';
|
import { SuccessResponse } from './base';
|
||||||
import { IsBoolean, IsInt, IsNotEmpty, Min } from 'class-validator';
|
import {
|
||||||
|
IsBoolean,
|
||||||
export type UploadFileResponse = SuccessResponse;
|
IsInt,
|
||||||
export type DeleteResponse = SuccessResponse;
|
IsNotEmpty,
|
||||||
export type CreateFileResponse = CreateFolderResponse;
|
IsOptional,
|
||||||
|
IsString,
|
||||||
|
Min
|
||||||
|
} from 'class-validator';
|
||||||
|
import { ValidateConstructor } from '../utils';
|
||||||
|
|
||||||
|
@ValidateConstructor
|
||||||
export class GetRootResponse extends SuccessResponse {
|
export class GetRootResponse extends SuccessResponse {
|
||||||
|
constructor(rootId: number) {
|
||||||
|
super();
|
||||||
|
this.rootId = rootId;
|
||||||
|
}
|
||||||
@IsInt()
|
@IsInt()
|
||||||
@Min(1)
|
@Min(1)
|
||||||
rootId: number;
|
rootId: number;
|
||||||
}
|
}
|
||||||
|
|
||||||
export class GetNodeResponse extends SuccessResponse {
|
export class GetNodeResponse extends SuccessResponse {
|
||||||
|
constructor(
|
||||||
|
id: number,
|
||||||
|
name: string,
|
||||||
|
isFile: boolean,
|
||||||
|
parent: number | null
|
||||||
|
) {
|
||||||
|
super();
|
||||||
|
this.id = id;
|
||||||
|
this.name = name;
|
||||||
|
this.isFile = isFile;
|
||||||
|
this.parent = parent;
|
||||||
|
}
|
||||||
|
|
||||||
@IsInt()
|
@IsInt()
|
||||||
@Min(1)
|
@Min(1)
|
||||||
id: number;
|
id: number;
|
||||||
@ -22,22 +44,46 @@ export class GetNodeResponse extends SuccessResponse {
|
|||||||
@IsBoolean()
|
@IsBoolean()
|
||||||
isFile: boolean;
|
isFile: boolean;
|
||||||
|
|
||||||
|
@IsOptional()
|
||||||
|
@IsInt()
|
||||||
|
@Min(1)
|
||||||
parent: number | null;
|
parent: number | null;
|
||||||
|
|
||||||
|
@IsOptional()
|
||||||
@IsInt({ each: true })
|
@IsInt({ each: true })
|
||||||
@Min(1, { each: true })
|
@Min(1, { each: true })
|
||||||
children?: number[];
|
children?: number[];
|
||||||
|
|
||||||
|
@IsOptional()
|
||||||
|
@IsInt()
|
||||||
|
@Min(0)
|
||||||
size?: number;
|
size?: number;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ValidateConstructor
|
||||||
export class GetPathResponse extends SuccessResponse {
|
export class GetPathResponse extends SuccessResponse {
|
||||||
|
constructor(path: string) {
|
||||||
|
super();
|
||||||
|
this.path = path;
|
||||||
|
}
|
||||||
|
|
||||||
@IsNotEmpty()
|
@IsNotEmpty()
|
||||||
|
@IsString()
|
||||||
path: string;
|
path: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ValidateConstructor
|
||||||
export class CreateFolderResponse extends SuccessResponse {
|
export class CreateFolderResponse extends SuccessResponse {
|
||||||
|
constructor(id: number) {
|
||||||
|
super();
|
||||||
|
this.id = id;
|
||||||
|
}
|
||||||
|
|
||||||
@IsInt()
|
@IsInt()
|
||||||
@Min(1)
|
@Min(1)
|
||||||
id: number;
|
id: number;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export class UploadFileResponse extends SuccessResponse {}
|
||||||
|
export class DeleteResponse extends SuccessResponse {}
|
||||||
|
export class CreateFileResponse extends CreateFolderResponse {}
|
||||||
|
35
dto/src/utils.ts
Normal file
35
dto/src/utils.ts
Normal file
@ -0,0 +1,35 @@
|
|||||||
|
import { validate, validateSync as _validateSync } from 'class-validator';
|
||||||
|
|
||||||
|
export function validateSync<T extends object>(data: T): void {
|
||||||
|
const errors = _validateSync(data);
|
||||||
|
if (errors.length > 0) {
|
||||||
|
console.error('Validation failed, errors: ', errors);
|
||||||
|
throw new Error('Validation failed');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function validateAsync<T extends object>(data: T): Promise<void> {
|
||||||
|
const errors = await validate(data);
|
||||||
|
if (errors.length > 0) {
|
||||||
|
console.error('Validation failed, errors: ', errors);
|
||||||
|
throw new Error('Validation failed');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function validateAsyncInline<T extends object>(
|
||||||
|
data: T
|
||||||
|
): Promise<T> {
|
||||||
|
await validateAsync(data);
|
||||||
|
return data;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function ValidateConstructor<T extends { new (...args: any[]): any }>(
|
||||||
|
constr: T
|
||||||
|
) {
|
||||||
|
return class extends constr {
|
||||||
|
constructor(...args: any[]) {
|
||||||
|
super(...args);
|
||||||
|
validateSync(this);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
@ -1,18 +1,18 @@
|
|||||||
{
|
{
|
||||||
"compilerOptions": {
|
"compilerOptions": {
|
||||||
"module": "commonjs",
|
"module": "commonjs",
|
||||||
"declaration": false,
|
"declaration": true,
|
||||||
"removeComments": true,
|
"removeComments": true,
|
||||||
"emitDecoratorMetadata": true,
|
"emitDecoratorMetadata": true,
|
||||||
"experimentalDecorators": true,
|
"experimentalDecorators": true,
|
||||||
"allowSyntheticDefaultImports": true,
|
"allowSyntheticDefaultImports": true,
|
||||||
"target": "es2017",
|
"target": "es2017",
|
||||||
"sourceMap": true,
|
"sourceMap": true,
|
||||||
"outDir": "./dist",
|
"outDir": "./lib",
|
||||||
"baseUrl": "./src",
|
"baseUrl": "./src",
|
||||||
"incremental": true,
|
"incremental": true,
|
||||||
"skipLibCheck": true,
|
"skipLibCheck": true,
|
||||||
"strictPropertyInitialization": false
|
"strictPropertyInitialization": false
|
||||||
},
|
},
|
||||||
"exclude": ["node_modules", "dist"]
|
"exclude": ["node_modules", "lib"]
|
||||||
}
|
}
|
||||||
|
@ -9,7 +9,7 @@ export const auth_login = (
|
|||||||
| Responses.Auth.TfaRequiredResponse
|
| Responses.Auth.TfaRequiredResponse
|
||||||
| Responses.ErrorResponse
|
| Responses.ErrorResponse
|
||||||
> =>
|
> =>
|
||||||
post<Requests.Auth.AuthLoginRequest>('/api/auth/login', {
|
post<Requests.Auth.LoginRequest>('/api/auth/login', {
|
||||||
username: username,
|
username: username,
|
||||||
password: password,
|
password: password,
|
||||||
otp: otp
|
otp: otp
|
||||||
@ -19,7 +19,7 @@ export const auth_signup = (
|
|||||||
username: string,
|
username: string,
|
||||||
password: string
|
password: string
|
||||||
): Promise<Responses.Auth.SignupResponse | Responses.ErrorResponse> =>
|
): Promise<Responses.Auth.SignupResponse | Responses.ErrorResponse> =>
|
||||||
post<Requests.Auth.AuthSignUpRequest>('/api/auth/signup', {
|
post<Requests.Auth.SignUpRequest>('/api/auth/signup', {
|
||||||
username: username,
|
username: username,
|
||||||
password: password
|
password: password
|
||||||
});
|
});
|
||||||
|
@ -1,7 +1,6 @@
|
|||||||
import axios from 'axios';
|
import axios from 'axios';
|
||||||
import { Requests, Responses } from 'dto';
|
import { Requests, Responses } from 'dto';
|
||||||
export { Requests, Responses };
|
export { Requests, Responses };
|
||||||
import { validateSync } from 'class-validator';
|
|
||||||
|
|
||||||
export const post = <T extends Requests.BaseRequest>(url: string, data: T) =>
|
export const post = <T extends Requests.BaseRequest>(url: string, data: T) =>
|
||||||
axios
|
axios
|
||||||
@ -11,17 +10,12 @@ export const post = <T extends Requests.BaseRequest>(url: string, data: T) =>
|
|||||||
.then((res) => res.data)
|
.then((res) => res.data)
|
||||||
.catch((err) => err.response.data);
|
.catch((err) => err.response.data);
|
||||||
|
|
||||||
export function post_token<T extends Requests.BaseRequest>(
|
export const post_token = <T extends Requests.BaseRequest>(
|
||||||
url: string,
|
url: string,
|
||||||
data: T,
|
data: T,
|
||||||
token: string
|
token: string
|
||||||
) {
|
) =>
|
||||||
const errors = validateSync(data);
|
axios
|
||||||
if (errors.length > 0) {
|
|
||||||
console.error('Validation failed, errors: ', errors);
|
|
||||||
throw new Error('Validation failed');
|
|
||||||
}
|
|
||||||
return axios
|
|
||||||
.post(url, data, {
|
.post(url, data, {
|
||||||
headers: {
|
headers: {
|
||||||
Authorization: 'Bearer ' + token,
|
Authorization: 'Bearer ' + token,
|
||||||
@ -30,7 +24,6 @@ export function post_token<T extends Requests.BaseRequest>(
|
|||||||
})
|
})
|
||||||
.then((res) => res.data)
|
.then((res) => res.data)
|
||||||
.catch((err) => err.response.data);
|
.catch((err) => err.response.data);
|
||||||
}
|
|
||||||
|
|
||||||
export const post_token_form = (
|
export const post_token_form = (
|
||||||
url: string,
|
url: string,
|
||||||
|
@ -4,18 +4,18 @@ import {
|
|||||||
Controller,
|
Controller,
|
||||||
Get,
|
Get,
|
||||||
HttpCode,
|
HttpCode,
|
||||||
ParseBoolPipe,
|
|
||||||
Post,
|
Post,
|
||||||
Query,
|
Query,
|
||||||
Redirect,
|
Redirect,
|
||||||
Request,
|
Request,
|
||||||
UnauthorizedException,
|
UnauthorizedException,
|
||||||
UseGuards
|
UseGuards,
|
||||||
|
ValidationPipe
|
||||||
} from '@nestjs/common';
|
} from '@nestjs/common';
|
||||||
import { AuthService } from '../services/auth';
|
import { AuthService } from '../services/auth';
|
||||||
import { AuthGuard } from '@nestjs/passport';
|
import { AuthGuard } from '@nestjs/passport';
|
||||||
import { Public } from '../authguards';
|
import { Public } from '../authguards';
|
||||||
import { Responses } from 'dto';
|
import { Responses, Requests } from 'dto';
|
||||||
import { tfaTypes } from '../entities';
|
import { tfaTypes } from '../entities';
|
||||||
import { toDataURL } from 'qrcode';
|
import { toDataURL } from 'qrcode';
|
||||||
import * as base32 from 'thirty-two';
|
import * as base32 from 'thirty-two';
|
||||||
@ -30,25 +30,22 @@ export default class AuthController {
|
|||||||
@HttpCode(200)
|
@HttpCode(200)
|
||||||
async login(
|
async login(
|
||||||
@Request() req,
|
@Request() req,
|
||||||
@Body('otp') otp?: string
|
@Body(new ValidationPipe()) data: Requests.Auth.LoginRequest
|
||||||
): Promise<
|
): Promise<
|
||||||
Responses.Auth.LoginResponse | Responses.Auth.TfaRequiredResponse
|
Responses.Auth.LoginResponse | Responses.Auth.TfaRequiredResponse
|
||||||
> {
|
> {
|
||||||
if (this.authService.requiresTfa(req.user)) {
|
if (this.authService.requiresTfa(req.user)) {
|
||||||
if (!otp) {
|
if (!data.otp) {
|
||||||
if (req.user.tfaType == tfaTypes.EMAIL)
|
if (req.user.tfaType == tfaTypes.EMAIL)
|
||||||
await this.authService.sendTfaMail(req.user);
|
await this.authService.sendTfaMail(req.user);
|
||||||
return {
|
return new Responses.Auth.TfaRequiredResponse();
|
||||||
statusCode: 200
|
|
||||||
};
|
|
||||||
}
|
}
|
||||||
if (!(await this.authService.verifyTfa(req.user, otp)))
|
if (!(await this.authService.verifyTfa(req.user, data.otp)))
|
||||||
throw new UnauthorizedException('Incorrect 2fa');
|
throw new UnauthorizedException('Incorrect 2fa');
|
||||||
}
|
}
|
||||||
return {
|
return new Responses.Auth.LoginResponse(
|
||||||
statusCode: 200,
|
await this.authService.login(req, req.user)
|
||||||
jwt: await this.authService.login(req, req.user)
|
);
|
||||||
};
|
|
||||||
}
|
}
|
||||||
|
|
||||||
async tfa(
|
async tfa(
|
||||||
@ -61,73 +58,61 @@ export default class AuthController {
|
|||||||
}
|
}
|
||||||
await this.authService.setTfaType(req.user, type);
|
await this.authService.setTfaType(req.user, type);
|
||||||
await this.authService.revokeAll(req.user);
|
await this.authService.revokeAll(req.user);
|
||||||
return {
|
return new Responses.Auth.TfaCompletedResponse();
|
||||||
statusCode: 200
|
|
||||||
};
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Post('2fa/complete/mail')
|
@Post('2fa/complete/mail')
|
||||||
async tfaMail(
|
async tfaMail(
|
||||||
@Request() req,
|
@Request() req,
|
||||||
@Body('code') code: string
|
@Body(new ValidationPipe()) data: Requests.Auth.TfaComplete
|
||||||
): Promise<Responses.Auth.TfaCompletedResponse> {
|
): Promise<Responses.Auth.TfaCompletedResponse> {
|
||||||
return await this.tfa(req, code, tfaTypes.EMAIL);
|
return await this.tfa(req, data.code, tfaTypes.EMAIL);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Post('2fa/complete/totp')
|
@Post('2fa/complete/totp')
|
||||||
async tfaTotp(
|
async tfaTotp(
|
||||||
@Request() req,
|
@Request() req,
|
||||||
@Body('code') code: string
|
@Body(new ValidationPipe()) data: Requests.Auth.TfaComplete
|
||||||
): Promise<Responses.Auth.TfaCompletedResponse> {
|
): Promise<Responses.Auth.TfaCompletedResponse> {
|
||||||
return await this.tfa(req, code, tfaTypes.TOTP);
|
return await this.tfa(req, data.code, tfaTypes.TOTP);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Get('2fa/setup')
|
@Get('2fa/setup')
|
||||||
async setupTotp(
|
async setupTotp(
|
||||||
@Request() req,
|
@Request() req,
|
||||||
@Body('mail', ParseBoolPipe) mail: boolean
|
@Body(new ValidationPipe()) data: Requests.Auth.TfaSetup
|
||||||
): Promise<
|
): Promise<
|
||||||
| Responses.Auth.RequestTotpTfaResponse
|
| Responses.Auth.RequestTotpTfaResponse
|
||||||
| Responses.Auth.RequestEmailTfaResponse
|
| Responses.Auth.RequestEmailTfaResponse
|
||||||
> {
|
> {
|
||||||
const secret = await this.authService.setupTfa(req.user);
|
const secret = await this.authService.setupTfa(req.user);
|
||||||
if (mail)
|
if (data.mail) return new Responses.Auth.RequestEmailTfaResponse();
|
||||||
return {
|
return new Responses.Auth.RequestTotpTfaResponse(
|
||||||
statusCode: 200
|
await toDataURL(
|
||||||
};
|
|
||||||
return {
|
|
||||||
statusCode: 200,
|
|
||||||
qrCode: await toDataURL(
|
|
||||||
`otpauth://totp/MFileserver:${req.user.name}?secret=${base32
|
`otpauth://totp/MFileserver:${req.user.name}?secret=${base32
|
||||||
.encode(secret)
|
.encode(secret)
|
||||||
.toString()}&issuer=MFileserver`
|
.toString()}&issuer=MFileserver`
|
||||||
),
|
),
|
||||||
secret
|
secret
|
||||||
};
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Public()
|
@Public()
|
||||||
@Post('signup')
|
@Post('signup')
|
||||||
async signup(
|
async signup(
|
||||||
@Body('username') username,
|
@Body(new ValidationPipe()) data: Requests.Auth.SignUpRequest
|
||||||
@Body('password') password
|
|
||||||
): Promise<Responses.Auth.SignupResponse> {
|
): Promise<Responses.Auth.SignupResponse> {
|
||||||
if ((await this.authService.findUser(username, false)) != null)
|
if ((await this.authService.findUser(data.username, false)) != null)
|
||||||
throw new BadRequestException('Username already taken');
|
throw new BadRequestException('Username already taken');
|
||||||
await this.authService.signup(username, password);
|
await this.authService.signup(data.username, data.password);
|
||||||
return {
|
return new Responses.Auth.SignupResponse();
|
||||||
statusCode: 200
|
|
||||||
};
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Post('refresh')
|
@Post('refresh')
|
||||||
async refresh(@Request() req): Promise<Responses.Auth.RefreshResponse> {
|
async refresh(@Request() req): Promise<Responses.Auth.RefreshResponse> {
|
||||||
const token = await this.authService.login(req, req.user);
|
const token = await this.authService.login(req, req.user);
|
||||||
await this.authService.revoke(req.token);
|
await this.authService.revoke(req.token);
|
||||||
return {
|
return await new Responses.Auth.RefreshResponse(token);
|
||||||
statusCode: 200,
|
|
||||||
jwt: token
|
|
||||||
};
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Public()
|
@Public()
|
||||||
|
@ -6,9 +6,10 @@ import {
|
|||||||
ParseIntPipe,
|
ParseIntPipe,
|
||||||
Post,
|
Post,
|
||||||
Request,
|
Request,
|
||||||
StreamableFile
|
StreamableFile,
|
||||||
|
ValidationPipe
|
||||||
} from '@nestjs/common';
|
} from '@nestjs/common';
|
||||||
import { Responses } from 'dto';
|
import { Responses, Requests, validateAsyncInline } from 'dto';
|
||||||
import FileSystemService from '../services/filesystem';
|
import FileSystemService from '../services/filesystem';
|
||||||
import { UserRole } from '../entities';
|
import { UserRole } from '../entities';
|
||||||
import { Role } from '../authguards';
|
import { Role } from '../authguards';
|
||||||
@ -20,10 +21,7 @@ export default class FileSystemController {
|
|||||||
@Get('root')
|
@Get('root')
|
||||||
@Role(UserRole.USER)
|
@Role(UserRole.USER)
|
||||||
async getRoot(@Request() req): Promise<Responses.FS.GetRootResponse> {
|
async getRoot(@Request() req): Promise<Responses.FS.GetRootResponse> {
|
||||||
return {
|
return new Responses.FS.GetRootResponse(req.user.rootId);
|
||||||
statusCode: 200,
|
|
||||||
rootId: req.user.rootId
|
|
||||||
};
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Get('node/:node')
|
@Get('node/:node')
|
||||||
@ -33,19 +31,19 @@ export default class FileSystemController {
|
|||||||
@Param('node', ParseIntPipe) nodeId
|
@Param('node', ParseIntPipe) nodeId
|
||||||
): Promise<Responses.FS.GetNodeResponse> {
|
): Promise<Responses.FS.GetNodeResponse> {
|
||||||
const node = await this.fsService.getNodeAndValidate(nodeId, req.user);
|
const node = await this.fsService.getNodeAndValidate(nodeId, req.user);
|
||||||
const data: Responses.FS.GetNodeResponse = {
|
const data = new Responses.FS.GetNodeResponse(
|
||||||
id: nodeId,
|
nodeId,
|
||||||
statusCode: 200,
|
node.name,
|
||||||
name: node.name,
|
node.isFile,
|
||||||
parent: node.parentId,
|
node.parentId
|
||||||
isFile: node.isFile
|
);
|
||||||
};
|
|
||||||
if (data.isFile) {
|
if (data.isFile) {
|
||||||
data.size = node.size;
|
data.size = node.size;
|
||||||
} else {
|
} else {
|
||||||
data.children = (await node.children).map((child) => child.id);
|
data.children = (await node.children).map((child) => child.id);
|
||||||
}
|
}
|
||||||
return data;
|
return validateAsyncInline(data);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Get('path/:node')
|
@Get('path/:node')
|
||||||
@ -54,62 +52,53 @@ export default class FileSystemController {
|
|||||||
@Request() req,
|
@Request() req,
|
||||||
@Param('node', ParseIntPipe) nodeId
|
@Param('node', ParseIntPipe) nodeId
|
||||||
): Promise<Responses.FS.GetPathResponse> {
|
): Promise<Responses.FS.GetPathResponse> {
|
||||||
return {
|
return new Responses.FS.GetPathResponse(
|
||||||
statusCode: 200,
|
await this.fsService.generatePath(
|
||||||
path: await this.fsService.generatePath(
|
|
||||||
await this.fsService.getNodeAndValidate(nodeId, req.user)
|
await this.fsService.getNodeAndValidate(nodeId, req.user)
|
||||||
)
|
)
|
||||||
};
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Post('createFolder')
|
@Post('createFolder')
|
||||||
@Role(UserRole.USER)
|
@Role(UserRole.USER)
|
||||||
async createFolder(
|
async createFolder(
|
||||||
@Request() req,
|
@Request() req,
|
||||||
@Body('parent', ParseIntPipe) parent,
|
@Body(new ValidationPipe()) data: Requests.FS.CreateFolderRequest
|
||||||
@Body('name') name
|
|
||||||
): Promise<Responses.FS.CreateFolderResponse> {
|
): Promise<Responses.FS.CreateFolderResponse> {
|
||||||
const newChild = await this.fsService.create(
|
const newChild = await this.fsService.create(
|
||||||
await this.fsService.getNodeAndValidate(parent, req.user),
|
await this.fsService.getNodeAndValidate(data.parent, req.user),
|
||||||
name,
|
data.name,
|
||||||
req.user,
|
req.user,
|
||||||
false
|
false
|
||||||
);
|
);
|
||||||
return {
|
return new Responses.FS.CreateFolderResponse(newChild.id);
|
||||||
statusCode: 200,
|
|
||||||
id: newChild.id
|
|
||||||
};
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Post('createFile')
|
@Post('createFile')
|
||||||
@Role(UserRole.USER)
|
@Role(UserRole.USER)
|
||||||
async createFile(
|
async createFile(
|
||||||
@Request() req,
|
@Request() req,
|
||||||
@Body('parent', ParseIntPipe) parent,
|
@Body(new ValidationPipe()) data: Requests.FS.CreateFileRequest
|
||||||
@Body('name') name
|
|
||||||
): Promise<Responses.FS.CreateFileResponse> {
|
): Promise<Responses.FS.CreateFileResponse> {
|
||||||
const newChild = await this.fsService.create(
|
const newChild = await this.fsService.create(
|
||||||
await this.fsService.getNodeAndValidate(parent, req.user),
|
await this.fsService.getNodeAndValidate(data.parent, req.user),
|
||||||
name,
|
data.name,
|
||||||
req.user,
|
req.user,
|
||||||
true
|
true
|
||||||
);
|
);
|
||||||
return {
|
return new Responses.FS.CreateFileResponse(newChild.id);
|
||||||
statusCode: 200,
|
|
||||||
id: newChild.id
|
|
||||||
};
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Post('delete')
|
@Post('delete')
|
||||||
@Role(UserRole.USER)
|
@Role(UserRole.USER)
|
||||||
async delete(
|
async delete(
|
||||||
@Request() req,
|
@Request() req,
|
||||||
@Body('node', ParseIntPipe) node_id
|
@Body(new ValidationPipe()) data: Requests.FS.DeleteRequest
|
||||||
): Promise<Responses.FS.DeleteResponse> {
|
): Promise<Responses.FS.DeleteResponse> {
|
||||||
await this.fsService.delete(
|
await this.fsService.delete(
|
||||||
await this.fsService.getNodeAndValidate(node_id, req.user)
|
await this.fsService.getNodeAndValidate(data.node, req.user)
|
||||||
);
|
);
|
||||||
return { statusCode: 200 };
|
return new Responses.FS.DeleteResponse();
|
||||||
}
|
}
|
||||||
|
|
||||||
@Post('upload/:node')
|
@Post('upload/:node')
|
||||||
@ -119,7 +108,7 @@ export default class FileSystemController {
|
|||||||
@Param('node', ParseIntPipe) nodeId
|
@Param('node', ParseIntPipe) nodeId
|
||||||
): Promise<Responses.FS.UploadFileResponse> {
|
): Promise<Responses.FS.UploadFileResponse> {
|
||||||
await this.fsService.uploadFile(await req.file(), nodeId, req.user);
|
await this.fsService.uploadFile(await req.file(), nodeId, req.user);
|
||||||
return { statusCode: 200 };
|
return new Responses.FS.UploadFileResponse();
|
||||||
}
|
}
|
||||||
|
|
||||||
@Post('download')
|
@Post('download')
|
||||||
|
Loading…
Reference in New Issue
Block a user