diff --git a/.gitignore b/.gitignore
index 0c9d47c18..6250dc478 100644
--- a/.gitignore
+++ b/.gitignore
@@ -2,3 +2,4 @@ build
.gradle
.idea
*.iml
+ci/variables.yml
\ No newline at end of file
diff --git a/.gradle/4.4/fileChanges/last-build.bin b/.gradle/4.4/fileChanges/last-build.bin
new file mode 100644
index 000000000..f76dd238a
Binary files /dev/null and b/.gradle/4.4/fileChanges/last-build.bin differ
diff --git a/.gradle/4.4/fileContent/annotation-processors.bin b/.gradle/4.4/fileContent/annotation-processors.bin
new file mode 100644
index 000000000..4ccb76e12
Binary files /dev/null and b/.gradle/4.4/fileContent/annotation-processors.bin differ
diff --git a/.gradle/4.4/fileContent/fileContent.lock b/.gradle/4.4/fileContent/fileContent.lock
new file mode 100644
index 000000000..b15af2766
Binary files /dev/null and b/.gradle/4.4/fileContent/fileContent.lock differ
diff --git a/.gradle/4.4/fileHashes/fileHashes.bin b/.gradle/4.4/fileHashes/fileHashes.bin
new file mode 100644
index 000000000..a8dc49062
Binary files /dev/null and b/.gradle/4.4/fileHashes/fileHashes.bin differ
diff --git a/.gradle/4.4/fileHashes/fileHashes.lock b/.gradle/4.4/fileHashes/fileHashes.lock
new file mode 100644
index 000000000..4cb1829af
Binary files /dev/null and b/.gradle/4.4/fileHashes/fileHashes.lock differ
diff --git a/.gradle/4.4/fileHashes/resourceHashesCache.bin b/.gradle/4.4/fileHashes/resourceHashesCache.bin
new file mode 100644
index 000000000..c9380f104
Binary files /dev/null and b/.gradle/4.4/fileHashes/resourceHashesCache.bin differ
diff --git a/.gradle/4.4/taskHistory/taskHistory.bin b/.gradle/4.4/taskHistory/taskHistory.bin
new file mode 100644
index 000000000..cf0250d4f
Binary files /dev/null and b/.gradle/4.4/taskHistory/taskHistory.bin differ
diff --git a/.gradle/4.4/taskHistory/taskHistory.lock b/.gradle/4.4/taskHistory/taskHistory.lock
new file mode 100644
index 000000000..3b57e7539
Binary files /dev/null and b/.gradle/4.4/taskHistory/taskHistory.lock differ
diff --git a/.gradle/buildOutputCleanup/buildOutputCleanup.lock b/.gradle/buildOutputCleanup/buildOutputCleanup.lock
new file mode 100644
index 000000000..0235f401e
Binary files /dev/null and b/.gradle/buildOutputCleanup/buildOutputCleanup.lock differ
diff --git a/.gradle/buildOutputCleanup/cache.properties b/.gradle/buildOutputCleanup/cache.properties
new file mode 100644
index 000000000..d6bb7e901
--- /dev/null
+++ b/.gradle/buildOutputCleanup/cache.properties
@@ -0,0 +1,2 @@
+#Fri Jul 27 01:08:31 SGT 2018
+gradle.version=4.4
diff --git a/.gradle/buildOutputCleanup/outputFiles.bin b/.gradle/buildOutputCleanup/outputFiles.bin
new file mode 100644
index 000000000..cc55b5c60
Binary files /dev/null and b/.gradle/buildOutputCleanup/outputFiles.bin differ
diff --git a/.idea/.name b/.idea/.name
new file mode 100644
index 000000000..3fdb09331
--- /dev/null
+++ b/.idea/.name
@@ -0,0 +1 @@
+pal-tracker
\ No newline at end of file
diff --git a/.idea/compiler.xml b/.idea/compiler.xml
new file mode 100644
index 000000000..fa227eb5a
--- /dev/null
+++ b/.idea/compiler.xml
@@ -0,0 +1,9 @@
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/.idea/gradle.xml b/.idea/gradle.xml
new file mode 100644
index 000000000..ba1ec5c7e
--- /dev/null
+++ b/.idea/gradle.xml
@@ -0,0 +1,16 @@
+
+
+
+
+
+
\ No newline at end of file
diff --git a/.idea/misc.xml b/.idea/misc.xml
new file mode 100644
index 000000000..bc8d0a3a6
--- /dev/null
+++ b/.idea/misc.xml
@@ -0,0 +1,7 @@
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/.idea/vcs.xml b/.idea/vcs.xml
new file mode 100644
index 000000000..94a25f7f4
--- /dev/null
+++ b/.idea/vcs.xml
@@ -0,0 +1,6 @@
+
+
+
+
+
+
\ No newline at end of file
diff --git a/.idea/workspace.xml b/.idea/workspace.xml
new file mode 100644
index 000000000..a89497314
--- /dev/null
+++ b/.idea/workspace.xml
@@ -0,0 +1,762 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ 1532624894965
+
+
+ 1532624894965
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/.travis.yml b/.travis.yml
new file mode 100644
index 000000000..a68a2c74c
--- /dev/null
+++ b/.travis.yml
@@ -0,0 +1,25 @@
+dist: trusty
+sudo: false
+notifications:
+ email: false
+env:
+ - RELEASE_TAG="release-$TRAVIS_BUILD_NUMBER"
+if: tag IS blank
+
+
+jobs:
+ include:
+ - stage: build and publish
+ language: java
+ jdk: oraclejdk8
+ install: skip
+ script: ./gradlew clean build
+ before_deploy:
+ - git config --local user.name "richard"
+ - git config --local user.email "fullydead@hotmail.com"
+ - git tag -f $RELEASE_TAG
+ deploy:
+ provider: releases
+ api_key: $GITHUB_OAUTH_TOKEN
+ file: "build/libs/pal-tracker.jar"
+ skip_cleanup: true
diff --git a/build.gradle b/build.gradle
new file mode 100644
index 000000000..310722b84
--- /dev/null
+++ b/build.gradle
@@ -0,0 +1,26 @@
+plugins {
+ id "java"
+ id "org.springframework.boot" version "1.5.4.RELEASE"
+}
+
+repositories {
+ mavenCentral()
+}
+
+dependencies {
+ compile("org.springframework.boot:spring-boot-starter-web")
+ compile("com.fasterxml.jackson.datatype:jackson-datatype-jsr310:2.9.1")
+
+ testCompile("junit:junit:4.12")
+ testCompile("org.mockito:mockito-core:2.20.1")
+ testCompile("org.assertj:assertj-core:3.8.0")
+ testCompile("org.springframework.boot:spring-boot-starter-test")
+}
+
+bootRun.environment([
+ "WELCOME_MESSAGE": "hello",
+])
+
+test.environment([
+ "WELCOME_MESSAGE": "Hello from test",
+])
\ No newline at end of file
diff --git a/build/classes/java/main/io/pivotal/pal/tracker/PalTrackerApplication.class b/build/classes/java/main/io/pivotal/pal/tracker/PalTrackerApplication.class
new file mode 100644
index 000000000..85fc292c3
Binary files /dev/null and b/build/classes/java/main/io/pivotal/pal/tracker/PalTrackerApplication.class differ
diff --git a/build/classes/java/main/io/pivotal/pal/tracker/WelcomeController.class b/build/classes/java/main/io/pivotal/pal/tracker/WelcomeController.class
new file mode 100644
index 000000000..c8c1555a8
Binary files /dev/null and b/build/classes/java/main/io/pivotal/pal/tracker/WelcomeController.class differ
diff --git a/build/libs/pal-tracker.jar b/build/libs/pal-tracker.jar
new file mode 100644
index 000000000..c55ae2396
Binary files /dev/null and b/build/libs/pal-tracker.jar differ
diff --git a/build/libs/pal-tracker.jar.original b/build/libs/pal-tracker.jar.original
new file mode 100644
index 000000000..80907a1e4
Binary files /dev/null and b/build/libs/pal-tracker.jar.original differ
diff --git a/build/tmp/jar/MANIFEST.MF b/build/tmp/jar/MANIFEST.MF
new file mode 100644
index 000000000..59499bce4
--- /dev/null
+++ b/build/tmp/jar/MANIFEST.MF
@@ -0,0 +1,2 @@
+Manifest-Version: 1.0
+
diff --git a/gradle/wrapper/gradle-wrapper.jar b/gradle/wrapper/gradle-wrapper.jar
new file mode 100644
index 000000000..01b8bf6b1
Binary files /dev/null and b/gradle/wrapper/gradle-wrapper.jar differ
diff --git a/gradle/wrapper/gradle-wrapper.properties b/gradle/wrapper/gradle-wrapper.properties
new file mode 100644
index 000000000..933b6473c
--- /dev/null
+++ b/gradle/wrapper/gradle-wrapper.properties
@@ -0,0 +1,5 @@
+distributionBase=GRADLE_USER_HOME
+distributionPath=wrapper/dists
+zipStoreBase=GRADLE_USER_HOME
+zipStorePath=wrapper/dists
+distributionUrl=https\://services.gradle.org/distributions/gradle-4.4-bin.zip
diff --git a/gradlew b/gradlew
new file mode 100755
index 000000000..cccdd3d51
--- /dev/null
+++ b/gradlew
@@ -0,0 +1,172 @@
+#!/usr/bin/env sh
+
+##############################################################################
+##
+## Gradle start up script for UN*X
+##
+##############################################################################
+
+# Attempt to set APP_HOME
+# Resolve links: $0 may be a link
+PRG="$0"
+# Need this for relative symlinks.
+while [ -h "$PRG" ] ; do
+ ls=`ls -ld "$PRG"`
+ link=`expr "$ls" : '.*-> \(.*\)$'`
+ if expr "$link" : '/.*' > /dev/null; then
+ PRG="$link"
+ else
+ PRG=`dirname "$PRG"`"/$link"
+ fi
+done
+SAVED="`pwd`"
+cd "`dirname \"$PRG\"`/" >/dev/null
+APP_HOME="`pwd -P`"
+cd "$SAVED" >/dev/null
+
+APP_NAME="Gradle"
+APP_BASE_NAME=`basename "$0"`
+
+# Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script.
+DEFAULT_JVM_OPTS=""
+
+# Use the maximum available, or set MAX_FD != -1 to use that value.
+MAX_FD="maximum"
+
+warn () {
+ echo "$*"
+}
+
+die () {
+ echo
+ echo "$*"
+ echo
+ exit 1
+}
+
+# OS specific support (must be 'true' or 'false').
+cygwin=false
+msys=false
+darwin=false
+nonstop=false
+case "`uname`" in
+ CYGWIN* )
+ cygwin=true
+ ;;
+ Darwin* )
+ darwin=true
+ ;;
+ MINGW* )
+ msys=true
+ ;;
+ NONSTOP* )
+ nonstop=true
+ ;;
+esac
+
+CLASSPATH=$APP_HOME/gradle/wrapper/gradle-wrapper.jar
+
+# Determine the Java command to use to start the JVM.
+if [ -n "$JAVA_HOME" ] ; then
+ if [ -x "$JAVA_HOME/jre/sh/java" ] ; then
+ # IBM's JDK on AIX uses strange locations for the executables
+ JAVACMD="$JAVA_HOME/jre/sh/java"
+ else
+ JAVACMD="$JAVA_HOME/bin/java"
+ fi
+ if [ ! -x "$JAVACMD" ] ; then
+ die "ERROR: JAVA_HOME is set to an invalid directory: $JAVA_HOME
+
+Please set the JAVA_HOME variable in your environment to match the
+location of your Java installation."
+ fi
+else
+ JAVACMD="java"
+ which java >/dev/null 2>&1 || die "ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH.
+
+Please set the JAVA_HOME variable in your environment to match the
+location of your Java installation."
+fi
+
+# Increase the maximum file descriptors if we can.
+if [ "$cygwin" = "false" -a "$darwin" = "false" -a "$nonstop" = "false" ] ; then
+ MAX_FD_LIMIT=`ulimit -H -n`
+ if [ $? -eq 0 ] ; then
+ if [ "$MAX_FD" = "maximum" -o "$MAX_FD" = "max" ] ; then
+ MAX_FD="$MAX_FD_LIMIT"
+ fi
+ ulimit -n $MAX_FD
+ if [ $? -ne 0 ] ; then
+ warn "Could not set maximum file descriptor limit: $MAX_FD"
+ fi
+ else
+ warn "Could not query maximum file descriptor limit: $MAX_FD_LIMIT"
+ fi
+fi
+
+# For Darwin, add options to specify how the application appears in the dock
+if $darwin; then
+ GRADLE_OPTS="$GRADLE_OPTS \"-Xdock:name=$APP_NAME\" \"-Xdock:icon=$APP_HOME/media/gradle.icns\""
+fi
+
+# For Cygwin, switch paths to Windows format before running java
+if $cygwin ; then
+ APP_HOME=`cygpath --path --mixed "$APP_HOME"`
+ CLASSPATH=`cygpath --path --mixed "$CLASSPATH"`
+ JAVACMD=`cygpath --unix "$JAVACMD"`
+
+ # We build the pattern for arguments to be converted via cygpath
+ ROOTDIRSRAW=`find -L / -maxdepth 1 -mindepth 1 -type d 2>/dev/null`
+ SEP=""
+ for dir in $ROOTDIRSRAW ; do
+ ROOTDIRS="$ROOTDIRS$SEP$dir"
+ SEP="|"
+ done
+ OURCYGPATTERN="(^($ROOTDIRS))"
+ # Add a user-defined pattern to the cygpath arguments
+ if [ "$GRADLE_CYGPATTERN" != "" ] ; then
+ OURCYGPATTERN="$OURCYGPATTERN|($GRADLE_CYGPATTERN)"
+ fi
+ # Now convert the arguments - kludge to limit ourselves to /bin/sh
+ i=0
+ for arg in "$@" ; do
+ CHECK=`echo "$arg"|egrep -c "$OURCYGPATTERN" -`
+ CHECK2=`echo "$arg"|egrep -c "^-"` ### Determine if an option
+
+ if [ $CHECK -ne 0 ] && [ $CHECK2 -eq 0 ] ; then ### Added a condition
+ eval `echo args$i`=`cygpath --path --ignore --mixed "$arg"`
+ else
+ eval `echo args$i`="\"$arg\""
+ fi
+ i=$((i+1))
+ done
+ case $i in
+ (0) set -- ;;
+ (1) set -- "$args0" ;;
+ (2) set -- "$args0" "$args1" ;;
+ (3) set -- "$args0" "$args1" "$args2" ;;
+ (4) set -- "$args0" "$args1" "$args2" "$args3" ;;
+ (5) set -- "$args0" "$args1" "$args2" "$args3" "$args4" ;;
+ (6) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" ;;
+ (7) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" ;;
+ (8) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" ;;
+ (9) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" "$args8" ;;
+ esac
+fi
+
+# Escape application args
+save () {
+ for i do printf %s\\n "$i" | sed "s/'/'\\\\''/g;1s/^/'/;\$s/\$/' \\\\/" ; done
+ echo " "
+}
+APP_ARGS=$(save "$@")
+
+# Collect all arguments for the java command, following the shell quoting and substitution rules
+eval set -- $DEFAULT_JVM_OPTS $JAVA_OPTS $GRADLE_OPTS "\"-Dorg.gradle.appname=$APP_BASE_NAME\"" -classpath "\"$CLASSPATH\"" org.gradle.wrapper.GradleWrapperMain "$APP_ARGS"
+
+# by default we should be in the correct project dir, but when run from Finder on Mac, the cwd is wrong
+if [ "$(uname)" = "Darwin" ] && [ "$HOME" = "$PWD" ]; then
+ cd "$(dirname "$0")"
+fi
+
+exec "$JAVACMD" "$@"
diff --git a/gradlew.bat b/gradlew.bat
new file mode 100644
index 000000000..e95643d6a
--- /dev/null
+++ b/gradlew.bat
@@ -0,0 +1,84 @@
+@if "%DEBUG%" == "" @echo off
+@rem ##########################################################################
+@rem
+@rem Gradle startup script for Windows
+@rem
+@rem ##########################################################################
+
+@rem Set local scope for the variables with windows NT shell
+if "%OS%"=="Windows_NT" setlocal
+
+set DIRNAME=%~dp0
+if "%DIRNAME%" == "" set DIRNAME=.
+set APP_BASE_NAME=%~n0
+set APP_HOME=%DIRNAME%
+
+@rem Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script.
+set DEFAULT_JVM_OPTS=
+
+@rem Find java.exe
+if defined JAVA_HOME goto findJavaFromJavaHome
+
+set JAVA_EXE=java.exe
+%JAVA_EXE% -version >NUL 2>&1
+if "%ERRORLEVEL%" == "0" goto init
+
+echo.
+echo ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH.
+echo.
+echo Please set the JAVA_HOME variable in your environment to match the
+echo location of your Java installation.
+
+goto fail
+
+:findJavaFromJavaHome
+set JAVA_HOME=%JAVA_HOME:"=%
+set JAVA_EXE=%JAVA_HOME%/bin/java.exe
+
+if exist "%JAVA_EXE%" goto init
+
+echo.
+echo ERROR: JAVA_HOME is set to an invalid directory: %JAVA_HOME%
+echo.
+echo Please set the JAVA_HOME variable in your environment to match the
+echo location of your Java installation.
+
+goto fail
+
+:init
+@rem Get command-line arguments, handling Windows variants
+
+if not "%OS%" == "Windows_NT" goto win9xME_args
+
+:win9xME_args
+@rem Slurp the command line arguments.
+set CMD_LINE_ARGS=
+set _SKIP=2
+
+:win9xME_args_slurp
+if "x%~1" == "x" goto execute
+
+set CMD_LINE_ARGS=%*
+
+:execute
+@rem Setup the command line
+
+set CLASSPATH=%APP_HOME%\gradle\wrapper\gradle-wrapper.jar
+
+@rem Execute Gradle
+"%JAVA_EXE%" %DEFAULT_JVM_OPTS% %JAVA_OPTS% %GRADLE_OPTS% "-Dorg.gradle.appname=%APP_BASE_NAME%" -classpath "%CLASSPATH%" org.gradle.wrapper.GradleWrapperMain %CMD_LINE_ARGS%
+
+:end
+@rem End local scope for the variables with windows NT shell
+if "%ERRORLEVEL%"=="0" goto mainEnd
+
+:fail
+rem Set variable GRADLE_EXIT_CONSOLE if you need the _script_ return code instead of
+rem the _cmd.exe /c_ return code!
+if not "" == "%GRADLE_EXIT_CONSOLE%" exit 1
+exit /b 1
+
+:mainEnd
+if "%OS%"=="Windows_NT" endlocal
+
+:omega
diff --git a/manifests.yml b/manifests.yml
new file mode 100644
index 000000000..7646099ee
--- /dev/null
+++ b/manifests.yml
@@ -0,0 +1,8 @@
+---
+applications:
+- name: pal-tracker
+ path: build/libs/pal-tracker.jar
+ routes:
+ - route: ps-pal-tracker-review.local.pcfdev.io
+ env:
+ WELCOME_MESSAGE: Hello from the review environment
\ No newline at end of file
diff --git a/pal-tracker-codebase.txt b/pal-tracker-codebase.txt
deleted file mode 100644
index 0943c3cd3..000000000
--- a/pal-tracker-codebase.txt
+++ /dev/null
@@ -1 +0,0 @@
-pal-tracker codebase
diff --git a/settings.gradle b/settings.gradle
new file mode 100644
index 000000000..ef961960e
--- /dev/null
+++ b/settings.gradle
@@ -0,0 +1 @@
+rootProject.name = "pal-tracker"
\ No newline at end of file
diff --git a/src/main/java/io/pivotal/pal/tracker/InMemoryTimeEntryRepository.java b/src/main/java/io/pivotal/pal/tracker/InMemoryTimeEntryRepository.java
new file mode 100644
index 000000000..e20907fc7
--- /dev/null
+++ b/src/main/java/io/pivotal/pal/tracker/InMemoryTimeEntryRepository.java
@@ -0,0 +1,62 @@
+package io.pivotal.pal.tracker;
+
+import java.util.ArrayList;
+import java.util.HashMap;
+import java.util.List;
+import java.util.Objects;
+
+public class InMemoryTimeEntryRepository implements TimeEntryRepository{
+ private HashMap timeEntries = new HashMap<>();
+
+ public InMemoryTimeEntryRepository() {
+ timeEntries = new HashMap<>();
+ }
+
+ public InMemoryTimeEntryRepository(HashMap timeEntries) {
+ this.timeEntries = timeEntries;
+ }
+
+ @Override
+ public TimeEntry create(TimeEntry timeEntry) {
+ Long id = timeEntries.size() + 1L;
+ TimeEntry newTimeEntry = new TimeEntry(
+ id,
+ timeEntry.getProjectId(),
+ timeEntry.getUserId(),
+ timeEntry.getDate(),
+ timeEntry.getHours()
+ );
+
+ timeEntries.put(id, newTimeEntry);
+ return newTimeEntry;
+ }
+
+ @Override
+ public TimeEntry find(Long id) {
+ return timeEntries.get(id);
+ }
+
+ @Override
+ public List list() {
+ return new ArrayList<>(timeEntries.values());
+ }
+
+ @Override
+ public TimeEntry update(Long id, TimeEntry timeEntry) {
+ TimeEntry updatedEntry = new TimeEntry(
+ id,
+ timeEntry.getProjectId(),
+ timeEntry.getUserId(),
+ timeEntry.getDate(),
+ timeEntry.getHours()
+ );
+
+ timeEntries.replace(id, updatedEntry);
+ return updatedEntry;
+ }
+
+ @Override
+ public void delete(Long id) {
+ timeEntries.remove(id);
+ }
+}
diff --git a/src/main/java/io/pivotal/pal/tracker/PalTrackerApplication.java b/src/main/java/io/pivotal/pal/tracker/PalTrackerApplication.java
new file mode 100644
index 000000000..56847d0b9
--- /dev/null
+++ b/src/main/java/io/pivotal/pal/tracker/PalTrackerApplication.java
@@ -0,0 +1,32 @@
+package io.pivotal.pal.tracker;
+
+import com.fasterxml.jackson.annotation.JsonInclude;
+import com.fasterxml.jackson.databind.ObjectMapper;
+import com.fasterxml.jackson.databind.SerializationFeature;
+import com.fasterxml.jackson.datatype.jsr310.JavaTimeModule;
+import org.springframework.boot.SpringApplication;
+import org.springframework.boot.autoconfigure.SpringBootApplication;
+import org.springframework.context.annotation.Bean;
+import org.springframework.http.converter.json.Jackson2ObjectMapperBuilder;
+
+@SpringBootApplication
+public class PalTrackerApplication {
+
+ @Bean
+ TimeEntryRepository timeEntryRepository() {
+ return new InMemoryTimeEntryRepository();
+ }
+
+ public static void main(String[] args) {
+ SpringApplication.run(PalTrackerApplication.class, args);
+ }
+
+ @Bean
+ public ObjectMapper jsonObjectMapper() {
+ return Jackson2ObjectMapperBuilder.json()
+ .serializationInclusion(JsonInclude.Include.NON_NULL) // Don’t include null values
+ .featuresToDisable(SerializationFeature.WRITE_DATES_AS_TIMESTAMPS) //ISODate
+ .modules(new JavaTimeModule())
+ .build();
+ }
+}
diff --git a/src/main/java/io/pivotal/pal/tracker/TimeEntry.java b/src/main/java/io/pivotal/pal/tracker/TimeEntry.java
new file mode 100644
index 000000000..128d328eb
--- /dev/null
+++ b/src/main/java/io/pivotal/pal/tracker/TimeEntry.java
@@ -0,0 +1,87 @@
+package io.pivotal.pal.tracker;
+
+import java.time.LocalDate;
+import java.util.Objects;
+
+public class TimeEntry {
+ private long id;
+ private long projectId;
+ private long userId;
+ private LocalDate date;
+ private int hours;
+
+ public TimeEntry() {
+ }
+
+ public TimeEntry(long projectId, long userId, LocalDate date, int hours) {
+ this.projectId = projectId;
+ this.userId = userId;
+ this.date = date;
+ this.hours = hours;
+ }
+
+ public TimeEntry(long id, long projectId, long userId, LocalDate date, int hours) {
+ this.id = id;
+ this.projectId = projectId;
+ this.userId = userId;
+ this.date = date;
+ this.hours = hours;
+ }
+
+ public long getId() {
+ return id;
+ }
+
+ public void setId(long id) {
+ this.id = id;
+ }
+
+ public long getProjectId() {
+ return projectId;
+ }
+
+ public void setProjectId(long projectId) {
+ this.projectId = projectId;
+ }
+
+ public long getUserId() {
+ return userId;
+ }
+
+ public void setUserId(long userId) {
+ this.userId = userId;
+ }
+
+ public LocalDate getDate() {
+ return date;
+ }
+
+ public void setDate(LocalDate date) {
+ this.date = date;
+ }
+
+ public int getHours() {
+ return hours;
+ }
+
+ public void setHours(int hours) {
+ this.hours = hours;
+ }
+
+ @Override
+ public boolean equals(Object o) {
+ if (this == o) return true;
+ if (o == null || getClass() != o.getClass()) return false;
+ TimeEntry timeEntry = (TimeEntry) o;
+ return id == timeEntry.id &&
+ projectId == timeEntry.projectId &&
+ userId == timeEntry.userId &&
+ hours == timeEntry.hours &&
+ Objects.equals(date, timeEntry.date);
+ }
+
+ @Override
+ public int hashCode() {
+ return Objects.hash(id, projectId, userId, date, hours);
+ }
+}
diff --git a/src/main/java/io/pivotal/pal/tracker/TimeEntryController.java b/src/main/java/io/pivotal/pal/tracker/TimeEntryController.java
new file mode 100644
index 000000000..a9b854f6c
--- /dev/null
+++ b/src/main/java/io/pivotal/pal/tracker/TimeEntryController.java
@@ -0,0 +1,57 @@
+package io.pivotal.pal.tracker;
+
+import org.springframework.http.HttpStatus;
+import org.springframework.http.ResponseEntity;
+import org.springframework.web.bind.annotation.*;
+
+import java.util.List;
+
+@RestController
+@RequestMapping("/time-entries")
+public class TimeEntryController {
+
+ private TimeEntryRepository timeEntriesRepo;
+
+ public TimeEntryController(TimeEntryRepository timeEntriesRepo) {
+ this.timeEntriesRepo = timeEntriesRepo;
+ }
+
+ @PostMapping
+ public ResponseEntity create(@RequestBody TimeEntry timeEntry) {
+ TimeEntry createdTimeEntry = timeEntriesRepo.create(timeEntry);
+
+ return new ResponseEntity<>(createdTimeEntry, HttpStatus.CREATED);
+ }
+
+ @GetMapping("{id}")
+ public ResponseEntity read(@PathVariable Long id) {
+ TimeEntry timeEntry = timeEntriesRepo.find(id);
+ if (timeEntry != null) {
+ return new ResponseEntity<>(timeEntry, HttpStatus.OK);
+ } else {
+ return new ResponseEntity<>(HttpStatus.NOT_FOUND);
+ }
+ }
+
+ @GetMapping
+ public ResponseEntity> list() {
+ return new ResponseEntity<>(timeEntriesRepo.list(), HttpStatus.OK);
+ }
+
+ @PutMapping("{id}")
+ public ResponseEntity update(@PathVariable Long id, @RequestBody TimeEntry timeEntry) {
+ TimeEntry updatedTimeEntry = timeEntriesRepo.update(id, timeEntry);
+ if (updatedTimeEntry != null) {
+ return new ResponseEntity<>(updatedTimeEntry, HttpStatus.OK);
+ } else {
+ return new ResponseEntity<>(HttpStatus.NOT_FOUND);
+ }
+ }
+
+ @DeleteMapping("{id}")
+ public ResponseEntity delete(@PathVariable Long id) {
+ timeEntriesRepo.delete(id);
+
+ return new ResponseEntity<>(HttpStatus.NO_CONTENT);
+ }
+}
diff --git a/src/main/java/io/pivotal/pal/tracker/TimeEntryRepository.java b/src/main/java/io/pivotal/pal/tracker/TimeEntryRepository.java
new file mode 100644
index 000000000..a3a649795
--- /dev/null
+++ b/src/main/java/io/pivotal/pal/tracker/TimeEntryRepository.java
@@ -0,0 +1,13 @@
+package io.pivotal.pal.tracker;
+
+import io.pivotal.pal.tracker.TimeEntry;
+
+import java.util.List;
+
+public interface TimeEntryRepository {
+ TimeEntry create(TimeEntry timeEntry);
+ TimeEntry find(Long id);
+ List list();
+ TimeEntry update(Long id, TimeEntry timeEntry);
+ void delete(Long id);
+}
diff --git a/src/main/java/io/pivotal/pal/tracker/WelcomeController.java b/src/main/java/io/pivotal/pal/tracker/WelcomeController.java
new file mode 100644
index 000000000..ba1a485c8
--- /dev/null
+++ b/src/main/java/io/pivotal/pal/tracker/WelcomeController.java
@@ -0,0 +1,20 @@
+package io.pivotal.pal.tracker;
+
+import org.springframework.beans.factory.annotation.Value;
+import org.springframework.web.bind.annotation.GetMapping;
+import org.springframework.web.bind.annotation.RestController;
+
+@RestController
+public class WelcomeController {
+
+ private String msg;
+
+ WelcomeController(@Value("${WELCOME_MESSAGE}") String msg){
+ this.msg = msg;
+ }
+
+ @GetMapping("/")
+ public String sayHello() {
+ return msg;
+ }
+}
diff --git a/src/test/java/test/pivotal/pal/tracker/InMemoryTimeEntryRepositoryTest.java b/src/test/java/test/pivotal/pal/tracker/InMemoryTimeEntryRepositoryTest.java
new file mode 100644
index 000000000..d0ae6cbe6
--- /dev/null
+++ b/src/test/java/test/pivotal/pal/tracker/InMemoryTimeEntryRepositoryTest.java
@@ -0,0 +1,71 @@
+package test.pivotal.pal.tracker;
+
+import io.pivotal.pal.tracker.InMemoryTimeEntryRepository;
+import io.pivotal.pal.tracker.TimeEntry;
+import org.junit.Test;
+
+import java.time.LocalDate;
+import java.util.List;
+
+import static java.util.Arrays.asList;
+import static org.assertj.core.api.Assertions.assertThat;
+
+public class InMemoryTimeEntryRepositoryTest {
+ @Test
+ public void create() throws Exception {
+ InMemoryTimeEntryRepository repo = new InMemoryTimeEntryRepository();
+ TimeEntry createdTimeEntry = repo.create(new TimeEntry(123L, 456L, LocalDate.parse("2017-01-08"), 8));
+
+ TimeEntry expected = new TimeEntry(1L, 123L, 456L, LocalDate.parse("2017-01-08"), 8);
+ assertThat(createdTimeEntry).isEqualTo(expected);
+
+ TimeEntry readEntry = repo.find(createdTimeEntry.getId());
+ assertThat(readEntry).isEqualTo(expected);
+ }
+
+ @Test
+ public void find() throws Exception {
+ InMemoryTimeEntryRepository repo = new InMemoryTimeEntryRepository();
+ repo.create(new TimeEntry(123L, 456L, LocalDate.parse("2017-01-08"), 8));
+
+ TimeEntry expected = new TimeEntry(1L, 123L, 456L, LocalDate.parse("2017-01-08"), 8);
+ TimeEntry readEntry = repo.find(1L);
+ assertThat(readEntry).isEqualTo(expected);
+ }
+
+ @Test
+ public void list() throws Exception {
+ InMemoryTimeEntryRepository repo = new InMemoryTimeEntryRepository();
+ repo.create(new TimeEntry(123L, 456L, LocalDate.parse("2017-01-08"), 8));
+ repo.create(new TimeEntry(789L, 654L, LocalDate.parse("2017-01-07"), 4));
+
+ List expected = asList(
+ new TimeEntry(1L, 123L, 456L, LocalDate.parse("2017-01-08"), 8),
+ new TimeEntry(2L, 789L, 654L, LocalDate.parse("2017-01-07"), 4)
+ );
+ assertThat(repo.list()).isEqualTo(expected);
+ }
+
+ @Test
+ public void update() throws Exception {
+ InMemoryTimeEntryRepository repo = new InMemoryTimeEntryRepository();
+ TimeEntry created = repo.create(new TimeEntry(123L, 456L, LocalDate.parse("2017-01-08"), 8));
+
+ TimeEntry updatedEntry = repo.update(
+ created.getId(),
+ new TimeEntry(321L, 654L, LocalDate.parse("2017-01-09"), 5));
+
+ TimeEntry expected = new TimeEntry(created.getId(), 321L, 654L, LocalDate.parse("2017-01-09"), 5);
+ assertThat(updatedEntry).isEqualTo(expected);
+ assertThat(repo.find(created.getId())).isEqualTo(expected);
+ }
+
+ @Test
+ public void delete() throws Exception {
+ InMemoryTimeEntryRepository repo = new InMemoryTimeEntryRepository();
+ TimeEntry created = repo.create(new TimeEntry(123L, 456L, LocalDate.parse("2017-01-08"), 8));
+
+ repo.delete(created.getId());
+ assertThat(repo.list()).isEmpty();
+ }
+}
diff --git a/src/test/java/test/pivotal/pal/tracker/TimeEntryControllerTest.java b/src/test/java/test/pivotal/pal/tracker/TimeEntryControllerTest.java
new file mode 100644
index 000000000..1e2bd6fad
--- /dev/null
+++ b/src/test/java/test/pivotal/pal/tracker/TimeEntryControllerTest.java
@@ -0,0 +1,116 @@
+package test.pivotal.pal.tracker;
+
+import io.pivotal.pal.tracker.TimeEntry;
+import io.pivotal.pal.tracker.TimeEntryController;
+import io.pivotal.pal.tracker.TimeEntryRepository;
+import org.junit.Before;
+import org.junit.Test;
+import org.springframework.http.HttpStatus;
+import org.springframework.http.ResponseEntity;
+
+import java.time.LocalDate;
+import java.util.List;
+
+import static java.util.Arrays.asList;
+import static org.assertj.core.api.Assertions.assertThat;
+import static org.mockito.Matchers.any;
+import static org.mockito.Matchers.eq;
+import static org.mockito.Mockito.*;
+
+public class TimeEntryControllerTest {
+ private TimeEntryRepository timeEntryRepository;
+ private TimeEntryController controller;
+
+ @Before
+ public void setUp() throws Exception {
+ timeEntryRepository = mock(TimeEntryRepository.class);
+ controller = new TimeEntryController(timeEntryRepository);
+ }
+
+ @Test
+ public void testCreate() throws Exception {
+ TimeEntry timeEntryToCreate = new TimeEntry(1L, 1L, LocalDate.parse("2017-01-08"), 8);
+ TimeEntry expectedResult = new TimeEntry(1L, 1L, LocalDate.parse("2017-01-08"), 8);
+ doReturn(expectedResult)
+ .when(timeEntryRepository)
+ .create(any(TimeEntry.class));
+
+
+ ResponseEntity response = controller.create(timeEntryToCreate);
+
+
+ verify(timeEntryRepository).create(timeEntryToCreate);
+ assertThat(response.getStatusCode()).isEqualTo(HttpStatus.CREATED);
+ assertThat(response.getBody()).isEqualTo(expectedResult);
+ }
+
+ @Test
+ public void testRead() throws Exception {
+ TimeEntry expected = new TimeEntry(1L, 1L, LocalDate.parse("2017-01-08"), 8);
+ doReturn(expected)
+ .when(timeEntryRepository)
+ .find(1L);
+
+ ResponseEntity response = controller.read(1L);
+
+ verify(timeEntryRepository).find(1L);
+ assertThat(response.getStatusCode()).isEqualTo(HttpStatus.OK);
+ assertThat(response.getBody()).isEqualTo(expected);
+ }
+
+ @Test
+ public void testRead_NotFound() throws Exception {
+ doReturn(null)
+ .when(timeEntryRepository)
+ .find(1L);
+
+ ResponseEntity response = controller.read(1L);
+ assertThat(response.getStatusCode()).isEqualTo(HttpStatus.NOT_FOUND);
+ }
+
+ @Test
+ public void testList() throws Exception {
+ List expected = asList(
+ new TimeEntry(1L, 1L, LocalDate.parse("2017-01-08"), 8),
+ new TimeEntry(2L, 2L, LocalDate.parse("2017-01-07"), 4)
+ );
+ doReturn(expected).when(timeEntryRepository).list();
+
+ ResponseEntity> response = controller.list();
+
+ verify(timeEntryRepository).list();
+ assertThat(response.getStatusCode()).isEqualTo(HttpStatus.OK);
+ assertThat(response.getBody()).isEqualTo(expected);
+ }
+
+ @Test
+ public void testUpdate() throws Exception {
+ TimeEntry expected = new TimeEntry(1L, 1L, LocalDate.parse("2017-01-07"), 4);
+ doReturn(expected)
+ .when(timeEntryRepository)
+ .update(eq(1L), any(TimeEntry.class));
+
+ ResponseEntity response = controller.update(1L, expected);
+
+ verify(timeEntryRepository).update(1L, expected);
+ assertThat(response.getStatusCode()).isEqualTo(HttpStatus.OK);
+ assertThat(response.getBody()).isEqualTo(expected);
+ }
+
+ @Test
+ public void testUpdate_NotFound() throws Exception {
+ doReturn(null)
+ .when(timeEntryRepository)
+ .update(eq(1L), any(TimeEntry.class));
+
+ ResponseEntity response = controller.update(1L, new TimeEntry());
+ assertThat(response.getStatusCode()).isEqualTo(HttpStatus.NOT_FOUND);
+ }
+
+ @Test
+ public void testDelete() throws Exception {
+ ResponseEntity response = controller.delete(1L);
+ verify(timeEntryRepository).delete(1L);
+ assertThat(response.getStatusCode()).isEqualTo(HttpStatus.NO_CONTENT);
+ }
+}
diff --git a/src/test/java/test/pivotal/pal/trackerapi/TimeEntryApiTest.java b/src/test/java/test/pivotal/pal/trackerapi/TimeEntryApiTest.java
new file mode 100644
index 000000000..91e271b45
--- /dev/null
+++ b/src/test/java/test/pivotal/pal/trackerapi/TimeEntryApiTest.java
@@ -0,0 +1,126 @@
+package test.pivotal.pal.trackerapi;
+
+import com.jayway.jsonpath.DocumentContext;
+import io.pivotal.pal.tracker.PalTrackerApplication;
+import io.pivotal.pal.tracker.TimeEntry;
+import org.junit.Test;
+import org.junit.runner.RunWith;
+import org.springframework.beans.factory.annotation.Autowired;
+import org.springframework.boot.test.context.SpringBootTest;
+import org.springframework.boot.test.web.client.TestRestTemplate;
+import org.springframework.http.HttpEntity;
+import org.springframework.http.HttpMethod;
+import org.springframework.http.HttpStatus;
+import org.springframework.http.ResponseEntity;
+import org.springframework.test.context.junit4.SpringRunner;
+
+import java.time.LocalDate;
+import java.util.Collection;
+
+import static com.jayway.jsonpath.JsonPath.parse;
+import static org.assertj.core.api.Assertions.assertThat;
+import static org.springframework.boot.test.context.SpringBootTest.WebEnvironment.RANDOM_PORT;
+
+@RunWith(SpringRunner.class)
+@SpringBootTest(classes = PalTrackerApplication.class, webEnvironment = RANDOM_PORT)
+public class TimeEntryApiTest {
+
+ @Autowired
+ private TestRestTemplate restTemplate;
+
+ private TimeEntry timeEntry = new TimeEntry(123L, 456L, LocalDate.parse("2017-01-08"), 8);
+
+ @Test
+ public void testCreate() throws Exception {
+ ResponseEntity createResponse = restTemplate.postForEntity("/time-entries", timeEntry, String.class);
+
+
+ assertThat(createResponse.getStatusCode()).isEqualTo(HttpStatus.CREATED);
+
+ DocumentContext createJson = parse(createResponse.getBody());
+ assertThat(createJson.read("$.id", Long.class)).isGreaterThan(0);
+ assertThat(createJson.read("$.projectId", Long.class)).isEqualTo(123L);
+ assertThat(createJson.read("$.userId", Long.class)).isEqualTo(456L);
+ assertThat(createJson.read("$.date", String.class)).isEqualTo("2017-01-08");
+ assertThat(createJson.read("$.hours", Long.class)).isEqualTo(8);
+ }
+
+ @Test
+ public void testList() throws Exception {
+ Long id = createTimeEntry();
+
+
+ ResponseEntity listResponse = restTemplate.getForEntity("/time-entries", String.class);
+
+
+ assertThat(listResponse.getStatusCode()).isEqualTo(HttpStatus.OK);
+
+ DocumentContext listJson = parse(listResponse.getBody());
+
+ Collection timeEntries = listJson.read("$[*]", Collection.class);
+ assertThat(timeEntries.size()).isEqualTo(1);
+
+ Long readId = listJson.read("$[0].id", Long.class);
+ assertThat(readId).isEqualTo(id);
+ }
+
+ @Test
+ public void testRead() throws Exception {
+ Long id = createTimeEntry();
+
+
+ ResponseEntity readResponse = this.restTemplate.getForEntity("/time-entries/" + id, String.class);
+
+
+ assertThat(readResponse.getStatusCode()).isEqualTo(HttpStatus.OK);
+ DocumentContext readJson = parse(readResponse.getBody());
+ assertThat(readJson.read("$.id", Long.class)).isEqualTo(id);
+ assertThat(readJson.read("$.projectId", Long.class)).isEqualTo(123L);
+ assertThat(readJson.read("$.userId", Long.class)).isEqualTo(456L);
+ assertThat(readJson.read("$.date", String.class)).isEqualTo("2017-01-08");
+ assertThat(readJson.read("$.hours", Long.class)).isEqualTo(8);
+ }
+
+ @Test
+ public void testUpdate() throws Exception {
+ Long id = createTimeEntry();
+ TimeEntry updatedTimeEntry = new TimeEntry(2L, 3L, LocalDate.parse("2017-01-09"), 9);
+
+
+ ResponseEntity updateResponse = restTemplate.exchange("/time-entries/" + id, HttpMethod.PUT, new HttpEntity<>(updatedTimeEntry, null), String.class);
+
+
+ assertThat(updateResponse.getStatusCode()).isEqualTo(HttpStatus.OK);
+
+ DocumentContext updateJson = parse(updateResponse.getBody());
+ assertThat(updateJson.read("$.id", Long.class)).isEqualTo(id);
+ assertThat(updateJson.read("$.projectId", Long.class)).isEqualTo(2L);
+ assertThat(updateJson.read("$.userId", Long.class)).isEqualTo(3L);
+ assertThat(updateJson.read("$.date", String.class)).isEqualTo("2017-01-09");
+ assertThat(updateJson.read("$.hours", Long.class)).isEqualTo(9);
+ }
+
+ @Test
+ public void testDelete() throws Exception {
+ Long id = createTimeEntry();
+
+
+ ResponseEntity deleteResponse = restTemplate.exchange("/time-entries/" + id, HttpMethod.DELETE, null, String.class);
+
+
+ assertThat(deleteResponse.getStatusCode()).isEqualTo(HttpStatus.NO_CONTENT);
+
+ ResponseEntity deletedReadResponse = this.restTemplate.getForEntity("/time-entries/" + id, String.class);
+ assertThat(deletedReadResponse.getStatusCode()).isEqualTo(HttpStatus.NOT_FOUND);
+ }
+
+ private Long createTimeEntry() {
+ HttpEntity entity = new HttpEntity<>(timeEntry);
+
+ ResponseEntity response = restTemplate.exchange("/time-entries", HttpMethod.POST, entity, TimeEntry.class);
+
+ assertThat(response.getStatusCode()).isEqualTo(HttpStatus.CREATED);
+
+ return response.getBody().getId();
+ }
+}