Rewritten using quasar framework
This commit is contained in:
		
							
								
								
									
										7
									
								
								backend/.gitattributes
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										7
									
								
								backend/.gitattributes
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,7 @@ | ||||
| # See https://git-scm.com/docs/gitattributes for more about git attribute files. | ||||
|  | ||||
| # Mark the database schema as having been generated. | ||||
| db/schema.rb linguist-generated | ||||
|  | ||||
| # Mark any vendored files as having been vendored. | ||||
| vendor/* linguist-vendored | ||||
							
								
								
									
										29
									
								
								backend/.gitignore
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										29
									
								
								backend/.gitignore
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,29 @@ | ||||
| # See https://help.github.com/articles/ignoring-files for more about ignoring files. | ||||
| # | ||||
| # If you find yourself ignoring temporary files generated by your text editor | ||||
| # or operating system, you probably want to add a global ignore instead: | ||||
| #   git config --global core.excludesfile '~/.gitignore_global' | ||||
|  | ||||
| # Ignore bundler config. | ||||
| /.bundle | ||||
|  | ||||
| # Ignore all logfiles and tempfiles. | ||||
| /log/* | ||||
| /tmp/* | ||||
| !/log/.keep | ||||
| !/tmp/.keep | ||||
|  | ||||
| # Ignore pidfiles, but keep the directory. | ||||
| /tmp/pids/* | ||||
| !/tmp/pids/ | ||||
| !/tmp/pids/.keep | ||||
|  | ||||
| # Ignore uploaded files in development. | ||||
| /storage/* | ||||
| !/storage/.keep | ||||
| /tmp/storage/* | ||||
| !/tmp/storage/ | ||||
| !/tmp/storage/.keep | ||||
|  | ||||
| # Ignore master key for decrypting credentials and more. | ||||
| /config/master.key | ||||
							
								
								
									
										1
									
								
								backend/.ruby-version
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								backend/.ruby-version
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1 @@ | ||||
| ruby-3.1.2 | ||||
							
								
								
									
										13
									
								
								backend/.vscode/extensions.json
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										13
									
								
								backend/.vscode/extensions.json
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,13 @@ | ||||
| { | ||||
|     "recommendations": [ | ||||
|         "kaiwood.endwise", | ||||
|         "bung87.rails", | ||||
|         "aki77.rails-db-schema", | ||||
|         "aki77.rails-routes", | ||||
|         "rebornix.ruby", | ||||
|         "castwide.solargraph", | ||||
|         "wingrunr21.vscode-ruby", | ||||
|         "sianglim.slim", | ||||
|         "bung87.vscode-gemfile" | ||||
|     ] | ||||
| } | ||||
							
								
								
									
										57
									
								
								backend/Gemfile
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										57
									
								
								backend/Gemfile
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,57 @@ | ||||
| source "https://rubygems.org" | ||||
| git_source(:github) { |repo| "https://github.com/#{repo}.git" } | ||||
|  | ||||
| ruby "3.1.2" | ||||
|  | ||||
| # Bundle edge Rails instead: gem "rails", github: "rails/rails", branch: "main" | ||||
| gem "rails", "~> 7.0.4" | ||||
|  | ||||
| # Use sqlite3 as the database for Active Record | ||||
| gem "sqlite3", "~> 1.4" | ||||
|  | ||||
| # Use the Puma web server [https://github.com/puma/puma] | ||||
| gem "puma", "~> 5.0" | ||||
|  | ||||
| # Build JSON APIs with ease [https://github.com/rails/jbuilder] | ||||
| # gem "jbuilder" | ||||
|  | ||||
| # Use Redis adapter to run Action Cable in production | ||||
| # gem "redis", "~> 4.0" | ||||
|  | ||||
| # Use Kredis to get higher-level data types in Redis [https://github.com/rails/kredis] | ||||
| # gem "kredis" | ||||
|  | ||||
| # Use Json Web Token (JWT) for token based authentication | ||||
| gem 'jwt' | ||||
|  | ||||
| gem 'pg' | ||||
|  | ||||
| # Use Active Model has_secure_password [https://guides.rubyonrails.org/active_model_basics.html#securepassword] | ||||
| gem "bcrypt", "~> 3.1.7" | ||||
|  | ||||
| # Windows does not include zoneinfo files, so bundle the tzinfo-data gem | ||||
| gem "tzinfo-data" | ||||
|  | ||||
| # Reduces boot times through caching; required in config/boot.rb | ||||
| gem "bootsnap", require: false | ||||
|  | ||||
| # Use Active Storage variants [https://guides.rubyonrails.org/active_storage_overview.html#transforming-images] | ||||
| # gem "image_processing", "~> 1.2" | ||||
|  | ||||
| # Use Rack CORS for handling Cross-Origin Resource Sharing (CORS), making cross-origin AJAX possible | ||||
| # gem "rack-cors" | ||||
|  | ||||
| group :development, :test do | ||||
|   # See https://guides.rubyonrails.org/debugging_rails_applications.html#debugging-with-the-debug-gem | ||||
|   gem "debug", platforms: %i[ mri mingw x64_mingw ] | ||||
|  | ||||
|   gem 'rubocop' | ||||
| end | ||||
|  | ||||
| group :development do | ||||
|   # Speed up commands on slow machines / big apps [https://github.com/rails/spring] | ||||
|   gem "spring" | ||||
|  | ||||
|   gem 'solargraph' | ||||
| end | ||||
|  | ||||
							
								
								
									
										230
									
								
								backend/Gemfile.lock
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										230
									
								
								backend/Gemfile.lock
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,230 @@ | ||||
| GEM | ||||
|   remote: https://rubygems.org/ | ||||
|   specs: | ||||
|     actioncable (7.0.4) | ||||
|       actionpack (= 7.0.4) | ||||
|       activesupport (= 7.0.4) | ||||
|       nio4r (~> 2.0) | ||||
|       websocket-driver (>= 0.6.1) | ||||
|     actionmailbox (7.0.4) | ||||
|       actionpack (= 7.0.4) | ||||
|       activejob (= 7.0.4) | ||||
|       activerecord (= 7.0.4) | ||||
|       activestorage (= 7.0.4) | ||||
|       activesupport (= 7.0.4) | ||||
|       mail (>= 2.7.1) | ||||
|       net-imap | ||||
|       net-pop | ||||
|       net-smtp | ||||
|     actionmailer (7.0.4) | ||||
|       actionpack (= 7.0.4) | ||||
|       actionview (= 7.0.4) | ||||
|       activejob (= 7.0.4) | ||||
|       activesupport (= 7.0.4) | ||||
|       mail (~> 2.5, >= 2.5.4) | ||||
|       net-imap | ||||
|       net-pop | ||||
|       net-smtp | ||||
|       rails-dom-testing (~> 2.0) | ||||
|     actionpack (7.0.4) | ||||
|       actionview (= 7.0.4) | ||||
|       activesupport (= 7.0.4) | ||||
|       rack (~> 2.0, >= 2.2.0) | ||||
|       rack-test (>= 0.6.3) | ||||
|       rails-dom-testing (~> 2.0) | ||||
|       rails-html-sanitizer (~> 1.0, >= 1.2.0) | ||||
|     actiontext (7.0.4) | ||||
|       actionpack (= 7.0.4) | ||||
|       activerecord (= 7.0.4) | ||||
|       activestorage (= 7.0.4) | ||||
|       activesupport (= 7.0.4) | ||||
|       globalid (>= 0.6.0) | ||||
|       nokogiri (>= 1.8.5) | ||||
|     actionview (7.0.4) | ||||
|       activesupport (= 7.0.4) | ||||
|       builder (~> 3.1) | ||||
|       erubi (~> 1.4) | ||||
|       rails-dom-testing (~> 2.0) | ||||
|       rails-html-sanitizer (~> 1.1, >= 1.2.0) | ||||
|     activejob (7.0.4) | ||||
|       activesupport (= 7.0.4) | ||||
|       globalid (>= 0.3.6) | ||||
|     activemodel (7.0.4) | ||||
|       activesupport (= 7.0.4) | ||||
|     activerecord (7.0.4) | ||||
|       activemodel (= 7.0.4) | ||||
|       activesupport (= 7.0.4) | ||||
|     activestorage (7.0.4) | ||||
|       actionpack (= 7.0.4) | ||||
|       activejob (= 7.0.4) | ||||
|       activerecord (= 7.0.4) | ||||
|       activesupport (= 7.0.4) | ||||
|       marcel (~> 1.0) | ||||
|       mini_mime (>= 1.1.0) | ||||
|     activesupport (7.0.4) | ||||
|       concurrent-ruby (~> 1.0, >= 1.0.2) | ||||
|       i18n (>= 1.6, < 2) | ||||
|       minitest (>= 5.1) | ||||
|       tzinfo (~> 2.0) | ||||
|     ast (2.4.2) | ||||
|     backport (1.2.0) | ||||
|     bcrypt (3.1.18) | ||||
|     benchmark (0.2.0) | ||||
|     bootsnap (1.13.0) | ||||
|       msgpack (~> 1.2) | ||||
|     builder (3.2.4) | ||||
|     concurrent-ruby (1.1.10) | ||||
|     crass (1.0.6) | ||||
|     debug (1.6.3) | ||||
|       irb (>= 1.3.6) | ||||
|       reline (>= 0.3.1) | ||||
|     diff-lcs (1.5.0) | ||||
|     e2mmap (0.1.0) | ||||
|     erubi (1.11.0) | ||||
|     globalid (1.0.0) | ||||
|       activesupport (>= 5.0) | ||||
|     i18n (1.12.0) | ||||
|       concurrent-ruby (~> 1.0) | ||||
|     io-console (0.5.11) | ||||
|     irb (1.4.2) | ||||
|       reline (>= 0.3.0) | ||||
|     jaro_winkler (1.5.4) | ||||
|     json (2.6.2) | ||||
|     jwt (2.5.0) | ||||
|     kramdown (2.4.0) | ||||
|       rexml | ||||
|     kramdown-parser-gfm (1.1.0) | ||||
|       kramdown (~> 2.0) | ||||
|     loofah (2.19.0) | ||||
|       crass (~> 1.0.2) | ||||
|       nokogiri (>= 1.5.9) | ||||
|     mail (2.7.1) | ||||
|       mini_mime (>= 0.1.1) | ||||
|     marcel (1.0.2) | ||||
|     method_source (1.0.0) | ||||
|     mini_mime (1.1.2) | ||||
|     minitest (5.16.3) | ||||
|     msgpack (1.6.0) | ||||
|     net-imap (0.3.1) | ||||
|       net-protocol | ||||
|     net-pop (0.1.2) | ||||
|       net-protocol | ||||
|     net-protocol (0.1.3) | ||||
|       timeout | ||||
|     net-smtp (0.3.3) | ||||
|       net-protocol | ||||
|     nio4r (2.5.8) | ||||
|     nokogiri (1.13.9-x64-mingw-ucrt) | ||||
|       racc (~> 1.4) | ||||
|     parallel (1.22.1) | ||||
|     parser (3.1.2.1) | ||||
|       ast (~> 2.4.1) | ||||
|     pg (1.4.4-x64-mingw-ucrt) | ||||
|     puma (5.6.5) | ||||
|       nio4r (~> 2.0) | ||||
|     racc (1.6.0) | ||||
|     rack (2.2.4) | ||||
|     rack-test (2.0.2) | ||||
|       rack (>= 1.3) | ||||
|     rails (7.0.4) | ||||
|       actioncable (= 7.0.4) | ||||
|       actionmailbox (= 7.0.4) | ||||
|       actionmailer (= 7.0.4) | ||||
|       actionpack (= 7.0.4) | ||||
|       actiontext (= 7.0.4) | ||||
|       actionview (= 7.0.4) | ||||
|       activejob (= 7.0.4) | ||||
|       activemodel (= 7.0.4) | ||||
|       activerecord (= 7.0.4) | ||||
|       activestorage (= 7.0.4) | ||||
|       activesupport (= 7.0.4) | ||||
|       bundler (>= 1.15.0) | ||||
|       railties (= 7.0.4) | ||||
|     rails-dom-testing (2.0.3) | ||||
|       activesupport (>= 4.2.0) | ||||
|       nokogiri (>= 1.6) | ||||
|     rails-html-sanitizer (1.4.3) | ||||
|       loofah (~> 2.3) | ||||
|     railties (7.0.4) | ||||
|       actionpack (= 7.0.4) | ||||
|       activesupport (= 7.0.4) | ||||
|       method_source | ||||
|       rake (>= 12.2) | ||||
|       thor (~> 1.0) | ||||
|       zeitwerk (~> 2.5) | ||||
|     rainbow (3.1.1) | ||||
|     rake (13.0.6) | ||||
|     regexp_parser (2.6.0) | ||||
|     reline (0.3.1) | ||||
|       io-console (~> 0.5) | ||||
|     reverse_markdown (2.1.1) | ||||
|       nokogiri | ||||
|     rexml (3.2.5) | ||||
|     rubocop (1.39.0) | ||||
|       json (~> 2.3) | ||||
|       parallel (~> 1.10) | ||||
|       parser (>= 3.1.2.1) | ||||
|       rainbow (>= 2.2.2, < 4.0) | ||||
|       regexp_parser (>= 1.8, < 3.0) | ||||
|       rexml (>= 3.2.5, < 4.0) | ||||
|       rubocop-ast (>= 1.23.0, < 2.0) | ||||
|       ruby-progressbar (~> 1.7) | ||||
|       unicode-display_width (>= 1.4.0, < 3.0) | ||||
|     rubocop-ast (1.23.0) | ||||
|       parser (>= 3.1.1.0) | ||||
|     ruby-progressbar (1.11.0) | ||||
|     solargraph (0.47.2) | ||||
|       backport (~> 1.2) | ||||
|       benchmark | ||||
|       bundler (>= 1.17.2) | ||||
|       diff-lcs (~> 1.4) | ||||
|       e2mmap | ||||
|       jaro_winkler (~> 1.5) | ||||
|       kramdown (~> 2.3) | ||||
|       kramdown-parser-gfm (~> 1.1) | ||||
|       parser (~> 3.0) | ||||
|       reverse_markdown (>= 1.0.5, < 3) | ||||
|       rubocop (>= 0.52) | ||||
|       thor (~> 1.0) | ||||
|       tilt (~> 2.0) | ||||
|       yard (~> 0.9, >= 0.9.24) | ||||
|     spring (4.1.0) | ||||
|     sqlite3 (1.5.3-x64-mingw-ucrt) | ||||
|     thor (1.2.1) | ||||
|     tilt (2.0.11) | ||||
|     timeout (0.3.0) | ||||
|     tzinfo (2.0.5) | ||||
|       concurrent-ruby (~> 1.0) | ||||
|     tzinfo-data (1.2022.6) | ||||
|       tzinfo (>= 1.0.0) | ||||
|     unicode-display_width (2.3.0) | ||||
|     webrick (1.7.0) | ||||
|     websocket-driver (0.7.5) | ||||
|       websocket-extensions (>= 0.1.0) | ||||
|     websocket-extensions (0.1.5) | ||||
|     yard (0.9.28) | ||||
|       webrick (~> 1.7.0) | ||||
|     zeitwerk (2.6.6) | ||||
|  | ||||
| PLATFORMS | ||||
|   x64-mingw-ucrt | ||||
|  | ||||
| DEPENDENCIES | ||||
|   bcrypt (~> 3.1.7) | ||||
|   bootsnap | ||||
|   debug | ||||
|   jwt | ||||
|   pg | ||||
|   puma (~> 5.0) | ||||
|   rails (~> 7.0.4) | ||||
|   rubocop | ||||
|   solargraph | ||||
|   spring | ||||
|   sqlite3 (~> 1.4) | ||||
|   tzinfo-data | ||||
|  | ||||
| RUBY VERSION | ||||
|    ruby 3.1.2p20 | ||||
|  | ||||
| BUNDLED WITH | ||||
|    2.3.7 | ||||
							
								
								
									
										24
									
								
								backend/README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										24
									
								
								backend/README.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,24 @@ | ||||
| # README | ||||
|  | ||||
| This README would normally document whatever steps are necessary to get the | ||||
| application up and running. | ||||
|  | ||||
| Things you may want to cover: | ||||
|  | ||||
| * Ruby version | ||||
|  | ||||
| * System dependencies | ||||
|  | ||||
| * Configuration | ||||
|  | ||||
| * Database creation | ||||
|  | ||||
| * Database initialization | ||||
|  | ||||
| * How to run the test suite | ||||
|  | ||||
| * Services (job queues, cache servers, search engines, etc.) | ||||
|  | ||||
| * Deployment instructions | ||||
|  | ||||
| * ... | ||||
							
								
								
									
										6
									
								
								backend/Rakefile
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										6
									
								
								backend/Rakefile
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,6 @@ | ||||
| # Add your own tasks in files placed in lib/tasks ending in .rake, | ||||
| # for example lib/tasks/capistrano.rake, and they will automatically be available to Rake. | ||||
|  | ||||
| require_relative "config/application" | ||||
|  | ||||
| Rails.application.load_tasks | ||||
							
								
								
									
										4
									
								
								backend/app/channels/application_cable/channel.rb
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										4
									
								
								backend/app/channels/application_cable/channel.rb
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,4 @@ | ||||
| module ApplicationCable | ||||
|   class Channel < ActionCable::Channel::Base | ||||
|   end | ||||
| end | ||||
							
								
								
									
										4
									
								
								backend/app/channels/application_cable/connection.rb
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										4
									
								
								backend/app/channels/application_cable/connection.rb
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,4 @@ | ||||
| module ApplicationCable | ||||
|   class Connection < ActionCable::Connection::Base | ||||
|   end | ||||
| end | ||||
							
								
								
									
										14
									
								
								backend/app/controllers/application_controller.rb
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										14
									
								
								backend/app/controllers/application_controller.rb
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,14 @@ | ||||
| class ApplicationController < ActionController::API | ||||
|   def authorize_request | ||||
|     header = request.headers['Authorization'] | ||||
|     header = header.split(' ').last if header | ||||
|     begin | ||||
|         @decoded = JsonWebToken.decode(header) | ||||
|         @current_user = User.find(@decoded[:user_id]) | ||||
|     rescue ActiveRecord::RecordNotFound => e | ||||
|         render json: { errors: e.message }, status: :unauthorized | ||||
|     rescue JWT::DecodeError => e | ||||
|         render json: { errors: e.message }, status: :unauthorized | ||||
|     end | ||||
|   end | ||||
| end | ||||
							
								
								
									
										19
									
								
								backend/app/controllers/auth/login_controller.rb
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										19
									
								
								backend/app/controllers/auth/login_controller.rb
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,19 @@ | ||||
| class Auth::LoginController < ApplicationController | ||||
|   def post | ||||
|     @user = User.find_by_email(params[:email]) | ||||
|     if @user&.authenticate(params[:password]) | ||||
|       token = JsonWebToken.encode(user_id: @user.id) | ||||
|       time = Time.now + 24.hours.to_i | ||||
|       render json: { token: token, exp: time.strftime("%m-%d-%Y %H:%M"), | ||||
|                      username: @user.username }, status: :ok | ||||
|     else | ||||
|       render json: { error: 'unauthorized' }, status: :unauthorized | ||||
|     end | ||||
|   end | ||||
|  | ||||
|   private | ||||
|  | ||||
|   def login_params | ||||
|     params.permit(:email, :password) | ||||
|   end | ||||
| end | ||||
							
								
								
									
										19
									
								
								backend/app/controllers/auth/register_controller.rb
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										19
									
								
								backend/app/controllers/auth/register_controller.rb
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,19 @@ | ||||
| class Auth::RegisterController < ApplicationController | ||||
|   def post | ||||
|     @user = User.new(user_params) | ||||
|     if @user.save | ||||
|       render status: :created | ||||
|     else | ||||
|       render json: { errors: @user.errors.full_messages }, | ||||
|              status: :unprocessable_entity | ||||
|     end | ||||
|   end | ||||
|  | ||||
|   private | ||||
|  | ||||
|   def user_params | ||||
|     params.permit( | ||||
|       :username, :email, :password, :password_confirmation | ||||
|     ) | ||||
|   end | ||||
| end | ||||
							
								
								
									
										0
									
								
								backend/app/controllers/concerns/.keep
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										0
									
								
								backend/app/controllers/concerns/.keep
									
									
									
									
									
										Normal file
									
								
							
							
								
								
									
										7
									
								
								backend/app/jobs/application_job.rb
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										7
									
								
								backend/app/jobs/application_job.rb
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,7 @@ | ||||
| class ApplicationJob < ActiveJob::Base | ||||
|   # Automatically retry jobs that encountered a deadlock | ||||
|   # retry_on ActiveRecord::Deadlocked | ||||
|  | ||||
|   # Most jobs are safe to ignore if the underlying records are no longer available | ||||
|   # discard_on ActiveJob::DeserializationError | ||||
| end | ||||
							
								
								
									
										13
									
								
								backend/app/lib/json_web_token.rb
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										13
									
								
								backend/app/lib/json_web_token.rb
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,13 @@ | ||||
| class JsonWebToken | ||||
|     SECRET_KEY = Rails.application.secrets.secret_key_base.to_s | ||||
|    | ||||
|     def self.encode(payload, exp = 24.hours.from_now) | ||||
|       payload[:exp] = exp.to_i | ||||
|       JWT.encode(payload, SECRET_KEY) | ||||
|     end | ||||
|    | ||||
|     def self.decode(token) | ||||
|       decoded = JWT.decode(token, SECRET_KEY)[0] | ||||
|       HashWithIndifferentAccess.new decoded | ||||
|     end | ||||
|   end | ||||
							
								
								
									
										4
									
								
								backend/app/mailers/application_mailer.rb
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										4
									
								
								backend/app/mailers/application_mailer.rb
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,4 @@ | ||||
| class ApplicationMailer < ActionMailer::Base | ||||
|   default from: "from@example.com" | ||||
|   layout "mailer" | ||||
| end | ||||
							
								
								
									
										3
									
								
								backend/app/models/application_record.rb
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								backend/app/models/application_record.rb
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,3 @@ | ||||
| class ApplicationRecord < ActiveRecord::Base | ||||
|   primary_abstract_class | ||||
| end | ||||
							
								
								
									
										0
									
								
								backend/app/models/concerns/.keep
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										0
									
								
								backend/app/models/concerns/.keep
									
									
									
									
									
										Normal file
									
								
							
							
								
								
									
										7
									
								
								backend/app/models/user.rb
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										7
									
								
								backend/app/models/user.rb
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,7 @@ | ||||
| class User < ApplicationRecord | ||||
|     has_secure_password | ||||
|     validates :email, presence: true, uniqueness: true | ||||
|     validates :email, format: { with: URI::MailTo::EMAIL_REGEXP } | ||||
|     validates :username, presence: true | ||||
|     validates :password, presence: true | ||||
| end | ||||
							
								
								
									
										13
									
								
								backend/app/views/layouts/mailer.html.erb
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										13
									
								
								backend/app/views/layouts/mailer.html.erb
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,13 @@ | ||||
| <!DOCTYPE html> | ||||
| <html> | ||||
|   <head> | ||||
|     <meta http-equiv="Content-Type" content="text/html; charset=utf-8" /> | ||||
|     <style> | ||||
|       /* Email styles need to be inline */ | ||||
|     </style> | ||||
|   </head> | ||||
|  | ||||
|   <body> | ||||
|     <%= yield %> | ||||
|   </body> | ||||
| </html> | ||||
							
								
								
									
										1
									
								
								backend/app/views/layouts/mailer.text.erb
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								backend/app/views/layouts/mailer.text.erb
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1 @@ | ||||
| <%= yield %> | ||||
							
								
								
									
										4
									
								
								backend/bin/rails
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										4
									
								
								backend/bin/rails
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,4 @@ | ||||
| #!/usr/bin/env ruby.exe | ||||
| APP_PATH = File.expand_path("../config/application", __dir__) | ||||
| require_relative "../config/boot" | ||||
| require "rails/commands" | ||||
							
								
								
									
										4
									
								
								backend/bin/rake
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										4
									
								
								backend/bin/rake
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,4 @@ | ||||
| #!/usr/bin/env ruby.exe | ||||
| require_relative "../config/boot" | ||||
| require "rake" | ||||
| Rake.application.run | ||||
							
								
								
									
										33
									
								
								backend/bin/setup
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										33
									
								
								backend/bin/setup
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,33 @@ | ||||
| #!/usr/bin/env ruby.exe | ||||
| require "fileutils" | ||||
|  | ||||
| # path to your application root. | ||||
| APP_ROOT = File.expand_path("..", __dir__) | ||||
|  | ||||
| def system!(*args) | ||||
|   system(*args) || abort("\n== Command #{args} failed ==") | ||||
| end | ||||
|  | ||||
| FileUtils.chdir APP_ROOT do | ||||
|   # This script is a way to set up or update your development environment automatically. | ||||
|   # This script is idempotent, so that you can run it at any time and get an expectable outcome. | ||||
|   # Add necessary setup steps to this file. | ||||
|  | ||||
|   puts "== Installing dependencies ==" | ||||
|   system! "gem install bundler --conservative" | ||||
|   system("bundle check") || system!("bundle install") | ||||
|  | ||||
|   # puts "\n== Copying sample files ==" | ||||
|   # unless File.exist?("config/database.yml") | ||||
|   #   FileUtils.cp "config/database.yml.sample", "config/database.yml" | ||||
|   # end | ||||
|  | ||||
|   puts "\n== Preparing database ==" | ||||
|   system! "bin/rails db:prepare" | ||||
|  | ||||
|   puts "\n== Removing old logs and tempfiles ==" | ||||
|   system! "bin/rails log:clear tmp:clear" | ||||
|  | ||||
|   puts "\n== Restarting application server ==" | ||||
|   system! "bin/rails restart" | ||||
| end | ||||
							
								
								
									
										6
									
								
								backend/config.ru
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										6
									
								
								backend/config.ru
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,6 @@ | ||||
| # This file is used by Rack-based servers to start the application. | ||||
|  | ||||
| require_relative "config/environment" | ||||
|  | ||||
| run Rails.application | ||||
| Rails.application.load_server | ||||
							
								
								
									
										27
									
								
								backend/config/application.rb
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										27
									
								
								backend/config/application.rb
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,27 @@ | ||||
| require_relative "boot" | ||||
|  | ||||
| require "rails/all" | ||||
|  | ||||
| # Require the gems listed in Gemfile, including any gems | ||||
| # you've limited to :test, :development, or :production. | ||||
| Bundler.require(*Rails.groups) | ||||
|  | ||||
| module Backend | ||||
|   class Application < Rails::Application | ||||
|     # Initialize configuration defaults for originally generated Rails version. | ||||
|     config.load_defaults 7.0 | ||||
|  | ||||
|     # Configuration for the application, engines, and railties goes here. | ||||
|     # | ||||
|     # These settings can be overridden in specific environments using the files | ||||
|     # in config/environments, which are processed later. | ||||
|     # | ||||
|     # config.time_zone = "Central Time (US & Canada)" | ||||
|     # config.eager_load_paths << Rails.root.join("extras") | ||||
|  | ||||
|     # Only loads a smaller set of middleware suitable for API only apps. | ||||
|     # Middleware like session, flash, cookies can be added back manually. | ||||
|     # Skip views, helpers and assets when generating a new resource. | ||||
|     config.api_only = true | ||||
|   end | ||||
| end | ||||
							
								
								
									
										4
									
								
								backend/config/boot.rb
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										4
									
								
								backend/config/boot.rb
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,4 @@ | ||||
| ENV["BUNDLE_GEMFILE"] ||= File.expand_path("../Gemfile", __dir__) | ||||
|  | ||||
| require "bundler/setup" # Set up gems listed in the Gemfile. | ||||
| require "bootsnap/setup" # Speed up boot time by caching expensive operations. | ||||
							
								
								
									
										10
									
								
								backend/config/cable.yml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										10
									
								
								backend/config/cable.yml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,10 @@ | ||||
| development: | ||||
|   adapter: async | ||||
|  | ||||
| test: | ||||
|   adapter: test | ||||
|  | ||||
| production: | ||||
|   adapter: redis | ||||
|   url: <%= ENV.fetch("REDIS_URL") { "redis://localhost:6379/1" } %> | ||||
|   channel_prefix: backend_production | ||||
							
								
								
									
										1
									
								
								backend/config/credentials.yml.enc
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								backend/config/credentials.yml.enc
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1 @@ | ||||
| 3BdTdAEBPYdAGuXAy5ivHHsZm3ebVg5K8s3E9OgqMu0v9BKKL8SB0PlTQXzccRCDb3RB+84XWJPK44saZM7KBYUZvHMr9N33++rdgJCv1b7UfavNGrxUy7oK8omDYiJ+X2bTdPLXMKbL4ZmZ0u9+8DyA7moHPpCh8ruA9EPi2siV3QBhyLPpH37HcOGkdln4Sds5C/7PgQlGec+LshA+aPDgRDzgMgQLTGvRJHKGhL2XX8TKF12aRq2H/d/xm46fZQvQJJdDb0ihU5jSMxcjxQ93PhOEyO8SbsScL+VI0J38HwqxESh4C/veZUavURLeuR9jBtM3FyToczE8F4F/7LGFe5IgW5CuI8o9BV709YPV5tFD9xP5Cazmmkw0uQ/I9H3vH4uu0MqsPOfPKcBmY+pRlGdE632zGUfe--j4vVMeAJAjYuZrN7--SUvABbgG70E38hhT537HNA== | ||||
							
								
								
									
										21
									
								
								backend/config/database.yml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										21
									
								
								backend/config/database.yml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,21 @@ | ||||
| default: &default | ||||
|   adapter: postgresql | ||||
|   encoding: unicode | ||||
|   adress: localhost | ||||
|   username: postgres | ||||
|   password: password | ||||
|   port: 5432 | ||||
|   pool: <%= ENV.fetch("RAILS_MAX_THREADS") { 5 } %> | ||||
|   timeout: 5000 | ||||
|  | ||||
| development: | ||||
|   <<: *default | ||||
|   database: development | ||||
|  | ||||
| test: | ||||
|   <<: *default | ||||
|   database: test | ||||
|  | ||||
| production: | ||||
|   <<: *default | ||||
|   database: production | ||||
							
								
								
									
										5
									
								
								backend/config/environment.rb
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								backend/config/environment.rb
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,5 @@ | ||||
| # Load the Rails application. | ||||
| require_relative "application" | ||||
|  | ||||
| # Initialize the Rails application. | ||||
| Rails.application.initialize! | ||||
							
								
								
									
										65
									
								
								backend/config/environments/development.rb
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										65
									
								
								backend/config/environments/development.rb
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,65 @@ | ||||
| require "active_support/core_ext/integer/time" | ||||
|  | ||||
| Rails.application.configure do | ||||
|   # Settings specified here will take precedence over those in config/application.rb. | ||||
|  | ||||
|   # In the development environment your application's code is reloaded any time | ||||
|   # it changes. This slows down response time but is perfect for development | ||||
|   # since you don't have to restart the web server when you make code changes. | ||||
|   config.cache_classes = false | ||||
|  | ||||
|   # Do not eager load code on boot. | ||||
|   config.eager_load = false | ||||
|  | ||||
|   # Show full error reports. | ||||
|   config.consider_all_requests_local = true | ||||
|  | ||||
|   # Enable server timing | ||||
|   config.server_timing = true | ||||
|  | ||||
|   # Enable/disable caching. By default caching is disabled. | ||||
|   # Run rails dev:cache to toggle caching. | ||||
|   if Rails.root.join("tmp/caching-dev.txt").exist? | ||||
|     config.cache_store = :memory_store | ||||
|     config.public_file_server.headers = { | ||||
|       "Cache-Control" => "public, max-age=#{2.days.to_i}" | ||||
|     } | ||||
|   else | ||||
|     config.action_controller.perform_caching = false | ||||
|  | ||||
|     config.cache_store = :null_store | ||||
|   end | ||||
|  | ||||
|   # Store uploaded files on the local file system (see config/storage.yml for options). | ||||
|   config.active_storage.service = :local | ||||
|  | ||||
|   # Don't care if the mailer can't send. | ||||
|   config.action_mailer.raise_delivery_errors = false | ||||
|  | ||||
|   config.action_mailer.perform_caching = false | ||||
|  | ||||
|   # Print deprecation notices to the Rails logger. | ||||
|   config.active_support.deprecation = :log | ||||
|  | ||||
|   # Raise exceptions for disallowed deprecations. | ||||
|   config.active_support.disallowed_deprecation = :raise | ||||
|  | ||||
|   # Tell Active Support which deprecation messages to disallow. | ||||
|   config.active_support.disallowed_deprecation_warnings = [] | ||||
|  | ||||
|   # Raise an error on page load if there are pending migrations. | ||||
|   config.active_record.migration_error = :page_load | ||||
|  | ||||
|   # Highlight code that triggered database queries in logs. | ||||
|   config.active_record.verbose_query_logs = true | ||||
|  | ||||
|  | ||||
|   # Raises error for missing translations. | ||||
|   # config.i18n.raise_on_missing_translations = true | ||||
|  | ||||
|   # Annotate rendered view with file names. | ||||
|   # config.action_view.annotate_rendered_view_with_filenames = true | ||||
|  | ||||
|   # Uncomment if you wish to allow Action Cable access from any origin. | ||||
|   # config.action_cable.disable_request_forgery_protection = true | ||||
| end | ||||
							
								
								
									
										86
									
								
								backend/config/environments/production.rb
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										86
									
								
								backend/config/environments/production.rb
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,86 @@ | ||||
| require "active_support/core_ext/integer/time" | ||||
|  | ||||
| Rails.application.configure do | ||||
|   # Settings specified here will take precedence over those in config/application.rb. | ||||
|  | ||||
|   # Code is not reloaded between requests. | ||||
|   config.cache_classes = true | ||||
|  | ||||
|   # Eager load code on boot. This eager loads most of Rails and | ||||
|   # your application in memory, allowing both threaded web servers | ||||
|   # and those relying on copy on write to perform better. | ||||
|   # Rake tasks automatically ignore this option for performance. | ||||
|   config.eager_load = true | ||||
|  | ||||
|   # Full error reports are disabled and caching is turned on. | ||||
|   config.consider_all_requests_local       = false | ||||
|  | ||||
|   # Ensures that a master key has been made available in either ENV["RAILS_MASTER_KEY"] | ||||
|   # or in config/master.key. This key is used to decrypt credentials (and other encrypted files). | ||||
|   # config.require_master_key = true | ||||
|  | ||||
|   # Disable serving static files from the `/public` folder by default since | ||||
|   # Apache or NGINX already handles this. | ||||
|   config.public_file_server.enabled = ENV["RAILS_SERVE_STATIC_FILES"].present? | ||||
|  | ||||
|   # Enable serving of images, stylesheets, and JavaScripts from an asset server. | ||||
|   # config.asset_host = "http://assets.example.com" | ||||
|  | ||||
|   # Specifies the header that your server uses for sending files. | ||||
|   # config.action_dispatch.x_sendfile_header = "X-Sendfile" # for Apache | ||||
|   # config.action_dispatch.x_sendfile_header = "X-Accel-Redirect" # for NGINX | ||||
|  | ||||
|   # Store uploaded files on the local file system (see config/storage.yml for options). | ||||
|   config.active_storage.service = :local | ||||
|  | ||||
|   # Mount Action Cable outside main process or domain. | ||||
|   # config.action_cable.mount_path = nil | ||||
|   # config.action_cable.url = "wss://example.com/cable" | ||||
|   # config.action_cable.allowed_request_origins = [ "http://example.com", /http:\/\/example.*/ ] | ||||
|  | ||||
|   # Force all access to the app over SSL, use Strict-Transport-Security, and use secure cookies. | ||||
|   # config.force_ssl = true | ||||
|  | ||||
|   # Include generic and useful information about system operation, but avoid logging too much | ||||
|   # information to avoid inadvertent exposure of personally identifiable information (PII). | ||||
|   config.log_level = :info | ||||
|  | ||||
|   # Prepend all log lines with the following tags. | ||||
|   config.log_tags = [ :request_id ] | ||||
|  | ||||
|   # Use a different cache store in production. | ||||
|   # config.cache_store = :mem_cache_store | ||||
|  | ||||
|   # Use a real queuing backend for Active Job (and separate queues per environment). | ||||
|   # config.active_job.queue_adapter     = :resque | ||||
|   # config.active_job.queue_name_prefix = "backend_production" | ||||
|  | ||||
|   config.action_mailer.perform_caching = false | ||||
|  | ||||
|   # Ignore bad email addresses and do not raise email delivery errors. | ||||
|   # Set this to true and configure the email server for immediate delivery to raise delivery errors. | ||||
|   # config.action_mailer.raise_delivery_errors = false | ||||
|  | ||||
|   # Enable locale fallbacks for I18n (makes lookups for any locale fall back to | ||||
|   # the I18n.default_locale when a translation cannot be found). | ||||
|   config.i18n.fallbacks = true | ||||
|  | ||||
|   # Don't log any deprecations. | ||||
|   config.active_support.report_deprecations = false | ||||
|  | ||||
|   # Use default logging formatter so that PID and timestamp are not suppressed. | ||||
|   config.log_formatter = ::Logger::Formatter.new | ||||
|  | ||||
|   # Use a different logger for distributed setups. | ||||
|   # require "syslog/logger" | ||||
|   # config.logger = ActiveSupport::TaggedLogging.new(Syslog::Logger.new "app-name") | ||||
|  | ||||
|   if ENV["RAILS_LOG_TO_STDOUT"].present? | ||||
|     logger           = ActiveSupport::Logger.new(STDOUT) | ||||
|     logger.formatter = config.log_formatter | ||||
|     config.logger    = ActiveSupport::TaggedLogging.new(logger) | ||||
|   end | ||||
|  | ||||
|   # Do not dump schema after migrations. | ||||
|   config.active_record.dump_schema_after_migration = false | ||||
| end | ||||
							
								
								
									
										60
									
								
								backend/config/environments/test.rb
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										60
									
								
								backend/config/environments/test.rb
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,60 @@ | ||||
| require "active_support/core_ext/integer/time" | ||||
|  | ||||
| # The test environment is used exclusively to run your application's | ||||
| # test suite. You never need to work with it otherwise. Remember that | ||||
| # your test database is "scratch space" for the test suite and is wiped | ||||
| # and recreated between test runs. Don't rely on the data there! | ||||
|  | ||||
| Rails.application.configure do | ||||
|   # Settings specified here will take precedence over those in config/application.rb. | ||||
|  | ||||
|   # Turn false under Spring and add config.action_view.cache_template_loading = true. | ||||
|   config.cache_classes = true | ||||
|  | ||||
|   # Eager loading loads your whole application. When running a single test locally, | ||||
|   # this probably isn't necessary. It's a good idea to do in a continuous integration | ||||
|   # system, or in some way before deploying your code. | ||||
|   config.eager_load = ENV["CI"].present? | ||||
|  | ||||
|   # Configure public file server for tests with Cache-Control for performance. | ||||
|   config.public_file_server.enabled = true | ||||
|   config.public_file_server.headers = { | ||||
|     "Cache-Control" => "public, max-age=#{1.hour.to_i}" | ||||
|   } | ||||
|  | ||||
|   # Show full error reports and disable caching. | ||||
|   config.consider_all_requests_local       = true | ||||
|   config.action_controller.perform_caching = false | ||||
|   config.cache_store = :null_store | ||||
|  | ||||
|   # Raise exceptions instead of rendering exception templates. | ||||
|   config.action_dispatch.show_exceptions = false | ||||
|  | ||||
|   # Disable request forgery protection in test environment. | ||||
|   config.action_controller.allow_forgery_protection = false | ||||
|  | ||||
|   # Store uploaded files on the local file system in a temporary directory. | ||||
|   config.active_storage.service = :test | ||||
|  | ||||
|   config.action_mailer.perform_caching = false | ||||
|  | ||||
|   # Tell Action Mailer not to deliver emails to the real world. | ||||
|   # The :test delivery method accumulates sent emails in the | ||||
|   # ActionMailer::Base.deliveries array. | ||||
|   config.action_mailer.delivery_method = :test | ||||
|  | ||||
|   # Print deprecation notices to the stderr. | ||||
|   config.active_support.deprecation = :stderr | ||||
|  | ||||
|   # Raise exceptions for disallowed deprecations. | ||||
|   config.active_support.disallowed_deprecation = :raise | ||||
|  | ||||
|   # Tell Active Support which deprecation messages to disallow. | ||||
|   config.active_support.disallowed_deprecation_warnings = [] | ||||
|  | ||||
|   # Raises error for missing translations. | ||||
|   # config.i18n.raise_on_missing_translations = true | ||||
|  | ||||
|   # Annotate rendered view with file names. | ||||
|   # config.action_view.annotate_rendered_view_with_filenames = true | ||||
| end | ||||
							
								
								
									
										16
									
								
								backend/config/initializers/cors.rb
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										16
									
								
								backend/config/initializers/cors.rb
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,16 @@ | ||||
| # Be sure to restart your server when you modify this file. | ||||
|  | ||||
| # Avoid CORS issues when API is called from the frontend app. | ||||
| # Handle Cross-Origin Resource Sharing (CORS) in order to accept cross-origin AJAX requests. | ||||
|  | ||||
| # Read more: https://github.com/cyu/rack-cors | ||||
|  | ||||
| # Rails.application.config.middleware.insert_before 0, Rack::Cors do | ||||
| #   allow do | ||||
| #     origins "example.com" | ||||
| # | ||||
| #     resource "*", | ||||
| #       headers: :any, | ||||
| #       methods: [:get, :post, :put, :patch, :delete, :options, :head] | ||||
| #   end | ||||
| # end | ||||
							
								
								
									
										8
									
								
								backend/config/initializers/filter_parameter_logging.rb
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										8
									
								
								backend/config/initializers/filter_parameter_logging.rb
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,8 @@ | ||||
| # Be sure to restart your server when you modify this file. | ||||
|  | ||||
| # Configure parameters to be filtered from the log file. Use this to limit dissemination of | ||||
| # sensitive information. See the ActiveSupport::ParameterFilter documentation for supported | ||||
| # notations and behaviors. | ||||
| Rails.application.config.filter_parameters += [ | ||||
|   :passw, :secret, :token, :_key, :crypt, :salt, :certificate, :otp, :ssn | ||||
| ] | ||||
							
								
								
									
										16
									
								
								backend/config/initializers/inflections.rb
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										16
									
								
								backend/config/initializers/inflections.rb
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,16 @@ | ||||
| # Be sure to restart your server when you modify this file. | ||||
|  | ||||
| # Add new inflection rules using the following format. Inflections | ||||
| # are locale specific, and you may define rules for as many different | ||||
| # locales as you wish. All of these examples are active by default: | ||||
| # ActiveSupport::Inflector.inflections(:en) do |inflect| | ||||
| #   inflect.plural /^(ox)$/i, "\\1en" | ||||
| #   inflect.singular /^(ox)en/i, "\\1" | ||||
| #   inflect.irregular "person", "people" | ||||
| #   inflect.uncountable %w( fish sheep ) | ||||
| # end | ||||
|  | ||||
| # These inflection rules are supported but not enabled by default: | ||||
| # ActiveSupport::Inflector.inflections(:en) do |inflect| | ||||
| #   inflect.acronym "RESTful" | ||||
| # end | ||||
							
								
								
									
										33
									
								
								backend/config/locales/en.yml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										33
									
								
								backend/config/locales/en.yml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,33 @@ | ||||
| # Files in the config/locales directory are used for internationalization | ||||
| # and are automatically loaded by Rails. If you want to use locales other | ||||
| # than English, add the necessary files in this directory. | ||||
| # | ||||
| # To use the locales, use `I18n.t`: | ||||
| # | ||||
| #     I18n.t "hello" | ||||
| # | ||||
| # In views, this is aliased to just `t`: | ||||
| # | ||||
| #     <%= t("hello") %> | ||||
| # | ||||
| # To use a different locale, set it with `I18n.locale`: | ||||
| # | ||||
| #     I18n.locale = :es | ||||
| # | ||||
| # This would use the information in config/locales/es.yml. | ||||
| # | ||||
| # The following keys must be escaped otherwise they will not be retrieved by | ||||
| # the default I18n backend: | ||||
| # | ||||
| # true, false, on, off, yes, no | ||||
| # | ||||
| # Instead, surround them with single quotes. | ||||
| # | ||||
| # en: | ||||
| #   "true": "foo" | ||||
| # | ||||
| # To learn more, please read the Rails Internationalization guide | ||||
| # available at https://guides.rubyonrails.org/i18n.html. | ||||
|  | ||||
| en: | ||||
|   hello: "Hello world" | ||||
							
								
								
									
										43
									
								
								backend/config/puma.rb
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										43
									
								
								backend/config/puma.rb
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,43 @@ | ||||
| # Puma can serve each request in a thread from an internal thread pool. | ||||
| # The `threads` method setting takes two numbers: a minimum and maximum. | ||||
| # Any libraries that use thread pools should be configured to match | ||||
| # the maximum value specified for Puma. Default is set to 5 threads for minimum | ||||
| # and maximum; this matches the default thread size of Active Record. | ||||
| # | ||||
| max_threads_count = ENV.fetch("RAILS_MAX_THREADS") { 5 } | ||||
| min_threads_count = ENV.fetch("RAILS_MIN_THREADS") { max_threads_count } | ||||
| threads min_threads_count, max_threads_count | ||||
|  | ||||
| # Specifies the `worker_timeout` threshold that Puma will use to wait before | ||||
| # terminating a worker in development environments. | ||||
| # | ||||
| worker_timeout 3600 if ENV.fetch("RAILS_ENV", "development") == "development" | ||||
|  | ||||
| # Specifies the `port` that Puma will listen on to receive requests; default is 3000. | ||||
| # | ||||
| port ENV.fetch("PORT") { 3000 } | ||||
|  | ||||
| # Specifies the `environment` that Puma will run in. | ||||
| # | ||||
| environment ENV.fetch("RAILS_ENV") { "development" } | ||||
|  | ||||
| # Specifies the `pidfile` that Puma will use. | ||||
| pidfile ENV.fetch("PIDFILE") { "tmp/pids/server.pid" } | ||||
|  | ||||
| # Specifies the number of `workers` to boot in clustered mode. | ||||
| # Workers are forked web server processes. If using threads and workers together | ||||
| # the concurrency of the application would be max `threads` * `workers`. | ||||
| # Workers do not work on JRuby or Windows (both of which do not support | ||||
| # processes). | ||||
| # | ||||
| # workers ENV.fetch("WEB_CONCURRENCY") { 2 } | ||||
|  | ||||
| # Use the `preload_app!` method when specifying a `workers` number. | ||||
| # This directive tells Puma to first boot the application and load code | ||||
| # before forking the application. This takes advantage of Copy On Write | ||||
| # process behavior so workers use less memory. | ||||
| # | ||||
| # preload_app! | ||||
|  | ||||
| # Allow puma to be restarted by `bin/rails restart` command. | ||||
| plugin :tmp_restart | ||||
							
								
								
									
										9
									
								
								backend/config/routes.rb
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										9
									
								
								backend/config/routes.rb
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,9 @@ | ||||
| Rails.application.routes.draw do | ||||
|   # Define your application routes per the DSL in https://guides.rubyonrails.org/routing.html | ||||
|  | ||||
|   # Defines the root path route ("/") | ||||
|   # root "articles#index" | ||||
|  | ||||
|   post '/auth/login', to: 'auth/login#post' | ||||
|   post '/auth/register', to: 'auth/register#post' | ||||
| end | ||||
							
								
								
									
										34
									
								
								backend/config/storage.yml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										34
									
								
								backend/config/storage.yml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,34 @@ | ||||
| test: | ||||
|   service: Disk | ||||
|   root: <%= Rails.root.join("tmp/storage") %> | ||||
|  | ||||
| local: | ||||
|   service: Disk | ||||
|   root: <%= Rails.root.join("storage") %> | ||||
|  | ||||
| # Use bin/rails credentials:edit to set the AWS secrets (as aws:access_key_id|secret_access_key) | ||||
| # amazon: | ||||
| #   service: S3 | ||||
| #   access_key_id: <%= Rails.application.credentials.dig(:aws, :access_key_id) %> | ||||
| #   secret_access_key: <%= Rails.application.credentials.dig(:aws, :secret_access_key) %> | ||||
| #   region: us-east-1 | ||||
| #   bucket: your_own_bucket-<%= Rails.env %> | ||||
|  | ||||
| # Remember not to checkin your GCS keyfile to a repository | ||||
| # google: | ||||
| #   service: GCS | ||||
| #   project: your_project | ||||
| #   credentials: <%= Rails.root.join("path/to/gcs.keyfile") %> | ||||
| #   bucket: your_own_bucket-<%= Rails.env %> | ||||
|  | ||||
| # Use bin/rails credentials:edit to set the Azure Storage secret (as azure_storage:storage_access_key) | ||||
| # microsoft: | ||||
| #   service: AzureStorage | ||||
| #   storage_account_name: your_account_name | ||||
| #   storage_access_key: <%= Rails.application.credentials.dig(:azure_storage, :storage_access_key) %> | ||||
| #   container: your_container_name-<%= Rails.env %> | ||||
|  | ||||
| # mirror: | ||||
| #   service: Mirror | ||||
| #   primary: local | ||||
| #   mirrors: [ amazon, google, microsoft ] | ||||
							
								
								
									
										11
									
								
								backend/db/migrate/20221114171435_create_users.rb
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										11
									
								
								backend/db/migrate/20221114171435_create_users.rb
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,11 @@ | ||||
| class CreateUsers < ActiveRecord::Migration[7.0] | ||||
|   def change | ||||
|     create_table :users do |t| | ||||
|       t.string :username | ||||
|       t.string :email | ||||
|       t.string :password_digest | ||||
|  | ||||
|       t.timestamps | ||||
|     end | ||||
|   end | ||||
| end | ||||
							
								
								
									
										25
									
								
								backend/db/schema.rb
									
									
									
										generated
									
									
									
										Normal file
									
								
							
							
						
						
									
										25
									
								
								backend/db/schema.rb
									
									
									
										generated
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,25 @@ | ||||
| # This file is auto-generated from the current state of the database. Instead | ||||
| # of editing this file, please use the migrations feature of Active Record to | ||||
| # incrementally modify your database, and then regenerate this schema definition. | ||||
| # | ||||
| # This file is the source Rails uses to define your schema when running `bin/rails | ||||
| # db:schema:load`. When creating a new database, `bin/rails db:schema:load` tends to | ||||
| # be faster and is potentially less error prone than running all of your | ||||
| # migrations from scratch. Old migrations may fail to apply correctly if those | ||||
| # migrations use external dependencies or application code. | ||||
| # | ||||
| # It's strongly recommended that you check this file into your version control system. | ||||
|  | ||||
| ActiveRecord::Schema[7.0].define(version: 2022_11_14_171435) do | ||||
|   # These are extensions that must be enabled in order to support this database | ||||
|   enable_extension "plpgsql" | ||||
|  | ||||
|   create_table "users", force: :cascade do |t| | ||||
|     t.string "username" | ||||
|     t.string "email" | ||||
|     t.string "password_digest" | ||||
|     t.datetime "created_at", null: false | ||||
|     t.datetime "updated_at", null: false | ||||
|   end | ||||
|  | ||||
| end | ||||
							
								
								
									
										7
									
								
								backend/db/seeds.rb
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										7
									
								
								backend/db/seeds.rb
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,7 @@ | ||||
| # This file should contain all the record creation needed to seed the database with its default values. | ||||
| # The data can then be loaded with the bin/rails db:seed command (or created alongside the database with db:setup). | ||||
| # | ||||
| # Examples: | ||||
| # | ||||
| #   movies = Movie.create([{ name: "Star Wars" }, { name: "Lord of the Rings" }]) | ||||
| #   Character.create(name: "Luke", movie: movies.first) | ||||
							
								
								
									
										0
									
								
								backend/lib/tasks/.keep
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										0
									
								
								backend/lib/tasks/.keep
									
									
									
									
									
										Normal file
									
								
							
							
								
								
									
										0
									
								
								backend/log/.keep
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										0
									
								
								backend/log/.keep
									
									
									
									
									
										Normal file
									
								
							
							
								
								
									
										1
									
								
								backend/public/robots.txt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								backend/public/robots.txt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1 @@ | ||||
| # See https://www.robotstxt.org/robotstxt.html for documentation on how to use the robots.txt file | ||||
							
								
								
									
										18
									
								
								backend/sample_requests.rest
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								backend/sample_requests.rest
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,18 @@ | ||||
| POST http://localhost:3000/auth/register | ||||
| content-type: application/json | ||||
|  | ||||
| { | ||||
|     "username": "nikki", | ||||
|     "email": "nikki@gmail.com", | ||||
|     "password": "password", | ||||
|     "password_confirmation": "password" | ||||
| } | ||||
|  | ||||
| ### | ||||
| POST http://localhost:3000/auth/login | ||||
| content-type: application/json | ||||
|  | ||||
| { | ||||
|     "email": "nikki@gmail.com", | ||||
|     "password": "password" | ||||
| } | ||||
							
								
								
									
										0
									
								
								backend/storage/.keep
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										0
									
								
								backend/storage/.keep
									
									
									
									
									
										Normal file
									
								
							
							
								
								
									
										11
									
								
								backend/test/channels/application_cable/connection_test.rb
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										11
									
								
								backend/test/channels/application_cable/connection_test.rb
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,11 @@ | ||||
| require "test_helper" | ||||
|  | ||||
| class ApplicationCable::ConnectionTest < ActionCable::Connection::TestCase | ||||
|   # test "connects with cookies" do | ||||
|   #   cookies.signed[:user_id] = 42 | ||||
|   # | ||||
|   #   connect | ||||
|   # | ||||
|   #   assert_equal connection.user_id, "42" | ||||
|   # end | ||||
| end | ||||
							
								
								
									
										0
									
								
								backend/test/controllers/.keep
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										0
									
								
								backend/test/controllers/.keep
									
									
									
									
									
										Normal file
									
								
							
							
								
								
									
										0
									
								
								backend/test/fixtures/files/.keep
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										0
									
								
								backend/test/fixtures/files/.keep
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
								
								
									
										11
									
								
								backend/test/fixtures/users.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										11
									
								
								backend/test/fixtures/users.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,11 @@ | ||||
| # Read about fixtures at https://api.rubyonrails.org/classes/ActiveRecord/FixtureSet.html | ||||
|  | ||||
| one: | ||||
|   username: MyString | ||||
|   email: MyString | ||||
|   password_digest: MyString | ||||
|  | ||||
| two: | ||||
|   username: MyString | ||||
|   email: MyString | ||||
|   password_digest: MyString | ||||
							
								
								
									
										0
									
								
								backend/test/integration/.keep
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										0
									
								
								backend/test/integration/.keep
									
									
									
									
									
										Normal file
									
								
							
							
								
								
									
										0
									
								
								backend/test/mailers/.keep
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										0
									
								
								backend/test/mailers/.keep
									
									
									
									
									
										Normal file
									
								
							
							
								
								
									
										0
									
								
								backend/test/models/.keep
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										0
									
								
								backend/test/models/.keep
									
									
									
									
									
										Normal file
									
								
							
							
								
								
									
										7
									
								
								backend/test/models/user_test.rb
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										7
									
								
								backend/test/models/user_test.rb
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,7 @@ | ||||
| require "test_helper" | ||||
|  | ||||
| class UserTest < ActiveSupport::TestCase | ||||
|   # test "the truth" do | ||||
|   #   assert true | ||||
|   # end | ||||
| end | ||||
							
								
								
									
										13
									
								
								backend/test/test_helper.rb
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										13
									
								
								backend/test/test_helper.rb
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,13 @@ | ||||
| ENV["RAILS_ENV"] ||= "test" | ||||
| require_relative "../config/environment" | ||||
| require "rails/test_help" | ||||
|  | ||||
| class ActiveSupport::TestCase | ||||
|   # Run tests in parallel with specified workers | ||||
|   parallelize(workers: :number_of_processors, with: :threads) | ||||
|  | ||||
|   # Setup all fixtures in test/fixtures/*.yml for all tests in alphabetical order. | ||||
|   fixtures :all | ||||
|  | ||||
|   # Add more helper methods to be used by all tests here... | ||||
| end | ||||
							
								
								
									
										0
									
								
								backend/tmp/.keep
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										0
									
								
								backend/tmp/.keep
									
									
									
									
									
										Normal file
									
								
							
							
								
								
									
										0
									
								
								backend/tmp/pids/.keep
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										0
									
								
								backend/tmp/pids/.keep
									
									
									
									
									
										Normal file
									
								
							
							
								
								
									
										0
									
								
								backend/tmp/storage/.keep
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										0
									
								
								backend/tmp/storage/.keep
									
									
									
									
									
										Normal file
									
								
							
							
								
								
									
										0
									
								
								backend/vendor/.keep
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										0
									
								
								backend/vendor/.keep
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
								
								
									
										9
									
								
								frontend/.editorconfig
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										9
									
								
								frontend/.editorconfig
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,9 @@ | ||||
| root = true | ||||
|  | ||||
| [*] | ||||
| charset = utf-8 | ||||
| indent_style = space | ||||
| indent_size = 2 | ||||
| end_of_line = lf | ||||
| insert_final_newline = true | ||||
| trim_trailing_whitespace = true | ||||
							
								
								
									
										7
									
								
								frontend/.eslintignore
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										7
									
								
								frontend/.eslintignore
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,7 @@ | ||||
| /dist | ||||
| /src-capacitor | ||||
| /src-cordova | ||||
| /.quasar | ||||
| /node_modules | ||||
| .eslintrc.js | ||||
| /src-ssr | ||||
							
								
								
									
										88
									
								
								frontend/.eslintrc.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										88
									
								
								frontend/.eslintrc.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,88 @@ | ||||
| module.exports = { | ||||
|   // https://eslint.org/docs/user-guide/configuring#configuration-cascading-and-hierarchy | ||||
|   // This option interrupts the configuration hierarchy at this file | ||||
|   // Remove this if you have an higher level ESLint config file (it usually happens into a monorepos) | ||||
|   root: true, | ||||
|  | ||||
|   // https://eslint.vuejs.org/user-guide/#how-to-use-a-custom-parser | ||||
|   // Must use parserOptions instead of "parser" to allow vue-eslint-parser to keep working | ||||
|   // `parser: 'vue-eslint-parser'` is already included with any 'plugin:vue/**' config and should be omitted | ||||
|   parserOptions: { | ||||
|     parser: require.resolve('@typescript-eslint/parser'), | ||||
|     extraFileExtensions: ['.vue'], | ||||
|   }, | ||||
|  | ||||
|   env: { | ||||
|     browser: true, | ||||
|     es2021: true, | ||||
|     node: true, | ||||
|     'vue/setup-compiler-macros': true, | ||||
|   }, | ||||
|  | ||||
|   // Rules order is important, please avoid shuffling them | ||||
|   extends: [ | ||||
|     // Base ESLint recommended rules | ||||
|     // 'eslint:recommended', | ||||
|  | ||||
|     // https://github.com/typescript-eslint/typescript-eslint/tree/master/packages/eslint-plugin#usage | ||||
|     // ESLint typescript rules | ||||
|     'plugin:@typescript-eslint/recommended', | ||||
|  | ||||
|     // Uncomment any of the lines below to choose desired strictness, | ||||
|     // but leave only one uncommented! | ||||
|     // See https://eslint.vuejs.org/rules/#available-rules | ||||
|     // 'plugin:vue/vue3-essential', // Priority A: Essential (Error Prevention) | ||||
|     // 'plugin:vue/vue3-strongly-recommended', // Priority B: Strongly Recommended (Improving Readability) | ||||
|     'plugin:vue/vue3-recommended', // Priority C: Recommended (Minimizing Arbitrary Choices and Cognitive Overhead) | ||||
|  | ||||
|     // https://github.com/prettier/eslint-config-prettier#installation | ||||
|     // usage with Prettier, provided by 'eslint-config-prettier'. | ||||
|     'prettier', | ||||
|   ], | ||||
|  | ||||
|   plugins: [ | ||||
|     // required to apply rules which need type information | ||||
|     '@typescript-eslint', | ||||
|  | ||||
|     // https://eslint.vuejs.org/user-guide/#why-doesn-t-it-work-on-vue-files | ||||
|     // required to lint *.vue files | ||||
|     'vue', | ||||
|  | ||||
|     // https://github.com/typescript-eslint/typescript-eslint/issues/389#issuecomment-509292674 | ||||
|     // Prettier has not been included as plugin to avoid performance impact | ||||
|     // add it as an extension for your IDE | ||||
|   ], | ||||
|  | ||||
|   globals: { | ||||
|     ga: 'readonly', // Google Analytics | ||||
|     cordova: 'readonly', | ||||
|     __statics: 'readonly', | ||||
|     __QUASAR_SSR__: 'readonly', | ||||
|     __QUASAR_SSR_SERVER__: 'readonly', | ||||
|     __QUASAR_SSR_CLIENT__: 'readonly', | ||||
|     __QUASAR_SSR_PWA__: 'readonly', | ||||
|     process: 'readonly', | ||||
|     Capacitor: 'readonly', | ||||
|     chrome: 'readonly', | ||||
|   }, | ||||
|  | ||||
|   // add your custom rules here | ||||
|   rules: { | ||||
|     'prefer-promise-reject-errors': 'off', | ||||
|  | ||||
|     quotes: ['warn', 'single', { avoidEscape: true }], | ||||
|  | ||||
|     // this rule, if on, would require explicit return type on the `render` function | ||||
|     '@typescript-eslint/explicit-function-return-type': 'off', | ||||
|  | ||||
|     // in plain CommonJS modules, you can't use `import foo = require('foo')` to pass this rule, so it has to be disabled | ||||
|     '@typescript-eslint/no-var-requires': 'off', | ||||
|  | ||||
|     // The core 'no-unused-vars' rules (in the eslint:recommended ruleset) | ||||
|     // does not work with type definitions | ||||
|     'no-unused-vars': 'off', | ||||
|  | ||||
|     // allow debugger during development only | ||||
|     'no-debugger': process.env.NODE_ENV === 'production' ? 'error' : 'off', | ||||
|   }, | ||||
| }; | ||||
							
								
								
									
										33
									
								
								frontend/.gitignore
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										33
									
								
								frontend/.gitignore
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,33 @@ | ||||
| .DS_Store | ||||
| .thumbs.db | ||||
| node_modules | ||||
|  | ||||
| # Quasar core related directories | ||||
| .quasar | ||||
| /dist | ||||
|  | ||||
| # Cordova related directories and files | ||||
| /src-cordova/node_modules | ||||
| /src-cordova/platforms | ||||
| /src-cordova/plugins | ||||
| /src-cordova/www | ||||
|  | ||||
| # Capacitor related directories and files | ||||
| /src-capacitor/www | ||||
| /src-capacitor/node_modules | ||||
|  | ||||
| # BEX related directories and files | ||||
| /src-bex/www | ||||
| /src-bex/js/core | ||||
|  | ||||
| # Log files | ||||
| npm-debug.log* | ||||
| yarn-debug.log* | ||||
| yarn-error.log* | ||||
|  | ||||
| # Editor directories and files | ||||
| .idea | ||||
| *.suo | ||||
| *.ntvs* | ||||
| *.njsproj | ||||
| *.sln | ||||
							
								
								
									
										3
									
								
								frontend/.npmrc
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								frontend/.npmrc
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,3 @@ | ||||
| # pnpm-related options | ||||
| shamefully-hoist=true | ||||
| strict-peer-dependencies=false | ||||
							
								
								
									
										4
									
								
								frontend/.prettierrc
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										4
									
								
								frontend/.prettierrc
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,4 @@ | ||||
| { | ||||
|   "singleQuote": true, | ||||
|   "semi": true | ||||
| } | ||||
							
								
								
									
										18
									
								
								frontend/.vscode/extensions.json
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								frontend/.vscode/extensions.json
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,18 @@ | ||||
| { | ||||
|   "recommendations": [ | ||||
|     "csstools.postcss", | ||||
|     "dbaeumer.vscode-eslint", | ||||
|     "esbenp.prettier-vscode", | ||||
|     "editorconfig.editorconfig", | ||||
|     "lokalise.i18n-ally", | ||||
|     "vue.volar", | ||||
|     "antfu.unocss", | ||||
|     "wayou.vscode-todo-highlight" | ||||
|   ], | ||||
|   "unwantedRecommendations": [ | ||||
|     "octref.vetur", | ||||
|     "hookyqr.beautify", | ||||
|     "dbaeumer.jshint", | ||||
|     "ms-vscode.vscode-typescript-tslint-plugin" | ||||
|   ] | ||||
| } | ||||
							
								
								
									
										16
									
								
								frontend/.vscode/settings.json
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										16
									
								
								frontend/.vscode/settings.json
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,16 @@ | ||||
| { | ||||
|   "editor.bracketPairColorization.enabled": true, | ||||
|   "editor.guides.bracketPairs": true, | ||||
|   "editor.formatOnSave": true, | ||||
|   "editor.defaultFormatter": "esbenp.prettier-vscode", | ||||
|   "editor.codeActionsOnSave": [ | ||||
|     "source.fixAll.eslint" | ||||
|   ], | ||||
|   "eslint.validate": [ | ||||
|     "javascript", | ||||
|     "javascriptreact", | ||||
|     "typescript", | ||||
|     "vue" | ||||
|   ], | ||||
|   "typescript.tsdk": "node_modules/typescript/lib" | ||||
| } | ||||
							
								
								
									
										41
									
								
								frontend/README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										41
									
								
								frontend/README.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,41 @@ | ||||
| # Quasar App (quasar-project) | ||||
|  | ||||
| A Quasar Project | ||||
|  | ||||
| ## Install the dependencies | ||||
| ```bash | ||||
| yarn | ||||
| # or | ||||
| npm install | ||||
| ``` | ||||
|  | ||||
| ### Start the app in development mode (hot-code reloading, error reporting, etc.) | ||||
| ```bash | ||||
| quasar dev | ||||
| ``` | ||||
|  | ||||
|  | ||||
| ### Lint the files | ||||
| ```bash | ||||
| yarn lint | ||||
| # or | ||||
| npm run lint | ||||
| ``` | ||||
|  | ||||
|  | ||||
| ### Format the files | ||||
| ```bash | ||||
| yarn format | ||||
| # or | ||||
| npm run format | ||||
| ``` | ||||
|  | ||||
|  | ||||
|  | ||||
| ### Build the app for production | ||||
| ```bash | ||||
| quasar build | ||||
| ``` | ||||
|  | ||||
| ### Customize the configuration | ||||
| See [Configuring quasar.config.js](https://v2.quasar.dev/quasar-cli-vite/quasar-config-js). | ||||
							
								
								
									
										44
									
								
								frontend/index.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										44
									
								
								frontend/index.html
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,44 @@ | ||||
| <!DOCTYPE html> | ||||
| <html> | ||||
|   <head> | ||||
|     <title><%= productName %></title> | ||||
|  | ||||
|     <meta charset="utf-8" /> | ||||
|     <meta name="description" content="<%= productDescription %>" /> | ||||
|     <meta name="format-detection" content="telephone=no" /> | ||||
|     <meta name="msapplication-tap-highlight" content="no" /> | ||||
|     <meta | ||||
|       name="viewport" | ||||
|       content="user-scalable=no, initial-scale=1, maximum-scale=1, minimum-scale=1, width=device-width<% if (ctx.mode.cordova || ctx.mode.capacitor) { %>, viewport-fit=cover<% } %>" | ||||
|     /> | ||||
|  | ||||
|     <link | ||||
|       rel="icon" | ||||
|       type="image/png" | ||||
|       sizes="128x128" | ||||
|       href="icons/favicon-128x128.png" | ||||
|     /> | ||||
|     <link | ||||
|       rel="icon" | ||||
|       type="image/png" | ||||
|       sizes="96x96" | ||||
|       href="icons/favicon-96x96.png" | ||||
|     /> | ||||
|     <link | ||||
|       rel="icon" | ||||
|       type="image/png" | ||||
|       sizes="32x32" | ||||
|       href="icons/favicon-32x32.png" | ||||
|     /> | ||||
|     <link | ||||
|       rel="icon" | ||||
|       type="image/png" | ||||
|       sizes="16x16" | ||||
|       href="icons/favicon-16x16.png" | ||||
|     /> | ||||
|     <link rel="icon" type="image/ico" href="favicon.ico" /> | ||||
|   </head> | ||||
|   <body> | ||||
|     <!-- quasar:entry-point --> | ||||
|   </body> | ||||
| </html> | ||||
							
								
								
									
										47
									
								
								frontend/package.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										47
									
								
								frontend/package.json
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,47 @@ | ||||
| { | ||||
|   "name": "quasar-project", | ||||
|   "version": "0.0.1", | ||||
|   "description": "A Quasar Project", | ||||
|   "productName": "Quasar App", | ||||
|   "author": "Nikki <nikki.arnette1337@gmail.com>", | ||||
|   "private": true, | ||||
|   "scripts": { | ||||
|     "dev": "quasar dev", | ||||
|     "build": "quasar build", | ||||
|     "build:pwa": "quasar build -m pwa", | ||||
|     "lint": "eslint --ext .js,.ts,.vue ./", | ||||
|     "format": "prettier --write \"**/*.{js,ts,vue,scss,html,md,json}\" --ignore-path .gitignore", | ||||
|     "test": "echo \"No test specified\" && exit 0" | ||||
|   }, | ||||
|   "dependencies": { | ||||
|     "@quasar/extras": "^1.15.6", | ||||
|     "@unocss/vite": "^0.47.1", | ||||
|     "axios": "^1.2.0", | ||||
|     "nprogress": "^0.2.0", | ||||
|     "pinia": "^2.0.27", | ||||
|     "quasar": "^2.10.2", | ||||
|     "unocss": "^0.47.1", | ||||
|     "vue": "^3.2.45", | ||||
|     "vue-i18n": "^9.2.2", | ||||
|     "vue-router": "^4.1.6" | ||||
|   }, | ||||
|   "devDependencies": { | ||||
|     "@intlify/vite-plugin-vue-i18n": "^6.0.3", | ||||
|     "@quasar/app-vite": "^1.1.3", | ||||
|     "@types/node": "^18.11.9", | ||||
|     "@types/nprogress": "^0.2.0", | ||||
|     "@typescript-eslint/eslint-plugin": "^5.45.0", | ||||
|     "@typescript-eslint/parser": "^5.45.0", | ||||
|     "autoprefixer": "^10.4.13", | ||||
|     "eslint": "^8.28.0", | ||||
|     "eslint-config-prettier": "^8.5.0", | ||||
|     "eslint-plugin-vue": "^9.8.0", | ||||
|     "prettier": "^2.8.0", | ||||
|     "typescript": "^4.9.3" | ||||
|   }, | ||||
|   "engines": { | ||||
|     "node": "^18 || ^16 || ^14.19", | ||||
|     "npm": ">= 6.13.4", | ||||
|     "yarn": ">= 1.21.1" | ||||
|   } | ||||
| } | ||||
							
								
								
									
										3782
									
								
								frontend/pnpm-lock.yaml
									
									
									
										generated
									
									
									
										Normal file
									
								
							
							
						
						
									
										3782
									
								
								frontend/pnpm-lock.yaml
									
									
									
										generated
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										27
									
								
								frontend/postcss.config.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										27
									
								
								frontend/postcss.config.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,27 @@ | ||||
| /* eslint-disable */ | ||||
| // https://github.com/michael-ciniawsky/postcss-load-config | ||||
|  | ||||
| module.exports = { | ||||
|   plugins: [ | ||||
|     // https://github.com/postcss/autoprefixer | ||||
|     require('autoprefixer')({ | ||||
|       overrideBrowserslist: [ | ||||
|         'last 4 Chrome versions', | ||||
|         'last 4 Firefox versions', | ||||
|         'last 4 Edge versions', | ||||
|         'last 4 Safari versions', | ||||
|         'last 4 Android versions', | ||||
|         'last 4 ChromeAndroid versions', | ||||
|         'last 4 FirefoxAndroid versions', | ||||
|         'last 4 iOS versions' | ||||
|       ] | ||||
|     }) | ||||
|  | ||||
|     // https://github.com/elchininet/postcss-rtlcss | ||||
|     // If you want to support RTL css, then | ||||
|     // 1. yarn/npm install postcss-rtlcss | ||||
|     // 2. optionally set quasar.config.js > framework > lang to an RTL language | ||||
|     // 3. uncomment the following line: | ||||
|     // require('postcss-rtlcss') | ||||
|   ] | ||||
| } | ||||
							
								
								
									
										
											BIN
										
									
								
								frontend/public/favicon.ico
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								frontend/public/favicon.ico
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 63 KiB | 
							
								
								
									
										
											BIN
										
									
								
								frontend/public/icons/favicon-128x128.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								frontend/public/icons/favicon-128x128.png
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 12 KiB | 
							
								
								
									
										
											BIN
										
									
								
								frontend/public/icons/favicon-16x16.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								frontend/public/icons/favicon-16x16.png
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 859 B | 
							
								
								
									
										
											BIN
										
									
								
								frontend/public/icons/favicon-32x32.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								frontend/public/icons/favicon-32x32.png
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 2.0 KiB | 
							
								
								
									
										
											BIN
										
									
								
								frontend/public/icons/favicon-96x96.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								frontend/public/icons/favicon-96x96.png
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 9.4 KiB | 
							
								
								
									
										217
									
								
								frontend/quasar.config.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										217
									
								
								frontend/quasar.config.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,217 @@ | ||||
| /* eslint-env node */ | ||||
|  | ||||
| /* | ||||
|  * This file runs in a Node context (it's NOT transpiled by Babel), so use only | ||||
|  * the ES6 features that are supported by your Node version. https://node.green/ | ||||
|  */ | ||||
|  | ||||
| // Configuration for your app | ||||
| // https://v2.quasar.dev/quasar-cli-vite/quasar-config-js | ||||
|  | ||||
| const { configure } = require('quasar/wrappers'); | ||||
| const path = require('path'); | ||||
| const UnoCSS = require('@unocss/vite').default; | ||||
| const { presetAttributify, presetUno } = require('unocss'); | ||||
|  | ||||
| module.exports = configure(function (/* ctx */) { | ||||
|   return { | ||||
|     eslint: { | ||||
|       // fix: true, | ||||
|       // include = [], | ||||
|       // exclude = [], | ||||
|       // rawOptions = {}, | ||||
|       warnings: true, | ||||
|       errors: true, | ||||
|     }, | ||||
|  | ||||
|     // https://v2.quasar.dev/quasar-cli-vite/prefetch-feature | ||||
|     // preFetch: true, | ||||
|  | ||||
|     // app boot file (/src/boot) | ||||
|     // --> boot files are part of "main.js" | ||||
|     // https://v2.quasar.dev/quasar-cli-vite/boot-files | ||||
|     boot: ['i18n', 'axios', 'eventBus', 'unocss'], | ||||
|  | ||||
|     // https://v2.quasar.dev/quasar-cli-vite/quasar-config-js#css | ||||
|     css: ['app.scss'], | ||||
|  | ||||
|     // https://github.com/quasarframework/quasar/tree/dev/extras | ||||
|     extras: [ | ||||
|       // 'ionicons-v4', | ||||
|       // 'mdi-v5', | ||||
|       // 'fontawesome-v6', | ||||
|       // 'eva-icons', | ||||
|       // 'themify', | ||||
|       // 'line-awesome', | ||||
|       // 'roboto-font-latin-ext', // this or either 'roboto-font', NEVER both! | ||||
|       // 'roboto-font', // optional, you are not bound to it | ||||
|       'material-icons', // optional, you are not bound to it | ||||
|     ], | ||||
|  | ||||
|     // Full list of options: https://v2.quasar.dev/quasar-cli-vite/quasar-config-js#build | ||||
|     build: { | ||||
|       target: { | ||||
|         browser: ['es2019', 'edge88', 'firefox78', 'chrome87', 'safari13.1'], | ||||
|         node: 'node16', | ||||
|       }, | ||||
|  | ||||
|       vueRouterMode: 'hash', // available values: 'hash', 'history' | ||||
|       // vueRouterBase, | ||||
|       // vueDevtools, | ||||
|       // vueOptionsAPI: false, | ||||
|  | ||||
|       // rebuildCache: true, // rebuilds Vite/linter/etc cache on startup | ||||
|  | ||||
|       // publicPath: '/', | ||||
|       // analyze: true, | ||||
|       // env: {}, | ||||
|       // rawDefine: {} | ||||
|       // ignorePublicFolder: true, | ||||
|       // minify: false, | ||||
|       // polyfillModulePreload: true, | ||||
|       // distDir | ||||
|  | ||||
|       extendViteConf(config) { | ||||
|         config.plugins.push( | ||||
|           ...UnoCSS({ | ||||
|             presets: [presetUno(), presetAttributify()], | ||||
|           }) | ||||
|         ); | ||||
|       }, | ||||
|       // viteVuePluginOptions: {}, | ||||
|  | ||||
|       vitePlugins: [ | ||||
|         [ | ||||
|           '@intlify/vite-plugin-vue-i18n', | ||||
|           { | ||||
|             // if you want to use Vue I18n Legacy API, you need to set `compositionOnly: false` | ||||
|             // compositionOnly: false, | ||||
|  | ||||
|             // you need to set i18n resource including paths ! | ||||
|             include: path.resolve(__dirname, './src/i18n/**'), | ||||
|           }, | ||||
|         ], | ||||
|       ], | ||||
|     }, | ||||
|  | ||||
|     // Full list of options: https://v2.quasar.dev/quasar-cli-vite/quasar-config-js#devServer | ||||
|     devServer: { | ||||
|       // https: true | ||||
|       open: true, // opens browser window automatically | ||||
|     }, | ||||
|  | ||||
|     // https://v2.quasar.dev/quasar-cli-vite/quasar-config-js#framework | ||||
|     framework: { | ||||
|       config: {}, | ||||
|  | ||||
|       // iconSet: 'material-icons', // Quasar icon set | ||||
|       // lang: 'en-US', // Quasar language pack | ||||
|  | ||||
|       // For special cases outside of where the auto-import strategy can have an impact | ||||
|       // (like functional components as one of the examples), | ||||
|       // you can manually specify Quasar components/directives to be available everywhere: | ||||
|       // | ||||
|       // components: [], | ||||
|       // directives: [], | ||||
|  | ||||
|       // Quasar plugins | ||||
|       plugins: ['Notify', 'Dialog', 'Dark'], | ||||
|     }, | ||||
|  | ||||
|     // animations: 'all', // --- includes all animations | ||||
|     // https://v2.quasar.dev/options/animations | ||||
|     animations: [], | ||||
|  | ||||
|     // https://v2.quasar.dev/quasar-cli-vite/quasar-config-js#sourcefiles | ||||
|     // sourceFiles: { | ||||
|     //   rootComponent: 'src/App.vue', | ||||
|     //   router: 'src/router/index', | ||||
|     //   store: 'src/store/index', | ||||
|     //   registerServiceWorker: 'src-pwa/register-service-worker', | ||||
|     //   serviceWorker: 'src-pwa/custom-service-worker', | ||||
|     //   pwaManifestFile: 'src-pwa/manifest.json', | ||||
|     //   electronMain: 'src-electron/electron-main', | ||||
|     //   electronPreload: 'src-electron/electron-preload' | ||||
|     // }, | ||||
|  | ||||
|     // https://v2.quasar.dev/quasar-cli-vite/developing-ssr/configuring-ssr | ||||
|     ssr: { | ||||
|       // ssrPwaHtmlFilename: 'offline.html', // do NOT use index.html as name! | ||||
|       // will mess up SSR | ||||
|  | ||||
|       // extendSSRWebserverConf (esbuildConf) {}, | ||||
|       // extendPackageJson (json) {}, | ||||
|  | ||||
|       pwa: false, | ||||
|  | ||||
|       // manualStoreHydration: true, | ||||
|       // manualPostHydrationTrigger: true, | ||||
|  | ||||
|       prodPort: 3000, // The default port that the production server should use | ||||
|       // (gets superseded if process.env.PORT is specified at runtime) | ||||
|  | ||||
|       middlewares: [ | ||||
|         'render', // keep this as last one | ||||
|       ], | ||||
|     }, | ||||
|  | ||||
|     // https://v2.quasar.dev/quasar-cli-vite/developing-pwa/configuring-pwa | ||||
|     pwa: { | ||||
|       workboxMode: 'generateSW', // or 'injectManifest' | ||||
|       injectPwaMetaTags: true, | ||||
|       swFilename: 'sw.js', | ||||
|       manifestFilename: 'manifest.json', | ||||
|       useCredentialsForManifestTag: false, | ||||
|       // useFilenameHashes: true, | ||||
|       // extendGenerateSWOptions (cfg) {} | ||||
|       // extendInjectManifestOptions (cfg) {}, | ||||
|       // extendManifestJson (json) {} | ||||
|       // extendPWACustomSWConf (esbuildConf) {} | ||||
|     }, | ||||
|  | ||||
|     // Full list of options: https://v2.quasar.dev/quasar-cli-vite/developing-cordova-apps/configuring-cordova | ||||
|     cordova: { | ||||
|       // noIosLegacyBuildFlag: true, // uncomment only if you know what you are doing | ||||
|     }, | ||||
|  | ||||
|     // Full list of options: https://v2.quasar.dev/quasar-cli-vite/developing-capacitor-apps/configuring-capacitor | ||||
|     capacitor: { | ||||
|       hideSplashscreen: true, | ||||
|     }, | ||||
|  | ||||
|     // Full list of options: https://v2.quasar.dev/quasar-cli-vite/developing-electron-apps/configuring-electron | ||||
|     electron: { | ||||
|       // extendElectronMainConf (esbuildConf) | ||||
|       // extendElectronPreloadConf (esbuildConf) | ||||
|  | ||||
|       inspectPort: 5858, | ||||
|  | ||||
|       bundler: 'packager', // 'packager' or 'builder' | ||||
|  | ||||
|       packager: { | ||||
|         // https://github.com/electron-userland/electron-packager/blob/master/docs/api.md#options | ||||
|         // OS X / Mac App Store | ||||
|         // appBundleId: '', | ||||
|         // appCategoryType: '', | ||||
|         // osxSign: '', | ||||
|         // protocol: 'myapp://path', | ||||
|         // Windows only | ||||
|         // win32metadata: { ... } | ||||
|       }, | ||||
|  | ||||
|       builder: { | ||||
|         // https://www.electron.build/configuration/configuration | ||||
|  | ||||
|         appId: 'quasar-project', | ||||
|       }, | ||||
|     }, | ||||
|  | ||||
|     // Full list of options: https://v2.quasar.dev/quasar-cli-vite/developing-browser-extensions/configuring-bex | ||||
|     bex: { | ||||
|       contentScripts: ['my-content-script'], | ||||
|  | ||||
|       // extendBexScriptsConf (esbuildConf) {} | ||||
|       // extendBexManifestJson (json) {} | ||||
|     }, | ||||
|   }; | ||||
| }); | ||||
							
								
								
									
										3
									
								
								frontend/quasar.extensions.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								frontend/quasar.extensions.json
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,3 @@ | ||||
| { | ||||
|   "vuelidate-rules": {} | ||||
| } | ||||
							
								
								
									
										11
									
								
								frontend/src/App.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										11
									
								
								frontend/src/App.vue
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,11 @@ | ||||
| <template> | ||||
|   <router-view /> | ||||
| </template> | ||||
|  | ||||
| <script setup lang="ts"> | ||||
| import { useQuasar } from 'quasar'; | ||||
|  | ||||
| const $q = useQuasar(); | ||||
|  | ||||
| $q.dark.set('auto'); | ||||
| </script> | ||||
							
								
								
									
										77
									
								
								frontend/src/api/authApi.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										77
									
								
								frontend/src/api/authApi.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,77 @@ | ||||
| import axios from 'axios'; | ||||
| import type { | ||||
|   GenericResponse, | ||||
|   ILoginInput, | ||||
|   ILoginResponse, | ||||
|   ISignUpInput, | ||||
|   IUserResponse, | ||||
| } from './types'; | ||||
|  | ||||
| const BASE_URL = 'http://localhost:8000/api/'; | ||||
|  | ||||
| export const authApi = axios.create({ | ||||
|   baseURL: BASE_URL, | ||||
|   withCredentials: true, | ||||
| }); | ||||
|  | ||||
| authApi.defaults.headers.common['Content-Type'] = 'application/json'; | ||||
|  | ||||
| export const refreshAccessToken = async () => { | ||||
|   const response = await authApi.get<ILoginResponse>('auth/refresh'); | ||||
|   return response.data; | ||||
| }; | ||||
|  | ||||
| authApi.interceptors.response.use( | ||||
|   (response) => { | ||||
|     return response; | ||||
|   }, | ||||
|   async (error) => { | ||||
|     const originalRequest = error.config; | ||||
|     const errMessage = error.response.data.message as string; | ||||
|     if ( | ||||
|       errMessage.includes('access token expired') && | ||||
|       !originalRequest._retry | ||||
|     ) { | ||||
|       originalRequest._retry = true; | ||||
|       await refreshAccessToken(); | ||||
|       return authApi(originalRequest); | ||||
|     } | ||||
|     if (errMessage.includes('access token invalid')) { | ||||
|       document.location.href = '/login'; | ||||
|     } | ||||
|     return Promise.reject(error); | ||||
|   } | ||||
| ); | ||||
|  | ||||
| export const signUpUser = async (user: ISignUpInput) => { | ||||
|   const response = await authApi.post<GenericResponse>('auth/register', user); | ||||
|   return response.data; | ||||
| }; | ||||
|  | ||||
| export const loginUser = async (user: ILoginInput) => { | ||||
|   const response = await authApi.post<ILoginResponse>('auth/login', user); | ||||
|   localStorage.setItem('accessToken', response.data.access_token); | ||||
|   return response.data; | ||||
| }; | ||||
|  | ||||
| export const verifyEmail = async (verificationCode: string) => { | ||||
|   const response = await authApi.get<GenericResponse>( | ||||
|     `auth/verifyemail/${verificationCode}` | ||||
|   ); | ||||
|   return response.data; | ||||
| }; | ||||
|  | ||||
| export const logoutUser = async () => { | ||||
|   const response = await authApi.get<GenericResponse>('auth/logout', { | ||||
|     headers: getAuthHeader(), | ||||
|   }); | ||||
|   localStorage.removeItem('accessToken'); | ||||
|   return response.data; | ||||
| }; | ||||
|  | ||||
| export const getMe = async () => { | ||||
|   const response = await authApi.get<IUserResponse>('users/me', { | ||||
|     headers: getAuthHeader(), | ||||
|   }); | ||||
|   return response.data; | ||||
| }; | ||||
							
								
								
									
										41
									
								
								frontend/src/api/postsApi.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										41
									
								
								frontend/src/api/postsApi.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,41 @@ | ||||
| import { authApi } from './authApi'; | ||||
| import type { GenericResponse, IPostResponse, IPostsResponse } from './types'; | ||||
|  | ||||
| export const getAllPosts = async () => { | ||||
|   const response = await authApi.get<IPostsResponse>('posts'); | ||||
|   return response.data; | ||||
| }; | ||||
|  | ||||
| export const getPost = async (id: string) => { | ||||
|   const response = await authApi.get<IPostResponse>(`posts/${id}`); | ||||
|   return response.data; | ||||
| }; | ||||
|  | ||||
| export const createPost = async (formData: FormData) => { | ||||
|   const response = await authApi.post<IPostResponse>('posts', formData, { | ||||
|     headers: { | ||||
|       'Content-Type': 'multipart/form-data', | ||||
|     }, | ||||
|   }); | ||||
|   return response.data; | ||||
| }; | ||||
|  | ||||
| export const updatePost = async ({ | ||||
|   id, | ||||
|   formData, | ||||
| }: { | ||||
|   id: string; | ||||
|   formData: FormData; | ||||
| }) => { | ||||
|   const response = await authApi.patch<IPostResponse>(`posts/${id}`, formData, { | ||||
|     headers: { | ||||
|       'Content-Type': 'multipart/form-data', | ||||
|     }, | ||||
|   }); | ||||
|   return response.data; | ||||
| }; | ||||
|  | ||||
| export const deletePost = async (id: string) => { | ||||
|   const response = await authApi.delete<GenericResponse>(`posts/${id}`); | ||||
|   return response.data; | ||||
| }; | ||||
							
								
								
									
										66
									
								
								frontend/src/api/types.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										66
									
								
								frontend/src/api/types.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,66 @@ | ||||
| export interface IUser { | ||||
|   id: string; | ||||
|   name: string; | ||||
|   email: string; | ||||
|   role: string; | ||||
|   photo: string; | ||||
| } | ||||
|  | ||||
| export interface GenericResponse { | ||||
|   status: string; | ||||
|   message: string; | ||||
| } | ||||
|  | ||||
| export interface ILoginInput { | ||||
|   email: string; | ||||
|   password: string; | ||||
| } | ||||
|  | ||||
| export interface ISignUpInput { | ||||
|   name: string; | ||||
|   email: string; | ||||
|   password: string; | ||||
|   password_confirm: string; | ||||
| } | ||||
|  | ||||
| export interface ILoginResponse { | ||||
|   status: string; | ||||
|   access_token: string; | ||||
| } | ||||
|  | ||||
| export interface ISignUpResponse { | ||||
|   status: string; | ||||
|   message: string; | ||||
| } | ||||
|  | ||||
| export interface IUserResponse { | ||||
|   status: string; | ||||
|   data: { | ||||
|     user: IUser; | ||||
|   }; | ||||
| } | ||||
|  | ||||
| export interface IPostRequest { | ||||
|   title: string; | ||||
|   content: string; | ||||
|   image: string; | ||||
|   user: string; | ||||
| } | ||||
|  | ||||
| export interface IPostResponse { | ||||
|   id: string; | ||||
|   title: string; | ||||
|   content: string; | ||||
|   image: string; | ||||
|   category: string; | ||||
|   user: IUser; | ||||
|   created_at: string; | ||||
|   updated_at: string; | ||||
| } | ||||
|  | ||||
| export interface IPostsResponse { | ||||
|   status: string; | ||||
|   data: { | ||||
|     posts: IPostResponse[]; | ||||
|   }; | ||||
| } | ||||
							
								
								
									
										8
									
								
								frontend/src/api/utils/authHeader.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										8
									
								
								frontend/src/api/utils/authHeader.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,8 @@ | ||||
| function getAuthHeader() { | ||||
|   const tokenLocalStorage: string | null = localStorage.getItem('accessToken'); | ||||
|   if (tokenLocalStorage) { | ||||
|     return { Authorization: 'Bearer ' + tokenLocalStorage }; | ||||
|   } else { | ||||
|     return {}; | ||||
|   } | ||||
| } | ||||
							
								
								
									
										15
									
								
								frontend/src/assets/quasar-logo-vertical.svg
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										15
									
								
								frontend/src/assets/quasar-logo-vertical.svg
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,15 @@ | ||||
| <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 356 360"> | ||||
| 	<path | ||||
| 		d="M43.4 303.4c0 3.8-2.3 6.3-7.1 6.3h-15v-22h14.4c4.3 0 6.2 2.2 6.2 5.2 0 2.6-1.5 4.4-3.4 5 2.8.4 4.9 2.5 4.9 5.5zm-8-13H24.1v6.9H35c2.1 0 4-1.3 4-3.8 0-2.2-1.3-3.1-3.7-3.1zm5.1 12.6c0-2.3-1.8-3.7-4-3.7H24.2v7.7h11.7c3.4 0 4.6-1.8 4.6-4zm36.3 4v2.7H56v-22h20.6v2.7H58.9v6.8h14.6v2.3H58.9v7.5h17.9zm23-5.8v8.5H97v-8.5l-11-13.4h3.4l8.9 11 8.8-11h3.4l-10.8 13.4zm19.1-1.8V298c0-7.9 5.2-10.7 12.7-10.7 7.5 0 13 2.8 13 10.7v1.4c0 7.9-5.5 10.8-13 10.8s-12.7-3-12.7-10.8zm22.7 0V298c0-5.7-3.9-8-10-8-6 0-9.8 2.3-9.8 8v1.4c0 5.8 3.8 8.1 9.8 8.1 6 0 10-2.3 10-8.1zm37.2-11.6v21.9h-2.9l-15.8-17.9v17.9h-2.8v-22h3l15.6 18v-18h2.9zm37.9 10.2v1.3c0 7.8-5.2 10.4-12.4 10.4H193v-22h11.2c7.2 0 12.4 2.8 12.4 10.3zm-3 0c0-5.3-3.3-7.6-9.4-7.6h-8.4V307h8.4c6 0 9.5-2 9.5-7.7V298zm50.8-7.6h-9.7v19.3h-3v-19.3h-9.7v-2.6h22.4v2.6zm34.4-2.6v21.9h-3v-10.1h-16.8v10h-2.8v-21.8h2.8v9.2H296v-9.2h2.9zm34.9 19.2v2.7h-20.7v-22h20.6v2.7H316v6.8h14.5v2.3H316v7.5h17.8zM24 340.2v7.3h13.9v2.4h-14v9.6H21v-22h20v2.7H24zm41.5 11.4h-9.8v7.9H53v-22h13.3c5.1 0 8 1.9 8 6.8 0 3.7-2 6.3-5.6 7l6 8.2h-3.3l-5.8-8zm-9.8-2.6H66c3.1 0 5.3-1.5 5.3-4.7 0-3.3-2.2-4.1-5.3-4.1H55.7v8.8zm47.9 6.2H89l-2 4.3h-3.2l10.7-22.2H98l10.7 22.2h-3.2l-2-4.3zm-1-2.3l-6.3-13-6 13h12.2zm46.3-15.3v21.9H146v-17.2L135.7 358h-2.1l-10.2-15.6v17h-2.8v-21.8h3l11 16.9 11.3-17h3zm35 19.3v2.6h-20.7v-22h20.6v2.7H166v6.8h14.5v2.3H166v7.6h17.8zm47-19.3l-8.3 22h-3l-7.1-18.6-7 18.6h-3l-8.2-22h3.3L204 356l6.8-18.5h3.4L221 356l6.6-18.5h3.3zm10 11.6v-1.4c0-7.8 5.2-10.7 12.7-10.7 7.6 0 13 2.9 13 10.7v1.4c0 7.9-5.4 10.8-13 10.8-7.5 0-12.7-3-12.7-10.8zm22.8 0v-1.4c0-5.7-4-8-10-8s-9.9 2.3-9.9 8v1.4c0 5.8 3.8 8.2 9.8 8.2 6.1 0 10-2.4 10-8.2zm28.3 2.4h-9.8v7.9h-2.8v-22h13.2c5.2 0 8 1.9 8 6.8 0 3.7-2 6.3-5.6 7l6 8.2h-3.3l-5.8-8zm-9.8-2.6h10.2c3 0 5.2-1.5 5.2-4.7 0-3.3-2.1-4.1-5.2-4.1h-10.2v8.8zm40.3-1.5l-6.8 5.6v6.4h-2.9v-22h2.9v12.3l15.2-12.2h3.7l-9.9 8.1 10.3 13.8h-3.6l-8.9-12z" /> | ||||
| 	<path fill="#050A14" | ||||
| 		d="M188.4 71.7a10.4 10.4 0 01-20.8 0 10.4 10.4 0 1120.8 0zM224.2 45c-2.2-3.9-5-7.5-8.2-10.7l-12 7c-3.7-3.2-8-5.7-12.6-7.3a49.4 49.4 0 00-9.7 13.9 59 59 0 0140.1 14l7.6-4.4a57 57 0 00-5.2-12.5zM178 125.1c4.5 0 9-.6 13.4-1.7v-14a40 40 0 0012.5-7.2 47.7 47.7 0 00-7.1-15.3 59 59 0 01-32.2 27.7v8.7c4.4 1.2 8.9 1.8 13.4 1.8zM131.8 45c-2.3 4-4 8.1-5.2 12.5l12 7a40 40 0 000 14.4c5.7 1.5 11.3 2 16.9 1.5a59 59 0 01-8-41.7l-7.5-4.3c-3.2 3.2-6 6.7-8.2 10.6z" /> | ||||
| 	<path fill="#00B4FF" | ||||
| 		d="M224.2 98.4c2.3-3.9 4-8 5.2-12.4l-12-7a40 40 0 000-14.5c-5.7-1.5-11.3-2-16.9-1.5a59 59 0 018 41.7l7.5 4.4c3.2-3.2 6-6.8 8.2-10.7zm-92.4 0c2.2 4 5 7.5 8.2 10.7l12-7a40 40 0 0012.6 7.3c4-4.1 7.3-8.8 9.7-13.8a59 59 0 01-40-14l-7.7 4.4c1.2 4.3 3 8.5 5.2 12.4zm46.2-80c-4.5 0-9 .5-13.4 1.7V34a40 40 0 00-12.5 7.2c1.5 5.7 4 10.8 7.1 15.4a59 59 0 0132.2-27.7V20a53.3 53.3 0 00-13.4-1.8z" /> | ||||
| 	<path fill="#00B4FF" | ||||
| 		d="M178 9.2a62.6 62.6 0 11-.1 125.2A62.6 62.6 0 01178 9.2m0-9.2a71.7 71.7 0 100 143.5A71.7 71.7 0 00178 0z" /> | ||||
| 	<path fill="#050A14" | ||||
| 		d="M96.6 212v4.3c-9.2-.8-15.4-5.8-15.4-17.8V180h4.6v18.4c0 8.6 4 12.6 10.8 13.5zm16-31.9v18.4c0 8.9-4.3 12.8-10.9 13.5v4.4c9.2-.7 15.5-5.6 15.5-18v-18.3h-4.7zM62.2 199v-2.2c0-12.7-8.8-17.4-21-17.4-12.1 0-20.7 4.7-20.7 17.4v2.2c0 12.8 8.6 17.6 20.7 17.6 1.5 0 3-.1 4.4-.3l11.8 6.2 2-3.3-8.2-4-6.4-3.1a32 32 0 01-3.6.2c-9.8 0-16-3.9-16-13.3v-2.2c0-9.3 6.2-13.1 16-13.1 9.9 0 16.3 3.8 16.3 13.1v2.2c0 5.3-2.1 8.7-5.6 10.8l4.8 2.4c3.4-2.8 5.5-7 5.5-13.2zM168 215.6h5.1L156 179.7h-4.8l17 36zM143 205l7.4-15.7-2.4-5-15.1 31.4h5.1l3.3-7h18.3l-1.8-3.7H143zm133.7 10.7h5.2l-17.3-35.9h-4.8l17 36zm-25-10.7l7.4-15.7-2.4-5-15.1 31.4h5.1l3.3-7h18.3l-1.7-3.7h-14.8zm73.8-2.5c6-1.2 9-5.4 9-11.4 0-8-4.5-10.9-12.9-10.9h-21.4v35.5h4.6v-31.3h16.5c5 0 8.5 1.4 8.5 6.7 0 5.2-3.5 7.7-8.5 7.7h-11.4v4.1h10.7l9.3 12.8h5.5l-9.9-13.2zm-117.4 9.9c-9.7 0-14.7-2.5-18.6-6.3l-2.2 3.8c5.1 5 11 6.7 21 6.7 1.6 0 3.1-.1 4.6-.3l-1.9-4h-3zm18.4-7c0-6.4-4.7-8.6-13.8-9.4l-10.1-1c-6.7-.7-9.3-2.2-9.3-5.6 0-2.5 1.4-4 4.6-5l-1.8-3.8c-4.7 1.4-7.5 4.2-7.5 8.9 0 5.2 3.4 8.7 13 9.6l11.3 1.2c6.4.6 8.9 2 8.9 5.4 0 2.7-2.1 4.7-6 5.8l1.8 3.9c5.3-1.6 8.9-4.7 8.9-10zm-20.3-21.9c7.9 0 13.3 1.8 18.1 5.7l1.8-3.9a30 30 0 00-19.6-5.9c-2 0-4 .1-5.7.3l1.9 4 3.5-.2z" /> | ||||
| 	<path fill="#00B4FF" | ||||
| 		d="M.5 251.9c29.6-.5 59.2-.8 88.8-1l88.7-.3 88.7.3 44.4.4 44.4.6-44.4.6-44.4.4-88.7.3-88.7-.3a7981 7981 0 01-88.8-1z" /> | ||||
| 	<path fill="none" d="M-565.2 324H-252v15.8h-313.2z" /> | ||||
| </svg> | ||||
| After Width: | Height: | Size: 4.4 KiB | 
							
								
								
									
										0
									
								
								frontend/src/boot/.gitkeep
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										0
									
								
								frontend/src/boot/.gitkeep
									
									
									
									
									
										Normal file
									
								
							
							
								
								
									
										30
									
								
								frontend/src/boot/axios.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										30
									
								
								frontend/src/boot/axios.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,30 @@ | ||||
| import { boot } from 'quasar/wrappers'; | ||||
| import axios, { AxiosInstance } from 'axios'; | ||||
|  | ||||
| declare module '@vue/runtime-core' { | ||||
|   interface ComponentCustomProperties { | ||||
|     $axios: AxiosInstance; | ||||
|   } | ||||
| } | ||||
|  | ||||
| // Be careful when using SSR for cross-request state pollution | ||||
| // due to creating a Singleton instance here; | ||||
| // If any client changes this (global) instance, it might be a | ||||
| // good idea to move this instance creation inside of the | ||||
| // "export default () => {}" function below (which runs individually | ||||
| // for each client) | ||||
| const api = axios.create({ baseURL: 'http://localhost:3000/api' }); | ||||
|  | ||||
| export default boot(({ app }) => { | ||||
|   // for use inside Vue files (Options API) through this.$axios and this.$api | ||||
|  | ||||
|   app.config.globalProperties.$axios = axios; | ||||
|   // ^ ^ ^ this will allow you to use this.$axios (for Vue Options API form) | ||||
|   //       so you won't necessarily have to import axios in each vue file | ||||
|  | ||||
|   app.config.globalProperties.$api = api; | ||||
|   // ^ ^ ^ this will allow you to use this.$api (for Vue Options API form) | ||||
|   //       so you can easily perform requests against your app's API | ||||
| }); | ||||
|  | ||||
| export { api }; | ||||
							
								
								
									
										12
									
								
								frontend/src/boot/eventBus.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										12
									
								
								frontend/src/boot/eventBus.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,12 @@ | ||||
| import { EventBus } from 'quasar'; | ||||
| import { boot } from 'quasar/wrappers'; | ||||
|  | ||||
| export default boot(({ app }) => { | ||||
|   const bus = new EventBus(); | ||||
|  | ||||
|   // for Options API | ||||
|   app.config.globalProperties.$bus = bus; | ||||
|  | ||||
|   // for Composition API | ||||
|   app.provide('bus', bus); | ||||
| }); | ||||
							
								
								
									
										33
									
								
								frontend/src/boot/i18n.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										33
									
								
								frontend/src/boot/i18n.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,33 @@ | ||||
| import { boot } from 'quasar/wrappers'; | ||||
| import { createI18n } from 'vue-i18n'; | ||||
|  | ||||
| import messages from 'src/i18n'; | ||||
|  | ||||
| export type MessageLanguages = keyof typeof messages; | ||||
| // Type-define 'en-US' as the master schema for the resource | ||||
| export type MessageSchema = typeof messages['en-US']; | ||||
|  | ||||
| // See https://vue-i18n.intlify.dev/guide/advanced/typescript.html#global-resource-schema-type-definition | ||||
| /* eslint-disable @typescript-eslint/no-empty-interface */ | ||||
| declare module 'vue-i18n' { | ||||
|   // define the locale messages schema | ||||
|   export interface DefineLocaleMessage extends MessageSchema {} | ||||
|  | ||||
|   // define the datetime format schema | ||||
|   export interface DefineDateTimeFormat {} | ||||
|  | ||||
|   // define the number format schema | ||||
|   export interface DefineNumberFormat {} | ||||
| } | ||||
| /* eslint-enable @typescript-eslint/no-empty-interface */ | ||||
|  | ||||
| export default boot(({ app }) => { | ||||
|   const i18n = createI18n({ | ||||
|     locale: 'en-US', | ||||
|     legacy: false, | ||||
|     messages, | ||||
|   }); | ||||
|  | ||||
|   // Set i18n instance on app | ||||
|   app.use(i18n); | ||||
| }); | ||||
							
								
								
									
										1
									
								
								frontend/src/boot/unocss.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								frontend/src/boot/unocss.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1 @@ | ||||
| import 'uno.css'; | ||||
							
								
								
									
										34
									
								
								frontend/src/components/authModal/AuthModal.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										34
									
								
								frontend/src/components/authModal/AuthModal.vue
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,34 @@ | ||||
| <template> | ||||
|   <q-card w-450px> | ||||
|     <q-tabs | ||||
|       v-model="selectedTab" | ||||
|       dense | ||||
|       active-color="primary" | ||||
|       indicator-color="primary" | ||||
|       align="justify" | ||||
|     > | ||||
|       <q-tab name="login" label="Войти" p-5 /> | ||||
|       <q-tab name="register" label="Зарегистрироваться" p-5 /> | ||||
|       <q-btn v-close-popup icon="close" flat round dense m-5 /> | ||||
|     </q-tabs> | ||||
|     <q-separator /> | ||||
|     <q-tab-panels v-model="selectedTab" animated> | ||||
|       <q-tab-panel name="login"> | ||||
|         <LoginForm></LoginForm> | ||||
|       </q-tab-panel> | ||||
|       <q-tab-panel name="register"> | ||||
|         <RegisterForm></RegisterForm> | ||||
|       </q-tab-panel> | ||||
|     </q-tab-panels> | ||||
|   </q-card> | ||||
| </template> | ||||
|  | ||||
| <script setup lang="ts"> | ||||
| import { ref } from 'vue'; | ||||
| import LoginForm from './forms/LoginForm.vue'; | ||||
| import RegisterForm from './forms/RegisterForm.vue'; | ||||
|  | ||||
| const selectedTab = ref('login'); | ||||
| </script> | ||||
|  | ||||
| <style lang="scss" scoped></style> | ||||
							
								
								
									
										52
									
								
								frontend/src/components/authModal/forms/LoginForm.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										52
									
								
								frontend/src/components/authModal/forms/LoginForm.vue
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,52 @@ | ||||
| <template> | ||||
|   <q-form class="q-gutter-md" @submit="onSubmit"> | ||||
|     <q-input | ||||
|       v-model="state.email" | ||||
|       filled | ||||
|       label="Электронная почта" | ||||
|       lazy-rules | ||||
|       :rules="[ | ||||
|         (val) => required.$validator(val) || 'Обязательное поле', | ||||
|         (val) => | ||||
|           email.$validator(val) || 'Некорректный формат электронной почты', | ||||
|       ]" | ||||
|     /> | ||||
|     <q-input | ||||
|       v-model="state.password" | ||||
|       filled | ||||
|       label="Пароль" | ||||
|       lazy-rules | ||||
|       :rules="[(val) => required.$validator(val) || 'Обязательное поле']" | ||||
|     /> | ||||
|     <q-separator /> | ||||
|     <q-btn label="Войти" type="submit" color="primary" /> | ||||
|     <q-checkbox v-model="checkbox" right-label label="Запомнить меня" /> | ||||
|   </q-form> | ||||
| </template> | ||||
|  | ||||
| <script setup lang="ts"> | ||||
| import { email, required } from '@vuelidate/validators'; | ||||
| import { EventBus, useQuasar } from 'quasar'; | ||||
| import { inject, reactive, ref } from 'vue'; | ||||
|  | ||||
| const $q = useQuasar(); | ||||
|  | ||||
| const checkbox = ref(false); | ||||
|  | ||||
| const state = reactive({ | ||||
|   email: <string>'', | ||||
|   password: <string>'', | ||||
| }); | ||||
|  | ||||
| const bus: EventBus | undefined = inject('bus'); | ||||
|  | ||||
| const onSubmit = () => { | ||||
|   bus?.emit('close-auth-modal'); | ||||
|   $q.notify({ | ||||
|     color: 'green-4', | ||||
|     textColor: 'white', | ||||
|     icon: 'cloud_done', | ||||
|     message: 'Submitted', | ||||
|   }); | ||||
| }; | ||||
| </script> | ||||
							
								
								
									
										79
									
								
								frontend/src/components/authModal/forms/RegisterForm.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										79
									
								
								frontend/src/components/authModal/forms/RegisterForm.vue
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,79 @@ | ||||
| <template> | ||||
|   <q-form class="q-gutter-md" @submit="onSubmit"> | ||||
|     <q-input | ||||
|       v-model="state.name" | ||||
|       filled | ||||
|       label="Имя пользователя" | ||||
|       lazy-rules | ||||
|       :rules="[(val) => required.$validator(val) || 'Обязательное поле']" | ||||
|     /> | ||||
|     <q-input | ||||
|       v-model="state.email" | ||||
|       filled | ||||
|       label="Электронная почта" | ||||
|       lazy-rules | ||||
|       :rules="[ | ||||
|         (val) => required.$validator(val) || 'Обязательное поле', | ||||
|         (val) => | ||||
|           email.$validator(val) || 'Некорректный формат электронной почты', | ||||
|       ]" | ||||
|     /> | ||||
|     <q-input | ||||
|       v-model="state.password" | ||||
|       filled | ||||
|       label="Пароль" | ||||
|       type="password" | ||||
|       :rules="[(val) => required.$validator(val) || 'Обязательное поле']" | ||||
|     /> | ||||
|     <q-input | ||||
|       v-model="state.passwordConfirm" | ||||
|       filled | ||||
|       label="Подтверждение пароля" | ||||
|       type="password" | ||||
|       :rules="[ | ||||
|         (val) => required.$validator(val) || 'Обязательное поле', | ||||
|         (val) => val == state.password || 'Пароли должны совпадать', | ||||
|       ]" | ||||
|     /> | ||||
|     <q-separator /> | ||||
|     <q-btn label="Зарегистрироваться" type="submit" color="primary" /> | ||||
|   </q-form> | ||||
| </template> | ||||
|  | ||||
| <script setup lang="ts"> | ||||
| import { EventBus, useQuasar } from 'quasar'; | ||||
| import { required, email } from '@vuelidate/validators'; | ||||
| import { inject, onMounted, reactive } from 'vue'; | ||||
| import { signUpUser } from 'src/api/authApi'; | ||||
|  | ||||
| const $q = useQuasar(); | ||||
|  | ||||
| const state = reactive({ | ||||
|   name: <string>'', | ||||
|   email: <string>'', | ||||
|   password: <string>'', | ||||
|   password_confirm: <string>'', | ||||
| }); | ||||
|  | ||||
| const bus: EventBus | undefined = inject('bus'); | ||||
|  | ||||
| const onSubmit = async () => { | ||||
|   const result = await signUpUser(state); | ||||
|   bus?.emit('close-auth-modal'); | ||||
|   if (result && result.status == 'ok') { | ||||
|     $q.notify({ | ||||
|       color: 'green-4', | ||||
|       textColor: 'white', | ||||
|       icon: 'cloud_done', | ||||
|       message: result.message, | ||||
|     }); | ||||
|   } else { | ||||
|     $q.notify({ | ||||
|       color: 'red-4', | ||||
|       textColor: 'white', | ||||
|       icon: 'cloud_done', | ||||
|       message: result.status, | ||||
|     }); | ||||
|   } | ||||
| }; | ||||
| </script> | ||||
							
								
								
									
										12
									
								
								frontend/src/css/app.scss
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										12
									
								
								frontend/src/css/app.scss
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,12 @@ | ||||
| // app global css in SCSS form | ||||
| @import './nprogress.scss'; | ||||
| @import url('https://fonts.googleapis.com/css2?family=Roboto&display=swap'); | ||||
|  | ||||
| html, | ||||
| body { | ||||
|   font-family: 'Roboto', sans-serif; | ||||
| } | ||||
|  | ||||
| #app { | ||||
|   font-family: 'Roboto', sans-serif; | ||||
| } | ||||
							
								
								
									
										84
									
								
								frontend/src/css/nprogress.scss
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										84
									
								
								frontend/src/css/nprogress.scss
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,84 @@ | ||||
| /* Make clicks pass-through */ | ||||
| #nprogress { | ||||
|   pointer-events: none; | ||||
| } | ||||
|  | ||||
| #nprogress .bar { | ||||
|   background: $primary; | ||||
|  | ||||
|   position: fixed; | ||||
|   z-index: 1031; | ||||
|   top: 0; | ||||
|   left: 0; | ||||
|  | ||||
|   width: 100%; | ||||
|   height: 2px; | ||||
| } | ||||
|  | ||||
| /* Fancy blur effect */ | ||||
| #nprogress .peg { | ||||
|   display: block; | ||||
|   position: absolute; | ||||
|   right: 0px; | ||||
|   width: 100px; | ||||
|   height: 100%; | ||||
|   box-shadow: 0 0 10px $primary, 0 0 5px $primary; | ||||
|   opacity: 1; | ||||
|  | ||||
|   -webkit-transform: rotate(3deg) translate(0px, -4px); | ||||
|   -ms-transform: rotate(3deg) translate(0px, -4px); | ||||
|   transform: rotate(3deg) translate(0px, -4px); | ||||
| } | ||||
|  | ||||
| /* Remove these to get rid of the spinner */ | ||||
| #nprogress .spinner { | ||||
|   display: block; | ||||
|   position: fixed; | ||||
|   z-index: 1031; | ||||
|   top: 15px; | ||||
|   right: 15px; | ||||
| } | ||||
|  | ||||
| #nprogress .spinner-icon { | ||||
|   width: 18px; | ||||
|   height: 18px; | ||||
|   box-sizing: border-box; | ||||
|  | ||||
|   border: solid 2px transparent; | ||||
|   border-top-color: $primary; | ||||
|   border-left-color: $primary; | ||||
|   border-radius: 50%; | ||||
|  | ||||
|   -webkit-animation: nprogress-spinner 400ms linear infinite; | ||||
|   animation: nprogress-spinner 400ms linear infinite; | ||||
| } | ||||
|  | ||||
| .nprogress-custom-parent { | ||||
|   overflow: hidden; | ||||
|   position: relative; | ||||
| } | ||||
|  | ||||
| .nprogress-custom-parent #nprogress .spinner, | ||||
| .nprogress-custom-parent #nprogress .bar { | ||||
|   position: absolute; | ||||
| } | ||||
|  | ||||
| @-webkit-keyframes nprogress-spinner { | ||||
|   0% { | ||||
|     -webkit-transform: rotate(0deg); | ||||
|   } | ||||
|  | ||||
|   100% { | ||||
|     -webkit-transform: rotate(360deg); | ||||
|   } | ||||
| } | ||||
|  | ||||
| @keyframes nprogress-spinner { | ||||
|   0% { | ||||
|     transform: rotate(0deg); | ||||
|   } | ||||
|  | ||||
|   100% { | ||||
|     transform: rotate(360deg); | ||||
|   } | ||||
| } | ||||
							
								
								
									
										25
									
								
								frontend/src/css/quasar.variables.scss
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										25
									
								
								frontend/src/css/quasar.variables.scss
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,25 @@ | ||||
| // Quasar SCSS (& Sass) Variables | ||||
| // -------------------------------------------------- | ||||
| // To customize the look and feel of this app, you can override | ||||
| // the Sass/SCSS variables found in Quasar's source Sass/SCSS files. | ||||
|  | ||||
| // Check documentation for full list of Quasar variables | ||||
|  | ||||
| // Your own variables (that are declared here) and Quasar's own | ||||
| // ones will be available out of the box in your .vue/.scss/.sass files | ||||
|  | ||||
| // It's highly recommended to change the default colors | ||||
| // to match your app's branding. | ||||
| // Tip: Use the "Theme Builder" on Quasar's documentation website. | ||||
|  | ||||
| $primary   : #1976D2; | ||||
| $secondary : #26A69A; | ||||
| $accent    : #9C27B0; | ||||
|  | ||||
| $dark      : #1D1D1D; | ||||
| $dark-page : #121212; | ||||
|  | ||||
| $positive  : #21BA45; | ||||
| $negative  : #C10015; | ||||
| $info      : #31CCEC; | ||||
| $warning   : #F2C037; | ||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user