├── .gitattributes ├── .github └── workflows │ └── ci.yml ├── .gitignore ├── README.md ├── app ├── build.gradle └── src │ ├── main │ └── java │ │ └── com │ │ └── codesoom │ │ └── assignment │ │ └── App.java │ └── test │ └── java │ └── com │ └── codesoom │ └── assignment │ └── AppTest.java ├── gradle └── wrapper │ ├── gradle-wrapper.jar │ └── gradle-wrapper.properties ├── gradlew ├── gradlew.bat ├── settings.gradle └── tests ├── .eslintrc.js ├── .gitignore ├── __tests__ └── api.spec.js ├── babel.config.js ├── jest.config.js ├── package-lock.json └── package.json /.gitattributes: -------------------------------------------------------------------------------- 1 | # 2 | # https://help.github.com/articles/dealing-with-line-endings/ 3 | # 4 | # These are explicitly windows files and should use crlf 5 | *.bat text eol=crlf 6 | 7 | -------------------------------------------------------------------------------- /.github/workflows/ci.yml: -------------------------------------------------------------------------------- 1 | name: CI 2 | 3 | on: [push, pull_request] 4 | 5 | jobs: 6 | build: 7 | runs-on: ubuntu-latest 8 | 9 | steps: 10 | - uses: actions/checkout@v2 11 | - name: Setup Java JDK 12 | uses: actions/setup-java@v1.4.3 13 | with: 14 | java-version: 15.0.1 15 | - name: Grant execute permission for gradlew 16 | run: chmod +x gradlew 17 | - name: Install 18 | working-directory: ./tests 19 | run: npm ci 20 | - name: Test 21 | working-directory: ./tests 22 | run: npm run ci 23 | -------------------------------------------------------------------------------- /.gitignore: -------------------------------------------------------------------------------- 1 | # Ignore Gradle project-specific cache directory 2 | .gradle 3 | **/build/ 4 | !src/**/build/ 5 | caches 6 | daemon 7 | native 8 | notifications 9 | 10 | # Ignore Gradle build output directory 11 | build 12 | 13 | # VS Code 14 | .vscode 15 | 16 | # Intellij IDEA 17 | .idea 18 | 19 | # macOS 20 | .DS_Store 21 | 22 | # Windows thumbnail cache files 23 | Thumbs.db 24 | Thumbs.db:encryptable 25 | ehthumbs.db 26 | ehthumbs_vista.db 27 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # ToDo REST API 만들기 2 | 3 | ## 과제 목표 4 | 5 | Java를 이용해서 ToDo REST API를 만들어봅니다. 6 | 7 | ![API](https://user-images.githubusercontent.com/14071105/103476206-0456f280-4df7-11eb-89c4-d61845ef45ec.png) 8 | 9 | - ToDo 목록 얻기 - `GET /tasks` 10 | - ToDo 상세 조회하기 - `GET /tasks/{id}` 11 | - ToDo 생성하기 - `POST /tasks` 12 | - ToDo 제목 수정하기 - `PUT/PATCH /tasks/{id}` 13 | - ToDo 삭제하기 - `DELETE /tasks/{id}` 14 | 15 | ## 요구 사항 16 | 17 | - 모든 API 테스트를 통과해야 합니다. 18 | 19 | ## 실행하기 20 | 21 | ```bash 22 | ./gradlew run 23 | ``` 24 | 25 | ## 테스트 26 | 27 | ### 설치하기 28 | 29 | ```bash 30 | $ cd tests 31 | $ npm install 32 | ``` 33 | 34 | ### 테스트 실행하기 35 | 36 | 테스트는 실제로 동작하는 서버에 테스트하므로 서버가 동작하고 있는 상태여야 올바르게 동작합니다. 37 | 38 | ```bash 39 | $ npm run test 40 | ``` 41 | -------------------------------------------------------------------------------- /app/build.gradle: -------------------------------------------------------------------------------- 1 | /* 2 | * This file was generated by the Gradle 'init' task. 3 | * 4 | * This generated file contains a sample Java application project to get you started. 5 | * For more details take a look at the 'Building Java & JVM projects' chapter in the Gradle 6 | * User Manual available at https://docs.gradle.org/6.7.1/userguide/building_java_projects.html 7 | */ 8 | 9 | plugins { 10 | // Apply the application plugin to add support for building a CLI application in Java. 11 | id 'application' 12 | } 13 | 14 | repositories { 15 | // Use JCenter for resolving dependencies. 16 | jcenter() 17 | } 18 | 19 | dependencies { 20 | // Use JUnit Jupiter API for testing. 21 | testImplementation 'org.junit.jupiter:junit-jupiter-api:5.6.2' 22 | 23 | // Use JUnit Jupiter Engine for testing. 24 | testRuntimeOnly 'org.junit.jupiter:junit-jupiter-engine' 25 | 26 | // This dependency is used by the application. 27 | implementation 'com.google.guava:guava:29.0-jre' 28 | 29 | // Jackson for JSON. 30 | implementation 'com.fasterxml.jackson.core:jackson-core:2.11.3' 31 | implementation 'com.fasterxml.jackson.core:jackson-databind:2.11.3' 32 | } 33 | 34 | application { 35 | // Define the main class for the application. 36 | mainClass = 'com.codesoom.assignment.App' 37 | } 38 | 39 | tasks.named('test') { 40 | // Use junit platform for unit tests. 41 | useJUnitPlatform() 42 | } 43 | -------------------------------------------------------------------------------- /app/src/main/java/com/codesoom/assignment/App.java: -------------------------------------------------------------------------------- 1 | package com.codesoom.assignment; 2 | 3 | public class App { 4 | public String getGreeting() { 5 | return "Hello World!"; 6 | } 7 | 8 | public static void main(String[] args) { 9 | } 10 | } 11 | -------------------------------------------------------------------------------- /app/src/test/java/com/codesoom/assignment/AppTest.java: -------------------------------------------------------------------------------- 1 | package com.codesoom.assignment; 2 | 3 | import org.junit.jupiter.api.Test; 4 | 5 | import static org.junit.jupiter.api.Assertions.assertNotNull; 6 | 7 | class AppTest { 8 | @Test 9 | void appHasAGreeting() { 10 | App classUnderTest = new App(); 11 | assertNotNull(classUnderTest.getGreeting(), "app should have a greeting"); 12 | } 13 | } 14 | -------------------------------------------------------------------------------- /gradle/wrapper/gradle-wrapper.jar: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/CodeSoom/spring-week1-assignment-1/ad741ee01bb624ee21db2f8d8c3322fd127f0ce6/gradle/wrapper/gradle-wrapper.jar -------------------------------------------------------------------------------- /gradle/wrapper/gradle-wrapper.properties: -------------------------------------------------------------------------------- 1 | distributionBase=GRADLE_USER_HOME 2 | distributionPath=wrapper/dists 3 | distributionUrl=https\://services.gradle.org/distributions/gradle-6.7.1-bin.zip 4 | zipStoreBase=GRADLE_USER_HOME 5 | zipStorePath=wrapper/dists 6 | -------------------------------------------------------------------------------- /gradlew: -------------------------------------------------------------------------------- 1 | #!/usr/bin/env sh 2 | 3 | # 4 | # Copyright 2015 the original author or authors. 5 | # 6 | # Licensed under the Apache License, Version 2.0 (the "License"); 7 | # you may not use this file except in compliance with the License. 8 | # You may obtain a copy of the License at 9 | # 10 | # https://www.apache.org/licenses/LICENSE-2.0 11 | # 12 | # Unless required by applicable law or agreed to in writing, software 13 | # distributed under the License is distributed on an "AS IS" BASIS, 14 | # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 15 | # See the License for the specific language governing permissions and 16 | # limitations under the License. 17 | # 18 | 19 | ############################################################################## 20 | ## 21 | ## Gradle start up script for UN*X 22 | ## 23 | ############################################################################## 24 | 25 | # Attempt to set APP_HOME 26 | # Resolve links: $0 may be a link 27 | PRG="$0" 28 | # Need this for relative symlinks. 29 | while [ -h "$PRG" ] ; do 30 | ls=`ls -ld "$PRG"` 31 | link=`expr "$ls" : '.*-> \(.*\)$'` 32 | if expr "$link" : '/.*' > /dev/null; then 33 | PRG="$link" 34 | else 35 | PRG=`dirname "$PRG"`"/$link" 36 | fi 37 | done 38 | SAVED="`pwd`" 39 | cd "`dirname \"$PRG\"`/" >/dev/null 40 | APP_HOME="`pwd -P`" 41 | cd "$SAVED" >/dev/null 42 | 43 | APP_NAME="Gradle" 44 | APP_BASE_NAME=`basename "$0"` 45 | 46 | # Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script. 47 | DEFAULT_JVM_OPTS='"-Xmx64m" "-Xms64m"' 48 | 49 | # Use the maximum available, or set MAX_FD != -1 to use that value. 50 | MAX_FD="maximum" 51 | 52 | warn () { 53 | echo "$*" 54 | } 55 | 56 | die () { 57 | echo 58 | echo "$*" 59 | echo 60 | exit 1 61 | } 62 | 63 | # OS specific support (must be 'true' or 'false'). 64 | cygwin=false 65 | msys=false 66 | darwin=false 67 | nonstop=false 68 | case "`uname`" in 69 | CYGWIN* ) 70 | cygwin=true 71 | ;; 72 | Darwin* ) 73 | darwin=true 74 | ;; 75 | MINGW* ) 76 | msys=true 77 | ;; 78 | NONSTOP* ) 79 | nonstop=true 80 | ;; 81 | esac 82 | 83 | CLASSPATH=$APP_HOME/gradle/wrapper/gradle-wrapper.jar 84 | 85 | 86 | # Determine the Java command to use to start the JVM. 87 | if [ -n "$JAVA_HOME" ] ; then 88 | if [ -x "$JAVA_HOME/jre/sh/java" ] ; then 89 | # IBM's JDK on AIX uses strange locations for the executables 90 | JAVACMD="$JAVA_HOME/jre/sh/java" 91 | else 92 | JAVACMD="$JAVA_HOME/bin/java" 93 | fi 94 | if [ ! -x "$JAVACMD" ] ; then 95 | die "ERROR: JAVA_HOME is set to an invalid directory: $JAVA_HOME 96 | 97 | Please set the JAVA_HOME variable in your environment to match the 98 | location of your Java installation." 99 | fi 100 | else 101 | JAVACMD="java" 102 | which java >/dev/null 2>&1 || die "ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH. 103 | 104 | Please set the JAVA_HOME variable in your environment to match the 105 | location of your Java installation." 106 | fi 107 | 108 | # Increase the maximum file descriptors if we can. 109 | if [ "$cygwin" = "false" -a "$darwin" = "false" -a "$nonstop" = "false" ] ; then 110 | MAX_FD_LIMIT=`ulimit -H -n` 111 | if [ $? -eq 0 ] ; then 112 | if [ "$MAX_FD" = "maximum" -o "$MAX_FD" = "max" ] ; then 113 | MAX_FD="$MAX_FD_LIMIT" 114 | fi 115 | ulimit -n $MAX_FD 116 | if [ $? -ne 0 ] ; then 117 | warn "Could not set maximum file descriptor limit: $MAX_FD" 118 | fi 119 | else 120 | warn "Could not query maximum file descriptor limit: $MAX_FD_LIMIT" 121 | fi 122 | fi 123 | 124 | # For Darwin, add options to specify how the application appears in the dock 125 | if $darwin; then 126 | GRADLE_OPTS="$GRADLE_OPTS \"-Xdock:name=$APP_NAME\" \"-Xdock:icon=$APP_HOME/media/gradle.icns\"" 127 | fi 128 | 129 | # For Cygwin or MSYS, switch paths to Windows format before running java 130 | if [ "$cygwin" = "true" -o "$msys" = "true" ] ; then 131 | APP_HOME=`cygpath --path --mixed "$APP_HOME"` 132 | CLASSPATH=`cygpath --path --mixed "$CLASSPATH"` 133 | 134 | JAVACMD=`cygpath --unix "$JAVACMD"` 135 | 136 | # We build the pattern for arguments to be converted via cygpath 137 | ROOTDIRSRAW=`find -L / -maxdepth 1 -mindepth 1 -type d 2>/dev/null` 138 | SEP="" 139 | for dir in $ROOTDIRSRAW ; do 140 | ROOTDIRS="$ROOTDIRS$SEP$dir" 141 | SEP="|" 142 | done 143 | OURCYGPATTERN="(^($ROOTDIRS))" 144 | # Add a user-defined pattern to the cygpath arguments 145 | if [ "$GRADLE_CYGPATTERN" != "" ] ; then 146 | OURCYGPATTERN="$OURCYGPATTERN|($GRADLE_CYGPATTERN)" 147 | fi 148 | # Now convert the arguments - kludge to limit ourselves to /bin/sh 149 | i=0 150 | for arg in "$@" ; do 151 | CHECK=`echo "$arg"|egrep -c "$OURCYGPATTERN" -` 152 | CHECK2=`echo "$arg"|egrep -c "^-"` ### Determine if an option 153 | 154 | if [ $CHECK -ne 0 ] && [ $CHECK2 -eq 0 ] ; then ### Added a condition 155 | eval `echo args$i`=`cygpath --path --ignore --mixed "$arg"` 156 | else 157 | eval `echo args$i`="\"$arg\"" 158 | fi 159 | i=`expr $i + 1` 160 | done 161 | case $i in 162 | 0) set -- ;; 163 | 1) set -- "$args0" ;; 164 | 2) set -- "$args0" "$args1" ;; 165 | 3) set -- "$args0" "$args1" "$args2" ;; 166 | 4) set -- "$args0" "$args1" "$args2" "$args3" ;; 167 | 5) set -- "$args0" "$args1" "$args2" "$args3" "$args4" ;; 168 | 6) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" ;; 169 | 7) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" ;; 170 | 8) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" ;; 171 | 9) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" "$args8" ;; 172 | esac 173 | fi 174 | 175 | # Escape application args 176 | save () { 177 | for i do printf %s\\n "$i" | sed "s/'/'\\\\''/g;1s/^/'/;\$s/\$/' \\\\/" ; done 178 | echo " " 179 | } 180 | APP_ARGS=`save "$@"` 181 | 182 | # Collect all arguments for the java command, following the shell quoting and substitution rules 183 | eval set -- $DEFAULT_JVM_OPTS $JAVA_OPTS $GRADLE_OPTS "\"-Dorg.gradle.appname=$APP_BASE_NAME\"" -classpath "\"$CLASSPATH\"" org.gradle.wrapper.GradleWrapperMain "$APP_ARGS" 184 | 185 | exec "$JAVACMD" "$@" 186 | -------------------------------------------------------------------------------- /gradlew.bat: -------------------------------------------------------------------------------- 1 | @rem 2 | @rem Copyright 2015 the original author or authors. 3 | @rem 4 | @rem Licensed under the Apache License, Version 2.0 (the "License"); 5 | @rem you may not use this file except in compliance with the License. 6 | @rem You may obtain a copy of the License at 7 | @rem 8 | @rem https://www.apache.org/licenses/LICENSE-2.0 9 | @rem 10 | @rem Unless required by applicable law or agreed to in writing, software 11 | @rem distributed under the License is distributed on an "AS IS" BASIS, 12 | @rem WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13 | @rem See the License for the specific language governing permissions and 14 | @rem limitations under the License. 15 | @rem 16 | 17 | @if "%DEBUG%" == "" @echo off 18 | @rem ########################################################################## 19 | @rem 20 | @rem Gradle startup script for Windows 21 | @rem 22 | @rem ########################################################################## 23 | 24 | @rem Set local scope for the variables with windows NT shell 25 | if "%OS%"=="Windows_NT" setlocal 26 | 27 | set DIRNAME=%~dp0 28 | if "%DIRNAME%" == "" set DIRNAME=. 29 | set APP_BASE_NAME=%~n0 30 | set APP_HOME=%DIRNAME% 31 | 32 | @rem Resolve any "." and ".." in APP_HOME to make it shorter. 33 | for %%i in ("%APP_HOME%") do set APP_HOME=%%~fi 34 | 35 | @rem Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script. 36 | set DEFAULT_JVM_OPTS="-Xmx64m" "-Xms64m" 37 | 38 | @rem Find java.exe 39 | if defined JAVA_HOME goto findJavaFromJavaHome 40 | 41 | set JAVA_EXE=java.exe 42 | %JAVA_EXE% -version >NUL 2>&1 43 | if "%ERRORLEVEL%" == "0" goto execute 44 | 45 | echo. 46 | echo ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH. 47 | echo. 48 | echo Please set the JAVA_HOME variable in your environment to match the 49 | echo location of your Java installation. 50 | 51 | goto fail 52 | 53 | :findJavaFromJavaHome 54 | set JAVA_HOME=%JAVA_HOME:"=% 55 | set JAVA_EXE=%JAVA_HOME%/bin/java.exe 56 | 57 | if exist "%JAVA_EXE%" goto execute 58 | 59 | echo. 60 | echo ERROR: JAVA_HOME is set to an invalid directory: %JAVA_HOME% 61 | echo. 62 | echo Please set the JAVA_HOME variable in your environment to match the 63 | echo location of your Java installation. 64 | 65 | goto fail 66 | 67 | :execute 68 | @rem Setup the command line 69 | 70 | set CLASSPATH=%APP_HOME%\gradle\wrapper\gradle-wrapper.jar 71 | 72 | 73 | @rem Execute Gradle 74 | "%JAVA_EXE%" %DEFAULT_JVM_OPTS% %JAVA_OPTS% %GRADLE_OPTS% "-Dorg.gradle.appname=%APP_BASE_NAME%" -classpath "%CLASSPATH%" org.gradle.wrapper.GradleWrapperMain %* 75 | 76 | :end 77 | @rem End local scope for the variables with windows NT shell 78 | if "%ERRORLEVEL%"=="0" goto mainEnd 79 | 80 | :fail 81 | rem Set variable GRADLE_EXIT_CONSOLE if you need the _script_ return code instead of 82 | rem the _cmd.exe /c_ return code! 83 | if not "" == "%GRADLE_EXIT_CONSOLE%" exit 1 84 | exit /b 1 85 | 86 | :mainEnd 87 | if "%OS%"=="Windows_NT" endlocal 88 | 89 | :omega 90 | -------------------------------------------------------------------------------- /settings.gradle: -------------------------------------------------------------------------------- 1 | /* 2 | * This file was generated by the Gradle 'init' task. 3 | * 4 | * The settings file is used to specify which projects to include in your build. 5 | * 6 | * Detailed information about configuring a multi-project build in Gradle can be found 7 | * in the user manual at https://docs.gradle.org/6.7.1/userguide/multi_project_builds.html 8 | */ 9 | 10 | rootProject.name = 'week1-assignment-1' 11 | include('app') 12 | -------------------------------------------------------------------------------- /tests/.eslintrc.js: -------------------------------------------------------------------------------- 1 | module.exports = { 2 | env: { 3 | es2021: true, 4 | node: true, 5 | jest: true, 6 | }, 7 | extends: [ 8 | 'airbnb-base', 9 | ], 10 | globals: { 11 | context: 'readonly', 12 | }, 13 | parserOptions: { 14 | ecmaVersion: 12, 15 | sourceType: 'module', 16 | }, 17 | rules: { 18 | }, 19 | }; 20 | -------------------------------------------------------------------------------- /tests/.gitignore: -------------------------------------------------------------------------------- 1 | node_modules 2 | -------------------------------------------------------------------------------- /tests/__tests__/api.spec.js: -------------------------------------------------------------------------------- 1 | import frisby from 'frisby'; 2 | 3 | frisby.baseUrl('http://localhost:8000'); 4 | 5 | describe('tasks', () => { 6 | const title = '책 읽기'; 7 | 8 | beforeEach(async () => { 9 | const res = await frisby.get('/tasks'); 10 | const tasks = JSON.parse(res.body); 11 | await Promise.all( 12 | tasks.map(({ id }) => frisby.del(`/tasks/${id}`)), 13 | ); 14 | }); 15 | 16 | describe('GET /tasks', () => { 17 | context('when tasks is empty', () => { 18 | it('responses empty array', async () => { 19 | await frisby.get('/tasks') 20 | .expect('status', 200) 21 | .expect('bodyContains', '[]'); 22 | }); 23 | }); 24 | 25 | context('when tasks is exist', () => { 26 | beforeEach(async () => { 27 | await frisby.post('/tasks', { title }); 28 | }); 29 | 30 | it('responses tasks', async () => { 31 | await frisby.get('/tasks') 32 | .expect('status', 200) 33 | .expect('bodyContains', `${title}`); 34 | }); 35 | }); 36 | }); 37 | 38 | describe('GET /tasks/:id', () => { 39 | let id; 40 | 41 | context('with existing task id', () => { 42 | beforeEach(async () => { 43 | const res = await frisby.post('/tasks', { title }); 44 | const task = JSON.parse(res.body); 45 | id = task.id; 46 | }); 47 | 48 | it('responses task', async () => { 49 | await frisby.get(`/tasks/${id}`) 50 | .expect('status', 200) 51 | .expect('bodyContains', title); 52 | }); 53 | }); 54 | 55 | context('with not existing task id', () => { 56 | beforeEach(() => { 57 | id = 0; 58 | }); 59 | 60 | it('responses 404 Not Found error', async () => { 61 | await frisby.get(`/tasks/${id}`) 62 | .expect('status', 404); 63 | }); 64 | }); 65 | }); 66 | 67 | describe('POST /tasks', () => { 68 | it('responses 201 Created', async () => { 69 | await frisby.post('/tasks', { title }) 70 | .expect('status', 201); 71 | }); 72 | }); 73 | 74 | describe('PUT /tasks/:id', () => { 75 | let id; 76 | 77 | context('with existing task id', () => { 78 | beforeEach(async () => { 79 | const res = await frisby.post('/tasks', { title }); 80 | const task = JSON.parse(res.body); 81 | id = task.id; 82 | }); 83 | 84 | it('responses updated task', async () => { 85 | const res = await frisby.put(`/tasks/${id}`, { title: '밥 먹기' }) 86 | .expect('status', 200); 87 | const task = JSON.parse(res.body); 88 | 89 | expect(task.title).toBe('밥 먹기'); 90 | }); 91 | }); 92 | 93 | context('with not existing task id', () => { 94 | beforeEach(async () => { 95 | id = 0; 96 | }); 97 | 98 | it('responses 404 Not Found error', async () => { 99 | await frisby.put(`/tasks/${id}`, { title: '밥 먹기' }) 100 | .expect('status', 404); 101 | }); 102 | }); 103 | }); 104 | 105 | describe('PATCH /tasks/:id', () => { 106 | let id; 107 | 108 | context('with existing task id', () => { 109 | beforeEach(async () => { 110 | const res = await frisby.post('/tasks', { title }); 111 | const task = JSON.parse(res.body); 112 | id = task.id; 113 | }); 114 | 115 | it('responses updated task', async () => { 116 | const res = await frisby.put(`/tasks/${id}`, { title: '밥 먹기' }) 117 | .expect('status', 200); 118 | const task = JSON.parse(res.body); 119 | 120 | expect(task.title).toBe('밥 먹기'); 121 | }); 122 | }); 123 | 124 | context('with not existing task id', () => { 125 | beforeEach(async () => { 126 | id = 0; 127 | }); 128 | 129 | it('responses 404 Not Found error', async () => { 130 | await frisby.put(`/tasks/${id}`, { title: '밥 먹기' }) 131 | .expect('status', 404); 132 | }); 133 | }); 134 | }); 135 | 136 | describe('DELETE /tasks/:id', () => { 137 | let id; 138 | 139 | context('with existing task id', () => { 140 | beforeEach(async () => { 141 | const res = await frisby.post('/tasks', { title }); 142 | const task = JSON.parse(res.body); 143 | id = task.id; 144 | }); 145 | 146 | it('deletes task', async () => { 147 | await frisby.delete(`/tasks/${id}`) 148 | .expect('status', 204); 149 | 150 | await frisby.get(`/tasks/${id}`) 151 | .expect('status', 404); 152 | }); 153 | }); 154 | 155 | context('with not existing task id', () => { 156 | beforeEach(async () => { 157 | id = 0; 158 | }); 159 | 160 | it('responses 404 Not Found error', async () => { 161 | await frisby.delete(`/tasks/${id}`) 162 | .expect('status', 404); 163 | }); 164 | }); 165 | }); 166 | }); 167 | -------------------------------------------------------------------------------- /tests/babel.config.js: -------------------------------------------------------------------------------- 1 | module.exports = { 2 | presets: [ 3 | [ 4 | '@babel/preset-env', 5 | { 6 | targets: { 7 | node: 'current', 8 | }, 9 | }, 10 | ], 11 | ], 12 | }; 13 | -------------------------------------------------------------------------------- /tests/jest.config.js: -------------------------------------------------------------------------------- 1 | module.exports = { 2 | setupFilesAfterEnv: [ 3 | 'jest-plugin-context/setup', 4 | ], 5 | transform: { 6 | '^.+\\.[t|j]sx?$': 'babel-jest', 7 | }, 8 | }; 9 | -------------------------------------------------------------------------------- /tests/package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "tests", 3 | "version": "1.0.0", 4 | "description": "", 5 | "main": "index.js", 6 | "scripts": { 7 | "test": "jest --runInBand", 8 | "lint": "eslint .", 9 | "ci": "start-server-and-test '../gradlew run -p ../' http-get://localhost:8000/tasks test" 10 | }, 11 | "author": "", 12 | "license": "ISC", 13 | "devDependencies": { 14 | "@babel/preset-env": "^7.12.11", 15 | "babel-jest": "^26.6.3", 16 | "eslint": "^7.17.0", 17 | "eslint-config-airbnb-base": "^14.2.1", 18 | "eslint-plugin-import": "^2.22.1", 19 | "frisby": "^2.1.3", 20 | "jest": "^26.6.3", 21 | "jest-plugin-context": "^2.9.0", 22 | "start-server-and-test": "^1.11.7" 23 | } 24 | } 25 | --------------------------------------------------------------------------------