├── LICENSE ├── backend ├── Pipfile ├── bootstrap.sh └── src │ ├── __init__.py │ ├── __pycache__ │ ├── __init__.cpython-36.pyc │ └── main.cpython-36.pyc │ ├── entities │ ├── __init__.py │ ├── __pycache__ │ │ ├── __init__.cpython-36.pyc │ │ ├── entity.cpython-36.pyc │ │ └── exam.cpython-36.pyc │ ├── entity.py │ └── exam.py │ └── main.py └── frontend ├── .angular-cli.json ├── .editorconfig ├── .gitignore ├── README.md ├── e2e ├── app.e2e-spec.ts ├── app.po.ts └── tsconfig.e2e.json ├── karma.conf.js ├── package-lock.json ├── package.json ├── protractor.conf.js ├── src ├── app │ ├── app.component.css │ ├── app.component.html │ ├── app.component.spec.ts │ ├── app.component.ts │ ├── app.module.ts │ ├── env.ts │ └── exams │ │ ├── exam.model.ts │ │ └── exams-api.service.ts ├── assets │ └── .gitkeep ├── environments │ ├── environment.prod.ts │ └── environment.ts ├── favicon.ico ├── index.html ├── main.ts ├── polyfills.ts ├── styles.css ├── test.ts ├── tsconfig.app.json ├── tsconfig.spec.json └── typings.d.ts ├── tsconfig.json └── tslint.json /LICENSE: -------------------------------------------------------------------------------- 1 | MIT License 2 | 3 | Copyright (c) 2018 Auth0 Blog Samples 4 | 5 | Permission is hereby granted, free of charge, to any person obtaining a copy 6 | of this software and associated documentation files (the "Software"), to deal 7 | in the Software without restriction, including without limitation the rights 8 | to use, copy, modify, merge, publish, distribute, sublicense, and/or sell 9 | copies of the Software, and to permit persons to whom the Software is 10 | furnished to do so, subject to the following conditions: 11 | 12 | The above copyright notice and this permission notice shall be included in all 13 | copies or substantial portions of the Software. 14 | 15 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR 16 | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, 17 | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE 18 | AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER 19 | LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, 20 | OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE 21 | SOFTWARE. 22 | -------------------------------------------------------------------------------- /backend/Pipfile: -------------------------------------------------------------------------------- 1 | [[source]] 2 | 3 | url = "https://pypi.python.org/simple" 4 | verify_ssl = true 5 | name = "pypi" 6 | 7 | 8 | [packages] 9 | 10 | sqlalchemy = "*" 11 | "psycopg2-binary" = "*" 12 | flask = "*" 13 | marshmallow = "*" 14 | flask-cors = "*" 15 | 16 | 17 | [dev-packages] 18 | 19 | 20 | 21 | [requires] 22 | 23 | python_version = "3.6" 24 | -------------------------------------------------------------------------------- /backend/bootstrap.sh: -------------------------------------------------------------------------------- 1 | #!/bin/bash 2 | export FLASK_APP=./src/main.py 3 | source $(pipenv --venv)/bin/activate 4 | flask run -h 0.0.0.0 5 | -------------------------------------------------------------------------------- /backend/src/__init__.py: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/auth0-blog/python-flask-angular-1/d35d7c76d88268d02ad9be1de2a226f8a1329d01/backend/src/__init__.py -------------------------------------------------------------------------------- /backend/src/__pycache__/__init__.cpython-36.pyc: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/auth0-blog/python-flask-angular-1/d35d7c76d88268d02ad9be1de2a226f8a1329d01/backend/src/__pycache__/__init__.cpython-36.pyc -------------------------------------------------------------------------------- /backend/src/__pycache__/main.cpython-36.pyc: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/auth0-blog/python-flask-angular-1/d35d7c76d88268d02ad9be1de2a226f8a1329d01/backend/src/__pycache__/main.cpython-36.pyc -------------------------------------------------------------------------------- /backend/src/entities/__init__.py: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/auth0-blog/python-flask-angular-1/d35d7c76d88268d02ad9be1de2a226f8a1329d01/backend/src/entities/__init__.py -------------------------------------------------------------------------------- /backend/src/entities/__pycache__/__init__.cpython-36.pyc: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/auth0-blog/python-flask-angular-1/d35d7c76d88268d02ad9be1de2a226f8a1329d01/backend/src/entities/__pycache__/__init__.cpython-36.pyc -------------------------------------------------------------------------------- /backend/src/entities/__pycache__/entity.cpython-36.pyc: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/auth0-blog/python-flask-angular-1/d35d7c76d88268d02ad9be1de2a226f8a1329d01/backend/src/entities/__pycache__/entity.cpython-36.pyc -------------------------------------------------------------------------------- /backend/src/entities/__pycache__/exam.cpython-36.pyc: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/auth0-blog/python-flask-angular-1/d35d7c76d88268d02ad9be1de2a226f8a1329d01/backend/src/entities/__pycache__/exam.cpython-36.pyc -------------------------------------------------------------------------------- /backend/src/entities/entity.py: -------------------------------------------------------------------------------- 1 | # coding=utf-8 2 | 3 | from datetime import datetime 4 | from sqlalchemy import create_engine, Column, String, Integer, DateTime 5 | from sqlalchemy.ext.declarative import declarative_base 6 | from sqlalchemy.orm import sessionmaker 7 | 8 | db_url = 'localhost:5432' 9 | db_name = 'online-exam' 10 | db_user = 'postgres' 11 | db_password = '0NLIN3-ex4m' 12 | engine = create_engine(f'postgresql://{db_user}:{db_password}@{db_url}/{db_name}') 13 | Session = sessionmaker(bind=engine) 14 | 15 | Base = declarative_base() 16 | 17 | 18 | class Entity(): 19 | id = Column(Integer, primary_key=True) 20 | created_at = Column(DateTime) 21 | updated_at = Column(DateTime) 22 | last_updated_by = Column(String) 23 | 24 | def __init__(self, created_by): 25 | self.created_at = datetime.now() 26 | self.updated_at = datetime.now() 27 | self.last_updated_by = created_by 28 | -------------------------------------------------------------------------------- /backend/src/entities/exam.py: -------------------------------------------------------------------------------- 1 | # coding=utf-8 2 | 3 | from marshmallow import Schema, fields 4 | from sqlalchemy import Column, String 5 | 6 | from .entity import Entity, Base 7 | 8 | 9 | class Exam(Entity, Base): 10 | __tablename__ = 'exams' 11 | 12 | title = Column(String) 13 | description = Column(String) 14 | 15 | def __init__(self, title, description, created_by): 16 | Entity.__init__(self, created_by) 17 | self.title = title 18 | self.description = description 19 | 20 | class ExamSchema(Schema): 21 | id = fields.Number() 22 | title = fields.Str() 23 | description = fields.Str() 24 | created_at = fields.DateTime() 25 | updated_at = fields.DateTime() 26 | last_updated_by = fields.Str() 27 | -------------------------------------------------------------------------------- /backend/src/main.py: -------------------------------------------------------------------------------- 1 | # coding=utf-8 2 | 3 | from flask_cors import CORS 4 | from flask import Flask, jsonify, request 5 | 6 | from .entities.entity import Session, engine, Base 7 | from .entities.exam import Exam, ExamSchema 8 | 9 | # creating the Flask application 10 | app = Flask(__name__) 11 | CORS(app) 12 | 13 | # if needed, generate database schema 14 | Base.metadata.create_all(engine) 15 | 16 | 17 | @app.route('/exams') 18 | def get_exams(): 19 | # fetching from the database 20 | session = Session() 21 | exam_objects = session.query(Exam).all() 22 | 23 | # transforming into JSON-serializable objects 24 | schema = ExamSchema(many=True) 25 | exams = schema.dump(exam_objects) 26 | 27 | # serializing as JSON 28 | session.close() 29 | return jsonify(exams.data) 30 | 31 | 32 | @app.route('/exams', methods=['POST']) 33 | def add_exam(): 34 | # mount exam object 35 | posted_exam = ExamSchema(only=('title', 'description'))\ 36 | .load(request.get_json()) 37 | 38 | exam = Exam(**posted_exam.data, created_by="HTTP post request") 39 | 40 | # persist exam 41 | session = Session() 42 | session.add(exam) 43 | session.commit() 44 | 45 | # return created exam 46 | new_exam = ExamSchema().dump(exam).data 47 | session.close() 48 | return jsonify(new_exam), 201 49 | -------------------------------------------------------------------------------- /frontend/.angular-cli.json: -------------------------------------------------------------------------------- 1 | { 2 | "$schema": "./node_modules/@angular/cli/lib/config/schema.json", 3 | "project": { 4 | "name": "frontend" 5 | }, 6 | "apps": [ 7 | { 8 | "root": "src", 9 | "outDir": "dist", 10 | "assets": [ 11 | "assets", 12 | "favicon.ico" 13 | ], 14 | "index": "index.html", 15 | "main": "main.ts", 16 | "polyfills": "polyfills.ts", 17 | "test": "test.ts", 18 | "tsconfig": "tsconfig.app.json", 19 | "testTsconfig": "tsconfig.spec.json", 20 | "prefix": "app", 21 | "styles": [ 22 | "styles.css" 23 | ], 24 | "scripts": [], 25 | "environmentSource": "environments/environment.ts", 26 | "environments": { 27 | "dev": "environments/environment.ts", 28 | "prod": "environments/environment.prod.ts" 29 | } 30 | } 31 | ], 32 | "e2e": { 33 | "protractor": { 34 | "config": "./protractor.conf.js" 35 | } 36 | }, 37 | "lint": [ 38 | { 39 | "project": "src/tsconfig.app.json", 40 | "exclude": "**/node_modules/**" 41 | }, 42 | { 43 | "project": "src/tsconfig.spec.json", 44 | "exclude": "**/node_modules/**" 45 | }, 46 | { 47 | "project": "e2e/tsconfig.e2e.json", 48 | "exclude": "**/node_modules/**" 49 | } 50 | ], 51 | "test": { 52 | "karma": { 53 | "config": "./karma.conf.js" 54 | } 55 | }, 56 | "defaults": { 57 | "styleExt": "css", 58 | "component": {} 59 | } 60 | } 61 | -------------------------------------------------------------------------------- /frontend/.editorconfig: -------------------------------------------------------------------------------- 1 | # Editor configuration, see http://editorconfig.org 2 | root = true 3 | 4 | [*] 5 | charset = utf-8 6 | indent_style = space 7 | indent_size = 2 8 | insert_final_newline = true 9 | trim_trailing_whitespace = true 10 | 11 | [*.md] 12 | max_line_length = off 13 | trim_trailing_whitespace = false 14 | -------------------------------------------------------------------------------- /frontend/.gitignore: -------------------------------------------------------------------------------- 1 | # See http://help.github.com/ignore-files/ for more about ignoring files. 2 | 3 | # compiled output 4 | /dist 5 | /dist-server 6 | /tmp 7 | /out-tsc 8 | 9 | # dependencies 10 | /node_modules 11 | 12 | # IDEs and editors 13 | /.idea 14 | .project 15 | .classpath 16 | .c9/ 17 | *.launch 18 | .settings/ 19 | *.sublime-workspace 20 | 21 | # IDE - VSCode 22 | .vscode/* 23 | !.vscode/settings.json 24 | !.vscode/tasks.json 25 | !.vscode/launch.json 26 | !.vscode/extensions.json 27 | 28 | # misc 29 | /.sass-cache 30 | /connect.lock 31 | /coverage 32 | /libpeerconnection.log 33 | npm-debug.log 34 | testem.log 35 | /typings 36 | 37 | # e2e 38 | /e2e/*.js 39 | /e2e/*.map 40 | 41 | # System Files 42 | .DS_Store 43 | Thumbs.db 44 | -------------------------------------------------------------------------------- /frontend/README.md: -------------------------------------------------------------------------------- 1 | # Frontend 2 | 3 | This project was generated with [Angular CLI](https://github.com/angular/angular-cli) version 1.7.0. 4 | 5 | ## Development server 6 | 7 | Run `ng serve` for a dev server. Navigate to `http://localhost:4200/`. The app will automatically reload if you change any of the source files. 8 | 9 | ## Code scaffolding 10 | 11 | Run `ng generate component component-name` to generate a new component. You can also use `ng generate directive|pipe|service|class|guard|interface|enum|module`. 12 | 13 | ## Build 14 | 15 | Run `ng build` to build the project. The build artifacts will be stored in the `dist/` directory. Use the `-prod` flag for a production build. 16 | 17 | ## Running unit tests 18 | 19 | Run `ng test` to execute the unit tests via [Karma](https://karma-runner.github.io). 20 | 21 | ## Running end-to-end tests 22 | 23 | Run `ng e2e` to execute the end-to-end tests via [Protractor](http://www.protractortest.org/). 24 | 25 | ## Further help 26 | 27 | To get more help on the Angular CLI use `ng help` or go check out the [Angular CLI README](https://github.com/angular/angular-cli/blob/master/README.md). 28 | -------------------------------------------------------------------------------- /frontend/e2e/app.e2e-spec.ts: -------------------------------------------------------------------------------- 1 | import { AppPage } from './app.po'; 2 | 3 | describe('frontend App', () => { 4 | let page: AppPage; 5 | 6 | beforeEach(() => { 7 | page = new AppPage(); 8 | }); 9 | 10 | it('should display welcome message', () => { 11 | page.navigateTo(); 12 | expect(page.getParagraphText()).toEqual('Welcome to app!'); 13 | }); 14 | }); 15 | -------------------------------------------------------------------------------- /frontend/e2e/app.po.ts: -------------------------------------------------------------------------------- 1 | import { browser, by, element } from 'protractor'; 2 | 3 | export class AppPage { 4 | navigateTo() { 5 | return browser.get('/'); 6 | } 7 | 8 | getParagraphText() { 9 | return element(by.css('app-root h1')).getText(); 10 | } 11 | } 12 | -------------------------------------------------------------------------------- /frontend/e2e/tsconfig.e2e.json: -------------------------------------------------------------------------------- 1 | { 2 | "extends": "../tsconfig.json", 3 | "compilerOptions": { 4 | "outDir": "../out-tsc/e2e", 5 | "baseUrl": "./", 6 | "module": "commonjs", 7 | "target": "es5", 8 | "types": [ 9 | "jasmine", 10 | "jasminewd2", 11 | "node" 12 | ] 13 | } 14 | } 15 | -------------------------------------------------------------------------------- /frontend/karma.conf.js: -------------------------------------------------------------------------------- 1 | // Karma configuration file, see link for more information 2 | // https://karma-runner.github.io/1.0/config/configuration-file.html 3 | 4 | module.exports = function (config) { 5 | config.set({ 6 | basePath: '', 7 | frameworks: ['jasmine', '@angular/cli'], 8 | plugins: [ 9 | require('karma-jasmine'), 10 | require('karma-chrome-launcher'), 11 | require('karma-jasmine-html-reporter'), 12 | require('karma-coverage-istanbul-reporter'), 13 | require('@angular/cli/plugins/karma') 14 | ], 15 | client:{ 16 | clearContext: false // leave Jasmine Spec Runner output visible in browser 17 | }, 18 | coverageIstanbulReporter: { 19 | reports: [ 'html', 'lcovonly' ], 20 | fixWebpackSourcePaths: true 21 | }, 22 | angularCli: { 23 | environment: 'dev' 24 | }, 25 | reporters: ['progress', 'kjhtml'], 26 | port: 9876, 27 | colors: true, 28 | logLevel: config.LOG_INFO, 29 | autoWatch: true, 30 | browsers: ['Chrome'], 31 | singleRun: false 32 | }); 33 | }; 34 | -------------------------------------------------------------------------------- /frontend/package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "frontend", 3 | "version": "0.0.0", 4 | "license": "MIT", 5 | "scripts": { 6 | "ng": "ng", 7 | "start": "ng serve", 8 | "build": "ng build --prod", 9 | "test": "ng test", 10 | "lint": "ng lint", 11 | "e2e": "ng e2e" 12 | }, 13 | "private": true, 14 | "dependencies": { 15 | "@angular/animations": "^5.2.0", 16 | "@angular/common": "^5.2.0", 17 | "@angular/compiler": "^5.2.0", 18 | "@angular/core": "^5.2.0", 19 | "@angular/forms": "^5.2.0", 20 | "@angular/http": "^5.2.0", 21 | "@angular/platform-browser": "^5.2.0", 22 | "@angular/platform-browser-dynamic": "^5.2.0", 23 | "@angular/router": "^5.2.0", 24 | "core-js": "^2.4.1", 25 | "rxjs": "^5.5.6", 26 | "zone.js": "^0.8.19" 27 | }, 28 | "devDependencies": { 29 | "@angular/cli": "~1.7.0", 30 | "@angular/compiler-cli": "^5.2.0", 31 | "@angular/language-service": "^5.2.0", 32 | "@types/jasmine": "~2.8.3", 33 | "@types/jasminewd2": "~2.0.2", 34 | "@types/node": "~6.0.60", 35 | "codelyzer": "^4.0.1", 36 | "jasmine-core": "~2.8.0", 37 | "jasmine-spec-reporter": "~4.2.1", 38 | "karma": "~2.0.0", 39 | "karma-chrome-launcher": "~2.2.0", 40 | "karma-coverage-istanbul-reporter": "^1.2.1", 41 | "karma-jasmine": "~1.1.0", 42 | "karma-jasmine-html-reporter": "^0.2.2", 43 | "protractor": "~5.1.2", 44 | "ts-node": "~4.1.0", 45 | "tslint": "~5.9.1", 46 | "typescript": "~2.5.3" 47 | } 48 | } 49 | -------------------------------------------------------------------------------- /frontend/protractor.conf.js: -------------------------------------------------------------------------------- 1 | // Protractor configuration file, see link for more information 2 | // https://github.com/angular/protractor/blob/master/lib/config.ts 3 | 4 | const { SpecReporter } = require('jasmine-spec-reporter'); 5 | 6 | exports.config = { 7 | allScriptsTimeout: 11000, 8 | specs: [ 9 | './e2e/**/*.e2e-spec.ts' 10 | ], 11 | capabilities: { 12 | 'browserName': 'chrome' 13 | }, 14 | directConnect: true, 15 | baseUrl: 'http://localhost:4200/', 16 | framework: 'jasmine', 17 | jasmineNodeOpts: { 18 | showColors: true, 19 | defaultTimeoutInterval: 30000, 20 | print: function() {} 21 | }, 22 | onPrepare() { 23 | require('ts-node').register({ 24 | project: 'e2e/tsconfig.e2e.json' 25 | }); 26 | jasmine.getEnv().addReporter(new SpecReporter({ spec: { displayStacktrace: true } })); 27 | } 28 | }; 29 | -------------------------------------------------------------------------------- /frontend/src/app/app.component.css: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/auth0-blog/python-flask-angular-1/d35d7c76d88268d02ad9be1de2a226f8a1329d01/frontend/src/app/app.component.css -------------------------------------------------------------------------------- /frontend/src/app/app.component.html: -------------------------------------------------------------------------------- 1 |