Everything is validated now
This commit is contained in:
		@@ -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')
 | 
				
			||||||
 
 | 
				
			|||||||
		Reference in New Issue
	
	Block a user