├── .gitignore ├── .vscode └── launch.json ├── LICENSE ├── README.md ├── week-1 ├── class-diagram.puml └── index.js ├── week-2 ├── .vscode │ └── launch.json ├── booking.js ├── driver.js ├── index.js ├── package-lock.json ├── package.json └── passenger.js ├── week-3 ├── .vscode │ └── launch.json ├── create-records.js ├── database │ ├── base-database.js │ ├── driver-database.js │ ├── driver.json │ ├── index.js │ ├── passenger-database.js │ └── passenger.json ├── index.js ├── lib │ └── print-booking-history.js ├── models │ ├── booking.js │ ├── driver.js │ └── passenger.js ├── package-lock.json └── package.json ├── week-4 ├── .vscode │ └── launch.json ├── create-records.js ├── database │ ├── base-database.js │ ├── driver-database.js │ ├── driver.json │ ├── index.js │ ├── passenger-database.js │ └── passenger.json ├── index.js ├── lib │ └── print-booking-history.js ├── models │ ├── booking.js │ ├── driver.js │ └── passenger.js ├── package-lock.json └── package.json ├── week-5 ├── .vscode │ └── launch.json ├── create-records.js ├── database │ ├── base-database.js │ ├── driver-database.js │ ├── driver.json │ ├── index.js │ ├── passenger-database.js │ └── passenger.json ├── fetch-records.js ├── index.js ├── lib │ └── print-booking-history.js ├── models │ ├── booking.js │ ├── driver.js │ └── passenger.js ├── package-lock.json ├── package.json ├── routes │ ├── index.js │ └── passengers.js └── views │ ├── index.pug │ ├── layout.pug │ ├── passenger.pug │ └── passengers.pug ├── week-6 ├── .vscode │ └── launch.json ├── create-records.js ├── database │ └── passenger-database.js ├── fetch-records.js ├── index.js ├── lib │ └── print-booking-history.js ├── models │ ├── booking.js │ ├── driver.js │ └── passenger.js ├── mongo-connection.js ├── package-lock.json ├── package.json ├── requests.http ├── routes │ ├── bookings.js │ ├── drivers.js │ ├── index.js │ └── passengers.js ├── services │ ├── base-service.js │ ├── booking-service.js │ ├── driver-service.js │ ├── driver.json │ ├── index.js │ ├── passenger-service.js │ └── passenger.json └── views │ ├── bookings.pug │ ├── driver.pug │ ├── drivers.pug │ ├── index.pug │ ├── layout.pug │ ├── passenger.pug │ └── passengers.pug ├── week-7 ├── .bin │ └── www ├── .vscode │ ├── launch.json │ └── settings.json ├── __mocks__ │ └── colors.js ├── create-records.js ├── fetch-records.js ├── index.js ├── jest.config.js ├── lib │ ├── __tests__ │ │ └── print-booking-history.test.js │ └── print-booking-history.js ├── models │ ├── booking.js │ ├── driver.js │ └── passenger.js ├── mongo-connection.js ├── package-lock.json ├── package.json ├── requests.http ├── routes │ ├── bookings.js │ ├── drivers.js │ ├── index.js │ └── passengers.js ├── services │ ├── base-service.js │ ├── booking-service.js │ ├── driver-service.js │ ├── driver.json │ ├── index.js │ ├── passenger-service.js │ └── passenger.json ├── tests │ ├── e2e │ │ └── passengers.test.js │ └── scenario │ │ └── bookings.test.js └── views │ ├── bookings.pug │ ├── driver.pug │ ├── drivers.pug │ ├── index.pug │ ├── layout.pug │ ├── passenger.pug │ └── passengers.pug ├── week-8 ├── .bin │ └── www ├── .vscode │ ├── launch.json │ └── settings.json ├── __mocks__ │ └── colors.js ├── create-records.js ├── fetch-records.js ├── frontend │ ├── .browserslistrc │ ├── .editorconfig │ ├── .eslintrc.js │ ├── .gitignore │ ├── README.md │ ├── babel.config.js │ ├── package-lock.json │ ├── package.json │ ├── public │ │ ├── favicon.ico │ │ ├── img │ │ │ └── icons │ │ │ │ ├── android-chrome-192x192.png │ │ │ │ ├── android-chrome-512x512.png │ │ │ │ ├── android-chrome-maskable-192x192.png │ │ │ │ ├── android-chrome-maskable-512x512.png │ │ │ │ ├── apple-touch-icon-120x120.png │ │ │ │ ├── apple-touch-icon-152x152.png │ │ │ │ ├── apple-touch-icon-180x180.png │ │ │ │ ├── apple-touch-icon-60x60.png │ │ │ │ ├── apple-touch-icon-76x76.png │ │ │ │ ├── apple-touch-icon.png │ │ │ │ ├── favicon-16x16.png │ │ │ │ ├── favicon-32x32.png │ │ │ │ ├── msapplication-icon-144x144.png │ │ │ │ ├── mstile-150x150.png │ │ │ │ └── safari-pinned-tab.svg │ │ ├── index.html │ │ └── robots.txt │ └── src │ │ ├── App.vue │ │ ├── assets │ │ └── logo.png │ │ ├── components │ │ └── HelloWorld.vue │ │ ├── main.js │ │ ├── registerServiceWorker.js │ │ ├── router │ │ └── index.js │ │ ├── store │ │ └── index.js │ │ └── views │ │ ├── About.vue │ │ ├── Passenger.vue │ │ └── Passengers.vue ├── index.js ├── jest.config.js ├── lib │ ├── __tests__ │ │ └── print-booking-history.test.js │ └── print-booking-history.js ├── models │ ├── booking.js │ ├── driver.js │ └── passenger.js ├── mongo-connection.js ├── package-lock.json ├── package.json ├── requests.http ├── routes │ ├── bookings.js │ ├── drivers.js │ ├── index.js │ └── passengers.js ├── services │ ├── base-service.js │ ├── booking-service.js │ ├── driver-service.js │ ├── driver.json │ ├── index.js │ ├── passenger-service.js │ └── passenger.json ├── tests │ ├── e2e │ │ └── passengers.test.js │ └── scenario │ │ └── bookings.test.js └── views │ ├── bookings.pug │ ├── driver.pug │ ├── drivers.pug │ ├── index.pug │ ├── layout.pug │ ├── passenger.pug │ └── passengers.pug └── week-9 ├── .gitignore ├── .vscode ├── launch.json └── settings.json ├── backend ├── .bin │ └── www ├── .dockerignore ├── Dockerfile ├── __mocks__ │ └── colors.js ├── create-records.js ├── fetch-records.js ├── index.js ├── jest.config.js ├── lib │ ├── __tests__ │ │ └── print-booking-history.test.js │ └── print-booking-history.js ├── models │ ├── booking.js │ ├── driver.js │ └── passenger.js ├── mongo-connection.js ├── package-lock.json ├── package.json ├── requests.http ├── routes │ ├── bookings.js │ ├── drivers.js │ ├── index.js │ └── passengers.js ├── services │ ├── base-service.js │ ├── booking-service.js │ ├── driver-service.js │ ├── driver.json │ ├── index.js │ ├── passenger-service.js │ └── passenger.json ├── tests │ ├── e2e │ │ └── passengers.test.js │ └── scenario │ │ └── bookings.test.js └── views │ ├── bookings.pug │ ├── driver.pug │ ├── drivers.pug │ ├── index.pug │ ├── layout.pug │ ├── passenger.pug │ └── passengers.pug ├── docker-compose.yml └── frontend ├── .browserslistrc ├── .dockerignore ├── .editorconfig ├── .eslintrc.js ├── .gitignore ├── Dockerfile ├── README.md ├── babel.config.js ├── package-lock.json ├── package.json ├── public ├── favicon.ico ├── img │ └── icons │ │ ├── android-chrome-192x192.png │ │ ├── android-chrome-512x512.png │ │ ├── android-chrome-maskable-192x192.png │ │ ├── android-chrome-maskable-512x512.png │ │ ├── apple-touch-icon-120x120.png │ │ ├── apple-touch-icon-152x152.png │ │ ├── apple-touch-icon-180x180.png │ │ ├── apple-touch-icon-60x60.png │ │ ├── apple-touch-icon-76x76.png │ │ ├── apple-touch-icon.png │ │ ├── favicon-16x16.png │ │ ├── favicon-32x32.png │ │ ├── msapplication-icon-144x144.png │ │ ├── mstile-150x150.png │ │ └── safari-pinned-tab.svg ├── index.html └── robots.txt └── src ├── App.vue ├── assets └── logo.png ├── components └── HelloWorld.vue ├── main.js ├── registerServiceWorker.js ├── router └── index.js ├── store └── index.js └── views ├── About.vue ├── Passenger.vue └── Passengers.vue /.gitignore: -------------------------------------------------------------------------------- 1 | # Logs 2 | logs 3 | *.log 4 | npm-debug.log* 5 | yarn-debug.log* 6 | yarn-error.log* 7 | lerna-debug.log* 8 | 9 | # Diagnostic reports (https://nodejs.org/api/report.html) 10 | report.[0-9]*.[0-9]*.[0-9]*.[0-9]*.json 11 | 12 | # Runtime data 13 | pids 14 | *.pid 15 | *.seed 16 | *.pid.lock 17 | 18 | # Directory for instrumented libs generated by jscoverage/JSCover 19 | lib-cov 20 | 21 | # Coverage directory used by tools like istanbul 22 | coverage 23 | *.lcov 24 | 25 | # nyc test coverage 26 | .nyc_output 27 | 28 | # Grunt intermediate storage (https://gruntjs.com/creating-plugins#storing-task-files) 29 | .grunt 30 | 31 | # Bower dependency directory (https://bower.io/) 32 | bower_components 33 | 34 | # node-waf configuration 35 | .lock-wscript 36 | 37 | # Compiled binary addons (https://nodejs.org/api/addons.html) 38 | build/Release 39 | 40 | # Dependency directories 41 | node_modules/ 42 | jspm_packages/ 43 | 44 | # TypeScript v1 declaration files 45 | typings/ 46 | 47 | # TypeScript cache 48 | *.tsbuildinfo 49 | 50 | # Optional npm cache directory 51 | .npm 52 | 53 | # Optional eslint cache 54 | .eslintcache 55 | 56 | # Microbundle cache 57 | .rpt2_cache/ 58 | .rts2_cache_cjs/ 59 | .rts2_cache_es/ 60 | .rts2_cache_umd/ 61 | 62 | # Optional REPL history 63 | .node_repl_history 64 | 65 | # Output of 'npm pack' 66 | *.tgz 67 | 68 | # Yarn Integrity file 69 | .yarn-integrity 70 | 71 | # dotenv environment variables file 72 | .env 73 | .env.test 74 | 75 | # parcel-bundler cache (https://parceljs.org/) 76 | .cache 77 | 78 | # Next.js build output 79 | .next 80 | 81 | # Nuxt.js build / generate output 82 | .nuxt 83 | dist 84 | 85 | # Gatsby files 86 | .cache/ 87 | # Comment in the public line in if your project uses Gatsby and *not* Next.js 88 | # https://nextjs.org/blog/next-9-1#public-directory-support 89 | # public 90 | 91 | # vuepress build output 92 | .vuepress/dist 93 | 94 | # Serverless directories 95 | .serverless/ 96 | 97 | # FuseBox cache 98 | .fusebox/ 99 | 100 | # DynamoDB Local files 101 | .dynamodb/ 102 | 103 | # TernJS port file 104 | .tern-port 105 | -------------------------------------------------------------------------------- /.vscode/launch.json: -------------------------------------------------------------------------------- 1 | { 2 | // Use IntelliSense to learn about possible attributes. 3 | // Hover to view descriptions of existing attributes. 4 | // For more information, visit: https://go.microsoft.com/fwlink/?linkid=830387 5 | "version": "0.2.0", 6 | "configurations": [ 7 | { 8 | "type": "node", 9 | "request": "launch", 10 | "name": "Launch Week 3", 11 | "skipFiles": [ 12 | "/**" 13 | ], 14 | "program": "${workspaceFolder}/week-3/index.js", 15 | "cwd": "${workspaceFolder}/week-3" 16 | } 17 | ] 18 | } 19 | -------------------------------------------------------------------------------- /LICENSE: -------------------------------------------------------------------------------- 1 | MIT License 2 | 3 | Copyright (c) 2020 Armagan Amcalar 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 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # nimble-js-1 -------------------------------------------------------------------------------- /week-1/class-diagram.puml: -------------------------------------------------------------------------------- 1 | @startuml "Class Diagram" 2 | 3 | class User <> { 4 | + name: String 5 | + email: String 6 | + phone: String 7 | + wallItems: [Wall] 8 | + findPT(): [PT] 9 | + createWall(): Wall 10 | } 11 | 12 | class PT <> { 13 | + price: Number 14 | + branch: Branch 15 | + locations: Location[] 16 | } 17 | 18 | class Athlete <> { 19 | + measurements: Measurement[] 20 | } 21 | 22 | enum Branch { 23 | FITNESS 24 | BODY_BUILDING 25 | YOGA 26 | } 27 | 28 | interface Measurement <> { 29 | + type: String 30 | + value: Number 31 | + unit: String 32 | } 33 | 34 | class BMI <> { 35 | + type: String 36 | + value: Number 37 | + unit: String 38 | } 39 | 40 | class Wall <> { 41 | + text: String 42 | + user: User 43 | + comments: [Wall] 44 | } 45 | 46 | class Location <> { 47 | + coords: Number[] 48 | + name: String 49 | } 50 | 51 | class Discount <> { 52 | + discountPercentage: Number 53 | } 54 | 55 | BMI -up-|> Measurement 56 | Athlete -up-|> User 57 | PT -up-|> User 58 | PT --> "1" Branch 59 | PT "1..n" o--> "0..n" Location 60 | User "0..n" *--> "1" Wall 61 | Athlete --> Discount 62 | 63 | @enduml 64 | -------------------------------------------------------------------------------- /week-1/index.js: -------------------------------------------------------------------------------- 1 | class User { 2 | constructor(name, email, phone) { 3 | this.name = name 4 | this.email = email 5 | this.phone = phone 6 | } 7 | 8 | findPT() { 9 | 10 | } 11 | 12 | bookPT(pt) { 13 | this.bookings.push(pt) 14 | } 15 | } 16 | 17 | class Athelete extends User { 18 | constructor(name, email, phone) { 19 | super(name, email, phone) 20 | 21 | this.measurements = [] 22 | } 23 | } 24 | 25 | const user = new Athlete('Armagan', 'armagan@nimble.dev', '+4914132') 26 | 27 | const pts = user.findPT() 28 | 29 | user.bookPT(pts[0]) 30 | -------------------------------------------------------------------------------- /week-2/.vscode/launch.json: -------------------------------------------------------------------------------- 1 | { 2 | // Use IntelliSense to learn about possible attributes. 3 | // Hover to view descriptions of existing attributes. 4 | // For more information, visit: https://go.microsoft.com/fwlink/?linkid=830387 5 | "version": "0.2.0", 6 | "configurations": [ 7 | { 8 | "type": "node", 9 | "request": "launch", 10 | "name": "Launch Program", 11 | "skipFiles": [ 12 | "/**" 13 | ], 14 | "program": "${workspaceFolder}/index.js" 15 | } 16 | ] 17 | } 18 | -------------------------------------------------------------------------------- /week-2/booking.js: -------------------------------------------------------------------------------- 1 | class Booking { 2 | constructor(driver, passenger, origin, destination) { 3 | this.driver = driver 4 | this.passenger = passenger 5 | this.origin = origin 6 | this.destination = destination 7 | } 8 | } 9 | 10 | module.exports = Booking 11 | -------------------------------------------------------------------------------- /week-2/driver.js: -------------------------------------------------------------------------------- 1 | class Driver { 2 | constructor(name, location) { 3 | this.name = name 4 | this.location = location 5 | } 6 | } 7 | 8 | module.exports = Driver 9 | -------------------------------------------------------------------------------- /week-2/index.js: -------------------------------------------------------------------------------- 1 | const colors = require('colors') 2 | 3 | const Passenger = require('./passenger') 4 | const Driver = require('./driver') 5 | 6 | const armagan = new Passenger('Armagan', 'Kreuzberg') 7 | const stefan = new Driver('Stefan', 'Treptower Park') 8 | 9 | armagan.book(stefan, 'Kreuzberg', 'Neukolln') 10 | armagan.book(stefan, 'Neukolln', 'Mitte') 11 | armagan.book(stefan, 'Mitte', 'Kreuzberg') 12 | armagan.book(stefan, 'Kreuzberg', 'SXF') 13 | 14 | function printBooking(booking) { 15 | console.log(`${colors.blue(booking.passenger.name)} booked ${colors.blue(booking.driver.name)} to travel from ${colors.bgRed.white(booking.origin)} to ${colors.bgRed.white(booking.destination)}`) 16 | } 17 | 18 | function printBookingHistory(passenger) { 19 | passenger.bookings.forEach(printBooking) 20 | } 21 | 22 | printBookingHistory(armagan) 23 | -------------------------------------------------------------------------------- /week-2/package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "week-2", 3 | "version": "1.0.0", 4 | "description": "In-class examples for week-2", 5 | "main": "index.js", 6 | "scripts": { 7 | "start": "node index.js", 8 | "watch": "nodemon", 9 | "test": "echo \"Error: no test specified\" && exit 1" 10 | }, 11 | "author": "Armagan Amcalar ", 12 | "license": "ISC", 13 | "dependencies": { 14 | "colors": "^1.4.0" 15 | }, 16 | "devDependencies": { 17 | "nodemon": "^2.0.2" 18 | } 19 | } 20 | -------------------------------------------------------------------------------- /week-2/passenger.js: -------------------------------------------------------------------------------- 1 | const Booking = require('./booking') 2 | 3 | class Passenger { 4 | constructor(name, location) { 5 | this.name = name 6 | this.location = location 7 | this.bookings = [] 8 | } 9 | 10 | book(driver, origin, destination) { 11 | const booking = new Booking(driver, this, origin, destination) 12 | 13 | this.bookings.push(booking) 14 | 15 | return booking 16 | } 17 | } 18 | 19 | module.exports = Passenger 20 | -------------------------------------------------------------------------------- /week-3/.vscode/launch.json: -------------------------------------------------------------------------------- 1 | { 2 | // Use IntelliSense to learn about possible attributes. 3 | // Hover to view descriptions of existing attributes. 4 | // For more information, visit: https://go.microsoft.com/fwlink/?linkid=830387 5 | "version": "0.2.0", 6 | "configurations": [ 7 | { 8 | "type": "node", 9 | "request": "launch", 10 | "name": "Launch Program", 11 | "skipFiles": [ 12 | "/**" 13 | ], 14 | "program": "${workspaceFolder}/index.js" 15 | } 16 | ] 17 | } 18 | -------------------------------------------------------------------------------- /week-3/create-records.js: -------------------------------------------------------------------------------- 1 | const Passenger = require('./models/passenger') 2 | const Driver = require('./models/driver') 3 | const passengerDatabase = require('./database/passenger-database') 4 | const driverDatabase = require('./database/driver-database') 5 | 6 | const printBookingHistory = require('./lib/print-booking-history') 7 | 8 | const armagan = Passenger.create({name: 'Armagan', location: 'Kreuzberg'}) 9 | const mert = Passenger.create({name: 'Mert', location: 'Mitte'}) 10 | const stefan = Driver.create({ name: 'Stefan', location: 'Treptower Park' }) 11 | 12 | armagan.book(stefan, 'Kreuzberg', 'Neukolln') 13 | armagan.book(stefan, 'Neukolln', 'Mitte') 14 | armagan.book(stefan, 'Mitte', 'Kreuzberg') 15 | armagan.book(stefan, 'Kreuzberg', 'SXF') 16 | mert.book(stefan, 'Mitte', 'Kreuzberg') 17 | 18 | passengerDatabase.save([armagan, mert]) 19 | driverDatabase.save([stefan]) 20 | 21 | const betul = Passenger.create({name: 'Betul', location: 'Tegel'}) 22 | 23 | passengerDatabase.insert(betul) 24 | 25 | const passengers = passengerDatabase.load() 26 | 27 | passengers.forEach(printBookingHistory) 28 | -------------------------------------------------------------------------------- /week-3/database/base-database.js: -------------------------------------------------------------------------------- 1 | const fs = require('fs') 2 | const flatted = require('flatted') 3 | 4 | class BaseDatabase { 5 | constructor(model) { 6 | this.model = model 7 | this.filename = model.name.toLowerCase() 8 | } 9 | 10 | save(objects) { 11 | fs.writeFileSync(`${__dirname}/${this.filename}.json`, flatted.stringify(objects, null, 2)) 12 | } 13 | 14 | load() { 15 | const file = fs.readFileSync(`${__dirname}/${this.filename}.json`, 'utf8') 16 | const objects = flatted.parse(file) 17 | 18 | return objects.map(this.model.create) 19 | } 20 | 21 | insert(object) { 22 | const objects = this.load() 23 | 24 | this.save(objects.concat(object)) 25 | } 26 | 27 | remove(index) { 28 | const objects = this.load() 29 | 30 | objects.splice(index, 1) 31 | this.save(objects) 32 | } 33 | 34 | update(object) { 35 | const objects = this.load() 36 | 37 | const index = objects.findIndex(o => o.id == object.id) 38 | 39 | if (index == -1) throw new Error(`Cannot find ${this.model.name} instance with id ${object.id}`) 40 | 41 | objects.splice(index, 1, object) 42 | this.save(objects) 43 | } 44 | 45 | find(id) { 46 | return this.load().find(o => o.id == id) 47 | } 48 | 49 | findBy(property, value) { 50 | return this.load().find(o => o[property] == value) 51 | } 52 | } 53 | 54 | module.exports = BaseDatabase 55 | -------------------------------------------------------------------------------- /week-3/database/driver-database.js: -------------------------------------------------------------------------------- 1 | const BaseDatabase = require('./base-database') 2 | const Driver = require('../models/driver') 3 | 4 | class DriverDatabase extends BaseDatabase { 5 | findByDriverName(name) { 6 | return this.findBy('name', name) 7 | } 8 | 9 | findByLocation(location) { 10 | return this.findBy('location', location) 11 | } 12 | } 13 | 14 | module.exports = new DriverDatabase(Driver) 15 | -------------------------------------------------------------------------------- /week-3/database/driver.json: -------------------------------------------------------------------------------- 1 | [[ 2 | "1" 3 | ],{ 4 | "name": "2", 5 | "location": "3" 6 | },"Stefan","Treptower Park"] -------------------------------------------------------------------------------- /week-3/database/index.js: -------------------------------------------------------------------------------- 1 | const passengerDatabase = require('./passenger-database') 2 | const driverDatabase = require('./driver-database') 3 | 4 | module.exports = { passengerDatabase, driverDatabase } 5 | -------------------------------------------------------------------------------- /week-3/database/passenger-database.js: -------------------------------------------------------------------------------- 1 | const BaseDatabase = require('./base-database') 2 | const Passenger = require('../models/passenger') 3 | 4 | class PassengerDatabase extends BaseDatabase { 5 | findByName(name) { 6 | const objects = this.load() 7 | 8 | return objects.find(o => o.name == name) 9 | } 10 | } 11 | 12 | module.exports = new PassengerDatabase(Passenger) 13 | -------------------------------------------------------------------------------- /week-3/database/passenger.json: -------------------------------------------------------------------------------- 1 | [[ 2 | "1", 3 | "2", 4 | "3" 5 | ],{ 6 | "id": "4", 7 | "name": "5", 8 | "location": "6", 9 | "bookings": "7" 10 | },{ 11 | "id": "8", 12 | "name": "9", 13 | "location": "10", 14 | "bookings": "11" 15 | },{ 16 | "id": "12", 17 | "name": "13", 18 | "location": "14", 19 | "bookings": "15" 20 | },"e4ee6ade-7cb0-4786-8de5-e5cad8f85c1e","Armagan","Kreuzberg",[ 21 | "16", 22 | "17", 23 | "18", 24 | "19", 25 | "20", 26 | "21" 27 | ],"394db551-3ae1-4118-a2a2-35498b592b16","Mert","Mitte",[ 28 | "22" 29 | ],"160b70a8-ce12-4906-bfcb-d2399caf803a","Betul","Tegel",[],{ 30 | "driver": "23", 31 | "passenger": "24", 32 | "origin": "6", 33 | "destination": "25" 34 | },{ 35 | "driver": "23", 36 | "passenger": "24", 37 | "origin": "25", 38 | "destination": "10" 39 | },{ 40 | "driver": "23", 41 | "passenger": "24", 42 | "origin": "10", 43 | "destination": "6" 44 | },{ 45 | "driver": "23", 46 | "passenger": "24", 47 | "origin": "6", 48 | "destination": "26" 49 | },{ 50 | "driver": "27", 51 | "passenger": "28", 52 | "origin": "6", 53 | "destination": "29" 54 | },{ 55 | "driver": "30", 56 | "passenger": "1", 57 | "origin": "6", 58 | "destination": "29" 59 | },{ 60 | "driver": "31", 61 | "passenger": "32", 62 | "origin": "10", 63 | "destination": "6" 64 | },{ 65 | "name": "33", 66 | "location": "34" 67 | },{ 68 | "id": "4", 69 | "name": "5", 70 | "location": "6", 71 | "bookings": "7" 72 | },"Neukolln","SXF",{ 73 | "name": "33", 74 | "location": "34" 75 | },{ 76 | "id": "4", 77 | "name": "5", 78 | "location": "6", 79 | "bookings": "7" 80 | },"Wannsee",{ 81 | "name": "33", 82 | "location": "34" 83 | },{ 84 | "name": "33", 85 | "location": "34" 86 | },{ 87 | "id": "8", 88 | "name": "9", 89 | "location": "10", 90 | "bookings": "11" 91 | },"Stefan","Treptower Park"] -------------------------------------------------------------------------------- /week-3/index.js: -------------------------------------------------------------------------------- 1 | const { passengerDatabase, driverDatabase } = require('./database') 2 | const printBookingHistory = require('./lib/print-booking-history') 3 | 4 | const stefan = driverDatabase.findBy('name', 'Stefan') 5 | const armagan = passengerDatabase.findByName('Armagan') 6 | 7 | armagan.book(stefan, 'Kreuzberg', 'Wannsee') 8 | passengerDatabase.update(armagan) 9 | 10 | printBookingHistory(armagan) 11 | 12 | console.log(passengerDatabase.findBy('location', 'Mitte')) 13 | -------------------------------------------------------------------------------- /week-3/lib/print-booking-history.js: -------------------------------------------------------------------------------- 1 | const colors = require('colors') 2 | 3 | function printBooking(booking) { 4 | console.log(`${colors.blue(booking.passenger.name)} booked ${colors.blue(booking.driver.name)} to travel from ${colors.bgRed.white(booking.origin)} to ${colors.bgRed.white(booking.destination)}`) 5 | } 6 | 7 | function printBookingHistory(passenger) { 8 | if (passenger.bookings.length == 0) 9 | return console.log(`${colors.blue(passenger.name)} has no bookings yet.`) 10 | 11 | passenger.bookings.forEach(printBooking) 12 | } 13 | 14 | module.exports = printBookingHistory 15 | -------------------------------------------------------------------------------- /week-3/models/booking.js: -------------------------------------------------------------------------------- 1 | class Booking { 2 | constructor(driver, passenger, origin, destination) { 3 | this.driver = driver 4 | this.passenger = passenger 5 | this.origin = origin 6 | this.destination = destination 7 | } 8 | } 9 | 10 | module.exports = Booking 11 | -------------------------------------------------------------------------------- /week-3/models/driver.js: -------------------------------------------------------------------------------- 1 | class Driver { 2 | constructor(name, location) { 3 | this.name = name 4 | this.location = location 5 | } 6 | 7 | static create({name, location}) { 8 | return new Driver(name, location) 9 | } 10 | } 11 | 12 | module.exports = Driver 13 | -------------------------------------------------------------------------------- /week-3/models/passenger.js: -------------------------------------------------------------------------------- 1 | const Booking = require('./booking') 2 | const uuid = require('uuid') 3 | 4 | class Passenger { 5 | constructor(id = uuid.v4(), name, location, bookings = []) { 6 | this.id = id 7 | 8 | this.name = name 9 | this.location = location 10 | this.bookings = bookings 11 | } 12 | 13 | book(driver, origin, destination) { 14 | const booking = new Booking(driver, this, origin, destination) 15 | 16 | this.bookings.push(booking) 17 | 18 | return booking 19 | } 20 | 21 | static create({id, name, location, bookings}) { 22 | return new Passenger(id, name, location, bookings) 23 | } 24 | } 25 | 26 | module.exports = Passenger 27 | -------------------------------------------------------------------------------- /week-3/package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "week-3", 3 | "version": "1.0.0", 4 | "description": "In-class examples for week-3", 5 | "main": "index.js", 6 | "scripts": { 7 | "start": "node index.js", 8 | "watch": "nodemon --ignore '*.json'", 9 | "test": "echo \"Error: no test specified\" && exit 1" 10 | }, 11 | "author": "Armagan Amcalar ", 12 | "license": "ISC", 13 | "dependencies": { 14 | "colors": "^1.4.0", 15 | "flatted": "^2.0.1", 16 | "uuid": "^7.0.2" 17 | }, 18 | "devDependencies": { 19 | "nodemon": "^2.0.2" 20 | } 21 | } 22 | -------------------------------------------------------------------------------- /week-4/.vscode/launch.json: -------------------------------------------------------------------------------- 1 | { 2 | // Use IntelliSense to learn about possible attributes. 3 | // Hover to view descriptions of existing attributes. 4 | // For more information, visit: https://go.microsoft.com/fwlink/?linkid=830387 5 | "version": "0.2.0", 6 | "configurations": [ 7 | { 8 | "type": "node", 9 | "request": "launch", 10 | "name": "Launch Current File", 11 | "program": "${file}", 12 | "skipFiles": [ 13 | "/**" 14 | ] 15 | }, 16 | { 17 | "type": "node", 18 | "request": "launch", 19 | "name": "Launch Program", 20 | "skipFiles": [ 21 | "/**" 22 | ], 23 | "program": "${workspaceFolder}/index.js" 24 | } 25 | ] 26 | } 27 | -------------------------------------------------------------------------------- /week-4/create-records.js: -------------------------------------------------------------------------------- 1 | const Passenger = require('./models/passenger') 2 | const Driver = require('./models/driver') 3 | const passengerDatabase = require('./database/passenger-database') 4 | const driverDatabase = require('./database/driver-database') 5 | 6 | const printBookingHistory = require('./lib/print-booking-history') 7 | 8 | const armagan = Passenger.create({name: 'Armagan', location: 'Kreuzberg'}) 9 | const mert = Passenger.create({name: 'Mert', location: 'Mitte'}) 10 | const stefan = Driver.create({ name: 'Stefan', location: 'Treptower Park' }) 11 | 12 | armagan.book(stefan, 'Kreuzberg', 'Neukolln') 13 | armagan.book(stefan, 'Neukolln', 'Mitte') 14 | armagan.book(stefan, 'Mitte', 'Kreuzberg') 15 | armagan.book(stefan, 'Kreuzberg', 'SXF') 16 | mert.book(stefan, 'Mitte', 'Kreuzberg') 17 | 18 | async function main() { 19 | try { 20 | await passengerDatabase.save([armagan, mert]) 21 | 22 | await driverDatabase.save([stefan]) 23 | 24 | const betul = Passenger.create({ name: 'Betul', location: 'Tegel' }) 25 | 26 | await passengerDatabase.insert(betul) 27 | const passengers = await passengerDatabase.load() 28 | passengers.forEach(printBookingHistory) 29 | } catch (e) { 30 | return console.log(e) 31 | } 32 | } 33 | 34 | main() 35 | -------------------------------------------------------------------------------- /week-4/database/base-database.js: -------------------------------------------------------------------------------- 1 | const fs = require('fs') 2 | const flatted = require('flatted') 3 | 4 | class BaseDatabase { 5 | constructor(model) { 6 | this.model = model 7 | this.filename = model.name.toLowerCase() 8 | } 9 | 10 | save(objects) { 11 | return new Promise((resolve, reject) => { 12 | fs.writeFile(`${__dirname}/${this.filename}.json`, flatted.stringify(objects, null, 2), (err) => { 13 | if (err) return reject(err) 14 | resolve() 15 | }) 16 | }) 17 | } 18 | 19 | load() { 20 | return new Promise((resolve, reject) => { 21 | fs.readFile(`${__dirname}/${this.filename}.json`, 'utf8', (err, file) => { 22 | if (err) return reject(err) 23 | 24 | const objects = flatted.parse(file) 25 | 26 | resolve(objects.map(this.model.create)) 27 | }) 28 | }) 29 | } 30 | 31 | async insert(object) { 32 | const objects = await this.load() 33 | return this.save(objects.concat(object)) 34 | } 35 | 36 | remove(index) { 37 | const objects = this.load() 38 | 39 | objects.splice(index, 1) 40 | this.save(objects) 41 | } 42 | 43 | update(object) { 44 | const objects = this.load() 45 | 46 | const index = objects.findIndex(o => o.id == object.id) 47 | 48 | if (index == -1) throw new Error(`Cannot find ${this.model.name} instance with id ${object.id}`) 49 | 50 | objects.splice(index, 1, object) 51 | this.save(objects) 52 | } 53 | 54 | async find(id) { 55 | const objects = await this.load() 56 | return objects.find(o => o.id == id) 57 | } 58 | 59 | async findBy(property, value) { 60 | return (await this.load()).find(o => o[property] == value) 61 | } 62 | } 63 | 64 | module.exports = BaseDatabase 65 | -------------------------------------------------------------------------------- /week-4/database/driver-database.js: -------------------------------------------------------------------------------- 1 | const BaseDatabase = require('./base-database') 2 | const Driver = require('../models/driver') 3 | 4 | class DriverDatabase extends BaseDatabase { 5 | findByDriverName(name) { 6 | return this.findBy('name', name) 7 | } 8 | 9 | findByLocation(location) { 10 | return this.findBy('location', location) 11 | } 12 | } 13 | 14 | module.exports = new DriverDatabase(Driver) 15 | -------------------------------------------------------------------------------- /week-4/database/driver.json: -------------------------------------------------------------------------------- 1 | [[ 2 | "1" 3 | ],{ 4 | "name": "2", 5 | "location": "3" 6 | },"Stefan","Treptower Park"] -------------------------------------------------------------------------------- /week-4/database/index.js: -------------------------------------------------------------------------------- 1 | const passengerDatabase = require('./passenger-database') 2 | const driverDatabase = require('./driver-database') 3 | 4 | module.exports = { passengerDatabase, driverDatabase } 5 | -------------------------------------------------------------------------------- /week-4/database/passenger-database.js: -------------------------------------------------------------------------------- 1 | const BaseDatabase = require('./base-database') 2 | const Passenger = require('../models/passenger') 3 | 4 | class PassengerDatabase extends BaseDatabase { 5 | findByName(name) { 6 | const objects = this.load() 7 | 8 | return objects.find(o => o.name == name) 9 | } 10 | } 11 | 12 | module.exports = new PassengerDatabase(Passenger) 13 | -------------------------------------------------------------------------------- /week-4/database/passenger.json: -------------------------------------------------------------------------------- 1 | [[ 2 | "1", 3 | "2", 4 | "3" 5 | ],{ 6 | "id": "4", 7 | "name": "5", 8 | "location": "6", 9 | "bookings": "7" 10 | },{ 11 | "id": "8", 12 | "name": "9", 13 | "location": "10", 14 | "bookings": "11" 15 | },{ 16 | "id": "12", 17 | "name": "13", 18 | "location": "14", 19 | "bookings": "15" 20 | },"367fef2a-4afe-4211-8d6d-901d2dd1b714","Armagan","Kreuzberg",[ 21 | "16", 22 | "17", 23 | "18", 24 | "19" 25 | ],"584ca022-c022-46ec-a3c0-758e478557ea","Mert","Mitte",[ 26 | "20" 27 | ],"75b7ff8f-9f54-4c9e-b000-5342296f7f88","Betul","Tegel",[],{ 28 | "driver": "21", 29 | "passenger": "22", 30 | "origin": "6", 31 | "destination": "23" 32 | },{ 33 | "driver": "21", 34 | "passenger": "22", 35 | "origin": "23", 36 | "destination": "10" 37 | },{ 38 | "driver": "21", 39 | "passenger": "22", 40 | "origin": "10", 41 | "destination": "6" 42 | },{ 43 | "driver": "21", 44 | "passenger": "22", 45 | "origin": "6", 46 | "destination": "24" 47 | },{ 48 | "driver": "21", 49 | "passenger": "25", 50 | "origin": "10", 51 | "destination": "6" 52 | },{ 53 | "name": "26", 54 | "location": "27" 55 | },{ 56 | "id": "4", 57 | "name": "5", 58 | "location": "6", 59 | "bookings": "7" 60 | },"Neukolln","SXF",{ 61 | "id": "8", 62 | "name": "9", 63 | "location": "10", 64 | "bookings": "11" 65 | },"Stefan","Treptower Park"] -------------------------------------------------------------------------------- /week-4/index.js: -------------------------------------------------------------------------------- 1 | const { passengerDatabase, driverDatabase } = require('./database') 2 | const printBookingHistory = require('./lib/print-booking-history') 3 | 4 | const stefan = driverDatabase.findBy('name', 'Stefan') 5 | const armagan = passengerDatabase.findByName('Armagan') 6 | 7 | armagan.book(stefan, 'Kreuzberg', 'Wannsee') 8 | passengerDatabase.update(armagan) 9 | 10 | printBookingHistory(armagan) 11 | 12 | console.log(passengerDatabase.findBy('location', 'Mitte')) 13 | -------------------------------------------------------------------------------- /week-4/lib/print-booking-history.js: -------------------------------------------------------------------------------- 1 | const colors = require('colors') 2 | 3 | function printBooking(booking) { 4 | console.log(`${colors.blue(booking.passenger.name)} booked ${colors.blue(booking.driver.name)} to travel from ${colors.bgRed.white(booking.origin)} to ${colors.bgRed.white(booking.destination)}`) 5 | } 6 | 7 | function printBookingHistory(passenger) { 8 | if (passenger.bookings.length == 0) 9 | return console.log(`${colors.blue(passenger.name)} has no bookings yet.`) 10 | 11 | passenger.bookings.forEach(printBooking) 12 | } 13 | 14 | module.exports = printBookingHistory 15 | -------------------------------------------------------------------------------- /week-4/models/booking.js: -------------------------------------------------------------------------------- 1 | class Booking { 2 | constructor(driver, passenger, origin, destination) { 3 | this.driver = driver 4 | this.passenger = passenger 5 | this.origin = origin 6 | this.destination = destination 7 | } 8 | } 9 | 10 | module.exports = Booking 11 | -------------------------------------------------------------------------------- /week-4/models/driver.js: -------------------------------------------------------------------------------- 1 | class Driver { 2 | constructor(name, location) { 3 | this.name = name 4 | this.location = location 5 | } 6 | 7 | static create({name, location}) { 8 | return new Driver(name, location) 9 | } 10 | } 11 | 12 | module.exports = Driver 13 | -------------------------------------------------------------------------------- /week-4/models/passenger.js: -------------------------------------------------------------------------------- 1 | const Booking = require('./booking') 2 | const uuid = require('uuid') 3 | 4 | class Passenger { 5 | constructor(id = uuid.v4(), name, location, bookings = []) { 6 | this.id = id 7 | 8 | this.name = name 9 | this.location = location 10 | this.bookings = bookings 11 | } 12 | 13 | book(driver, origin, destination) { 14 | const booking = new Booking(driver, this, origin, destination) 15 | 16 | this.bookings.push(booking) 17 | 18 | return booking 19 | } 20 | 21 | static create({id, name, location, bookings}) { 22 | return new Passenger(id, name, location, bookings) 23 | } 24 | } 25 | 26 | module.exports = Passenger 27 | -------------------------------------------------------------------------------- /week-4/package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "week-3", 3 | "version": "1.0.0", 4 | "description": "In-class examples for week-3", 5 | "main": "index.js", 6 | "scripts": { 7 | "start": "node index.js", 8 | "watch": "nodemon --ignore '*.json'", 9 | "test": "echo \"Error: no test specified\" && exit 1" 10 | }, 11 | "author": "Armagan Amcalar ", 12 | "license": "ISC", 13 | "dependencies": { 14 | "colors": "^1.4.0", 15 | "flatted": "^2.0.1", 16 | "uuid": "^7.0.2" 17 | }, 18 | "devDependencies": { 19 | "nodemon": "^2.0.2" 20 | } 21 | } 22 | -------------------------------------------------------------------------------- /week-5/.vscode/launch.json: -------------------------------------------------------------------------------- 1 | { 2 | // Use IntelliSense to learn about possible attributes. 3 | // Hover to view descriptions of existing attributes. 4 | // For more information, visit: https://go.microsoft.com/fwlink/?linkid=830387 5 | "version": "0.2.0", 6 | "configurations": [ 7 | { 8 | "type": "node", 9 | "request": "launch", 10 | "name": "Launch Current File", 11 | "program": "${file}", 12 | "skipFiles": [ 13 | "/**" 14 | ] 15 | }, 16 | { 17 | "type": "node", 18 | "request": "launch", 19 | "name": "Launch Program", 20 | "skipFiles": [ 21 | "/**" 22 | ], 23 | "program": "${workspaceFolder}/index.js" 24 | } 25 | ] 26 | } 27 | -------------------------------------------------------------------------------- /week-5/create-records.js: -------------------------------------------------------------------------------- 1 | const Passenger = require('./models/passenger') 2 | const Driver = require('./models/driver') 3 | const passengerDatabase = require('./database/passenger-database') 4 | const driverDatabase = require('./database/driver-database') 5 | 6 | const printBookingHistory = require('./lib/print-booking-history') 7 | 8 | const armagan = Passenger.create({name: 'Armagan', location: 'Kreuzberg'}) 9 | const mert = Passenger.create({name: 'Mert', location: 'Mitte'}) 10 | const stefan = Driver.create({ name: 'Stefan', location: 'Treptower Park' }) 11 | 12 | armagan.book(stefan, 'Kreuzberg', 'Neukolln') 13 | armagan.book(stefan, 'Neukolln', 'Mitte') 14 | armagan.book(stefan, 'Mitte', 'Kreuzberg') 15 | armagan.book(stefan, 'Kreuzberg', 'SXF') 16 | mert.book(stefan, 'Mitte', 'Kreuzberg') 17 | 18 | async function main() { 19 | try { 20 | await passengerDatabase.save([armagan, mert]) 21 | 22 | await driverDatabase.save([stefan]) 23 | 24 | const betul = Passenger.create({ name: 'Betul', location: 'Tegel' }) 25 | 26 | await passengerDatabase.insert(betul) 27 | const passengers = await passengerDatabase.load() 28 | passengers.forEach(printBookingHistory) 29 | } catch (e) { 30 | return console.log(e) 31 | } 32 | } 33 | 34 | main() 35 | -------------------------------------------------------------------------------- /week-5/database/base-database.js: -------------------------------------------------------------------------------- 1 | const fs = require('fs') 2 | const flatted = require('flatted') 3 | 4 | class BaseDatabase { 5 | constructor(model) { 6 | this.model = model 7 | this.filename = model.name.toLowerCase() 8 | } 9 | 10 | save(objects) { 11 | return new Promise((resolve, reject) => { 12 | fs.writeFile(`${__dirname}/${this.filename}.json`, flatted.stringify(objects, null, 2), (err) => { 13 | if (err) return reject(err) 14 | resolve() 15 | }) 16 | }) 17 | } 18 | 19 | load() { 20 | return new Promise((resolve, reject) => { 21 | fs.readFile(`${__dirname}/${this.filename}.json`, 'utf8', (err, file) => { 22 | if (err) return reject(err) 23 | 24 | const objects = flatted.parse(file) 25 | 26 | resolve(objects.map(this.model.create)) 27 | }) 28 | }) 29 | } 30 | 31 | async insert(object) { 32 | const objects = await this.load() 33 | 34 | if (!(object instanceof this.model)) { 35 | object = this.model.create(object) 36 | } 37 | 38 | await this.save(objects.concat(object)) 39 | 40 | return object 41 | } 42 | 43 | async remove(index) { 44 | const objects = await this.load() 45 | 46 | objects.splice(index, 1) 47 | await this.save(objects) 48 | } 49 | 50 | async removeBy(property, value) { 51 | const objects = await this.load() 52 | 53 | const index = objects.findIndex(o => o[property] == value) 54 | 55 | if (index == -1) throw new Error(`Cannot find ${this.model.name} instance with ${property} ${value}`) 56 | 57 | objects.splice(index, 1) 58 | await this.save(objects) 59 | } 60 | 61 | async update(object) { 62 | const objects = await this.load() 63 | 64 | const index = objects.findIndex(o => o.id == object.id) 65 | 66 | if (index == -1) throw new Error(`Cannot find ${this.model.name} instance with id ${object.id}`) 67 | 68 | objects.splice(index, 1, object) 69 | await this.save(objects) 70 | } 71 | 72 | async find(id) { 73 | const objects = await this.load() 74 | return objects.find(o => o.id == id) 75 | } 76 | 77 | async findBy(property, value) { 78 | return (await this.load()).find(o => o[property] == value) 79 | } 80 | } 81 | 82 | module.exports = BaseDatabase 83 | -------------------------------------------------------------------------------- /week-5/database/driver-database.js: -------------------------------------------------------------------------------- 1 | const BaseDatabase = require('./base-database') 2 | const Driver = require('../models/driver') 3 | 4 | class DriverDatabase extends BaseDatabase { 5 | findByDriverName(name) { 6 | return this.findBy('name', name) 7 | } 8 | 9 | findByLocation(location) { 10 | return this.findBy('location', location) 11 | } 12 | } 13 | 14 | module.exports = new DriverDatabase(Driver) 15 | -------------------------------------------------------------------------------- /week-5/database/driver.json: -------------------------------------------------------------------------------- 1 | [[ 2 | "1" 3 | ],{ 4 | "id": "2", 5 | "name": "3", 6 | "location": "4" 7 | },"33722c67-7b62-4dbb-81cb-e64a9aba6713","Stefan","Treptower Park"] -------------------------------------------------------------------------------- /week-5/database/index.js: -------------------------------------------------------------------------------- 1 | const passengerDatabase = require('./passenger-database') 2 | const driverDatabase = require('./driver-database') 3 | 4 | module.exports = { passengerDatabase, driverDatabase } 5 | -------------------------------------------------------------------------------- /week-5/database/passenger-database.js: -------------------------------------------------------------------------------- 1 | const BaseDatabase = require('./base-database') 2 | const Passenger = require('../models/passenger') 3 | 4 | class PassengerDatabase extends BaseDatabase { 5 | async findByName(name) { 6 | const objects = await this.load() 7 | 8 | return objects.find(o => o.name == name) 9 | } 10 | } 11 | 12 | module.exports = new PassengerDatabase(Passenger) 13 | -------------------------------------------------------------------------------- /week-5/database/passenger.json: -------------------------------------------------------------------------------- 1 | [[ 2 | "1", 3 | "2", 4 | "3" 5 | ],{ 6 | "id": "4", 7 | "name": "5", 8 | "location": "6", 9 | "bookings": "7" 10 | },{ 11 | "id": "8", 12 | "name": "9", 13 | "location": "10", 14 | "bookings": "11" 15 | },{ 16 | "id": "12", 17 | "name": "13", 18 | "location": "14", 19 | "bookings": "15" 20 | },"01ed8529-ad31-4351-b092-21cabb931e16","Armagan","Kreuzberg",[ 21 | "16", 22 | "17", 23 | "18", 24 | "19" 25 | ],"68e6a08c-f4cf-486e-b1df-dac92576bc48","Mert","Mitte",[ 26 | "20" 27 | ],"db80ba9d-c327-4222-b62c-f5f230a3e0b6","Betul","Tegel",[ 28 | "21", 29 | "22", 30 | "23" 31 | ],{ 32 | "driver": "24", 33 | "passenger": "25", 34 | "origin": "6", 35 | "destination": "26" 36 | },{ 37 | "driver": "24", 38 | "passenger": "25", 39 | "origin": "26", 40 | "destination": "10" 41 | },{ 42 | "driver": "24", 43 | "passenger": "25", 44 | "origin": "10", 45 | "destination": "6" 46 | },{ 47 | "driver": "24", 48 | "passenger": "25", 49 | "origin": "6", 50 | "destination": "27" 51 | },{ 52 | "driver": "24", 53 | "passenger": "28", 54 | "origin": "10", 55 | "destination": "6" 56 | },{ 57 | "driver": "29", 58 | "passenger": "30", 59 | "origin": "27", 60 | "destination": "6" 61 | },{ 62 | "driver": "31", 63 | "passenger": "32", 64 | "origin": "6", 65 | "destination": "33" 66 | },{ 67 | "driver": "34", 68 | "passenger": "3", 69 | "origin": "33", 70 | "destination": "10" 71 | },{ 72 | "id": "35", 73 | "name": "36", 74 | "location": "37" 75 | },{ 76 | "id": "4", 77 | "name": "5", 78 | "location": "6", 79 | "bookings": "7" 80 | },"Neukolln","SXF",{ 81 | "id": "8", 82 | "name": "9", 83 | "location": "10", 84 | "bookings": "11" 85 | },{ 86 | "id": "35", 87 | "name": "36", 88 | "location": "37" 89 | },{ 90 | "id": "12", 91 | "name": "13", 92 | "location": "14", 93 | "bookings": "15" 94 | },{ 95 | "id": "35", 96 | "name": "36", 97 | "location": "37" 98 | },{ 99 | "id": "12", 100 | "name": "13", 101 | "location": "14", 102 | "bookings": "15" 103 | },"TXL",{ 104 | "id": "35", 105 | "name": "36", 106 | "location": "37" 107 | },"33722c67-7b62-4dbb-81cb-e64a9aba6713","Stefan","Treptower Park"] -------------------------------------------------------------------------------- /week-5/fetch-records.js: -------------------------------------------------------------------------------- 1 | const { passengerDatabase, driverDatabase } = require('./database') 2 | const printBookingHistory = require('./lib/print-booking-history') 3 | 4 | async function main() { 5 | const stefan = await driverDatabase.findBy('name', 'Stefan') 6 | const armagan = await passengerDatabase.findByName('Armagan') 7 | 8 | armagan.book(stefan, 'Kreuzberg', 'Wannsee') 9 | passengerDatabase.update(armagan) 10 | 11 | printBookingHistory(armagan) 12 | 13 | console.log(await passengerDatabase.findBy('location', 'Mitte')) 14 | } 15 | 16 | main() 17 | -------------------------------------------------------------------------------- /week-5/index.js: -------------------------------------------------------------------------------- 1 | const express = require('express') 2 | const bodyParser = require('body-parser') 3 | const passengersRouter = require('./routes/passengers') 4 | const indexRouter = require('./routes/index') 5 | 6 | const app = express() 7 | app.use(bodyParser.json()) 8 | 9 | app.set('view engine', 'pug') 10 | 11 | app.use('/passengers', passengersRouter) 12 | app.use('/', indexRouter) 13 | 14 | app.listen(3000, () => { 15 | console.log('started listening on 3000') 16 | }) 17 | -------------------------------------------------------------------------------- /week-5/lib/print-booking-history.js: -------------------------------------------------------------------------------- 1 | const colors = require('colors') 2 | 3 | function printBooking(booking) { 4 | console.log(`${colors.blue(booking.passenger.name)} booked ${colors.blue(booking.driver.name)} to travel from ${colors.bgRed.white(booking.origin)} to ${colors.bgRed.white(booking.destination)}`) 5 | } 6 | 7 | function printBookingHistory(passenger) { 8 | if (passenger.bookings.length == 0) 9 | return console.log(`${colors.blue(passenger.name)} has no bookings yet.`) 10 | 11 | passenger.bookings.forEach(printBooking) 12 | } 13 | 14 | module.exports = printBookingHistory 15 | -------------------------------------------------------------------------------- /week-5/models/booking.js: -------------------------------------------------------------------------------- 1 | class Booking { 2 | constructor(driver, passenger, origin, destination) { 3 | this.driver = driver 4 | this.passenger = passenger 5 | this.origin = origin 6 | this.destination = destination 7 | } 8 | } 9 | 10 | module.exports = Booking 11 | -------------------------------------------------------------------------------- /week-5/models/driver.js: -------------------------------------------------------------------------------- 1 | const uuid = require('uuid') 2 | 3 | class Driver { 4 | constructor(id = uuid.v4(), name, location) { 5 | this.id = id 6 | 7 | this.name = name 8 | this.location = location 9 | } 10 | 11 | static create({id, name, location}) { 12 | return new Driver(id, name, location) 13 | } 14 | } 15 | 16 | module.exports = Driver 17 | -------------------------------------------------------------------------------- /week-5/models/passenger.js: -------------------------------------------------------------------------------- 1 | const Booking = require('./booking') 2 | const uuid = require('uuid') 3 | 4 | class Passenger { 5 | constructor(id = uuid.v4(), name, location, bookings = []) { 6 | this.id = id 7 | 8 | this.name = name 9 | this.location = location 10 | this.bookings = bookings 11 | } 12 | 13 | book(driver, origin, destination) { 14 | const booking = new Booking(driver, this, origin, destination) 15 | 16 | this.bookings.push(booking) 17 | 18 | return booking 19 | } 20 | 21 | static create({id, name, location, bookings}) { 22 | return new Passenger(id, name, location, bookings) 23 | } 24 | } 25 | 26 | module.exports = Passenger 27 | -------------------------------------------------------------------------------- /week-5/package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "week-5", 3 | "version": "1.0.0", 4 | "description": "In-class examples for week-5", 5 | "main": "index.js", 6 | "scripts": { 7 | "start": "node index.js", 8 | "watch": "nodemon --ignore '*.json'", 9 | "test": "echo \"Error: no test specified\" && exit 1" 10 | }, 11 | "author": "Armagan Amcalar ", 12 | "license": "ISC", 13 | "dependencies": { 14 | "body-parser": "^1.19.0", 15 | "colors": "^1.4.0", 16 | "express": "^4.17.1", 17 | "flatted": "^2.0.1", 18 | "pug": "^2.0.4", 19 | "uuid": "^7.0.2" 20 | }, 21 | "devDependencies": { 22 | "nodemon": "^2.0.2" 23 | } 24 | } 25 | -------------------------------------------------------------------------------- /week-5/routes/index.js: -------------------------------------------------------------------------------- 1 | const router = require('express').Router() 2 | 3 | router.get('/', (req, res) => { 4 | res.render('index') 5 | }) 6 | 7 | module.exports = router 8 | -------------------------------------------------------------------------------- /week-5/routes/passengers.js: -------------------------------------------------------------------------------- 1 | const { passengerDatabase, driverDatabase } = require('../database') 2 | const flatted = require('flatted') 3 | 4 | const router = require('express').Router() 5 | 6 | router.get('/', async (req, res) => { 7 | const passengers = await passengerDatabase.load() 8 | 9 | res.render('passengers', { passengers }) 10 | }) 11 | 12 | router.post('/', async (req, res) => { 13 | const passenger = await passengerDatabase.insert(req.body) 14 | 15 | res.send(passenger) 16 | }) 17 | 18 | router.delete('/:passengerId', async (req, res) => { 19 | await passengerDatabase.removeBy('id', req.params.passengerId) 20 | 21 | res.send('OK') 22 | }) 23 | 24 | router.get('/:passengerId', async (req, res) => { 25 | const passenger = await passengerDatabase.find(req.params.passengerId) 26 | if (!passenger) return res.status(404).send('Cannot find passenger') 27 | res.render('passenger', { passenger }) 28 | }) 29 | 30 | router.post('/:passengerId/bookings', async (req, res) => { 31 | const { passengerId } = req.params 32 | const { driverId, origin, destination } = req.body 33 | 34 | const passenger = await passengerDatabase.find(passengerId) 35 | const driver = await driverDatabase.find(driverId) 36 | 37 | passenger.book(driver, origin, destination) 38 | 39 | await passengerDatabase.update(passenger) 40 | 41 | res.send('OK') 42 | }) 43 | 44 | module.exports = router 45 | -------------------------------------------------------------------------------- /week-5/views/index.pug: -------------------------------------------------------------------------------- 1 | extends layout 2 | block content 3 | p With this app, you can book a taxi from one place to another. 4 | -------------------------------------------------------------------------------- /week-5/views/layout.pug: -------------------------------------------------------------------------------- 1 | 2 | html(lang="en") 3 | head 4 | meta(charset="UTF-8") 5 | meta(name="viewport", content="width=device-width, initial-scale=1.0") 6 | title Taxi-booking app 7 | style. 8 | body { font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, Oxygen, Ubuntu, Cantarell, 'Open Sans', 'Helvetica Neue', sans-serif;} 9 | 10 | body 11 | h1 Hi! Welcome to taxi-booking app 12 | 13 | block content 14 | 15 | script(src="https://unpkg.com/axios/dist/axios.min.js") 16 | -------------------------------------------------------------------------------- /week-5/views/passenger.pug: -------------------------------------------------------------------------------- 1 | extends layout 2 | block content 3 | h2 Passenger Detail 4 | p #{passenger.name} 5 | 6 | h3 Booking History 7 | if passenger.bookings.length 8 | ol 9 | each booking in passenger.bookings 10 | li From #{booking.origin} to #{booking.destination} 11 | else 12 | p No bookings 13 | -------------------------------------------------------------------------------- /week-5/views/passengers.pug: -------------------------------------------------------------------------------- 1 | extends layout 2 | block content 3 | h2 Passengers 4 | p There are #{passengers.length} passengers waiting. 5 | 6 | ol 7 | each passenger in passengers 8 | li 9 | a(href=`/passengers/${passenger.id}`) #{passenger.name} 10 | -------------------------------------------------------------------------------- /week-6/.vscode/launch.json: -------------------------------------------------------------------------------- 1 | { 2 | // Use IntelliSense to learn about possible attributes. 3 | // Hover to view descriptions of existing attributes. 4 | // For more information, visit: https://go.microsoft.com/fwlink/?linkid=830387 5 | "version": "0.2.0", 6 | "configurations": [ 7 | { 8 | "type": "node", 9 | "request": "launch", 10 | "name": "Launch Current File", 11 | "program": "${file}", 12 | "skipFiles": [ 13 | "/**" 14 | ] 15 | }, 16 | { 17 | "type": "node", 18 | "request": "launch", 19 | "name": "Launch Program", 20 | "skipFiles": [ 21 | "/**" 22 | ], 23 | "program": "${workspaceFolder}/index.js" 24 | } 25 | ] 26 | } 27 | -------------------------------------------------------------------------------- /week-6/create-records.js: -------------------------------------------------------------------------------- 1 | const Passenger = require('./models/passenger') 2 | const Driver = require('./models/driver') 3 | const passengerService = require('./services/passenger-services') 4 | const driverService = require('./services/driver-services') 5 | 6 | const printBookingHistory = require('./lib/print-booking-history') 7 | 8 | const armagan = Passenger.create({name: 'Armagan', location: 'Kreuzberg'}) 9 | const mert = Passenger.create({name: 'Mert', location: 'Mitte'}) 10 | const stefan = Driver.create({ name: 'Stefan', location: 'Treptower Park' }) 11 | 12 | armagan.book(stefan, 'Kreuzberg', 'Neukolln') 13 | armagan.book(stefan, 'Neukolln', 'Mitte') 14 | armagan.book(stefan, 'Mitte', 'Kreuzberg') 15 | armagan.book(stefan, 'Kreuzberg', 'SXF') 16 | mert.book(stefan, 'Mitte', 'Kreuzberg') 17 | 18 | async function main() { 19 | try { 20 | await passengerService.save([armagan, mert]) 21 | 22 | await driverService.save([stefan]) 23 | 24 | const betul = Passenger.create({ name: 'Betul', location: 'Tegel' }) 25 | 26 | await passengerService.insert(betul) 27 | const passengers = await passengerService.load() 28 | passengers.forEach(printBookingHistory) 29 | } catch (e) { 30 | return console.log(e) 31 | } 32 | } 33 | 34 | main() 35 | -------------------------------------------------------------------------------- /week-6/database/passenger-database.js: -------------------------------------------------------------------------------- 1 | const BaseDatabase = require('./base-service') 2 | const Passenger = require('../models/passenger') 3 | const bookingDatabase = require('./booking-service') 4 | const driverDatabase = require('./driver-service') 5 | 6 | class PassengerDatabase extends BaseDatabase { 7 | async findByName(name) { 8 | return this.findBy('name', name) 9 | } 10 | 11 | async book(driverId, passengerId, origin, destination) { 12 | const passenger = await this.find(passengerId) 13 | const driver = await driverDatabase.find(driverId) 14 | 15 | const booking = await bookingDatabase.insert({ driver, passenger, origin, destination }) 16 | passenger.bookings.push(booking) 17 | 18 | await passenger.save() 19 | 20 | return booking 21 | } 22 | } 23 | 24 | module.exports = new PassengerDatabase(Passenger) 25 | -------------------------------------------------------------------------------- /week-6/fetch-records.js: -------------------------------------------------------------------------------- 1 | const { passengerService, driverService } = require('./services') 2 | const printBookingHistory = require('./lib/print-booking-history') 3 | 4 | async function main() { 5 | const stefan = await driverService.findBy('name', 'Stefan') 6 | const armagan = await passengerService.findByName('Armagan') 7 | 8 | armagan.book(stefan, 'Kreuzberg', 'Wannsee') 9 | passengerService.update(armagan) 10 | 11 | printBookingHistory(armagan) 12 | 13 | console.log(await passengerService.findBy('location', 'Mitte')) 14 | } 15 | 16 | main() 17 | -------------------------------------------------------------------------------- /week-6/index.js: -------------------------------------------------------------------------------- 1 | const express = require('express') 2 | const bodyParser = require('body-parser') 3 | const passengersRouter = require('./routes/passengers') 4 | const driversRouter = require('./routes/drivers') 5 | const bookingsRouter = require('./routes/bookings') 6 | const indexRouter = require('./routes/index') 7 | require('./mongo-connection') 8 | 9 | const app = express() 10 | app.use(bodyParser.json()) 11 | 12 | app.set('view engine', 'pug') 13 | 14 | app.use('/passengers', passengersRouter) 15 | app.use('/drivers', driversRouter) 16 | app.use('/bookings', bookingsRouter) 17 | app.use('/', indexRouter) 18 | 19 | app.listen(3000, () => { 20 | console.log('started listening on 3000') 21 | }) 22 | -------------------------------------------------------------------------------- /week-6/lib/print-booking-history.js: -------------------------------------------------------------------------------- 1 | const colors = require('colors') 2 | 3 | function printBooking(booking) { 4 | console.log(`${colors.blue(booking.passenger.name)} booked ${colors.blue(booking.driver.name)} to travel from ${colors.bgRed.white(booking.origin)} to ${colors.bgRed.white(booking.destination)}`) 5 | } 6 | 7 | function printBookingHistory(passenger) { 8 | if (passenger.bookings.length == 0) 9 | return console.log(`${colors.blue(passenger.name)} has no bookings yet.`) 10 | 11 | passenger.bookings.forEach(printBooking) 12 | } 13 | 14 | module.exports = printBookingHistory 15 | -------------------------------------------------------------------------------- /week-6/models/booking.js: -------------------------------------------------------------------------------- 1 | const mongoose = require('mongoose') 2 | 3 | const BookingSchema = new mongoose.Schema({ 4 | driver: { 5 | type: mongoose.Schema.Types.ObjectId, 6 | ref: 'Driver', 7 | autopopulate: { maxDepth: 1 } 8 | }, 9 | passenger: { 10 | type: mongoose.Schema.Types.ObjectId, 11 | ref: 'Passenger', 12 | autopopulate: { maxDepth: 1 } 13 | }, 14 | origin: String, 15 | destination: String 16 | }) 17 | 18 | BookingSchema.plugin(require('mongoose-autopopulate')) 19 | 20 | module.exports = mongoose.model('Booking', BookingSchema) 21 | -------------------------------------------------------------------------------- /week-6/models/driver.js: -------------------------------------------------------------------------------- 1 | const mongoose = require('mongoose') 2 | 3 | const DriverSchema = new mongoose.Schema({ 4 | name: String, 5 | location: String, 6 | age: { type: Number, required: true, min: 18 }, 7 | }) 8 | 9 | module.exports = mongoose.model('Driver', DriverSchema) 10 | -------------------------------------------------------------------------------- /week-6/models/passenger.js: -------------------------------------------------------------------------------- 1 | const mongoose = require('mongoose') 2 | 3 | const PassengerSchema = new mongoose.Schema({ 4 | name: { type: String, required: true, minlength: 2 }, 5 | location: String, 6 | bookings: [{ 7 | type: mongoose.Schema.Types.ObjectId, 8 | ref: 'Booking', 9 | autopopulate: { maxDepth: 2 } 10 | }] 11 | }) 12 | 13 | PassengerSchema.plugin(require('mongoose-autopopulate')) 14 | 15 | module.exports = mongoose.model('Passenger', PassengerSchema) 16 | -------------------------------------------------------------------------------- /week-6/mongo-connection.js: -------------------------------------------------------------------------------- 1 | const mongoose = require('mongoose') 2 | mongoose.connect('mongodb://localhost/week-6', { useNewUrlParser: true, useUnifiedTopology: true }) 3 | 4 | var db = mongoose.connection 5 | db.on('error', console.error.bind(console, 'connection error:')) 6 | db.once('open', function () { 7 | console.log('we are connected to mongodb!') 8 | }) 9 | -------------------------------------------------------------------------------- /week-6/package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "week-6", 3 | "version": "1.0.0", 4 | "description": "In-class examples for week-6", 5 | "main": "index.js", 6 | "scripts": { 7 | "start": "node index.js", 8 | "watch": "nodemon --ignore '*.json'", 9 | "test": "echo \"Error: no test specified\" && exit 1" 10 | }, 11 | "author": "Armagan Amcalar ", 12 | "license": "ISC", 13 | "dependencies": { 14 | "body-parser": "^1.19.0", 15 | "colors": "^1.4.0", 16 | "express": "^4.17.1", 17 | "flatted": "^2.0.1", 18 | "mongoose": "^5.9.9", 19 | "mongoose-autopopulate": "^0.12.2", 20 | "pug": "^2.0.4", 21 | "uuid": "^7.0.2" 22 | }, 23 | "devDependencies": { 24 | "nodemon": "^2.0.2" 25 | } 26 | } 27 | -------------------------------------------------------------------------------- /week-6/requests.http: -------------------------------------------------------------------------------- 1 | POST http://localhost:3000/passengers HTTP/1.1 2 | content-type: application/json 3 | 4 | {"name":"armagan", "location": "kuzguncuk"} 5 | 6 | ### 7 | 8 | GET http://localhost:3000/passengers/5ea70b28bbf7cb710503f2d4 HTTP/1.1 9 | 10 | ### 11 | DELETE http://localhost:3000/passengers/5e9de11c0b0ac272d7866b1b HTTP/1.1 12 | 13 | ### 14 | 15 | PATCH http://localhost:3000/passengers/5e9de4cd9f11eb7402e56941 HTTP/1.1 16 | content-type: application/json 17 | 18 | {"name": "new name4"} 19 | 20 | ### 21 | 22 | POST http://localhost:3000/drivers HTTP/1.1 23 | content-type: application/json 24 | 25 | {"name":"older driver", "age": 35, "location": "kuzguncuk"} 26 | 27 | ### 28 | DELETE http://localhost:3000/drivers/5ea70bf063b0877147bd3e6b HTTP/1.1 29 | 30 | ### 31 | POST http://localhost:3000/passengers/5ea70b28bbf7cb710503f2d4/bookings HTTP/1.1 32 | content-type: application/json 33 | 34 | {"driverId": "5ea724416e3b607ac41df63a", "origin": "Kabatas", "destination": "Pendik"} 35 | 36 | ### 37 | 38 | PATCH http://localhost:3000/drivers/5ea70c8063b0877147bd3e6c HTTP/1.1 39 | content-type: application/json 40 | 41 | {"name": "a newer surname"} 42 | -------------------------------------------------------------------------------- /week-6/routes/bookings.js: -------------------------------------------------------------------------------- 1 | const { bookingService } = require('../services') 2 | 3 | const router = require('express').Router() 4 | 5 | router.get('/', async (req, res) => { 6 | const bookings = await bookingService.load() 7 | 8 | res.render('bookings', { bookings }) 9 | }) 10 | 11 | router.get('/search', async (req, res) => { 12 | const origin = req.query.origin 13 | const destination = req.query.destination 14 | 15 | const query = {} 16 | 17 | if (origin) query.origin = origin 18 | if (destination) query.destination = destination 19 | 20 | const bookings = await bookingService.query(query) 21 | 22 | res.render('bookings', { bookings }) 23 | }) 24 | 25 | module.exports = router 26 | -------------------------------------------------------------------------------- /week-6/routes/drivers.js: -------------------------------------------------------------------------------- 1 | const { driverService } = require('../services') 2 | 3 | const router = require('express').Router() 4 | 5 | router.get('/', async (req, res) => { 6 | const drivers = await driverService.load() 7 | 8 | res.render('drivers', { drivers }) 9 | }) 10 | 11 | router.post('/', async (req, res) => { 12 | const driver = await driverService.insert(req.body) 13 | 14 | res.send(driver) 15 | }) 16 | 17 | router.delete('/:driverId', async (req, res) => { 18 | await driverService.removeBy('_id', req.params.driverId) 19 | 20 | res.send('OK') 21 | }) 22 | 23 | router.get('/young-drivers', async (req, res) => { 24 | const drivers = await driverService.findYoungDrivers() 25 | 26 | res.render('drivers', { drivers }) 27 | }) 28 | 29 | router.get('/:driverId', async (req, res) => { 30 | const driver = await driverService.find(req.params.driverId) 31 | if (!driver) return res.status(404).send('Cannot find driver') 32 | res.render('driver', { driver }) 33 | }) 34 | 35 | router.patch('/:driverId', async (req, res) => { 36 | const { driverId } = req.params 37 | const { name } = req.body 38 | 39 | await driverService.update(driverId, { name }) 40 | }) 41 | 42 | module.exports = router 43 | -------------------------------------------------------------------------------- /week-6/routes/index.js: -------------------------------------------------------------------------------- 1 | const router = require('express').Router() 2 | 3 | router.get('/', (req, res) => { 4 | res.render('index') 5 | }) 6 | 7 | module.exports = router 8 | -------------------------------------------------------------------------------- /week-6/routes/passengers.js: -------------------------------------------------------------------------------- 1 | const { passengerService, bookingService } = require('../services') 2 | 3 | const router = require('express').Router() 4 | 5 | router.get('/', async (req, res) => { 6 | const passengers = await passengerService.load() 7 | 8 | res.render('passengers', { passengers }) 9 | }) 10 | 11 | router.post('/', async (req, res, next) => { 12 | try { 13 | const passenger = await passengerService.insert(req.body) 14 | res.send(passenger) 15 | } catch(e) { 16 | next(e) 17 | } 18 | }) 19 | 20 | router.delete('/:passengerId', async (req, res) => { 21 | await passengerService.removeBy('_id', req.params.passengerId) 22 | 23 | res.send('OK') 24 | }) 25 | 26 | router.get('/:passengerId', async (req, res) => { 27 | const passenger = await passengerService.find(req.params.passengerId) 28 | 29 | if (!passenger) return res.status(404).send('Cannot find passenger') 30 | res.render('passenger', { passenger }) 31 | }) 32 | 33 | router.post('/:passengerId/bookings', async (req, res) => { 34 | const { passengerId } = req.params 35 | const { driverId, origin, destination } = req.body 36 | 37 | const booking = await bookingService.book(driverId, passengerId, origin, destination) 38 | 39 | res.send(booking) 40 | }) 41 | 42 | router.patch('/:passengerId', async (req, res) => { 43 | const { passengerId } = req.params 44 | const { name } = req.body 45 | 46 | await passengerService.update(passengerId, { name }) 47 | }) 48 | 49 | module.exports = router 50 | -------------------------------------------------------------------------------- /week-6/services/base-service.js: -------------------------------------------------------------------------------- 1 | class BaseService { 2 | constructor(model) { 3 | this.model = model 4 | } 5 | 6 | save(objects) { 7 | return this.model.insertMany(objects) 8 | } 9 | 10 | load() { 11 | return this.model.find() 12 | } 13 | 14 | async insert(object) { 15 | return await this.model.create(object) 16 | } 17 | 18 | async removeBy(property, value) { 19 | return this.model.deleteOne({ [property]: value }) 20 | } 21 | 22 | async update(id, object) { 23 | return this.model.findByIdAndUpdate(id, object) 24 | } 25 | 26 | async find(id) { 27 | return this.model.findById(id) 28 | } 29 | 30 | async query(obj) { 31 | return this.model.find(obj) 32 | } 33 | 34 | async findBy(property, value) { 35 | return this.model.find({ [property]: value }) 36 | } 37 | } 38 | 39 | module.exports = BaseService 40 | -------------------------------------------------------------------------------- /week-6/services/booking-service.js: -------------------------------------------------------------------------------- 1 | const BaseService = require('./base-service') 2 | const Booking = require('../models/booking') 3 | const driverService = require('./driver-service') 4 | const passengerService = require('./passenger-service') 5 | 6 | class BookingService extends BaseService { 7 | async findByPassengerId(passengerId) { 8 | return this.findBy('passenger', passengerId) 9 | } 10 | 11 | async findByDriverId(driverId) { 12 | return this.findBy('driver', driverId) 13 | } 14 | 15 | async book(driverId, passengerId, origin, destination) { 16 | const passenger = await passengerService.find(passengerId) 17 | const driver = await driverService.find(driverId) 18 | 19 | const booking = await this.insert({ driver, passenger, origin, destination }) 20 | passenger.bookings.push(booking) 21 | 22 | await passenger.save() 23 | 24 | return booking 25 | } 26 | } 27 | 28 | module.exports = new BookingService(Booking) 29 | -------------------------------------------------------------------------------- /week-6/services/driver-service.js: -------------------------------------------------------------------------------- 1 | const BaseService = require('./base-service') 2 | const Driver = require('../models/driver') 3 | 4 | class DriverService extends BaseService { 5 | async findByDriverName(name) { 6 | return this.findBy('name', name) 7 | } 8 | 9 | async findByLocation(location) { 10 | return this.findBy('location', location) 11 | } 12 | 13 | async findYoungDrivers() { 14 | return this.query({ 15 | age: { 16 | $lt: 30, 17 | $gte: 18 18 | } 19 | }) 20 | } 21 | } 22 | 23 | module.exports = new DriverService(Driver) 24 | -------------------------------------------------------------------------------- /week-6/services/driver.json: -------------------------------------------------------------------------------- 1 | [[ 2 | "1" 3 | ],{ 4 | "id": "2", 5 | "name": "3", 6 | "location": "4" 7 | },"33722c67-7b62-4dbb-81cb-e64a9aba6713","Stefan","Treptower Park"] -------------------------------------------------------------------------------- /week-6/services/index.js: -------------------------------------------------------------------------------- 1 | const passengerService = require('./passenger-service') 2 | const driverService = require('./driver-service') 3 | const bookingService = require('./booking-service') 4 | 5 | module.exports = { passengerService, driverService, bookingService } 6 | -------------------------------------------------------------------------------- /week-6/services/passenger-service.js: -------------------------------------------------------------------------------- 1 | const BaseService = require('./base-service') 2 | const Passenger = require('../models/passenger') 3 | 4 | class PassengerService extends BaseService { 5 | async findByName(name) { 6 | return this.findBy('name', name) 7 | } 8 | } 9 | 10 | module.exports = new PassengerService(Passenger) 11 | -------------------------------------------------------------------------------- /week-6/services/passenger.json: -------------------------------------------------------------------------------- 1 | [[ 2 | "1", 3 | "2", 4 | "3", 5 | "4", 6 | "5", 7 | "6", 8 | "7", 9 | "8", 10 | "9", 11 | "10", 12 | "11", 13 | "12", 14 | "13", 15 | "14", 16 | "15", 17 | "16", 18 | "17", 19 | "18", 20 | "19", 21 | "20" 22 | ],{ 23 | "id": "21", 24 | "name": "22", 25 | "location": "23", 26 | "bookings": "24" 27 | },{ 28 | "id": "25", 29 | "name": "26", 30 | "location": "27", 31 | "bookings": "28" 32 | },{ 33 | "id": "29", 34 | "name": "30", 35 | "location": "31", 36 | "bookings": "32" 37 | },{ 38 | "id": "33", 39 | "bookings": "34" 40 | },{ 41 | "id": "35", 42 | "bookings": "36" 43 | },{ 44 | "id": "37", 45 | "bookings": "38" 46 | },{ 47 | "id": "39", 48 | "bookings": "40" 49 | },{ 50 | "id": "41", 51 | "name": "42", 52 | "location": "43", 53 | "bookings": "44" 54 | },{ 55 | "id": "45", 56 | "bookings": "46" 57 | },{ 58 | "id": "47", 59 | "name": "48", 60 | "location": "49", 61 | "bookings": "50" 62 | },{ 63 | "id": "51", 64 | "name": "48", 65 | "location": "49", 66 | "bookings": "52" 67 | },{ 68 | "id": "53", 69 | "name": "48", 70 | "location": "49", 71 | "bookings": "54" 72 | },{ 73 | "id": "55", 74 | "name": "48", 75 | "location": "49", 76 | "bookings": "56" 77 | },{ 78 | "id": "57", 79 | "name": "48", 80 | "location": "49", 81 | "bookings": "58" 82 | },{ 83 | "id": "59", 84 | "name": "48", 85 | "location": "49", 86 | "bookings": "60" 87 | },{ 88 | "id": "61", 89 | "name": "48", 90 | "location": "49", 91 | "bookings": "62" 92 | },{ 93 | "id": "63", 94 | "name": "48", 95 | "location": "49", 96 | "bookings": "64" 97 | },{ 98 | "id": "65", 99 | "name": "48", 100 | "location": "49", 101 | "bookings": "66" 102 | },{ 103 | "id": "67", 104 | "name": "48", 105 | "location": "49", 106 | "bookings": "68" 107 | },{ 108 | "id": "69", 109 | "name": "48", 110 | "location": "49", 111 | "bookings": "70" 112 | },"01ed8529-ad31-4351-b092-21cabb931e16","Armagan","Kreuzberg",[ 113 | "71", 114 | "72", 115 | "73", 116 | "74" 117 | ],"68e6a08c-f4cf-486e-b1df-dac92576bc48","Mert","Mitte",[ 118 | "75" 119 | ],"db80ba9d-c327-4222-b62c-f5f230a3e0b6","Betul","Tegel",[ 120 | "76", 121 | "77", 122 | "78" 123 | ],"92f71ad7-a806-41f6-bd33-59fcc754162f",[],"269e3f80-42bf-4409-af1e-b897dedfde4c",[],"59be8bc5-f28a-48e3-a98b-a7a7d40c4dfd",[],"7fa6f1dc-76b5-450b-8aba-f73ea0104bd9",[],"66f47f82-063c-43d7-9c6e-5311f9d31fa2","new passenger","Moda",[],"b8ca32d3-f610-4e41-8534-b53afb5c01fd",[],"ede41daa-a3d4-427f-96aa-94dda0c0299a","weird client","kuzguncuk",[],"88b04838-742d-4aa2-9e9f-b06a363d454c",[],"c2c00469-2022-4f27-b3e9-5705535c2a42",[],"1d569e2b-d8a3-410f-a957-85e55896ca11",[],"437f691b-a097-4daf-ba56-ffff33bc2935",[],"91effbd5-e751-4403-8bd6-aec6fe335806",[],"05a1e9a3-da4a-46a3-8086-65933d7cc1e3",[],"d50628c1-6922-43ef-beeb-df9073211b5a",[],"55c0329f-60c3-4810-9388-a26f096a6449",[],"e36afcbc-8349-43a4-b8e8-445ab464050a",[],"5d5c3d85-b328-457e-8598-4d0fe69c252c",[],{ 124 | "driver": "79", 125 | "passenger": "80", 126 | "origin": "23", 127 | "destination": "81" 128 | },{ 129 | "driver": "79", 130 | "passenger": "80", 131 | "origin": "81", 132 | "destination": "27" 133 | },{ 134 | "driver": "79", 135 | "passenger": "80", 136 | "origin": "27", 137 | "destination": "23" 138 | },{ 139 | "driver": "79", 140 | "passenger": "80", 141 | "origin": "23", 142 | "destination": "82" 143 | },{ 144 | "driver": "79", 145 | "passenger": "83", 146 | "origin": "27", 147 | "destination": "23" 148 | },{ 149 | "driver": "84", 150 | "passenger": "85", 151 | "origin": "82", 152 | "destination": "23" 153 | },{ 154 | "driver": "86", 155 | "passenger": "87", 156 | "origin": "23", 157 | "destination": "88" 158 | },{ 159 | "driver": "89", 160 | "passenger": "90", 161 | "origin": "88", 162 | "destination": "27" 163 | },{ 164 | "id": "91", 165 | "name": "92", 166 | "location": "93" 167 | },{ 168 | "id": "21", 169 | "name": "22", 170 | "location": "23", 171 | "bookings": "24" 172 | },"Neukolln","SXF",{ 173 | "id": "25", 174 | "name": "26", 175 | "location": "27", 176 | "bookings": "28" 177 | },{ 178 | "id": "91", 179 | "name": "92", 180 | "location": "93" 181 | },{ 182 | "id": "29", 183 | "name": "30", 184 | "location": "31", 185 | "bookings": "32" 186 | },{ 187 | "id": "91", 188 | "name": "92", 189 | "location": "93" 190 | },{ 191 | "id": "29", 192 | "name": "30", 193 | "location": "31", 194 | "bookings": "32" 195 | },"TXL",{ 196 | "id": "91", 197 | "name": "92", 198 | "location": "93" 199 | },{ 200 | "id": "29", 201 | "name": "30", 202 | "location": "31", 203 | "bookings": "32" 204 | },"33722c67-7b62-4dbb-81cb-e64a9aba6713","Stefan","Treptower Park"] -------------------------------------------------------------------------------- /week-6/views/bookings.pug: -------------------------------------------------------------------------------- 1 | extends layout 2 | block content 3 | h2 Bookings 4 | p There are #{bookings.length} bookings. 5 | 6 | ol 7 | each booking in bookings 8 | li 9 | p Passenger: 10 | a(href=`/passengers/${booking.passenger.id}`) #{booking.passenger.name} 11 | p Driver: 12 | a(href=`/drivers/${booking.driver.id}`) #{booking.driver.name} 13 | p Origin: #{booking.origin} 14 | p Destination: #{booking.destination} 15 | -------------------------------------------------------------------------------- /week-6/views/driver.pug: -------------------------------------------------------------------------------- 1 | extends layout 2 | block content 3 | h2 Driver Detail 4 | p #{driver.name} 5 | p #{driver.age} 6 | p #{driver.location} 7 | -------------------------------------------------------------------------------- /week-6/views/drivers.pug: -------------------------------------------------------------------------------- 1 | extends layout 2 | block content 3 | h2 Drivers 4 | p There are #{drivers.length} drivers waiting. 5 | 6 | ol 7 | each driver in drivers 8 | li 9 | a(href=`/drivers/${driver.id}`) #{driver.name} 10 | -------------------------------------------------------------------------------- /week-6/views/index.pug: -------------------------------------------------------------------------------- 1 | extends layout 2 | block content 3 | p With this app, you can book a taxi from one place to another. 4 | -------------------------------------------------------------------------------- /week-6/views/layout.pug: -------------------------------------------------------------------------------- 1 | 2 | html(lang="en") 3 | head 4 | meta(charset="UTF-8") 5 | meta(name="viewport", content="width=device-width, initial-scale=1.0") 6 | title Taxi-booking app 7 | style. 8 | body { font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, Oxygen, Ubuntu, Cantarell, 'Open Sans', 'Helvetica Neue', sans-serif;} 9 | 10 | body 11 | h1 Hi! Welcome to taxi-booking app 12 | 13 | block content 14 | 15 | script(src="https://unpkg.com/axios/dist/axios.min.js") 16 | -------------------------------------------------------------------------------- /week-6/views/passenger.pug: -------------------------------------------------------------------------------- 1 | extends layout 2 | block content 3 | h2 Passenger Detail 4 | p #{passenger.name} 5 | 6 | h3 Booking History 7 | if passenger.bookings.length 8 | ol 9 | each booking in passenger.bookings 10 | li From #{booking.origin} to #{booking.destination} with #{booking.driver.name} 11 | else 12 | p No bookings 13 | -------------------------------------------------------------------------------- /week-6/views/passengers.pug: -------------------------------------------------------------------------------- 1 | extends layout 2 | block content 3 | h2 Passengers 4 | p There are #{passengers.length} passengers waiting. 5 | 6 | ol 7 | each passenger in passengers 8 | li 9 | a(href=`/passengers/${passenger.id}`) #{passenger.name} 10 | -------------------------------------------------------------------------------- /week-7/.bin/www: -------------------------------------------------------------------------------- 1 | const app = require('../') 2 | 3 | app.listen(3000, () => { 4 | console.log('started listening on 3000') 5 | }) 6 | -------------------------------------------------------------------------------- /week-7/.vscode/launch.json: -------------------------------------------------------------------------------- 1 | { 2 | // Use IntelliSense to learn about possible attributes. 3 | // Hover to view descriptions of existing attributes. 4 | // For more information, visit: https://go.microsoft.com/fwlink/?linkid=830387 5 | "version": "0.2.0", 6 | "configurations": [ 7 | { 8 | "type": "node", 9 | "request": "launch", 10 | "name": "Launch Current File", 11 | "program": "${file}", 12 | "skipFiles": [ 13 | "/**" 14 | ] 15 | }, 16 | { 17 | "type": "node", 18 | "request": "launch", 19 | "name": "Launch Program", 20 | "skipFiles": [ 21 | "/**" 22 | ], 23 | "program": "${workspaceFolder}/index.js" 24 | } 25 | ] 26 | } 27 | -------------------------------------------------------------------------------- /week-7/.vscode/settings.json: -------------------------------------------------------------------------------- 1 | 123,125 -------------------------------------------------------------------------------- /week-7/__mocks__/colors.js: -------------------------------------------------------------------------------- 1 | const colors = jest.createMockFromModule('colors') 2 | 3 | const unity = s => s 4 | 5 | colors.bgRed = {} 6 | colors.blue = unity 7 | colors.bgRed.white = unity 8 | 9 | module.exports = colors 10 | -------------------------------------------------------------------------------- /week-7/create-records.js: -------------------------------------------------------------------------------- 1 | const Passenger = require('./models/passenger') 2 | const Driver = require('./models/driver') 3 | const passengerService = require('./services/passenger-services') 4 | const driverService = require('./services/driver-services') 5 | 6 | const printBookingHistory = require('./lib/print-booking-history') 7 | 8 | const armagan = Passenger.create({name: 'Armagan', location: 'Kreuzberg'}) 9 | const mert = Passenger.create({name: 'Mert', location: 'Mitte'}) 10 | const stefan = Driver.create({ name: 'Stefan', location: 'Treptower Park' }) 11 | 12 | armagan.book(stefan, 'Kreuzberg', 'Neukolln') 13 | armagan.book(stefan, 'Neukolln', 'Mitte') 14 | armagan.book(stefan, 'Mitte', 'Kreuzberg') 15 | armagan.book(stefan, 'Kreuzberg', 'SXF') 16 | mert.book(stefan, 'Mitte', 'Kreuzberg') 17 | 18 | async function main() { 19 | try { 20 | await passengerService.save([armagan, mert]) 21 | 22 | await driverService.save([stefan]) 23 | 24 | const betul = Passenger.create({ name: 'Betul', location: 'Tegel' }) 25 | 26 | await passengerService.insert(betul) 27 | const passengers = await passengerService.load() 28 | passengers.forEach(printBookingHistory) 29 | } catch (e) { 30 | return console.log(e) 31 | } 32 | } 33 | 34 | main() 35 | -------------------------------------------------------------------------------- /week-7/fetch-records.js: -------------------------------------------------------------------------------- 1 | const { passengerService, driverService } = require('./services') 2 | const printBookingHistory = require('./lib/print-booking-history') 3 | 4 | async function main() { 5 | const stefan = await driverService.findBy('name', 'Stefan') 6 | const armagan = await passengerService.findByName('Armagan') 7 | 8 | armagan.book(stefan, 'Kreuzberg', 'Wannsee') 9 | passengerService.update(armagan) 10 | 11 | printBookingHistory(armagan) 12 | 13 | console.log(await passengerService.findBy('location', 'Mitte')) 14 | } 15 | 16 | main() 17 | -------------------------------------------------------------------------------- /week-7/index.js: -------------------------------------------------------------------------------- 1 | const express = require('express') 2 | const bodyParser = require('body-parser') 3 | const passengersRouter = require('./routes/passengers') 4 | const driversRouter = require('./routes/drivers') 5 | const bookingsRouter = require('./routes/bookings') 6 | const indexRouter = require('./routes/index') 7 | require('./mongo-connection') 8 | 9 | const app = express() 10 | app.use(bodyParser.json()) 11 | 12 | app.set('view engine', 'pug') 13 | 14 | app.use('/passengers', passengersRouter) 15 | app.use('/drivers', driversRouter) 16 | app.use('/bookings', bookingsRouter) 17 | app.use('/', indexRouter) 18 | 19 | module.exports = app 20 | -------------------------------------------------------------------------------- /week-7/jest.config.js: -------------------------------------------------------------------------------- 1 | module.exports = { 2 | testEnvironment: 'node' 3 | }; 4 | -------------------------------------------------------------------------------- /week-7/lib/__tests__/print-booking-history.test.js: -------------------------------------------------------------------------------- 1 | const printBookingHistory = require('../print-booking-history') 2 | 3 | test('prints passenger bookings when a passenger has a booking', () => { 4 | // initialization 5 | const passenger = { 6 | name: 'Armagan', 7 | bookings: [{ 8 | passenger: { name: 'Armagan' }, 9 | driver: { name: 'Stefan' }, 10 | origin: 'Kreuzberg', 11 | destination: 'Neukolln' 12 | }] 13 | } 14 | const consoleSpy = jest.spyOn(console, 'log') 15 | 16 | // test 17 | printBookingHistory(passenger) 18 | 19 | // assertions 20 | expect(consoleSpy).toHaveBeenCalledWith('Armagan booked Stefan to travel from Kreuzberg to Neukolln') 21 | 22 | // teardown 23 | consoleSpy.mockRestore() 24 | }) 25 | 26 | test('prints warning message when a passenger has no bookings', () => { 27 | // initialization 28 | const passenger = { 29 | name: 'Armagan', 30 | bookings: [] 31 | } 32 | const consoleSpy = jest.spyOn(console, 'log') 33 | 34 | // test 35 | printBookingHistory(passenger) 36 | 37 | // assertions 38 | expect(consoleSpy).toHaveBeenCalledWith('Armagan has no bookings yet.') 39 | 40 | // teardown 41 | consoleSpy.mockRestore() 42 | }) 43 | -------------------------------------------------------------------------------- /week-7/lib/print-booking-history.js: -------------------------------------------------------------------------------- 1 | const colors = require('colors') 2 | 3 | function printBooking(booking) { 4 | console.log(`${colors.blue(booking.passenger.name)} booked ${colors.blue(booking.driver.name)} to travel from ${colors.bgRed.white(booking.origin)} to ${colors.bgRed.white(booking.destination)}`) 5 | } 6 | 7 | function printBookingHistory(passenger) { 8 | if (passenger.bookings.length == 0) 9 | return console.log(`${colors.blue(passenger.name)} has no bookings yet.`) 10 | 11 | passenger.bookings.forEach(printBooking) 12 | } 13 | 14 | module.exports = printBookingHistory 15 | -------------------------------------------------------------------------------- /week-7/models/booking.js: -------------------------------------------------------------------------------- 1 | const mongoose = require('mongoose') 2 | 3 | const BookingSchema = new mongoose.Schema({ 4 | driver: { 5 | type: mongoose.Schema.Types.ObjectId, 6 | ref: 'Driver', 7 | autopopulate: { maxDepth: 1 } 8 | }, 9 | passenger: { 10 | type: mongoose.Schema.Types.ObjectId, 11 | ref: 'Passenger', 12 | autopopulate: { maxDepth: 1 } 13 | }, 14 | origin: String, 15 | destination: String 16 | }) 17 | 18 | BookingSchema.plugin(require('mongoose-autopopulate')) 19 | 20 | module.exports = mongoose.model('Booking', BookingSchema) 21 | -------------------------------------------------------------------------------- /week-7/models/driver.js: -------------------------------------------------------------------------------- 1 | const mongoose = require('mongoose') 2 | 3 | const DriverSchema = new mongoose.Schema({ 4 | name: String, 5 | location: String, 6 | age: { type: Number, required: true, min: 18 }, 7 | }) 8 | 9 | module.exports = mongoose.model('Driver', DriverSchema) 10 | -------------------------------------------------------------------------------- /week-7/models/passenger.js: -------------------------------------------------------------------------------- 1 | const mongoose = require('mongoose') 2 | 3 | const PassengerSchema = new mongoose.Schema({ 4 | name: { type: String, required: true, minlength: 2 }, 5 | location: String, 6 | bookings: [{ 7 | type: mongoose.Schema.Types.ObjectId, 8 | ref: 'Booking', 9 | autopopulate: { maxDepth: 2 } 10 | }] 11 | }) 12 | 13 | PassengerSchema.plugin(require('mongoose-autopopulate')) 14 | 15 | module.exports = mongoose.model('Passenger', PassengerSchema) 16 | -------------------------------------------------------------------------------- /week-7/mongo-connection.js: -------------------------------------------------------------------------------- 1 | const mongoose = require('mongoose') 2 | mongoose.connect('mongodb://localhost/week-7', { useNewUrlParser: true, useUnifiedTopology: true }) 3 | 4 | var db = mongoose.connection 5 | db.on('error', console.error.bind(console, 'connection error:')) 6 | db.once('open', function () { 7 | console.log('we are connected to mongodb!') 8 | }) 9 | -------------------------------------------------------------------------------- /week-7/package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "week-7", 3 | "version": "1.0.0", 4 | "description": "In-class examples for week-7", 5 | "main": "index.js", 6 | "scripts": { 7 | "start": "node .bin/www", 8 | "watch": "nodemon --ignore '*.json' .bin/www", 9 | "test": "jest --coverage --verbose", 10 | "test-watch": "jest --coverage --verbose --watch" 11 | }, 12 | "author": "Armagan Amcalar ", 13 | "license": "ISC", 14 | "dependencies": { 15 | "body-parser": "^1.19.0", 16 | "colors": "^1.4.0", 17 | "express": "^4.17.1", 18 | "flatted": "^2.0.1", 19 | "mongoose": "^5.9.9", 20 | "mongoose-autopopulate": "^0.12.2", 21 | "pug": "^2.0.4", 22 | "uuid": "^7.0.2" 23 | }, 24 | "devDependencies": { 25 | "jest": "^26.0.1", 26 | "nodemon": "^2.0.2", 27 | "supertest": "^4.0.2" 28 | } 29 | } 30 | -------------------------------------------------------------------------------- /week-7/requests.http: -------------------------------------------------------------------------------- 1 | POST http://localhost:3000/passengers HTTP/1.1 2 | content-type: application/json 3 | 4 | {"name":"armagan", "location": "kuzguncuk"} 5 | 6 | ### 7 | 8 | GET http://localhost:3000/passengers/5ea70b28bbf7cb710503f2d4 HTTP/1.1 9 | 10 | ### 11 | DELETE http://localhost:3000/passengers/5e9de11c0b0ac272d7866b1b HTTP/1.1 12 | 13 | ### 14 | 15 | PATCH http://localhost:3000/passengers/5e9de4cd9f11eb7402e56941 HTTP/1.1 16 | content-type: application/json 17 | 18 | {"name": "new name4"} 19 | 20 | ### 21 | 22 | POST http://localhost:3000/drivers HTTP/1.1 23 | content-type: application/json 24 | 25 | {"name":"older driver", "age": 35, "location": "kuzguncuk"} 26 | 27 | ### 28 | DELETE http://localhost:3000/drivers/5ea70bf063b0877147bd3e6b HTTP/1.1 29 | 30 | ### 31 | POST http://localhost:3000/passengers/5ea70b28bbf7cb710503f2d4/bookings HTTP/1.1 32 | content-type: application/json 33 | 34 | {"driverId": "5ea724416e3b607ac41df63a", "origin": "Kabatas", "destination": "Pendik"} 35 | 36 | ### 37 | 38 | PATCH http://localhost:3000/drivers/5ea70c8063b0877147bd3e6c HTTP/1.1 39 | content-type: application/json 40 | 41 | {"name": "a newer surname"} 42 | -------------------------------------------------------------------------------- /week-7/routes/bookings.js: -------------------------------------------------------------------------------- 1 | const { bookingService } = require('../services') 2 | 3 | const router = require('express').Router() 4 | 5 | router.get('/', async (req, res) => { 6 | const bookings = await bookingService.load() 7 | 8 | res.render('bookings', { bookings }) 9 | }) 10 | 11 | router.get('/search', async (req, res) => { 12 | const origin = req.query.origin 13 | const destination = req.query.destination 14 | 15 | const query = {} 16 | 17 | if (origin) query.origin = origin 18 | if (destination) query.destination = destination 19 | 20 | const bookings = await bookingService.query(query) 21 | 22 | res.render('bookings', { bookings }) 23 | }) 24 | 25 | module.exports = router 26 | -------------------------------------------------------------------------------- /week-7/routes/drivers.js: -------------------------------------------------------------------------------- 1 | const { driverService } = require('../services') 2 | 3 | const router = require('express').Router() 4 | 5 | router.get('/', async (req, res) => { 6 | const drivers = await driverService.load() 7 | 8 | res.render('drivers', { drivers }) 9 | }) 10 | 11 | router.post('/', async (req, res) => { 12 | const driver = await driverService.insert(req.body) 13 | 14 | res.send(driver) 15 | }) 16 | 17 | router.delete('/:driverId', async (req, res) => { 18 | await driverService.removeBy('_id', req.params.driverId) 19 | 20 | res.send('OK') 21 | }) 22 | 23 | router.get('/young-drivers', async (req, res) => { 24 | const drivers = await driverService.findYoungDrivers() 25 | 26 | res.render('drivers', { drivers }) 27 | }) 28 | 29 | router.get('/:driverId', async (req, res) => { 30 | const driver = await driverService.find(req.params.driverId) 31 | if (!driver) return res.status(404).send('Cannot find driver') 32 | res.render('driver', { driver }) 33 | }) 34 | 35 | router.patch('/:driverId', async (req, res) => { 36 | const { driverId } = req.params 37 | const { name } = req.body 38 | 39 | await driverService.update(driverId, { name }) 40 | }) 41 | 42 | module.exports = router 43 | -------------------------------------------------------------------------------- /week-7/routes/index.js: -------------------------------------------------------------------------------- 1 | const router = require('express').Router() 2 | 3 | router.get('/', (req, res) => { 4 | res.render('index') 5 | }) 6 | 7 | module.exports = router 8 | -------------------------------------------------------------------------------- /week-7/routes/passengers.js: -------------------------------------------------------------------------------- 1 | const { passengerService, bookingService } = require('../services') 2 | 3 | const router = require('express').Router() 4 | 5 | router.get('/', async (req, res) => { 6 | const passengers = await passengerService.load() 7 | 8 | res.render('passengers', { passengers }) 9 | }) 10 | 11 | router.post('/', async (req, res, next) => { 12 | try { 13 | const passenger = await passengerService.insert(req.body) 14 | res.send(passenger) 15 | } catch(e) { 16 | next(e) 17 | } 18 | }) 19 | 20 | router.delete('/:passengerId', async (req, res) => { 21 | await passengerService.removeBy('_id', req.params.passengerId) 22 | 23 | res.send('OK') 24 | }) 25 | 26 | router.get('/:passengerId', async (req, res) => { 27 | const passenger = await passengerService.find(req.params.passengerId) 28 | 29 | if (!passenger) return res.status(404).send('Cannot find passenger') 30 | res.render('passenger', { passenger }) 31 | }) 32 | 33 | router.post('/:passengerId/bookings', async (req, res) => { 34 | const { passengerId } = req.params 35 | const { driverId, origin, destination } = req.body 36 | 37 | const booking = await bookingService.book(driverId, passengerId, origin, destination) 38 | 39 | res.send(booking) 40 | }) 41 | 42 | router.patch('/:passengerId', async (req, res) => { 43 | const { passengerId } = req.params 44 | const { name } = req.body 45 | 46 | await passengerService.update(passengerId, { name }) 47 | }) 48 | 49 | module.exports = router 50 | -------------------------------------------------------------------------------- /week-7/services/base-service.js: -------------------------------------------------------------------------------- 1 | class BaseService { 2 | constructor(model) { 3 | this.model = model 4 | } 5 | 6 | save(objects) { 7 | return this.model.insertMany(objects) 8 | } 9 | 10 | load() { 11 | return this.model.find() 12 | } 13 | 14 | async insert(object) { 15 | return await this.model.create(object) 16 | } 17 | 18 | async removeBy(property, value) { 19 | return this.model.deleteOne({ [property]: value }) 20 | } 21 | 22 | async update(id, object) { 23 | return this.model.findByIdAndUpdate(id, object) 24 | } 25 | 26 | async find(id) { 27 | return this.model.findById(id) 28 | } 29 | 30 | async query(obj) { 31 | return this.model.find(obj) 32 | } 33 | 34 | async findBy(property, value) { 35 | return this.model.find({ [property]: value }) 36 | } 37 | } 38 | 39 | module.exports = BaseService 40 | -------------------------------------------------------------------------------- /week-7/services/booking-service.js: -------------------------------------------------------------------------------- 1 | const BaseService = require('./base-service') 2 | const Booking = require('../models/booking') 3 | const driverService = require('./driver-service') 4 | const passengerService = require('./passenger-service') 5 | 6 | class BookingService extends BaseService { 7 | async findByPassengerId(passengerId) { 8 | return this.findBy('passenger', passengerId) 9 | } 10 | 11 | async findByDriverId(driverId) { 12 | return this.findBy('driver', driverId) 13 | } 14 | 15 | async book(driverId, passengerId, origin, destination) { 16 | const passenger = await passengerService.find(passengerId) 17 | const driver = await driverService.find(driverId) 18 | 19 | const booking = await this.insert({ driver, passenger, origin, destination }) 20 | passenger.bookings.push(booking) 21 | 22 | await passenger.save() 23 | 24 | return booking 25 | } 26 | } 27 | 28 | module.exports = new BookingService(Booking) 29 | -------------------------------------------------------------------------------- /week-7/services/driver-service.js: -------------------------------------------------------------------------------- 1 | const BaseService = require('./base-service') 2 | const Driver = require('../models/driver') 3 | 4 | class DriverService extends BaseService { 5 | async findByDriverName(name) { 6 | return this.findBy('name', name) 7 | } 8 | 9 | async findByLocation(location) { 10 | return this.findBy('location', location) 11 | } 12 | 13 | async findYoungDrivers() { 14 | return this.query({ 15 | age: { 16 | $lt: 30, 17 | $gte: 18 18 | } 19 | }) 20 | } 21 | } 22 | 23 | module.exports = new DriverService(Driver) 24 | -------------------------------------------------------------------------------- /week-7/services/driver.json: -------------------------------------------------------------------------------- 1 | [[ 2 | "1" 3 | ],{ 4 | "id": "2", 5 | "name": "3", 6 | "location": "4" 7 | },"33722c67-7b62-4dbb-81cb-e64a9aba6713","Stefan","Treptower Park"] -------------------------------------------------------------------------------- /week-7/services/index.js: -------------------------------------------------------------------------------- 1 | const passengerService = require('./passenger-service') 2 | const driverService = require('./driver-service') 3 | const bookingService = require('./booking-service') 4 | 5 | module.exports = { passengerService, driverService, bookingService } 6 | -------------------------------------------------------------------------------- /week-7/services/passenger-service.js: -------------------------------------------------------------------------------- 1 | const BaseService = require('./base-service') 2 | const Passenger = require('../models/passenger') 3 | 4 | class PassengerService extends BaseService { 5 | async findByName(name) { 6 | return this.findBy('name', name) 7 | } 8 | } 9 | 10 | module.exports = new PassengerService(Passenger) 11 | -------------------------------------------------------------------------------- /week-7/services/passenger.json: -------------------------------------------------------------------------------- 1 | [[ 2 | "1", 3 | "2", 4 | "3", 5 | "4", 6 | "5", 7 | "6", 8 | "7", 9 | "8", 10 | "9", 11 | "10", 12 | "11", 13 | "12", 14 | "13", 15 | "14", 16 | "15", 17 | "16", 18 | "17", 19 | "18", 20 | "19", 21 | "20" 22 | ],{ 23 | "id": "21", 24 | "name": "22", 25 | "location": "23", 26 | "bookings": "24" 27 | },{ 28 | "id": "25", 29 | "name": "26", 30 | "location": "27", 31 | "bookings": "28" 32 | },{ 33 | "id": "29", 34 | "name": "30", 35 | "location": "31", 36 | "bookings": "32" 37 | },{ 38 | "id": "33", 39 | "bookings": "34" 40 | },{ 41 | "id": "35", 42 | "bookings": "36" 43 | },{ 44 | "id": "37", 45 | "bookings": "38" 46 | },{ 47 | "id": "39", 48 | "bookings": "40" 49 | },{ 50 | "id": "41", 51 | "name": "42", 52 | "location": "43", 53 | "bookings": "44" 54 | },{ 55 | "id": "45", 56 | "bookings": "46" 57 | },{ 58 | "id": "47", 59 | "name": "48", 60 | "location": "49", 61 | "bookings": "50" 62 | },{ 63 | "id": "51", 64 | "name": "48", 65 | "location": "49", 66 | "bookings": "52" 67 | },{ 68 | "id": "53", 69 | "name": "48", 70 | "location": "49", 71 | "bookings": "54" 72 | },{ 73 | "id": "55", 74 | "name": "48", 75 | "location": "49", 76 | "bookings": "56" 77 | },{ 78 | "id": "57", 79 | "name": "48", 80 | "location": "49", 81 | "bookings": "58" 82 | },{ 83 | "id": "59", 84 | "name": "48", 85 | "location": "49", 86 | "bookings": "60" 87 | },{ 88 | "id": "61", 89 | "name": "48", 90 | "location": "49", 91 | "bookings": "62" 92 | },{ 93 | "id": "63", 94 | "name": "48", 95 | "location": "49", 96 | "bookings": "64" 97 | },{ 98 | "id": "65", 99 | "name": "48", 100 | "location": "49", 101 | "bookings": "66" 102 | },{ 103 | "id": "67", 104 | "name": "48", 105 | "location": "49", 106 | "bookings": "68" 107 | },{ 108 | "id": "69", 109 | "name": "48", 110 | "location": "49", 111 | "bookings": "70" 112 | },"01ed8529-ad31-4351-b092-21cabb931e16","Armagan","Kreuzberg",[ 113 | "71", 114 | "72", 115 | "73", 116 | "74" 117 | ],"68e6a08c-f4cf-486e-b1df-dac92576bc48","Mert","Mitte",[ 118 | "75" 119 | ],"db80ba9d-c327-4222-b62c-f5f230a3e0b6","Betul","Tegel",[ 120 | "76", 121 | "77", 122 | "78" 123 | ],"92f71ad7-a806-41f6-bd33-59fcc754162f",[],"269e3f80-42bf-4409-af1e-b897dedfde4c",[],"59be8bc5-f28a-48e3-a98b-a7a7d40c4dfd",[],"7fa6f1dc-76b5-450b-8aba-f73ea0104bd9",[],"66f47f82-063c-43d7-9c6e-5311f9d31fa2","new passenger","Moda",[],"b8ca32d3-f610-4e41-8534-b53afb5c01fd",[],"ede41daa-a3d4-427f-96aa-94dda0c0299a","weird client","kuzguncuk",[],"88b04838-742d-4aa2-9e9f-b06a363d454c",[],"c2c00469-2022-4f27-b3e9-5705535c2a42",[],"1d569e2b-d8a3-410f-a957-85e55896ca11",[],"437f691b-a097-4daf-ba56-ffff33bc2935",[],"91effbd5-e751-4403-8bd6-aec6fe335806",[],"05a1e9a3-da4a-46a3-8086-65933d7cc1e3",[],"d50628c1-6922-43ef-beeb-df9073211b5a",[],"55c0329f-60c3-4810-9388-a26f096a6449",[],"e36afcbc-8349-43a4-b8e8-445ab464050a",[],"5d5c3d85-b328-457e-8598-4d0fe69c252c",[],{ 124 | "driver": "79", 125 | "passenger": "80", 126 | "origin": "23", 127 | "destination": "81" 128 | },{ 129 | "driver": "79", 130 | "passenger": "80", 131 | "origin": "81", 132 | "destination": "27" 133 | },{ 134 | "driver": "79", 135 | "passenger": "80", 136 | "origin": "27", 137 | "destination": "23" 138 | },{ 139 | "driver": "79", 140 | "passenger": "80", 141 | "origin": "23", 142 | "destination": "82" 143 | },{ 144 | "driver": "79", 145 | "passenger": "83", 146 | "origin": "27", 147 | "destination": "23" 148 | },{ 149 | "driver": "84", 150 | "passenger": "85", 151 | "origin": "82", 152 | "destination": "23" 153 | },{ 154 | "driver": "86", 155 | "passenger": "87", 156 | "origin": "23", 157 | "destination": "88" 158 | },{ 159 | "driver": "89", 160 | "passenger": "90", 161 | "origin": "88", 162 | "destination": "27" 163 | },{ 164 | "id": "91", 165 | "name": "92", 166 | "location": "93" 167 | },{ 168 | "id": "21", 169 | "name": "22", 170 | "location": "23", 171 | "bookings": "24" 172 | },"Neukolln","SXF",{ 173 | "id": "25", 174 | "name": "26", 175 | "location": "27", 176 | "bookings": "28" 177 | },{ 178 | "id": "91", 179 | "name": "92", 180 | "location": "93" 181 | },{ 182 | "id": "29", 183 | "name": "30", 184 | "location": "31", 185 | "bookings": "32" 186 | },{ 187 | "id": "91", 188 | "name": "92", 189 | "location": "93" 190 | },{ 191 | "id": "29", 192 | "name": "30", 193 | "location": "31", 194 | "bookings": "32" 195 | },"TXL",{ 196 | "id": "91", 197 | "name": "92", 198 | "location": "93" 199 | },{ 200 | "id": "29", 201 | "name": "30", 202 | "location": "31", 203 | "bookings": "32" 204 | },"33722c67-7b62-4dbb-81cb-e64a9aba6713","Stefan","Treptower Park"] -------------------------------------------------------------------------------- /week-7/tests/e2e/passengers.test.js: -------------------------------------------------------------------------------- 1 | const app = require('../..') 2 | const request = require('supertest')(app) 3 | 4 | test('creates a new passenger', async (done) => { 5 | const passengerToCreate = { 6 | name: 'Test passenger', 7 | location: 'Moda' 8 | } 9 | 10 | const response = await request 11 | .post('/passengers') 12 | .send(passengerToCreate) 13 | .expect(200) 14 | 15 | const passengerCreated = response.body 16 | 17 | expect(passengerCreated).toMatchObject(passengerToCreate) 18 | expect(passengerCreated.bookings).toEqual([]) 19 | 20 | done() 21 | }) 22 | -------------------------------------------------------------------------------- /week-7/tests/scenario/bookings.test.js: -------------------------------------------------------------------------------- 1 | const app = require('../..') 2 | const request = require('supertest')(app) 3 | 4 | test('creates a new booking', async (done) => { 5 | const passengerToCreate = { 6 | name: 'Test passenger', 7 | location: 'Moda' 8 | } 9 | 10 | const driverToCreate = { 11 | name: 'Test driver', 12 | location: 'Moda', 13 | age: 18 14 | } 15 | 16 | const origin = 'Moda' 17 | const destination = 'Bostanci' 18 | 19 | const passengerResponse = await request 20 | .post('/passengers') 21 | .send(passengerToCreate) 22 | .expect(200) 23 | 24 | const driverResponse = await request 25 | .post('/drivers') 26 | .send(driverToCreate) 27 | .expect(200) 28 | 29 | const bookingResponse = await request 30 | .post(`/passengers/${passengerResponse.body._id}/bookings`) 31 | .send({ 32 | driverId: driverResponse.body._id, 33 | origin, 34 | destination 35 | }) 36 | .expect(200) 37 | 38 | const bookingCreated = bookingResponse.body 39 | 40 | expect(bookingCreated).toMatchObject({ 41 | driver: driverResponse.body, 42 | passenger: passengerResponse.body, 43 | origin, 44 | destination 45 | }) 46 | 47 | done() 48 | }) 49 | -------------------------------------------------------------------------------- /week-7/views/bookings.pug: -------------------------------------------------------------------------------- 1 | extends layout 2 | block content 3 | h2 Bookings 4 | p There are #{bookings.length} bookings. 5 | 6 | ol 7 | each booking in bookings 8 | li 9 | p Passenger: 10 | a(href=`/passengers/${booking.passenger.id}`) #{booking.passenger.name} 11 | p Driver: 12 | a(href=`/drivers/${booking.driver.id}`) #{booking.driver.name} 13 | p Origin: #{booking.origin} 14 | p Destination: #{booking.destination} 15 | -------------------------------------------------------------------------------- /week-7/views/driver.pug: -------------------------------------------------------------------------------- 1 | extends layout 2 | block content 3 | h2 Driver Detail 4 | p #{driver.name} 5 | p #{driver.age} 6 | p #{driver.location} 7 | -------------------------------------------------------------------------------- /week-7/views/drivers.pug: -------------------------------------------------------------------------------- 1 | extends layout 2 | block content 3 | h2 Drivers 4 | p There are #{drivers.length} drivers waiting. 5 | 6 | ol 7 | each driver in drivers 8 | li 9 | a(href=`/drivers/${driver.id}`) #{driver.name} 10 | -------------------------------------------------------------------------------- /week-7/views/index.pug: -------------------------------------------------------------------------------- 1 | extends layout 2 | block content 3 | p With this app, you can book a taxi from one place to another. 4 | -------------------------------------------------------------------------------- /week-7/views/layout.pug: -------------------------------------------------------------------------------- 1 | 2 | html(lang="en") 3 | head 4 | meta(charset="UTF-8") 5 | meta(name="viewport", content="width=device-width, initial-scale=1.0") 6 | title Taxi-booking app 7 | style. 8 | body { font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, Oxygen, Ubuntu, Cantarell, 'Open Sans', 'Helvetica Neue', sans-serif;} 9 | 10 | body 11 | h1 Hi! Welcome to taxi-booking app 12 | 13 | block content 14 | 15 | script(src="https://unpkg.com/axios/dist/axios.min.js") 16 | -------------------------------------------------------------------------------- /week-7/views/passenger.pug: -------------------------------------------------------------------------------- 1 | extends layout 2 | block content 3 | h2 Passenger Detail 4 | p #{passenger.name} 5 | 6 | h3 Booking History 7 | if passenger.bookings.length 8 | ol 9 | each booking in passenger.bookings 10 | li From #{booking.origin} to #{booking.destination} with #{booking.driver.name} 11 | else 12 | p No bookings 13 | -------------------------------------------------------------------------------- /week-7/views/passengers.pug: -------------------------------------------------------------------------------- 1 | extends layout 2 | block content 3 | h2 Passengers 4 | p There are #{passengers.length} passengers waiting. 5 | 6 | ol 7 | each passenger in passengers 8 | li 9 | a(href=`/passengers/${passenger.id}`) #{passenger.name} 10 | -------------------------------------------------------------------------------- /week-8/.bin/www: -------------------------------------------------------------------------------- 1 | const app = require('../') 2 | 3 | app.listen(3000, () => { 4 | console.log('started listening on 3000') 5 | }) 6 | -------------------------------------------------------------------------------- /week-8/.vscode/launch.json: -------------------------------------------------------------------------------- 1 | { 2 | // Use IntelliSense to learn about possible attributes. 3 | // Hover to view descriptions of existing attributes. 4 | // For more information, visit: https://go.microsoft.com/fwlink/?linkid=830387 5 | "version": "0.2.0", 6 | "configurations": [ 7 | { 8 | "type": "node", 9 | "request": "launch", 10 | "name": "Launch Current File", 11 | "program": "${file}", 12 | "skipFiles": [ 13 | "/**" 14 | ] 15 | }, 16 | { 17 | "type": "node", 18 | "request": "launch", 19 | "name": "Launch Program", 20 | "skipFiles": [ 21 | "/**" 22 | ], 23 | "program": "${workspaceFolder}/index.js" 24 | } 25 | ] 26 | } 27 | -------------------------------------------------------------------------------- /week-8/.vscode/settings.json: -------------------------------------------------------------------------------- 1 | 123,125 -------------------------------------------------------------------------------- /week-8/__mocks__/colors.js: -------------------------------------------------------------------------------- 1 | const colors = jest.createMockFromModule('colors') 2 | 3 | const unity = s => s 4 | 5 | colors.bgRed = {} 6 | colors.blue = unity 7 | colors.bgRed.white = unity 8 | 9 | module.exports = colors 10 | -------------------------------------------------------------------------------- /week-8/create-records.js: -------------------------------------------------------------------------------- 1 | const Passenger = require('./models/passenger') 2 | const Driver = require('./models/driver') 3 | const passengerService = require('./services/passenger-services') 4 | const driverService = require('./services/driver-services') 5 | 6 | const printBookingHistory = require('./lib/print-booking-history') 7 | 8 | const armagan = Passenger.create({name: 'Armagan', location: 'Kreuzberg'}) 9 | const mert = Passenger.create({name: 'Mert', location: 'Mitte'}) 10 | const stefan = Driver.create({ name: 'Stefan', location: 'Treptower Park' }) 11 | 12 | armagan.book(stefan, 'Kreuzberg', 'Neukolln') 13 | armagan.book(stefan, 'Neukolln', 'Mitte') 14 | armagan.book(stefan, 'Mitte', 'Kreuzberg') 15 | armagan.book(stefan, 'Kreuzberg', 'SXF') 16 | mert.book(stefan, 'Mitte', 'Kreuzberg') 17 | 18 | async function main() { 19 | try { 20 | await passengerService.save([armagan, mert]) 21 | 22 | await driverService.save([stefan]) 23 | 24 | const betul = Passenger.create({ name: 'Betul', location: 'Tegel' }) 25 | 26 | await passengerService.insert(betul) 27 | const passengers = await passengerService.load() 28 | passengers.forEach(printBookingHistory) 29 | } catch (e) { 30 | return console.log(e) 31 | } 32 | } 33 | 34 | main() 35 | -------------------------------------------------------------------------------- /week-8/fetch-records.js: -------------------------------------------------------------------------------- 1 | const { passengerService, driverService } = require('./services') 2 | const printBookingHistory = require('./lib/print-booking-history') 3 | 4 | async function main() { 5 | const stefan = await driverService.findBy('name', 'Stefan') 6 | const armagan = await passengerService.findByName('Armagan') 7 | 8 | armagan.book(stefan, 'Kreuzberg', 'Wannsee') 9 | passengerService.update(armagan) 10 | 11 | printBookingHistory(armagan) 12 | 13 | console.log(await passengerService.findBy('location', 'Mitte')) 14 | } 15 | 16 | main() 17 | -------------------------------------------------------------------------------- /week-8/frontend/.browserslistrc: -------------------------------------------------------------------------------- 1 | > 1% 2 | last 2 versions 3 | not dead 4 | -------------------------------------------------------------------------------- /week-8/frontend/.editorconfig: -------------------------------------------------------------------------------- 1 | [*.{js,jsx,ts,tsx,vue}] 2 | indent_style = space 3 | indent_size = 2 4 | trim_trailing_whitespace = true 5 | insert_final_newline = true 6 | -------------------------------------------------------------------------------- /week-8/frontend/.eslintrc.js: -------------------------------------------------------------------------------- 1 | module.exports = { 2 | root: true, 3 | env: { 4 | node: true 5 | }, 6 | extends: [ 7 | 'plugin:vue/essential', 8 | '@vue/standard' 9 | ], 10 | parserOptions: { 11 | parser: 'babel-eslint' 12 | }, 13 | rules: { 14 | 'no-console': process.env.NODE_ENV === 'production' ? 'warn' : 'off', 15 | 'no-debugger': process.env.NODE_ENV === 'production' ? 'warn' : 'off' 16 | } 17 | } 18 | -------------------------------------------------------------------------------- /week-8/frontend/.gitignore: -------------------------------------------------------------------------------- 1 | .DS_Store 2 | node_modules 3 | /dist 4 | 5 | # local env files 6 | .env.local 7 | .env.*.local 8 | 9 | # Log files 10 | npm-debug.log* 11 | yarn-debug.log* 12 | yarn-error.log* 13 | pnpm-debug.log* 14 | 15 | # Editor directories and files 16 | .idea 17 | .vscode 18 | *.suo 19 | *.ntvs* 20 | *.njsproj 21 | *.sln 22 | *.sw? 23 | -------------------------------------------------------------------------------- /week-8/frontend/README.md: -------------------------------------------------------------------------------- 1 | # frontend 2 | 3 | ## Project setup 4 | ``` 5 | npm install 6 | ``` 7 | 8 | ### Compiles and hot-reloads for development 9 | ``` 10 | npm run serve 11 | ``` 12 | 13 | ### Compiles and minifies for production 14 | ``` 15 | npm run build 16 | ``` 17 | 18 | ### Lints and fixes files 19 | ``` 20 | npm run lint 21 | ``` 22 | 23 | ### Customize configuration 24 | See [Configuration Reference](https://cli.vuejs.org/config/). 25 | -------------------------------------------------------------------------------- /week-8/frontend/babel.config.js: -------------------------------------------------------------------------------- 1 | module.exports = { 2 | presets: [ 3 | '@vue/cli-plugin-babel/preset' 4 | ] 5 | } 6 | -------------------------------------------------------------------------------- /week-8/frontend/package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "frontend", 3 | "version": "0.1.0", 4 | "private": true, 5 | "scripts": { 6 | "serve": "vue-cli-service serve", 7 | "build": "vue-cli-service build", 8 | "lint": "vue-cli-service lint" 9 | }, 10 | "dependencies": { 11 | "axios": "^0.19.2", 12 | "core-js": "^3.6.5", 13 | "register-service-worker": "^1.7.1", 14 | "vue": "^2.6.11", 15 | "vue-router": "^3.2.0", 16 | "vuex": "^3.4.0" 17 | }, 18 | "devDependencies": { 19 | "@vue/cli-plugin-babel": "~4.4.0", 20 | "@vue/cli-plugin-eslint": "~4.4.0", 21 | "@vue/cli-plugin-pwa": "~4.4.0", 22 | "@vue/cli-plugin-router": "~4.4.0", 23 | "@vue/cli-plugin-vuex": "~4.4.0", 24 | "@vue/cli-service": "~4.4.0", 25 | "@vue/eslint-config-standard": "^5.1.2", 26 | "babel-eslint": "^10.1.0", 27 | "eslint": "^6.7.2", 28 | "eslint-plugin-import": "^2.20.2", 29 | "eslint-plugin-node": "^11.1.0", 30 | "eslint-plugin-promise": "^4.2.1", 31 | "eslint-plugin-standard": "^4.0.0", 32 | "eslint-plugin-vue": "^6.2.2", 33 | "lint-staged": "^9.5.0", 34 | "node-sass": "^4.12.0", 35 | "pug": "^3.0.0", 36 | "pug-plain-loader": "^1.0.0", 37 | "sass-loader": "^8.0.2", 38 | "vue-template-compiler": "^2.6.11" 39 | }, 40 | "gitHooks": { 41 | "pre-commit": "lint-staged" 42 | }, 43 | "lint-staged": { 44 | "*.{js,jsx,vue}": [ 45 | "vue-cli-service lint", 46 | "git add" 47 | ] 48 | } 49 | } 50 | -------------------------------------------------------------------------------- /week-8/frontend/public/favicon.ico: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/nimbleturkiye/js-classes/cb8bc70d5df6e49f92a1c2e04ad86c8e2f4bc695/week-8/frontend/public/favicon.ico -------------------------------------------------------------------------------- /week-8/frontend/public/img/icons/android-chrome-192x192.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/nimbleturkiye/js-classes/cb8bc70d5df6e49f92a1c2e04ad86c8e2f4bc695/week-8/frontend/public/img/icons/android-chrome-192x192.png -------------------------------------------------------------------------------- /week-8/frontend/public/img/icons/android-chrome-512x512.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/nimbleturkiye/js-classes/cb8bc70d5df6e49f92a1c2e04ad86c8e2f4bc695/week-8/frontend/public/img/icons/android-chrome-512x512.png -------------------------------------------------------------------------------- /week-8/frontend/public/img/icons/android-chrome-maskable-192x192.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/nimbleturkiye/js-classes/cb8bc70d5df6e49f92a1c2e04ad86c8e2f4bc695/week-8/frontend/public/img/icons/android-chrome-maskable-192x192.png -------------------------------------------------------------------------------- /week-8/frontend/public/img/icons/android-chrome-maskable-512x512.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/nimbleturkiye/js-classes/cb8bc70d5df6e49f92a1c2e04ad86c8e2f4bc695/week-8/frontend/public/img/icons/android-chrome-maskable-512x512.png -------------------------------------------------------------------------------- /week-8/frontend/public/img/icons/apple-touch-icon-120x120.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/nimbleturkiye/js-classes/cb8bc70d5df6e49f92a1c2e04ad86c8e2f4bc695/week-8/frontend/public/img/icons/apple-touch-icon-120x120.png -------------------------------------------------------------------------------- /week-8/frontend/public/img/icons/apple-touch-icon-152x152.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/nimbleturkiye/js-classes/cb8bc70d5df6e49f92a1c2e04ad86c8e2f4bc695/week-8/frontend/public/img/icons/apple-touch-icon-152x152.png -------------------------------------------------------------------------------- /week-8/frontend/public/img/icons/apple-touch-icon-180x180.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/nimbleturkiye/js-classes/cb8bc70d5df6e49f92a1c2e04ad86c8e2f4bc695/week-8/frontend/public/img/icons/apple-touch-icon-180x180.png -------------------------------------------------------------------------------- /week-8/frontend/public/img/icons/apple-touch-icon-60x60.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/nimbleturkiye/js-classes/cb8bc70d5df6e49f92a1c2e04ad86c8e2f4bc695/week-8/frontend/public/img/icons/apple-touch-icon-60x60.png -------------------------------------------------------------------------------- /week-8/frontend/public/img/icons/apple-touch-icon-76x76.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/nimbleturkiye/js-classes/cb8bc70d5df6e49f92a1c2e04ad86c8e2f4bc695/week-8/frontend/public/img/icons/apple-touch-icon-76x76.png -------------------------------------------------------------------------------- /week-8/frontend/public/img/icons/apple-touch-icon.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/nimbleturkiye/js-classes/cb8bc70d5df6e49f92a1c2e04ad86c8e2f4bc695/week-8/frontend/public/img/icons/apple-touch-icon.png -------------------------------------------------------------------------------- /week-8/frontend/public/img/icons/favicon-16x16.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/nimbleturkiye/js-classes/cb8bc70d5df6e49f92a1c2e04ad86c8e2f4bc695/week-8/frontend/public/img/icons/favicon-16x16.png -------------------------------------------------------------------------------- /week-8/frontend/public/img/icons/favicon-32x32.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/nimbleturkiye/js-classes/cb8bc70d5df6e49f92a1c2e04ad86c8e2f4bc695/week-8/frontend/public/img/icons/favicon-32x32.png -------------------------------------------------------------------------------- /week-8/frontend/public/img/icons/msapplication-icon-144x144.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/nimbleturkiye/js-classes/cb8bc70d5df6e49f92a1c2e04ad86c8e2f4bc695/week-8/frontend/public/img/icons/msapplication-icon-144x144.png -------------------------------------------------------------------------------- /week-8/frontend/public/img/icons/mstile-150x150.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/nimbleturkiye/js-classes/cb8bc70d5df6e49f92a1c2e04ad86c8e2f4bc695/week-8/frontend/public/img/icons/mstile-150x150.png -------------------------------------------------------------------------------- /week-8/frontend/public/img/icons/safari-pinned-tab.svg: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | -------------------------------------------------------------------------------- /week-8/frontend/public/index.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | <%= htmlWebpackPlugin.options.title %> 9 | 10 | 11 | 14 |
15 | 16 | 17 | 18 | -------------------------------------------------------------------------------- /week-8/frontend/public/robots.txt: -------------------------------------------------------------------------------- 1 | User-agent: * 2 | Disallow: 3 | -------------------------------------------------------------------------------- /week-8/frontend/src/App.vue: -------------------------------------------------------------------------------- 1 | 10 | 11 | 33 | -------------------------------------------------------------------------------- /week-8/frontend/src/assets/logo.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/nimbleturkiye/js-classes/cb8bc70d5df6e49f92a1c2e04ad86c8e2f4bc695/week-8/frontend/src/assets/logo.png -------------------------------------------------------------------------------- /week-8/frontend/src/components/HelloWorld.vue: -------------------------------------------------------------------------------- 1 | 16 | 17 | 24 | 25 | 27 | -------------------------------------------------------------------------------- /week-8/frontend/src/main.js: -------------------------------------------------------------------------------- 1 | import Vue from 'vue' 2 | import App from './App.vue' 3 | import './registerServiceWorker' 4 | import router from './router' 5 | import store from './store' 6 | 7 | Vue.config.productionTip = false 8 | 9 | new Vue({ 10 | router, 11 | store, 12 | render: h => h(App) 13 | }).$mount('#app') 14 | -------------------------------------------------------------------------------- /week-8/frontend/src/registerServiceWorker.js: -------------------------------------------------------------------------------- 1 | /* eslint-disable no-console */ 2 | 3 | import { register } from 'register-service-worker' 4 | 5 | if (process.env.NODE_ENV === 'production') { 6 | register(`${process.env.BASE_URL}service-worker.js`, { 7 | ready () { 8 | console.log( 9 | 'App is being served from cache by a service worker.\n' + 10 | 'For more details, visit https://goo.gl/AFskqB' 11 | ) 12 | }, 13 | registered () { 14 | console.log('Service worker has been registered.') 15 | }, 16 | cached () { 17 | console.log('Content has been cached for offline use.') 18 | }, 19 | updatefound () { 20 | console.log('New content is downloading.') 21 | }, 22 | updated () { 23 | console.log('New content is available; please refresh.') 24 | }, 25 | offline () { 26 | console.log('No internet connection found. App is running in offline mode.') 27 | }, 28 | error (error) { 29 | console.error('Error during service worker registration:', error) 30 | } 31 | }) 32 | } 33 | -------------------------------------------------------------------------------- /week-8/frontend/src/router/index.js: -------------------------------------------------------------------------------- 1 | import Vue from 'vue' 2 | import VueRouter from 'vue-router' 3 | import Passengers from '../views/Passengers.vue' 4 | import PassengerView from '../views/Passenger.vue' 5 | 6 | Vue.use(VueRouter) 7 | 8 | const routes = [ 9 | { 10 | path: '/', 11 | name: 'Passengers', 12 | component: Passengers 13 | }, 14 | { 15 | path: '/passengers/:passengerId', 16 | name: 'PassengerView', 17 | component: PassengerView 18 | }, 19 | { 20 | path: '/about', 21 | name: 'About', 22 | // route level code-splitting 23 | // this generates a separate chunk (about.[hash].js) for this route 24 | // which is lazy-loaded when the route is visited. 25 | component: () => import(/* webpackChunkName: "about" */ '../views/About.vue') 26 | } 27 | ] 28 | 29 | const router = new VueRouter({ 30 | mode: 'history', 31 | base: process.env.BASE_URL, 32 | routes 33 | }) 34 | 35 | export default router 36 | -------------------------------------------------------------------------------- /week-8/frontend/src/store/index.js: -------------------------------------------------------------------------------- 1 | import Vue from 'vue' 2 | import Vuex from 'vuex' 3 | import axios from 'axios' 4 | 5 | axios.defaults.baseURL = 'http://localhost:3000' 6 | 7 | Vue.use(Vuex) 8 | 9 | const Mutations = { 10 | INCREMENT: 'INCREMENT', 11 | DECREMENT: 'DECREMENT' 12 | } 13 | 14 | export default new Vuex.Store({ 15 | state: { 16 | countHome: 0, 17 | countAbout: 0 18 | }, 19 | mutations: { 20 | [Mutations.INCREMENT] (state, type) { 21 | type === 'countHome' ? state.countHome++ : state.countAbout++ 22 | }, 23 | [Mutations.DECREMENT] (state, type) { 24 | if (state[type] === 0) return 25 | 26 | type === 'countHome' ? state.countHome-- : state.countAbout-- 27 | } 28 | }, 29 | actions: { 30 | increment ({ commit }, type) { 31 | commit(Mutations.INCREMENT, type) 32 | }, 33 | decrement ({ commit }, type) { 34 | commit(Mutations.DECREMENT, type) 35 | }, 36 | async fetchPassengers () { 37 | const request = await axios.get('/passengers') 38 | return request.data 39 | }, 40 | async fetchDrivers () { 41 | const request = await axios.get('/drivers') 42 | return request.data 43 | }, 44 | async fetchPassenger ({ state }, passengerId) { 45 | const request = await axios.get(`/passengers/${passengerId}`) 46 | return request.data 47 | }, 48 | async bookDriver ({ dispatch }, { passengerId, driverId, origin, destination }) { 49 | const request = await axios.post(`/passengers/${passengerId}/bookings`, { 50 | driverId, origin, destination 51 | }) 52 | 53 | return request.data 54 | } 55 | } 56 | }) 57 | -------------------------------------------------------------------------------- /week-8/frontend/src/views/About.vue: -------------------------------------------------------------------------------- 1 | 15 | 16 | 21 | -------------------------------------------------------------------------------- /week-8/frontend/src/views/Passenger.vue: -------------------------------------------------------------------------------- 1 | 36 | 37 | 60 | -------------------------------------------------------------------------------- /week-8/frontend/src/views/Passengers.vue: -------------------------------------------------------------------------------- 1 | 21 | 22 | 33 | -------------------------------------------------------------------------------- /week-8/index.js: -------------------------------------------------------------------------------- 1 | const express = require('express') 2 | const bodyParser = require('body-parser') 3 | const cors = require('cors') 4 | 5 | const passengersRouter = require('./routes/passengers') 6 | const driversRouter = require('./routes/drivers') 7 | const bookingsRouter = require('./routes/bookings') 8 | const indexRouter = require('./routes/index') 9 | require('./mongo-connection') 10 | 11 | const app = express() 12 | app.use(bodyParser.json()) 13 | app.use(cors()) 14 | 15 | app.set('view engine', 'pug') 16 | 17 | app.use('/passengers', passengersRouter) 18 | app.use('/drivers', driversRouter) 19 | app.use('/bookings', bookingsRouter) 20 | app.use('/', indexRouter) 21 | 22 | module.exports = app 23 | -------------------------------------------------------------------------------- /week-8/jest.config.js: -------------------------------------------------------------------------------- 1 | module.exports = { 2 | testEnvironment: 'node' 3 | }; 4 | -------------------------------------------------------------------------------- /week-8/lib/__tests__/print-booking-history.test.js: -------------------------------------------------------------------------------- 1 | const printBookingHistory = require('../print-booking-history') 2 | 3 | test('prints passenger bookings when a passenger has a booking', () => { 4 | // initialization 5 | const passenger = { 6 | name: 'Armagan', 7 | bookings: [{ 8 | passenger: { name: 'Armagan' }, 9 | driver: { name: 'Stefan' }, 10 | origin: 'Kreuzberg', 11 | destination: 'Neukolln' 12 | }] 13 | } 14 | const consoleSpy = jest.spyOn(console, 'log') 15 | 16 | // test 17 | printBookingHistory(passenger) 18 | 19 | // assertions 20 | expect(consoleSpy).toHaveBeenCalledWith('Armagan booked Stefan to travel from Kreuzberg to Neukolln') 21 | 22 | // teardown 23 | consoleSpy.mockRestore() 24 | }) 25 | 26 | test('prints warning message when a passenger has no bookings', () => { 27 | // initialization 28 | const passenger = { 29 | name: 'Armagan', 30 | bookings: [] 31 | } 32 | const consoleSpy = jest.spyOn(console, 'log') 33 | 34 | // test 35 | printBookingHistory(passenger) 36 | 37 | // assertions 38 | expect(consoleSpy).toHaveBeenCalledWith('Armagan has no bookings yet.') 39 | 40 | // teardown 41 | consoleSpy.mockRestore() 42 | }) 43 | -------------------------------------------------------------------------------- /week-8/lib/print-booking-history.js: -------------------------------------------------------------------------------- 1 | const colors = require('colors') 2 | 3 | function printBooking(booking) { 4 | console.log(`${colors.blue(booking.passenger.name)} booked ${colors.blue(booking.driver.name)} to travel from ${colors.bgRed.white(booking.origin)} to ${colors.bgRed.white(booking.destination)}`) 5 | } 6 | 7 | function printBookingHistory(passenger) { 8 | if (passenger.bookings.length == 0) 9 | return console.log(`${colors.blue(passenger.name)} has no bookings yet.`) 10 | 11 | passenger.bookings.forEach(printBooking) 12 | } 13 | 14 | module.exports = printBookingHistory 15 | -------------------------------------------------------------------------------- /week-8/models/booking.js: -------------------------------------------------------------------------------- 1 | const mongoose = require('mongoose') 2 | 3 | const BookingSchema = new mongoose.Schema({ 4 | driver: { 5 | type: mongoose.Schema.Types.ObjectId, 6 | ref: 'Driver', 7 | autopopulate: { maxDepth: 1 } 8 | }, 9 | passenger: { 10 | type: mongoose.Schema.Types.ObjectId, 11 | ref: 'Passenger', 12 | autopopulate: { maxDepth: 1 } 13 | }, 14 | origin: String, 15 | destination: String 16 | }) 17 | 18 | BookingSchema.plugin(require('mongoose-autopopulate')) 19 | 20 | module.exports = mongoose.model('Booking', BookingSchema) 21 | -------------------------------------------------------------------------------- /week-8/models/driver.js: -------------------------------------------------------------------------------- 1 | const mongoose = require('mongoose') 2 | 3 | const DriverSchema = new mongoose.Schema({ 4 | name: String, 5 | location: String, 6 | age: { type: Number, required: true, min: 18 }, 7 | }) 8 | 9 | module.exports = mongoose.model('Driver', DriverSchema) 10 | -------------------------------------------------------------------------------- /week-8/models/passenger.js: -------------------------------------------------------------------------------- 1 | const mongoose = require('mongoose') 2 | 3 | const PassengerSchema = new mongoose.Schema({ 4 | name: { type: String, required: true, minlength: 2 }, 5 | location: String, 6 | bookings: [{ 7 | type: mongoose.Schema.Types.ObjectId, 8 | ref: 'Booking', 9 | autopopulate: { maxDepth: 2 } 10 | }] 11 | }) 12 | 13 | PassengerSchema.plugin(require('mongoose-autopopulate')) 14 | 15 | module.exports = mongoose.model('Passenger', PassengerSchema) 16 | -------------------------------------------------------------------------------- /week-8/mongo-connection.js: -------------------------------------------------------------------------------- 1 | const mongoose = require('mongoose') 2 | mongoose.connect('mongodb://localhost/week-8', { useNewUrlParser: true, useUnifiedTopology: true }) 3 | 4 | var db = mongoose.connection 5 | db.on('error', console.error.bind(console, 'connection error:')) 6 | db.once('open', function () { 7 | console.log('we are connected to mongodb!') 8 | }) 9 | -------------------------------------------------------------------------------- /week-8/package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "week-8", 3 | "version": "1.0.0", 4 | "description": "In-class examples for week-8", 5 | "main": "index.js", 6 | "scripts": { 7 | "start": "node .bin/www", 8 | "watch": "nodemon --ignore '*.json' .bin/www", 9 | "test": "jest --coverage --verbose", 10 | "test-watch": "jest --coverage --verbose --watch" 11 | }, 12 | "author": "Armagan Amcalar ", 13 | "license": "ISC", 14 | "dependencies": { 15 | "body-parser": "^1.19.0", 16 | "colors": "^1.4.0", 17 | "cors": "^2.8.5", 18 | "express": "^4.17.1", 19 | "flatted": "^2.0.1", 20 | "mongoose": "^5.9.9", 21 | "mongoose-autopopulate": "^0.12.2", 22 | "pug": "^2.0.4", 23 | "uuid": "^7.0.2" 24 | }, 25 | "devDependencies": { 26 | "jest": "^26.0.1", 27 | "nodemon": "^2.0.2", 28 | "supertest": "^4.0.2" 29 | } 30 | } 31 | -------------------------------------------------------------------------------- /week-8/requests.http: -------------------------------------------------------------------------------- 1 | POST http://localhost:3000/passengers HTTP/1.1 2 | content-type: application/json 3 | 4 | {"name":"armagan", "location": "kuzguncuk"} 5 | 6 | ### 7 | 8 | GET http://localhost:3000/passengers/5ea70b28bbf7cb710503f2d4 HTTP/1.1 9 | 10 | ### 11 | DELETE http://localhost:3000/passengers/5e9de11c0b0ac272d7866b1b HTTP/1.1 12 | 13 | ### 14 | 15 | PATCH http://localhost:3000/passengers/5e9de4cd9f11eb7402e56941 HTTP/1.1 16 | content-type: application/json 17 | 18 | {"name": "new name4"} 19 | 20 | ### 21 | 22 | POST http://localhost:3000/drivers HTTP/1.1 23 | content-type: application/json 24 | 25 | {"name":"older driver", "age": 35, "location": "kuzguncuk"} 26 | 27 | ### 28 | DELETE http://localhost:3000/drivers/5ea70bf063b0877147bd3e6b HTTP/1.1 29 | 30 | ### 31 | POST http://localhost:3000/passengers/5ea70b28bbf7cb710503f2d4/bookings HTTP/1.1 32 | content-type: application/json 33 | 34 | {"driverId": "5ea724416e3b607ac41df63a", "origin": "Kabatas", "destination": "Pendik"} 35 | 36 | ### 37 | 38 | PATCH http://localhost:3000/drivers/5ea70c8063b0877147bd3e6c HTTP/1.1 39 | content-type: application/json 40 | 41 | {"name": "a newer surname"} 42 | -------------------------------------------------------------------------------- /week-8/routes/bookings.js: -------------------------------------------------------------------------------- 1 | const { bookingService } = require('../services') 2 | 3 | const router = require('express').Router() 4 | 5 | router.get('/', async (req, res) => { 6 | const bookings = await bookingService.load() 7 | 8 | res.render('bookings', { bookings }) 9 | }) 10 | 11 | router.get('/search', async (req, res) => { 12 | const origin = req.query.origin 13 | const destination = req.query.destination 14 | 15 | const query = {} 16 | 17 | if (origin) query.origin = origin 18 | if (destination) query.destination = destination 19 | 20 | const bookings = await bookingService.query(query) 21 | 22 | res.render('bookings', { bookings }) 23 | }) 24 | 25 | module.exports = router 26 | -------------------------------------------------------------------------------- /week-8/routes/drivers.js: -------------------------------------------------------------------------------- 1 | const { driverService } = require('../services') 2 | 3 | const router = require('express').Router() 4 | 5 | router.get('/', async (req, res) => { 6 | res.send(await driverService.load()) 7 | }) 8 | 9 | router.post('/', async (req, res) => { 10 | const driver = await driverService.insert(req.body) 11 | 12 | res.send(driver) 13 | }) 14 | 15 | router.delete('/:driverId', async (req, res) => { 16 | await driverService.removeBy('_id', req.params.driverId) 17 | 18 | res.send('OK') 19 | }) 20 | 21 | router.get('/young-drivers', async (req, res) => { 22 | const drivers = await driverService.findYoungDrivers() 23 | 24 | res.render('drivers', { drivers }) 25 | }) 26 | 27 | router.get('/:driverId', async (req, res) => { 28 | const driver = await driverService.find(req.params.driverId) 29 | if (!driver) return res.status(404).send('Cannot find driver') 30 | res.render('driver', { driver }) 31 | }) 32 | 33 | router.patch('/:driverId', async (req, res) => { 34 | const { driverId } = req.params 35 | const { name } = req.body 36 | 37 | await driverService.update(driverId, { name }) 38 | }) 39 | 40 | module.exports = router 41 | -------------------------------------------------------------------------------- /week-8/routes/index.js: -------------------------------------------------------------------------------- 1 | const router = require('express').Router() 2 | 3 | router.get('/', (req, res) => { 4 | res.render('index') 5 | }) 6 | 7 | module.exports = router 8 | -------------------------------------------------------------------------------- /week-8/routes/passengers.js: -------------------------------------------------------------------------------- 1 | const { passengerService, bookingService } = require('../services') 2 | 3 | const router = require('express').Router() 4 | 5 | router.get('/', async (req, res) => { 6 | res.send(await passengerService.load()) 7 | }) 8 | 9 | router.post('/', async (req, res, next) => { 10 | try { 11 | const passenger = await passengerService.insert(req.body) 12 | res.send(passenger) 13 | } catch(e) { 14 | next(e) 15 | } 16 | }) 17 | 18 | router.delete('/:passengerId', async (req, res) => { 19 | await passengerService.removeBy('_id', req.params.passengerId) 20 | 21 | res.send('OK') 22 | }) 23 | 24 | router.get('/:passengerId', async (req, res) => { 25 | const passenger = await passengerService.find(req.params.passengerId) 26 | 27 | if (!passenger) return res.status(404) 28 | res.send(passenger) 29 | }) 30 | 31 | router.post('/:passengerId/bookings', async (req, res) => { 32 | const { passengerId } = req.params 33 | const { driverId, origin, destination } = req.body 34 | 35 | const booking = await bookingService.book(driverId, passengerId, origin, destination) 36 | 37 | res.send(booking) 38 | }) 39 | 40 | router.patch('/:passengerId', async (req, res) => { 41 | const { passengerId } = req.params 42 | const { name } = req.body 43 | 44 | await passengerService.update(passengerId, { name }) 45 | }) 46 | 47 | module.exports = router 48 | -------------------------------------------------------------------------------- /week-8/services/base-service.js: -------------------------------------------------------------------------------- 1 | class BaseService { 2 | constructor(model) { 3 | this.model = model 4 | } 5 | 6 | save(objects) { 7 | return this.model.insertMany(objects) 8 | } 9 | 10 | load() { 11 | return this.model.find() 12 | } 13 | 14 | async insert(object) { 15 | return await this.model.create(object) 16 | } 17 | 18 | async removeBy(property, value) { 19 | return this.model.deleteOne({ [property]: value }) 20 | } 21 | 22 | async update(id, object) { 23 | return this.model.findByIdAndUpdate(id, object) 24 | } 25 | 26 | async find(id) { 27 | return this.model.findById(id) 28 | } 29 | 30 | async query(obj) { 31 | return this.model.find(obj) 32 | } 33 | 34 | async findBy(property, value) { 35 | return this.model.find({ [property]: value }) 36 | } 37 | } 38 | 39 | module.exports = BaseService 40 | -------------------------------------------------------------------------------- /week-8/services/booking-service.js: -------------------------------------------------------------------------------- 1 | const BaseService = require('./base-service') 2 | const Booking = require('../models/booking') 3 | const driverService = require('./driver-service') 4 | const passengerService = require('./passenger-service') 5 | 6 | class BookingService extends BaseService { 7 | async findByPassengerId(passengerId) { 8 | return this.findBy('passenger', passengerId) 9 | } 10 | 11 | async findByDriverId(driverId) { 12 | return this.findBy('driver', driverId) 13 | } 14 | 15 | async book(driverId, passengerId, origin, destination) { 16 | const passenger = await passengerService.find(passengerId) 17 | const driver = await driverService.find(driverId) 18 | 19 | const booking = await this.insert({ driver, passenger, origin, destination }) 20 | passenger.bookings.push(booking) 21 | 22 | await passenger.save() 23 | 24 | return booking 25 | } 26 | } 27 | 28 | module.exports = new BookingService(Booking) 29 | -------------------------------------------------------------------------------- /week-8/services/driver-service.js: -------------------------------------------------------------------------------- 1 | const BaseService = require('./base-service') 2 | const Driver = require('../models/driver') 3 | 4 | class DriverService extends BaseService { 5 | async findByDriverName(name) { 6 | return this.findBy('name', name) 7 | } 8 | 9 | async findByLocation(location) { 10 | return this.findBy('location', location) 11 | } 12 | 13 | async findYoungDrivers() { 14 | return this.query({ 15 | age: { 16 | $lt: 30, 17 | $gte: 18 18 | } 19 | }) 20 | } 21 | } 22 | 23 | module.exports = new DriverService(Driver) 24 | -------------------------------------------------------------------------------- /week-8/services/driver.json: -------------------------------------------------------------------------------- 1 | [[ 2 | "1" 3 | ],{ 4 | "id": "2", 5 | "name": "3", 6 | "location": "4" 7 | },"33722c67-7b62-4dbb-81cb-e64a9aba6713","Stefan","Treptower Park"] -------------------------------------------------------------------------------- /week-8/services/index.js: -------------------------------------------------------------------------------- 1 | const passengerService = require('./passenger-service') 2 | const driverService = require('./driver-service') 3 | const bookingService = require('./booking-service') 4 | 5 | module.exports = { passengerService, driverService, bookingService } 6 | -------------------------------------------------------------------------------- /week-8/services/passenger-service.js: -------------------------------------------------------------------------------- 1 | const BaseService = require('./base-service') 2 | const Passenger = require('../models/passenger') 3 | 4 | class PassengerService extends BaseService { 5 | async findByName(name) { 6 | return this.findBy('name', name) 7 | } 8 | } 9 | 10 | module.exports = new PassengerService(Passenger) 11 | -------------------------------------------------------------------------------- /week-8/services/passenger.json: -------------------------------------------------------------------------------- 1 | [[ 2 | "1", 3 | "2", 4 | "3", 5 | "4", 6 | "5", 7 | "6", 8 | "7", 9 | "8", 10 | "9", 11 | "10", 12 | "11", 13 | "12", 14 | "13", 15 | "14", 16 | "15", 17 | "16", 18 | "17", 19 | "18", 20 | "19", 21 | "20" 22 | ],{ 23 | "id": "21", 24 | "name": "22", 25 | "location": "23", 26 | "bookings": "24" 27 | },{ 28 | "id": "25", 29 | "name": "26", 30 | "location": "27", 31 | "bookings": "28" 32 | },{ 33 | "id": "29", 34 | "name": "30", 35 | "location": "31", 36 | "bookings": "32" 37 | },{ 38 | "id": "33", 39 | "bookings": "34" 40 | },{ 41 | "id": "35", 42 | "bookings": "36" 43 | },{ 44 | "id": "37", 45 | "bookings": "38" 46 | },{ 47 | "id": "39", 48 | "bookings": "40" 49 | },{ 50 | "id": "41", 51 | "name": "42", 52 | "location": "43", 53 | "bookings": "44" 54 | },{ 55 | "id": "45", 56 | "bookings": "46" 57 | },{ 58 | "id": "47", 59 | "name": "48", 60 | "location": "49", 61 | "bookings": "50" 62 | },{ 63 | "id": "51", 64 | "name": "48", 65 | "location": "49", 66 | "bookings": "52" 67 | },{ 68 | "id": "53", 69 | "name": "48", 70 | "location": "49", 71 | "bookings": "54" 72 | },{ 73 | "id": "55", 74 | "name": "48", 75 | "location": "49", 76 | "bookings": "56" 77 | },{ 78 | "id": "57", 79 | "name": "48", 80 | "location": "49", 81 | "bookings": "58" 82 | },{ 83 | "id": "59", 84 | "name": "48", 85 | "location": "49", 86 | "bookings": "60" 87 | },{ 88 | "id": "61", 89 | "name": "48", 90 | "location": "49", 91 | "bookings": "62" 92 | },{ 93 | "id": "63", 94 | "name": "48", 95 | "location": "49", 96 | "bookings": "64" 97 | },{ 98 | "id": "65", 99 | "name": "48", 100 | "location": "49", 101 | "bookings": "66" 102 | },{ 103 | "id": "67", 104 | "name": "48", 105 | "location": "49", 106 | "bookings": "68" 107 | },{ 108 | "id": "69", 109 | "name": "48", 110 | "location": "49", 111 | "bookings": "70" 112 | },"01ed8529-ad31-4351-b092-21cabb931e16","Armagan","Kreuzberg",[ 113 | "71", 114 | "72", 115 | "73", 116 | "74" 117 | ],"68e6a08c-f4cf-486e-b1df-dac92576bc48","Mert","Mitte",[ 118 | "75" 119 | ],"db80ba9d-c327-4222-b62c-f5f230a3e0b6","Betul","Tegel",[ 120 | "76", 121 | "77", 122 | "78" 123 | ],"92f71ad7-a806-41f6-bd33-59fcc754162f",[],"269e3f80-42bf-4409-af1e-b897dedfde4c",[],"59be8bc5-f28a-48e3-a98b-a7a7d40c4dfd",[],"7fa6f1dc-76b5-450b-8aba-f73ea0104bd9",[],"66f47f82-063c-43d7-9c6e-5311f9d31fa2","new passenger","Moda",[],"b8ca32d3-f610-4e41-8534-b53afb5c01fd",[],"ede41daa-a3d4-427f-96aa-94dda0c0299a","weird client","kuzguncuk",[],"88b04838-742d-4aa2-9e9f-b06a363d454c",[],"c2c00469-2022-4f27-b3e9-5705535c2a42",[],"1d569e2b-d8a3-410f-a957-85e55896ca11",[],"437f691b-a097-4daf-ba56-ffff33bc2935",[],"91effbd5-e751-4403-8bd6-aec6fe335806",[],"05a1e9a3-da4a-46a3-8086-65933d7cc1e3",[],"d50628c1-6922-43ef-beeb-df9073211b5a",[],"55c0329f-60c3-4810-9388-a26f096a6449",[],"e36afcbc-8349-43a4-b8e8-445ab464050a",[],"5d5c3d85-b328-457e-8598-4d0fe69c252c",[],{ 124 | "driver": "79", 125 | "passenger": "80", 126 | "origin": "23", 127 | "destination": "81" 128 | },{ 129 | "driver": "79", 130 | "passenger": "80", 131 | "origin": "81", 132 | "destination": "27" 133 | },{ 134 | "driver": "79", 135 | "passenger": "80", 136 | "origin": "27", 137 | "destination": "23" 138 | },{ 139 | "driver": "79", 140 | "passenger": "80", 141 | "origin": "23", 142 | "destination": "82" 143 | },{ 144 | "driver": "79", 145 | "passenger": "83", 146 | "origin": "27", 147 | "destination": "23" 148 | },{ 149 | "driver": "84", 150 | "passenger": "85", 151 | "origin": "82", 152 | "destination": "23" 153 | },{ 154 | "driver": "86", 155 | "passenger": "87", 156 | "origin": "23", 157 | "destination": "88" 158 | },{ 159 | "driver": "89", 160 | "passenger": "90", 161 | "origin": "88", 162 | "destination": "27" 163 | },{ 164 | "id": "91", 165 | "name": "92", 166 | "location": "93" 167 | },{ 168 | "id": "21", 169 | "name": "22", 170 | "location": "23", 171 | "bookings": "24" 172 | },"Neukolln","SXF",{ 173 | "id": "25", 174 | "name": "26", 175 | "location": "27", 176 | "bookings": "28" 177 | },{ 178 | "id": "91", 179 | "name": "92", 180 | "location": "93" 181 | },{ 182 | "id": "29", 183 | "name": "30", 184 | "location": "31", 185 | "bookings": "32" 186 | },{ 187 | "id": "91", 188 | "name": "92", 189 | "location": "93" 190 | },{ 191 | "id": "29", 192 | "name": "30", 193 | "location": "31", 194 | "bookings": "32" 195 | },"TXL",{ 196 | "id": "91", 197 | "name": "92", 198 | "location": "93" 199 | },{ 200 | "id": "29", 201 | "name": "30", 202 | "location": "31", 203 | "bookings": "32" 204 | },"33722c67-7b62-4dbb-81cb-e64a9aba6713","Stefan","Treptower Park"] -------------------------------------------------------------------------------- /week-8/tests/e2e/passengers.test.js: -------------------------------------------------------------------------------- 1 | const app = require('../..') 2 | const request = require('supertest')(app) 3 | 4 | test('creates a new passenger', async (done) => { 5 | const passengerToCreate = { 6 | name: 'Test passenger', 7 | location: 'Moda' 8 | } 9 | 10 | const response = await request 11 | .post('/passengers') 12 | .send(passengerToCreate) 13 | .expect(200) 14 | 15 | const passengerCreated = response.body 16 | 17 | expect(passengerCreated).toMatchObject(passengerToCreate) 18 | expect(passengerCreated.bookings).toEqual([]) 19 | 20 | done() 21 | }) 22 | -------------------------------------------------------------------------------- /week-8/tests/scenario/bookings.test.js: -------------------------------------------------------------------------------- 1 | const app = require('../..') 2 | const request = require('supertest')(app) 3 | 4 | test('creates a new booking', async (done) => { 5 | const passengerToCreate = { 6 | name: 'Test passenger', 7 | location: 'Moda' 8 | } 9 | 10 | const driverToCreate = { 11 | name: 'Test driver', 12 | location: 'Moda', 13 | age: 18 14 | } 15 | 16 | const origin = 'Moda' 17 | const destination = 'Bostanci' 18 | 19 | const passengerResponse = await request 20 | .post('/passengers') 21 | .send(passengerToCreate) 22 | .expect(200) 23 | 24 | const driverResponse = await request 25 | .post('/drivers') 26 | .send(driverToCreate) 27 | .expect(200) 28 | 29 | const bookingResponse = await request 30 | .post(`/passengers/${passengerResponse.body._id}/bookings`) 31 | .send({ 32 | driverId: driverResponse.body._id, 33 | origin, 34 | destination 35 | }) 36 | .expect(200) 37 | 38 | const bookingCreated = bookingResponse.body 39 | 40 | expect(bookingCreated).toMatchObject({ 41 | driver: driverResponse.body, 42 | passenger: passengerResponse.body, 43 | origin, 44 | destination 45 | }) 46 | 47 | done() 48 | }) 49 | -------------------------------------------------------------------------------- /week-8/views/bookings.pug: -------------------------------------------------------------------------------- 1 | extends layout 2 | block content 3 | h2 Bookings 4 | p There are #{bookings.length} bookings. 5 | 6 | ol 7 | each booking in bookings 8 | li 9 | p Passenger: 10 | a(href=`/passengers/${booking.passenger.id}`) #{booking.passenger.name} 11 | p Driver: 12 | a(href=`/drivers/${booking.driver.id}`) #{booking.driver.name} 13 | p Origin: #{booking.origin} 14 | p Destination: #{booking.destination} 15 | -------------------------------------------------------------------------------- /week-8/views/driver.pug: -------------------------------------------------------------------------------- 1 | extends layout 2 | block content 3 | h2 Driver Detail 4 | p #{driver.name} 5 | p #{driver.age} 6 | p #{driver.location} 7 | -------------------------------------------------------------------------------- /week-8/views/drivers.pug: -------------------------------------------------------------------------------- 1 | extends layout 2 | block content 3 | h2 Drivers 4 | p There are #{drivers.length} drivers waiting. 5 | 6 | ol 7 | each driver in drivers 8 | li 9 | a(href=`/drivers/${driver.id}`) #{driver.name} 10 | -------------------------------------------------------------------------------- /week-8/views/index.pug: -------------------------------------------------------------------------------- 1 | extends layout 2 | block content 3 | p With this app, you can book a taxi from one place to another. 4 | -------------------------------------------------------------------------------- /week-8/views/layout.pug: -------------------------------------------------------------------------------- 1 | 2 | html(lang="en") 3 | head 4 | meta(charset="UTF-8") 5 | meta(name="viewport", content="width=device-width, initial-scale=1.0") 6 | title Taxi-booking app 7 | style. 8 | body { font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, Oxygen, Ubuntu, Cantarell, 'Open Sans', 'Helvetica Neue', sans-serif;} 9 | 10 | body 11 | h1 Hi! Welcome to taxi-booking app 12 | 13 | block content 14 | 15 | script(src="https://unpkg.com/axios/dist/axios.min.js") 16 | -------------------------------------------------------------------------------- /week-8/views/passenger.pug: -------------------------------------------------------------------------------- 1 | extends layout 2 | block content 3 | h2 Passenger Detail 4 | p #{passenger.name} 5 | 6 | h3 Booking History 7 | if passenger.bookings.length 8 | ol 9 | each booking in passenger.bookings 10 | li From #{booking.origin} to #{booking.destination} with #{booking.driver.name} 11 | else 12 | p No bookings 13 | -------------------------------------------------------------------------------- /week-8/views/passengers.pug: -------------------------------------------------------------------------------- 1 | extends layout 2 | block content 3 | h2 Passengers 4 | p There are #{passengers.length} passengers waiting. 5 | 6 | ol 7 | each passenger in passengers 8 | li 9 | a(href=`/passengers/${passenger.id}`) #{passenger.name} 10 | -------------------------------------------------------------------------------- /week-9/.gitignore: -------------------------------------------------------------------------------- 1 | data 2 | -------------------------------------------------------------------------------- /week-9/.vscode/launch.json: -------------------------------------------------------------------------------- 1 | { 2 | // Use IntelliSense to learn about possible attributes. 3 | // Hover to view descriptions of existing attributes. 4 | // For more information, visit: https://go.microsoft.com/fwlink/?linkid=830387 5 | "version": "0.2.0", 6 | "configurations": [ 7 | { 8 | "type": "node", 9 | "request": "launch", 10 | "name": "Launch Current File", 11 | "program": "${file}", 12 | "skipFiles": [ 13 | "/**" 14 | ] 15 | }, 16 | { 17 | "type": "node", 18 | "request": "launch", 19 | "name": "Launch Program", 20 | "skipFiles": [ 21 | "/**" 22 | ], 23 | "program": "${workspaceFolder}/index.js" 24 | } 25 | ] 26 | } 27 | -------------------------------------------------------------------------------- /week-9/.vscode/settings.json: -------------------------------------------------------------------------------- 1 | 123,125 -------------------------------------------------------------------------------- /week-9/backend/.bin/www: -------------------------------------------------------------------------------- 1 | const app = require('../') 2 | 3 | app.listen(3000, () => { 4 | console.log('started listening on 3000') 5 | }) 6 | -------------------------------------------------------------------------------- /week-9/backend/.dockerignore: -------------------------------------------------------------------------------- 1 | node_modules 2 | coverage 3 | tests 4 | requests.http 5 | __mocks__ 6 | jest.config.jest 7 | Dockerfile 8 | -------------------------------------------------------------------------------- /week-9/backend/Dockerfile: -------------------------------------------------------------------------------- 1 | FROM node:14.4.0-alpine3.11 2 | 3 | WORKDIR /app 4 | 5 | COPY package.json package-lock.json ./ 6 | 7 | RUN npm install 8 | 9 | COPY . . 10 | 11 | EXPOSE 3000 12 | 13 | CMD [ "npm", "start" ] 14 | -------------------------------------------------------------------------------- /week-9/backend/__mocks__/colors.js: -------------------------------------------------------------------------------- 1 | const colors = jest.createMockFromModule('colors') 2 | 3 | const unity = s => s 4 | 5 | colors.bgRed = {} 6 | colors.blue = unity 7 | colors.bgRed.white = unity 8 | 9 | module.exports = colors 10 | -------------------------------------------------------------------------------- /week-9/backend/create-records.js: -------------------------------------------------------------------------------- 1 | const Passenger = require('./models/passenger') 2 | const Driver = require('./models/driver') 3 | const passengerService = require('./services/passenger-services') 4 | const driverService = require('./services/driver-services') 5 | 6 | const printBookingHistory = require('./lib/print-booking-history') 7 | 8 | const armagan = Passenger.create({name: 'Armagan', location: 'Kreuzberg'}) 9 | const mert = Passenger.create({name: 'Mert', location: 'Mitte'}) 10 | const stefan = Driver.create({ name: 'Stefan', location: 'Treptower Park' }) 11 | 12 | armagan.book(stefan, 'Kreuzberg', 'Neukolln') 13 | armagan.book(stefan, 'Neukolln', 'Mitte') 14 | armagan.book(stefan, 'Mitte', 'Kreuzberg') 15 | armagan.book(stefan, 'Kreuzberg', 'SXF') 16 | mert.book(stefan, 'Mitte', 'Kreuzberg') 17 | 18 | async function main() { 19 | try { 20 | await passengerService.save([armagan, mert]) 21 | 22 | await driverService.save([stefan]) 23 | 24 | const betul = Passenger.create({ name: 'Betul', location: 'Tegel' }) 25 | 26 | await passengerService.insert(betul) 27 | const passengers = await passengerService.load() 28 | passengers.forEach(printBookingHistory) 29 | } catch (e) { 30 | return console.log(e) 31 | } 32 | } 33 | 34 | main() 35 | -------------------------------------------------------------------------------- /week-9/backend/fetch-records.js: -------------------------------------------------------------------------------- 1 | const { passengerService, driverService } = require('./services') 2 | const printBookingHistory = require('./lib/print-booking-history') 3 | 4 | async function main() { 5 | const stefan = await driverService.findBy('name', 'Stefan') 6 | const armagan = await passengerService.findByName('Armagan') 7 | 8 | armagan.book(stefan, 'Kreuzberg', 'Wannsee') 9 | passengerService.update(armagan) 10 | 11 | printBookingHistory(armagan) 12 | 13 | console.log(await passengerService.findBy('location', 'Mitte')) 14 | } 15 | 16 | main() 17 | -------------------------------------------------------------------------------- /week-9/backend/index.js: -------------------------------------------------------------------------------- 1 | const express = require('express') 2 | const bodyParser = require('body-parser') 3 | const cors = require('cors') 4 | 5 | const passengersRouter = require('./routes/passengers') 6 | const driversRouter = require('./routes/drivers') 7 | const bookingsRouter = require('./routes/bookings') 8 | const indexRouter = require('./routes/index') 9 | require('./mongo-connection') 10 | 11 | const app = express() 12 | app.use(bodyParser.json()) 13 | app.use(cors()) 14 | 15 | app.set('view engine', 'pug') 16 | 17 | app.use('/passengers', passengersRouter) 18 | app.use('/drivers', driversRouter) 19 | app.use('/bookings', bookingsRouter) 20 | app.use('/', indexRouter) 21 | 22 | module.exports = app 23 | -------------------------------------------------------------------------------- /week-9/backend/jest.config.js: -------------------------------------------------------------------------------- 1 | module.exports = { 2 | testEnvironment: 'node' 3 | }; 4 | -------------------------------------------------------------------------------- /week-9/backend/lib/__tests__/print-booking-history.test.js: -------------------------------------------------------------------------------- 1 | const printBookingHistory = require('../print-booking-history') 2 | 3 | test('prints passenger bookings when a passenger has a booking', () => { 4 | // initialization 5 | const passenger = { 6 | name: 'Armagan', 7 | bookings: [{ 8 | passenger: { name: 'Armagan' }, 9 | driver: { name: 'Stefan' }, 10 | origin: 'Kreuzberg', 11 | destination: 'Neukolln' 12 | }] 13 | } 14 | const consoleSpy = jest.spyOn(console, 'log') 15 | 16 | // test 17 | printBookingHistory(passenger) 18 | 19 | // assertions 20 | expect(consoleSpy).toHaveBeenCalledWith('Armagan booked Stefan to travel from Kreuzberg to Neukolln') 21 | 22 | // teardown 23 | consoleSpy.mockRestore() 24 | }) 25 | 26 | test('prints warning message when a passenger has no bookings', () => { 27 | // initialization 28 | const passenger = { 29 | name: 'Armagan', 30 | bookings: [] 31 | } 32 | const consoleSpy = jest.spyOn(console, 'log') 33 | 34 | // test 35 | printBookingHistory(passenger) 36 | 37 | // assertions 38 | expect(consoleSpy).toHaveBeenCalledWith('Armagan has no bookings yet.') 39 | 40 | // teardown 41 | consoleSpy.mockRestore() 42 | }) 43 | -------------------------------------------------------------------------------- /week-9/backend/lib/print-booking-history.js: -------------------------------------------------------------------------------- 1 | const colors = require('colors') 2 | 3 | function printBooking(booking) { 4 | console.log(`${colors.blue(booking.passenger.name)} booked ${colors.blue(booking.driver.name)} to travel from ${colors.bgRed.white(booking.origin)} to ${colors.bgRed.white(booking.destination)}`) 5 | } 6 | 7 | function printBookingHistory(passenger) { 8 | if (passenger.bookings.length == 0) 9 | return console.log(`${colors.blue(passenger.name)} has no bookings yet.`) 10 | 11 | passenger.bookings.forEach(printBooking) 12 | } 13 | 14 | module.exports = printBookingHistory 15 | -------------------------------------------------------------------------------- /week-9/backend/models/booking.js: -------------------------------------------------------------------------------- 1 | const mongoose = require('mongoose') 2 | 3 | const BookingSchema = new mongoose.Schema({ 4 | driver: { 5 | type: mongoose.Schema.Types.ObjectId, 6 | ref: 'Driver', 7 | autopopulate: { maxDepth: 1 } 8 | }, 9 | passenger: { 10 | type: mongoose.Schema.Types.ObjectId, 11 | ref: 'Passenger', 12 | autopopulate: { maxDepth: 1 } 13 | }, 14 | origin: String, 15 | destination: String 16 | }) 17 | 18 | BookingSchema.plugin(require('mongoose-autopopulate')) 19 | 20 | module.exports = mongoose.model('Booking', BookingSchema) 21 | -------------------------------------------------------------------------------- /week-9/backend/models/driver.js: -------------------------------------------------------------------------------- 1 | const mongoose = require('mongoose') 2 | 3 | const DriverSchema = new mongoose.Schema({ 4 | name: String, 5 | location: String, 6 | age: { type: Number, required: true, min: 18 }, 7 | }) 8 | 9 | module.exports = mongoose.model('Driver', DriverSchema) 10 | -------------------------------------------------------------------------------- /week-9/backend/models/passenger.js: -------------------------------------------------------------------------------- 1 | const mongoose = require('mongoose') 2 | 3 | const PassengerSchema = new mongoose.Schema({ 4 | name: { type: String, required: true, minlength: 2 }, 5 | location: String, 6 | bookings: [{ 7 | type: mongoose.Schema.Types.ObjectId, 8 | ref: 'Booking', 9 | autopopulate: { maxDepth: 2 } 10 | }] 11 | }) 12 | 13 | PassengerSchema.plugin(require('mongoose-autopopulate')) 14 | 15 | module.exports = mongoose.model('Passenger', PassengerSchema) 16 | -------------------------------------------------------------------------------- /week-9/backend/mongo-connection.js: -------------------------------------------------------------------------------- 1 | const mongoose = require('mongoose') 2 | 3 | const connectionString = process.env.MONGODB_CONNECTION_STRING || 'mongodb://mongodb/week-9' 4 | 5 | mongoose.connect(connectionString, { useNewUrlParser: true, useUnifiedTopology: true }) 6 | 7 | var db = mongoose.connection 8 | db.on('error', console.error.bind(console, 'connection error:')) 9 | db.once('open', function () { 10 | console.log('we are connected to mongodb!') 11 | }) 12 | -------------------------------------------------------------------------------- /week-9/backend/package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "week-9", 3 | "version": "1.0.0", 4 | "description": "In-class examples for week-9", 5 | "main": "index.js", 6 | "scripts": { 7 | "start": "node .bin/www", 8 | "watch": "nodemon --ignore '*.json' .bin/www", 9 | "test": "jest --coverage --verbose", 10 | "test-watch": "jest --coverage --verbose --watch" 11 | }, 12 | "author": "Armagan Amcalar ", 13 | "license": "ISC", 14 | "dependencies": { 15 | "body-parser": "^1.19.0", 16 | "colors": "^1.4.0", 17 | "cors": "^2.8.5", 18 | "express": "^4.17.1", 19 | "flatted": "^2.0.1", 20 | "mongoose": "^5.9.9", 21 | "mongoose-autopopulate": "^0.12.2", 22 | "pug": "^2.0.4", 23 | "uuid": "^7.0.2" 24 | }, 25 | "devDependencies": { 26 | "jest": "^26.0.1", 27 | "nodemon": "^2.0.2", 28 | "supertest": "^4.0.2" 29 | } 30 | } 31 | -------------------------------------------------------------------------------- /week-9/backend/requests.http: -------------------------------------------------------------------------------- 1 | POST http://localhost:3000/passengers HTTP/1.1 2 | content-type: application/json 3 | 4 | {"name":"armagan", "location": "kuzguncuk"} 5 | 6 | ### 7 | 8 | GET http://localhost:3000/passengers/5ea70b28bbf7cb710503f2d4 HTTP/1.1 9 | 10 | ### 11 | DELETE http://localhost:3000/passengers/5e9de11c0b0ac272d7866b1b HTTP/1.1 12 | 13 | ### 14 | 15 | PATCH http://localhost:3000/passengers/5e9de4cd9f11eb7402e56941 HTTP/1.1 16 | content-type: application/json 17 | 18 | {"name": "new name4"} 19 | 20 | ### 21 | 22 | POST http://localhost:3000/drivers HTTP/1.1 23 | content-type: application/json 24 | 25 | {"name":"older driver", "age": 35, "location": "kuzguncuk"} 26 | 27 | ### 28 | DELETE http://localhost:3000/drivers/5ea70bf063b0877147bd3e6b HTTP/1.1 29 | 30 | ### 31 | POST http://localhost:3000/passengers/5ea70b28bbf7cb710503f2d4/bookings HTTP/1.1 32 | content-type: application/json 33 | 34 | {"driverId": "5ea724416e3b607ac41df63a", "origin": "Kabatas", "destination": "Pendik"} 35 | 36 | ### 37 | 38 | PATCH http://localhost:3000/drivers/5ea70c8063b0877147bd3e6c HTTP/1.1 39 | content-type: application/json 40 | 41 | {"name": "a newer surname"} 42 | -------------------------------------------------------------------------------- /week-9/backend/routes/bookings.js: -------------------------------------------------------------------------------- 1 | const { bookingService } = require('../services') 2 | 3 | const router = require('express').Router() 4 | 5 | router.get('/', async (req, res) => { 6 | const bookings = await bookingService.load() 7 | 8 | res.render('bookings', { bookings }) 9 | }) 10 | 11 | router.get('/search', async (req, res) => { 12 | const origin = req.query.origin 13 | const destination = req.query.destination 14 | 15 | const query = {} 16 | 17 | if (origin) query.origin = origin 18 | if (destination) query.destination = destination 19 | 20 | const bookings = await bookingService.query(query) 21 | 22 | res.render('bookings', { bookings }) 23 | }) 24 | 25 | module.exports = router 26 | -------------------------------------------------------------------------------- /week-9/backend/routes/drivers.js: -------------------------------------------------------------------------------- 1 | const { driverService } = require('../services') 2 | 3 | const router = require('express').Router() 4 | 5 | router.get('/', async (req, res) => { 6 | res.send(await driverService.load()) 7 | }) 8 | 9 | router.post('/', async (req, res) => { 10 | const driver = await driverService.insert(req.body) 11 | 12 | res.send(driver) 13 | }) 14 | 15 | router.delete('/:driverId', async (req, res) => { 16 | await driverService.removeBy('_id', req.params.driverId) 17 | 18 | res.send('OK') 19 | }) 20 | 21 | router.get('/young-drivers', async (req, res) => { 22 | const drivers = await driverService.findYoungDrivers() 23 | 24 | res.render('drivers', { drivers }) 25 | }) 26 | 27 | router.get('/:driverId', async (req, res) => { 28 | const driver = await driverService.find(req.params.driverId) 29 | if (!driver) return res.status(404).send('Cannot find driver') 30 | res.render('driver', { driver }) 31 | }) 32 | 33 | router.patch('/:driverId', async (req, res) => { 34 | const { driverId } = req.params 35 | const { name } = req.body 36 | 37 | await driverService.update(driverId, { name }) 38 | }) 39 | 40 | module.exports = router 41 | -------------------------------------------------------------------------------- /week-9/backend/routes/index.js: -------------------------------------------------------------------------------- 1 | const router = require('express').Router() 2 | 3 | router.get('/', (req, res) => { 4 | res.render('index') 5 | }) 6 | 7 | module.exports = router 8 | -------------------------------------------------------------------------------- /week-9/backend/routes/passengers.js: -------------------------------------------------------------------------------- 1 | const { passengerService, bookingService } = require('../services') 2 | 3 | const router = require('express').Router() 4 | 5 | router.get('/', async (req, res) => { 6 | res.send(await passengerService.load()) 7 | }) 8 | 9 | router.post('/', async (req, res, next) => { 10 | try { 11 | const passenger = await passengerService.insert(req.body) 12 | res.send(passenger) 13 | } catch(e) { 14 | next(e) 15 | } 16 | }) 17 | 18 | router.delete('/:passengerId', async (req, res) => { 19 | await passengerService.removeBy('_id', req.params.passengerId) 20 | 21 | res.send('OK') 22 | }) 23 | 24 | router.get('/:passengerId', async (req, res) => { 25 | const passenger = await passengerService.find(req.params.passengerId) 26 | 27 | if (!passenger) return res.status(404) 28 | res.send(passenger) 29 | }) 30 | 31 | router.post('/:passengerId/bookings', async (req, res) => { 32 | const { passengerId } = req.params 33 | const { driverId, origin, destination } = req.body 34 | 35 | const booking = await bookingService.book(driverId, passengerId, origin, destination) 36 | 37 | res.send(booking) 38 | }) 39 | 40 | router.patch('/:passengerId', async (req, res) => { 41 | const { passengerId } = req.params 42 | const { name } = req.body 43 | 44 | await passengerService.update(passengerId, { name }) 45 | }) 46 | 47 | module.exports = router 48 | -------------------------------------------------------------------------------- /week-9/backend/services/base-service.js: -------------------------------------------------------------------------------- 1 | class BaseService { 2 | constructor(model) { 3 | this.model = model 4 | } 5 | 6 | save(objects) { 7 | return this.model.insertMany(objects) 8 | } 9 | 10 | load() { 11 | return this.model.find() 12 | } 13 | 14 | async insert(object) { 15 | return await this.model.create(object) 16 | } 17 | 18 | async removeBy(property, value) { 19 | return this.model.deleteOne({ [property]: value }) 20 | } 21 | 22 | async update(id, object) { 23 | return this.model.findByIdAndUpdate(id, object) 24 | } 25 | 26 | async find(id) { 27 | return this.model.findById(id) 28 | } 29 | 30 | async query(obj) { 31 | return this.model.find(obj) 32 | } 33 | 34 | async findBy(property, value) { 35 | return this.model.find({ [property]: value }) 36 | } 37 | } 38 | 39 | module.exports = BaseService 40 | -------------------------------------------------------------------------------- /week-9/backend/services/booking-service.js: -------------------------------------------------------------------------------- 1 | const BaseService = require('./base-service') 2 | const Booking = require('../models/booking') 3 | const driverService = require('./driver-service') 4 | const passengerService = require('./passenger-service') 5 | 6 | class BookingService extends BaseService { 7 | async findByPassengerId(passengerId) { 8 | return this.findBy('passenger', passengerId) 9 | } 10 | 11 | async findByDriverId(driverId) { 12 | return this.findBy('driver', driverId) 13 | } 14 | 15 | async book(driverId, passengerId, origin, destination) { 16 | const passenger = await passengerService.find(passengerId) 17 | const driver = await driverService.find(driverId) 18 | 19 | const booking = await this.insert({ driver, passenger, origin, destination }) 20 | passenger.bookings.push(booking) 21 | 22 | await passenger.save() 23 | 24 | return booking 25 | } 26 | } 27 | 28 | module.exports = new BookingService(Booking) 29 | -------------------------------------------------------------------------------- /week-9/backend/services/driver-service.js: -------------------------------------------------------------------------------- 1 | const BaseService = require('./base-service') 2 | const Driver = require('../models/driver') 3 | 4 | class DriverService extends BaseService { 5 | async findByDriverName(name) { 6 | return this.findBy('name', name) 7 | } 8 | 9 | async findByLocation(location) { 10 | return this.findBy('location', location) 11 | } 12 | 13 | async findYoungDrivers() { 14 | return this.query({ 15 | age: { 16 | $lt: 30, 17 | $gte: 18 18 | } 19 | }) 20 | } 21 | } 22 | 23 | module.exports = new DriverService(Driver) 24 | -------------------------------------------------------------------------------- /week-9/backend/services/driver.json: -------------------------------------------------------------------------------- 1 | [[ 2 | "1" 3 | ],{ 4 | "id": "2", 5 | "name": "3", 6 | "location": "4" 7 | },"33722c67-7b62-4dbb-81cb-e64a9aba6713","Stefan","Treptower Park"] -------------------------------------------------------------------------------- /week-9/backend/services/index.js: -------------------------------------------------------------------------------- 1 | const passengerService = require('./passenger-service') 2 | const driverService = require('./driver-service') 3 | const bookingService = require('./booking-service') 4 | 5 | module.exports = { passengerService, driverService, bookingService } 6 | -------------------------------------------------------------------------------- /week-9/backend/services/passenger-service.js: -------------------------------------------------------------------------------- 1 | const BaseService = require('./base-service') 2 | const Passenger = require('../models/passenger') 3 | 4 | class PassengerService extends BaseService { 5 | async findByName(name) { 6 | return this.findBy('name', name) 7 | } 8 | } 9 | 10 | module.exports = new PassengerService(Passenger) 11 | -------------------------------------------------------------------------------- /week-9/backend/services/passenger.json: -------------------------------------------------------------------------------- 1 | [[ 2 | "1", 3 | "2", 4 | "3", 5 | "4", 6 | "5", 7 | "6", 8 | "7", 9 | "8", 10 | "9", 11 | "10", 12 | "11", 13 | "12", 14 | "13", 15 | "14", 16 | "15", 17 | "16", 18 | "17", 19 | "18", 20 | "19", 21 | "20" 22 | ],{ 23 | "id": "21", 24 | "name": "22", 25 | "location": "23", 26 | "bookings": "24" 27 | },{ 28 | "id": "25", 29 | "name": "26", 30 | "location": "27", 31 | "bookings": "28" 32 | },{ 33 | "id": "29", 34 | "name": "30", 35 | "location": "31", 36 | "bookings": "32" 37 | },{ 38 | "id": "33", 39 | "bookings": "34" 40 | },{ 41 | "id": "35", 42 | "bookings": "36" 43 | },{ 44 | "id": "37", 45 | "bookings": "38" 46 | },{ 47 | "id": "39", 48 | "bookings": "40" 49 | },{ 50 | "id": "41", 51 | "name": "42", 52 | "location": "43", 53 | "bookings": "44" 54 | },{ 55 | "id": "45", 56 | "bookings": "46" 57 | },{ 58 | "id": "47", 59 | "name": "48", 60 | "location": "49", 61 | "bookings": "50" 62 | },{ 63 | "id": "51", 64 | "name": "48", 65 | "location": "49", 66 | "bookings": "52" 67 | },{ 68 | "id": "53", 69 | "name": "48", 70 | "location": "49", 71 | "bookings": "54" 72 | },{ 73 | "id": "55", 74 | "name": "48", 75 | "location": "49", 76 | "bookings": "56" 77 | },{ 78 | "id": "57", 79 | "name": "48", 80 | "location": "49", 81 | "bookings": "58" 82 | },{ 83 | "id": "59", 84 | "name": "48", 85 | "location": "49", 86 | "bookings": "60" 87 | },{ 88 | "id": "61", 89 | "name": "48", 90 | "location": "49", 91 | "bookings": "62" 92 | },{ 93 | "id": "63", 94 | "name": "48", 95 | "location": "49", 96 | "bookings": "64" 97 | },{ 98 | "id": "65", 99 | "name": "48", 100 | "location": "49", 101 | "bookings": "66" 102 | },{ 103 | "id": "67", 104 | "name": "48", 105 | "location": "49", 106 | "bookings": "68" 107 | },{ 108 | "id": "69", 109 | "name": "48", 110 | "location": "49", 111 | "bookings": "70" 112 | },"01ed8529-ad31-4351-b092-21cabb931e16","Armagan","Kreuzberg",[ 113 | "71", 114 | "72", 115 | "73", 116 | "74" 117 | ],"68e6a08c-f4cf-486e-b1df-dac92576bc48","Mert","Mitte",[ 118 | "75" 119 | ],"db80ba9d-c327-4222-b62c-f5f230a3e0b6","Betul","Tegel",[ 120 | "76", 121 | "77", 122 | "78" 123 | ],"92f71ad7-a806-41f6-bd33-59fcc754162f",[],"269e3f80-42bf-4409-af1e-b897dedfde4c",[],"59be8bc5-f28a-48e3-a98b-a7a7d40c4dfd",[],"7fa6f1dc-76b5-450b-8aba-f73ea0104bd9",[],"66f47f82-063c-43d7-9c6e-5311f9d31fa2","new passenger","Moda",[],"b8ca32d3-f610-4e41-8534-b53afb5c01fd",[],"ede41daa-a3d4-427f-96aa-94dda0c0299a","weird client","kuzguncuk",[],"88b04838-742d-4aa2-9e9f-b06a363d454c",[],"c2c00469-2022-4f27-b3e9-5705535c2a42",[],"1d569e2b-d8a3-410f-a957-85e55896ca11",[],"437f691b-a097-4daf-ba56-ffff33bc2935",[],"91effbd5-e751-4403-8bd6-aec6fe335806",[],"05a1e9a3-da4a-46a3-8086-65933d7cc1e3",[],"d50628c1-6922-43ef-beeb-df9073211b5a",[],"55c0329f-60c3-4810-9388-a26f096a6449",[],"e36afcbc-8349-43a4-b8e8-445ab464050a",[],"5d5c3d85-b328-457e-8598-4d0fe69c252c",[],{ 124 | "driver": "79", 125 | "passenger": "80", 126 | "origin": "23", 127 | "destination": "81" 128 | },{ 129 | "driver": "79", 130 | "passenger": "80", 131 | "origin": "81", 132 | "destination": "27" 133 | },{ 134 | "driver": "79", 135 | "passenger": "80", 136 | "origin": "27", 137 | "destination": "23" 138 | },{ 139 | "driver": "79", 140 | "passenger": "80", 141 | "origin": "23", 142 | "destination": "82" 143 | },{ 144 | "driver": "79", 145 | "passenger": "83", 146 | "origin": "27", 147 | "destination": "23" 148 | },{ 149 | "driver": "84", 150 | "passenger": "85", 151 | "origin": "82", 152 | "destination": "23" 153 | },{ 154 | "driver": "86", 155 | "passenger": "87", 156 | "origin": "23", 157 | "destination": "88" 158 | },{ 159 | "driver": "89", 160 | "passenger": "90", 161 | "origin": "88", 162 | "destination": "27" 163 | },{ 164 | "id": "91", 165 | "name": "92", 166 | "location": "93" 167 | },{ 168 | "id": "21", 169 | "name": "22", 170 | "location": "23", 171 | "bookings": "24" 172 | },"Neukolln","SXF",{ 173 | "id": "25", 174 | "name": "26", 175 | "location": "27", 176 | "bookings": "28" 177 | },{ 178 | "id": "91", 179 | "name": "92", 180 | "location": "93" 181 | },{ 182 | "id": "29", 183 | "name": "30", 184 | "location": "31", 185 | "bookings": "32" 186 | },{ 187 | "id": "91", 188 | "name": "92", 189 | "location": "93" 190 | },{ 191 | "id": "29", 192 | "name": "30", 193 | "location": "31", 194 | "bookings": "32" 195 | },"TXL",{ 196 | "id": "91", 197 | "name": "92", 198 | "location": "93" 199 | },{ 200 | "id": "29", 201 | "name": "30", 202 | "location": "31", 203 | "bookings": "32" 204 | },"33722c67-7b62-4dbb-81cb-e64a9aba6713","Stefan","Treptower Park"] -------------------------------------------------------------------------------- /week-9/backend/tests/e2e/passengers.test.js: -------------------------------------------------------------------------------- 1 | const app = require('../..') 2 | const request = require('supertest')(app) 3 | 4 | test('creates a new passenger', async (done) => { 5 | const passengerToCreate = { 6 | name: 'Test passenger', 7 | location: 'Moda' 8 | } 9 | 10 | const response = await request 11 | .post('/passengers') 12 | .send(passengerToCreate) 13 | .expect(200) 14 | 15 | const passengerCreated = response.body 16 | 17 | expect(passengerCreated).toMatchObject(passengerToCreate) 18 | expect(passengerCreated.bookings).toEqual([]) 19 | 20 | done() 21 | }) 22 | -------------------------------------------------------------------------------- /week-9/backend/tests/scenario/bookings.test.js: -------------------------------------------------------------------------------- 1 | const app = require('../..') 2 | const request = require('supertest')(app) 3 | 4 | test('creates a new booking', async (done) => { 5 | const passengerToCreate = { 6 | name: 'Test passenger', 7 | location: 'Moda' 8 | } 9 | 10 | const driverToCreate = { 11 | name: 'Test driver', 12 | location: 'Moda', 13 | age: 18 14 | } 15 | 16 | const origin = 'Moda' 17 | const destination = 'Bostanci' 18 | 19 | const passengerResponse = await request 20 | .post('/passengers') 21 | .send(passengerToCreate) 22 | .expect(200) 23 | 24 | const driverResponse = await request 25 | .post('/drivers') 26 | .send(driverToCreate) 27 | .expect(200) 28 | 29 | const bookingResponse = await request 30 | .post(`/passengers/${passengerResponse.body._id}/bookings`) 31 | .send({ 32 | driverId: driverResponse.body._id, 33 | origin, 34 | destination 35 | }) 36 | .expect(200) 37 | 38 | const bookingCreated = bookingResponse.body 39 | 40 | expect(bookingCreated).toMatchObject({ 41 | driver: driverResponse.body, 42 | passenger: passengerResponse.body, 43 | origin, 44 | destination 45 | }) 46 | 47 | done() 48 | }) 49 | -------------------------------------------------------------------------------- /week-9/backend/views/bookings.pug: -------------------------------------------------------------------------------- 1 | extends layout 2 | block content 3 | h2 Bookings 4 | p There are #{bookings.length} bookings. 5 | 6 | ol 7 | each booking in bookings 8 | li 9 | p Passenger: 10 | a(href=`/passengers/${booking.passenger.id}`) #{booking.passenger.name} 11 | p Driver: 12 | a(href=`/drivers/${booking.driver.id}`) #{booking.driver.name} 13 | p Origin: #{booking.origin} 14 | p Destination: #{booking.destination} 15 | -------------------------------------------------------------------------------- /week-9/backend/views/driver.pug: -------------------------------------------------------------------------------- 1 | extends layout 2 | block content 3 | h2 Driver Detail 4 | p #{driver.name} 5 | p #{driver.age} 6 | p #{driver.location} 7 | -------------------------------------------------------------------------------- /week-9/backend/views/drivers.pug: -------------------------------------------------------------------------------- 1 | extends layout 2 | block content 3 | h2 Drivers 4 | p There are #{drivers.length} drivers waiting. 5 | 6 | ol 7 | each driver in drivers 8 | li 9 | a(href=`/drivers/${driver.id}`) #{driver.name} 10 | -------------------------------------------------------------------------------- /week-9/backend/views/index.pug: -------------------------------------------------------------------------------- 1 | extends layout 2 | block content 3 | p With this app, you can book a taxi from one place to another. 4 | -------------------------------------------------------------------------------- /week-9/backend/views/layout.pug: -------------------------------------------------------------------------------- 1 | 2 | html(lang="en") 3 | head 4 | meta(charset="UTF-8") 5 | meta(name="viewport", content="width=device-width, initial-scale=1.0") 6 | title Taxi-booking app 7 | style. 8 | body { font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, Oxygen, Ubuntu, Cantarell, 'Open Sans', 'Helvetica Neue', sans-serif;} 9 | 10 | body 11 | h1 Hi! Welcome to taxi-booking app 12 | 13 | block content 14 | 15 | script(src="https://unpkg.com/axios/dist/axios.min.js") 16 | -------------------------------------------------------------------------------- /week-9/backend/views/passenger.pug: -------------------------------------------------------------------------------- 1 | extends layout 2 | block content 3 | h2 Passenger Detail 4 | p #{passenger.name} 5 | 6 | h3 Booking History 7 | if passenger.bookings.length 8 | ol 9 | each booking in passenger.bookings 10 | li From #{booking.origin} to #{booking.destination} with #{booking.driver.name} 11 | else 12 | p No bookings 13 | -------------------------------------------------------------------------------- /week-9/backend/views/passengers.pug: -------------------------------------------------------------------------------- 1 | extends layout 2 | block content 3 | h2 Passengers 4 | p There are #{passengers.length} passengers waiting. 5 | 6 | ol 7 | each passenger in passengers 8 | li 9 | a(href=`/passengers/${passenger.id}`) #{passenger.name} 10 | -------------------------------------------------------------------------------- /week-9/docker-compose.yml: -------------------------------------------------------------------------------- 1 | version: "3.8" 2 | services: 3 | frontend: 4 | build: frontend 5 | ports: 6 | - 8080:8080 7 | backend: 8 | build: backend 9 | environment: 10 | MONGODB_CONNECTION_STRING: "mongodb://mongodb/taxi-booking-app" 11 | ports: 12 | - 3000:3000 13 | mongodb: 14 | image: mongo:4.2 15 | ports: 16 | - 37017:27017 17 | volumes: 18 | - ./data:/data/db 19 | -------------------------------------------------------------------------------- /week-9/frontend/.browserslistrc: -------------------------------------------------------------------------------- 1 | > 1% 2 | last 2 versions 3 | not dead 4 | -------------------------------------------------------------------------------- /week-9/frontend/.dockerignore: -------------------------------------------------------------------------------- 1 | node_modules 2 | .editorconfig 3 | .eslintrc 4 | Dockerfile 5 | -------------------------------------------------------------------------------- /week-9/frontend/.editorconfig: -------------------------------------------------------------------------------- 1 | [*.{js,jsx,ts,tsx,vue}] 2 | indent_style = space 3 | indent_size = 2 4 | trim_trailing_whitespace = true 5 | insert_final_newline = true 6 | -------------------------------------------------------------------------------- /week-9/frontend/.eslintrc.js: -------------------------------------------------------------------------------- 1 | module.exports = { 2 | root: true, 3 | env: { 4 | node: true 5 | }, 6 | extends: [ 7 | 'plugin:vue/essential', 8 | '@vue/standard' 9 | ], 10 | parserOptions: { 11 | parser: 'babel-eslint' 12 | }, 13 | rules: { 14 | 'no-console': process.env.NODE_ENV === 'production' ? 'warn' : 'off', 15 | 'no-debugger': process.env.NODE_ENV === 'production' ? 'warn' : 'off' 16 | } 17 | } 18 | -------------------------------------------------------------------------------- /week-9/frontend/.gitignore: -------------------------------------------------------------------------------- 1 | .DS_Store 2 | node_modules 3 | /dist 4 | 5 | # local env files 6 | .env.local 7 | .env.*.local 8 | 9 | # Log files 10 | npm-debug.log* 11 | yarn-debug.log* 12 | yarn-error.log* 13 | pnpm-debug.log* 14 | 15 | # Editor directories and files 16 | .idea 17 | .vscode 18 | *.suo 19 | *.ntvs* 20 | *.njsproj 21 | *.sln 22 | *.sw? 23 | -------------------------------------------------------------------------------- /week-9/frontend/Dockerfile: -------------------------------------------------------------------------------- 1 | FROM node:14.4.0-alpine3.11 2 | 3 | WORKDIR /app 4 | 5 | COPY package.json package-lock.json ./ 6 | 7 | RUN npm install 8 | 9 | COPY . . 10 | 11 | EXPOSE 8080 12 | 13 | CMD [ "npm", "run", "serve" ] 14 | -------------------------------------------------------------------------------- /week-9/frontend/README.md: -------------------------------------------------------------------------------- 1 | # frontend 2 | 3 | ## Project setup 4 | ``` 5 | npm install 6 | ``` 7 | 8 | ### Compiles and hot-reloads for development 9 | ``` 10 | npm run serve 11 | ``` 12 | 13 | ### Compiles and minifies for production 14 | ``` 15 | npm run build 16 | ``` 17 | 18 | ### Lints and fixes files 19 | ``` 20 | npm run lint 21 | ``` 22 | 23 | ### Customize configuration 24 | See [Configuration Reference](https://cli.vuejs.org/config/). 25 | -------------------------------------------------------------------------------- /week-9/frontend/babel.config.js: -------------------------------------------------------------------------------- 1 | module.exports = { 2 | presets: [ 3 | '@vue/cli-plugin-babel/preset' 4 | ] 5 | } 6 | -------------------------------------------------------------------------------- /week-9/frontend/package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "frontend", 3 | "version": "0.1.0", 4 | "private": true, 5 | "scripts": { 6 | "serve": "vue-cli-service serve", 7 | "build": "vue-cli-service build", 8 | "lint": "vue-cli-service lint" 9 | }, 10 | "dependencies": { 11 | "axios": "^0.19.2", 12 | "core-js": "^3.6.5", 13 | "register-service-worker": "^1.7.1", 14 | "vue": "^2.6.11", 15 | "vue-router": "^3.2.0", 16 | "vuex": "^3.4.0" 17 | }, 18 | "devDependencies": { 19 | "@vue/cli-plugin-babel": "~4.4.0", 20 | "@vue/cli-plugin-eslint": "~4.4.0", 21 | "@vue/cli-plugin-pwa": "~4.4.0", 22 | "@vue/cli-plugin-router": "~4.4.0", 23 | "@vue/cli-plugin-vuex": "~4.4.0", 24 | "@vue/cli-service": "~4.4.0", 25 | "@vue/eslint-config-standard": "^5.1.2", 26 | "babel-eslint": "^10.1.0", 27 | "eslint": "^6.7.2", 28 | "eslint-plugin-import": "^2.20.2", 29 | "eslint-plugin-node": "^11.1.0", 30 | "eslint-plugin-promise": "^4.2.1", 31 | "eslint-plugin-standard": "^4.0.0", 32 | "eslint-plugin-vue": "^6.2.2", 33 | "lint-staged": "^9.5.0", 34 | "node-sass": "^4.12.0", 35 | "pug": "^3.0.0", 36 | "pug-plain-loader": "^1.0.0", 37 | "sass-loader": "^8.0.2", 38 | "vue-template-compiler": "^2.6.11" 39 | }, 40 | "gitHooks": { 41 | "pre-commit": "lint-staged" 42 | }, 43 | "lint-staged": { 44 | "*.{js,jsx,vue}": [ 45 | "vue-cli-service lint", 46 | "git add" 47 | ] 48 | } 49 | } 50 | -------------------------------------------------------------------------------- /week-9/frontend/public/favicon.ico: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/nimbleturkiye/js-classes/cb8bc70d5df6e49f92a1c2e04ad86c8e2f4bc695/week-9/frontend/public/favicon.ico -------------------------------------------------------------------------------- /week-9/frontend/public/img/icons/android-chrome-192x192.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/nimbleturkiye/js-classes/cb8bc70d5df6e49f92a1c2e04ad86c8e2f4bc695/week-9/frontend/public/img/icons/android-chrome-192x192.png -------------------------------------------------------------------------------- /week-9/frontend/public/img/icons/android-chrome-512x512.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/nimbleturkiye/js-classes/cb8bc70d5df6e49f92a1c2e04ad86c8e2f4bc695/week-9/frontend/public/img/icons/android-chrome-512x512.png -------------------------------------------------------------------------------- /week-9/frontend/public/img/icons/android-chrome-maskable-192x192.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/nimbleturkiye/js-classes/cb8bc70d5df6e49f92a1c2e04ad86c8e2f4bc695/week-9/frontend/public/img/icons/android-chrome-maskable-192x192.png -------------------------------------------------------------------------------- /week-9/frontend/public/img/icons/android-chrome-maskable-512x512.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/nimbleturkiye/js-classes/cb8bc70d5df6e49f92a1c2e04ad86c8e2f4bc695/week-9/frontend/public/img/icons/android-chrome-maskable-512x512.png -------------------------------------------------------------------------------- /week-9/frontend/public/img/icons/apple-touch-icon-120x120.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/nimbleturkiye/js-classes/cb8bc70d5df6e49f92a1c2e04ad86c8e2f4bc695/week-9/frontend/public/img/icons/apple-touch-icon-120x120.png -------------------------------------------------------------------------------- /week-9/frontend/public/img/icons/apple-touch-icon-152x152.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/nimbleturkiye/js-classes/cb8bc70d5df6e49f92a1c2e04ad86c8e2f4bc695/week-9/frontend/public/img/icons/apple-touch-icon-152x152.png -------------------------------------------------------------------------------- /week-9/frontend/public/img/icons/apple-touch-icon-180x180.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/nimbleturkiye/js-classes/cb8bc70d5df6e49f92a1c2e04ad86c8e2f4bc695/week-9/frontend/public/img/icons/apple-touch-icon-180x180.png -------------------------------------------------------------------------------- /week-9/frontend/public/img/icons/apple-touch-icon-60x60.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/nimbleturkiye/js-classes/cb8bc70d5df6e49f92a1c2e04ad86c8e2f4bc695/week-9/frontend/public/img/icons/apple-touch-icon-60x60.png -------------------------------------------------------------------------------- /week-9/frontend/public/img/icons/apple-touch-icon-76x76.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/nimbleturkiye/js-classes/cb8bc70d5df6e49f92a1c2e04ad86c8e2f4bc695/week-9/frontend/public/img/icons/apple-touch-icon-76x76.png -------------------------------------------------------------------------------- /week-9/frontend/public/img/icons/apple-touch-icon.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/nimbleturkiye/js-classes/cb8bc70d5df6e49f92a1c2e04ad86c8e2f4bc695/week-9/frontend/public/img/icons/apple-touch-icon.png -------------------------------------------------------------------------------- /week-9/frontend/public/img/icons/favicon-16x16.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/nimbleturkiye/js-classes/cb8bc70d5df6e49f92a1c2e04ad86c8e2f4bc695/week-9/frontend/public/img/icons/favicon-16x16.png -------------------------------------------------------------------------------- /week-9/frontend/public/img/icons/favicon-32x32.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/nimbleturkiye/js-classes/cb8bc70d5df6e49f92a1c2e04ad86c8e2f4bc695/week-9/frontend/public/img/icons/favicon-32x32.png -------------------------------------------------------------------------------- /week-9/frontend/public/img/icons/msapplication-icon-144x144.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/nimbleturkiye/js-classes/cb8bc70d5df6e49f92a1c2e04ad86c8e2f4bc695/week-9/frontend/public/img/icons/msapplication-icon-144x144.png -------------------------------------------------------------------------------- /week-9/frontend/public/img/icons/mstile-150x150.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/nimbleturkiye/js-classes/cb8bc70d5df6e49f92a1c2e04ad86c8e2f4bc695/week-9/frontend/public/img/icons/mstile-150x150.png -------------------------------------------------------------------------------- /week-9/frontend/public/img/icons/safari-pinned-tab.svg: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | -------------------------------------------------------------------------------- /week-9/frontend/public/index.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | <%= htmlWebpackPlugin.options.title %> 9 | 10 | 11 | 14 |
15 | 16 | 17 | 18 | -------------------------------------------------------------------------------- /week-9/frontend/public/robots.txt: -------------------------------------------------------------------------------- 1 | User-agent: * 2 | Disallow: 3 | -------------------------------------------------------------------------------- /week-9/frontend/src/App.vue: -------------------------------------------------------------------------------- 1 | 10 | 11 | 33 | -------------------------------------------------------------------------------- /week-9/frontend/src/assets/logo.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/nimbleturkiye/js-classes/cb8bc70d5df6e49f92a1c2e04ad86c8e2f4bc695/week-9/frontend/src/assets/logo.png -------------------------------------------------------------------------------- /week-9/frontend/src/components/HelloWorld.vue: -------------------------------------------------------------------------------- 1 | 16 | 17 | 24 | 25 | 27 | -------------------------------------------------------------------------------- /week-9/frontend/src/main.js: -------------------------------------------------------------------------------- 1 | import Vue from 'vue' 2 | import App from './App.vue' 3 | import './registerServiceWorker' 4 | import router from './router' 5 | import store from './store' 6 | 7 | Vue.config.productionTip = false 8 | 9 | new Vue({ 10 | router, 11 | store, 12 | render: h => h(App) 13 | }).$mount('#app') 14 | -------------------------------------------------------------------------------- /week-9/frontend/src/registerServiceWorker.js: -------------------------------------------------------------------------------- 1 | /* eslint-disable no-console */ 2 | 3 | import { register } from 'register-service-worker' 4 | 5 | if (process.env.NODE_ENV === 'production') { 6 | register(`${process.env.BASE_URL}service-worker.js`, { 7 | ready () { 8 | console.log( 9 | 'App is being served from cache by a service worker.\n' + 10 | 'For more details, visit https://goo.gl/AFskqB' 11 | ) 12 | }, 13 | registered () { 14 | console.log('Service worker has been registered.') 15 | }, 16 | cached () { 17 | console.log('Content has been cached for offline use.') 18 | }, 19 | updatefound () { 20 | console.log('New content is downloading.') 21 | }, 22 | updated () { 23 | console.log('New content is available; please refresh.') 24 | }, 25 | offline () { 26 | console.log('No internet connection found. App is running in offline mode.') 27 | }, 28 | error (error) { 29 | console.error('Error during service worker registration:', error) 30 | } 31 | }) 32 | } 33 | -------------------------------------------------------------------------------- /week-9/frontend/src/router/index.js: -------------------------------------------------------------------------------- 1 | import Vue from 'vue' 2 | import VueRouter from 'vue-router' 3 | import Passengers from '../views/Passengers.vue' 4 | import PassengerView from '../views/Passenger.vue' 5 | 6 | Vue.use(VueRouter) 7 | 8 | const routes = [ 9 | { 10 | path: '/', 11 | name: 'Passengers', 12 | component: Passengers 13 | }, 14 | { 15 | path: '/passengers/:passengerId', 16 | name: 'PassengerView', 17 | component: PassengerView 18 | }, 19 | { 20 | path: '/about', 21 | name: 'About', 22 | // route level code-splitting 23 | // this generates a separate chunk (about.[hash].js) for this route 24 | // which is lazy-loaded when the route is visited. 25 | component: () => import(/* webpackChunkName: "about" */ '../views/About.vue') 26 | } 27 | ] 28 | 29 | const router = new VueRouter({ 30 | mode: 'history', 31 | base: process.env.BASE_URL, 32 | routes 33 | }) 34 | 35 | export default router 36 | -------------------------------------------------------------------------------- /week-9/frontend/src/store/index.js: -------------------------------------------------------------------------------- 1 | import Vue from 'vue' 2 | import Vuex from 'vuex' 3 | import axios from 'axios' 4 | 5 | axios.defaults.baseURL = 'http://localhost:3000' 6 | 7 | Vue.use(Vuex) 8 | 9 | const Mutations = { 10 | INCREMENT: 'INCREMENT', 11 | DECREMENT: 'DECREMENT' 12 | } 13 | 14 | export default new Vuex.Store({ 15 | state: { 16 | countHome: 0, 17 | countAbout: 0 18 | }, 19 | mutations: { 20 | [Mutations.INCREMENT] (state, type) { 21 | type === 'countHome' ? state.countHome++ : state.countAbout++ 22 | }, 23 | [Mutations.DECREMENT] (state, type) { 24 | if (state[type] === 0) return 25 | 26 | type === 'countHome' ? state.countHome-- : state.countAbout-- 27 | } 28 | }, 29 | actions: { 30 | increment ({ commit }, type) { 31 | commit(Mutations.INCREMENT, type) 32 | }, 33 | decrement ({ commit }, type) { 34 | commit(Mutations.DECREMENT, type) 35 | }, 36 | async fetchPassengers () { 37 | const request = await axios.get('/passengers') 38 | return request.data 39 | }, 40 | async fetchDrivers () { 41 | const request = await axios.get('/drivers') 42 | return request.data 43 | }, 44 | async fetchPassenger ({ state }, passengerId) { 45 | const request = await axios.get(`/passengers/${passengerId}`) 46 | return request.data 47 | }, 48 | async bookDriver ({ dispatch }, { passengerId, driverId, origin, destination }) { 49 | const request = await axios.post(`/passengers/${passengerId}/bookings`, { 50 | driverId, origin, destination 51 | }) 52 | 53 | return request.data 54 | } 55 | } 56 | }) 57 | -------------------------------------------------------------------------------- /week-9/frontend/src/views/About.vue: -------------------------------------------------------------------------------- 1 | 15 | 16 | 21 | -------------------------------------------------------------------------------- /week-9/frontend/src/views/Passenger.vue: -------------------------------------------------------------------------------- 1 | 36 | 37 | 60 | -------------------------------------------------------------------------------- /week-9/frontend/src/views/Passengers.vue: -------------------------------------------------------------------------------- 1 | 21 | 22 | 33 | --------------------------------------------------------------------------------