Compare commits
1 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| 32337ec308 |
@@ -1,6 +1,24 @@
|
|||||||
# Gradle
|
# Java
|
||||||
build/
|
.mtj.tmp/
|
||||||
.gradle
|
*.class
|
||||||
|
*.jar
|
||||||
|
*.war
|
||||||
|
*.ear
|
||||||
|
*.nar
|
||||||
|
hs_err_pid*
|
||||||
|
|
||||||
|
# Maven
|
||||||
|
target/
|
||||||
|
pom.xml.tag
|
||||||
|
pom.xml.releaseBackup
|
||||||
|
pom.xml.versionsBackup
|
||||||
|
pom.xml.next
|
||||||
|
pom.xml.bak
|
||||||
|
release.properties
|
||||||
|
dependency-reduced-pom.xml
|
||||||
|
buildNumber.properties
|
||||||
|
.mvn/timing.properties
|
||||||
|
.mvn/wrapper/maven-wrapper.jar
|
||||||
|
|
||||||
# IntelliJ
|
# IntelliJ
|
||||||
out/
|
out/
|
||||||
@@ -10,8 +28,11 @@ out/
|
|||||||
*.ipr
|
*.ipr
|
||||||
*.iws
|
*.iws
|
||||||
|
|
||||||
|
# Vue
|
||||||
|
node_modules
|
||||||
|
/dist
|
||||||
|
|
||||||
# Local env files
|
# Local env files
|
||||||
.env
|
|
||||||
.env.local
|
.env.local
|
||||||
.env.*.local
|
.env.*.local
|
||||||
|
|
||||||
@@ -28,13 +49,85 @@ pids
|
|||||||
*.seed
|
*.seed
|
||||||
*.pid.lock
|
*.pid.lock
|
||||||
|
|
||||||
|
# Directory for instrumented libs generated by jscoverage/JSCover
|
||||||
|
lib-cov
|
||||||
|
|
||||||
|
# Coverage directory used by tools like istanbul
|
||||||
|
coverage
|
||||||
|
|
||||||
|
# nyc test coverage
|
||||||
|
.nyc_output
|
||||||
|
|
||||||
|
# Grunt intermediate storage (http://gruntjs.com/creating-plugins#storing-task-files)
|
||||||
|
.grunt
|
||||||
|
|
||||||
|
# Bower dependency directory (https://bower.io/)
|
||||||
|
bower_components
|
||||||
|
|
||||||
|
# node-waf configuration
|
||||||
|
.lock-wscript
|
||||||
|
|
||||||
|
# Compiled binary addons (https://nodejs.org/api/addons.html)
|
||||||
|
build/Release
|
||||||
|
|
||||||
# Dependency directories
|
# Dependency directories
|
||||||
node_modules/
|
node_modules/
|
||||||
jspm_packages/
|
jspm_packages/
|
||||||
|
|
||||||
|
# TypeScript v1 declaration files
|
||||||
|
typings/
|
||||||
|
|
||||||
|
# Optional npm cache directory
|
||||||
|
.npm
|
||||||
|
|
||||||
|
# Optional eslint cache
|
||||||
|
.eslintcache
|
||||||
|
|
||||||
|
# Optional REPL history
|
||||||
|
.node_repl_history
|
||||||
|
|
||||||
|
# Output of 'npm pack'
|
||||||
|
*.tgz
|
||||||
|
|
||||||
# Yarn Integrity file
|
# Yarn Integrity file
|
||||||
.yarn-integrity
|
.yarn-integrity
|
||||||
|
|
||||||
|
# dotenv environment variables file
|
||||||
|
.env
|
||||||
|
|
||||||
|
# parcel-bundler cache (https://parceljs.org/)
|
||||||
|
.cache
|
||||||
|
|
||||||
|
# next.js build output
|
||||||
|
.next
|
||||||
|
|
||||||
|
# nuxt.js build output
|
||||||
|
.nuxt
|
||||||
|
|
||||||
|
# Nuxt generate
|
||||||
|
dist
|
||||||
|
|
||||||
|
# vuepress build output
|
||||||
|
.vuepress/dist
|
||||||
|
|
||||||
|
# Serverless directories
|
||||||
|
.serverless
|
||||||
|
|
||||||
|
# IDE / Editor
|
||||||
|
.idea
|
||||||
|
|
||||||
|
# Service worker
|
||||||
|
sw.*
|
||||||
|
*.private.env.json
|
||||||
|
|
||||||
|
# Certificates
|
||||||
|
data/
|
||||||
|
letsencrypt/
|
||||||
|
|
||||||
|
# generated resources
|
||||||
|
app/src/main/resources/css-manifest.json
|
||||||
|
app/src/main/resources/static/styles*
|
||||||
|
|
||||||
# h2 db
|
# h2 db
|
||||||
*.db
|
*.db
|
||||||
|
|
||||||
@@ -43,4 +136,3 @@ jspm_packages/
|
|||||||
|
|
||||||
# python
|
# python
|
||||||
__pycache__
|
__pycache__
|
||||||
|
|
||||||
|
|||||||
@@ -1,3 +1,25 @@
|
|||||||
|
FROM maven:3.6.3-jdk-14 as builder
|
||||||
|
|
||||||
|
WORKDIR /tmp
|
||||||
|
|
||||||
|
# Cache dependencies
|
||||||
|
COPY pom.xml .
|
||||||
|
COPY app/pom.xml app/pom.xml
|
||||||
|
COPY domain/pom.xml domain/pom.xml
|
||||||
|
COPY persistance/pom.xml persistance/pom.xml
|
||||||
|
COPY shared/pom.xml shared/pom.xml
|
||||||
|
COPY search/pom.xml search/pom.xml
|
||||||
|
|
||||||
|
RUN mvn verify clean --fail-never
|
||||||
|
|
||||||
|
COPY app/src app/src
|
||||||
|
COPY domain/src domain/src
|
||||||
|
COPY persistance/src persistance/src
|
||||||
|
COPY shared/src shared/src
|
||||||
|
COPY search/src search/src
|
||||||
|
|
||||||
|
RUN mvn -Dstyle.color=always package
|
||||||
|
|
||||||
FROM openjdk:14-alpine as jdkbuilder
|
FROM openjdk:14-alpine as jdkbuilder
|
||||||
|
|
||||||
RUN apk add --no-cache binutils
|
RUN apk add --no-cache binutils
|
||||||
@@ -20,21 +42,8 @@ RUN chown -R $APPLICATION_USER /app
|
|||||||
|
|
||||||
USER $APPLICATION_USER
|
USER $APPLICATION_USER
|
||||||
|
|
||||||
|
COPY --from=builder /tmp/app/target/app-*.jar /app/app.jar
|
||||||
COPY --from=jdkbuilder /myjdk /myjdk
|
COPY --from=jdkbuilder /myjdk /myjdk
|
||||||
COPY simplenotes-app/build/libs/app-*-all.jar /app/simplenotes.jar
|
|
||||||
WORKDIR /app
|
WORKDIR /app
|
||||||
|
|
||||||
CMD [ \
|
CMD ["/myjdk/bin/java", "-server", "-XX:+UnlockExperimentalVMOptions", "-Xms64m", "-Xmx256m", "-XX:+UseG1GC", "-XX:MaxGCPauseMillis=100", "-XX:+UseStringDeduplication", "-jar", "app.jar"]
|
||||||
"/myjdk/bin/java", \
|
|
||||||
"--add-opens", \
|
|
||||||
"java.base/java.nio=ALL-UNNAMED", \
|
|
||||||
"-server", \
|
|
||||||
"-XX:+UnlockExperimentalVMOptions", \
|
|
||||||
"-Xms64m", \
|
|
||||||
"-Xmx256m", \
|
|
||||||
"-XX:+UseG1GC", \
|
|
||||||
"-XX:MaxGCPauseMillis=100", \
|
|
||||||
"-XX:+UseStringDeduplication", \
|
|
||||||
"-jar", \
|
|
||||||
"simplenotes.jar" \
|
|
||||||
]
|
|
||||||
|
|||||||
@@ -0,0 +1,191 @@
|
|||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
|
||||||
|
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
|
||||||
|
<parent>
|
||||||
|
<artifactId>parent</artifactId>
|
||||||
|
<groupId>be.simplenotes</groupId>
|
||||||
|
<version>1.0-SNAPSHOT</version>
|
||||||
|
</parent>
|
||||||
|
<modelVersion>4.0.0</modelVersion>
|
||||||
|
|
||||||
|
<artifactId>app</artifactId>
|
||||||
|
|
||||||
|
<properties>
|
||||||
|
<http4k.version>3.268.0</http4k.version>
|
||||||
|
</properties>
|
||||||
|
|
||||||
|
<dependencies>
|
||||||
|
<dependency>
|
||||||
|
<groupId>be.simplenotes</groupId>
|
||||||
|
<artifactId>persistance</artifactId>
|
||||||
|
<version>1.0-SNAPSHOT</version>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>be.simplenotes</groupId>
|
||||||
|
<artifactId>search</artifactId>
|
||||||
|
<version>1.0-SNAPSHOT</version>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>be.simplenotes</groupId>
|
||||||
|
<artifactId>domain</artifactId>
|
||||||
|
<version>1.0-SNAPSHOT</version>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>be.simplenotes</groupId>
|
||||||
|
<artifactId>shared</artifactId>
|
||||||
|
<version>1.0-SNAPSHOT</version>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>org.http4k</groupId>
|
||||||
|
<artifactId>http4k-core</artifactId>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>org.http4k</groupId>
|
||||||
|
<artifactId>http4k-server-jetty</artifactId>
|
||||||
|
<exclusions>
|
||||||
|
<exclusion>
|
||||||
|
<groupId>org.eclipse.jetty.websocket</groupId>
|
||||||
|
<artifactId>javax-websocket-server-impl</artifactId>
|
||||||
|
</exclusion>
|
||||||
|
</exclusions>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>org.jetbrains.kotlinx</groupId>
|
||||||
|
<artifactId>kotlinx-html-jvm</artifactId>
|
||||||
|
<version>0.7.1</version>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>org.jetbrains.kotlinx</groupId>
|
||||||
|
<artifactId>kotlinx-serialization-json-jvm</artifactId>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>org.ocpsoft.prettytime</groupId>
|
||||||
|
<artifactId>prettytime</artifactId>
|
||||||
|
<version>4.0.5.Final</version>
|
||||||
|
</dependency>
|
||||||
|
|
||||||
|
<dependency>
|
||||||
|
<groupId>ch.qos.logback</groupId>
|
||||||
|
<artifactId>logback-classic</artifactId>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>org.assertj</groupId>
|
||||||
|
<artifactId>assertj-core</artifactId>
|
||||||
|
<scope>test</scope>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>org.junit.jupiter</groupId>
|
||||||
|
<artifactId>junit-jupiter</artifactId>
|
||||||
|
<scope>test</scope>
|
||||||
|
</dependency>
|
||||||
|
|
||||||
|
<dependency>
|
||||||
|
<groupId>be.simplenotes</groupId>
|
||||||
|
<artifactId>shared</artifactId>
|
||||||
|
<version>1.0-SNAPSHOT</version>
|
||||||
|
<type>test-jar</type>
|
||||||
|
<scope>test</scope>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>org.http4k</groupId>
|
||||||
|
<artifactId>http4k-testing-hamkrest</artifactId>
|
||||||
|
<scope>test</scope>
|
||||||
|
</dependency>
|
||||||
|
|
||||||
|
<dependency>
|
||||||
|
<groupId>me.liuwj.ktorm</groupId>
|
||||||
|
<artifactId>ktorm-core</artifactId>
|
||||||
|
</dependency>
|
||||||
|
</dependencies>
|
||||||
|
|
||||||
|
<dependencyManagement>
|
||||||
|
<dependencies>
|
||||||
|
<dependency>
|
||||||
|
<groupId>org.http4k</groupId>
|
||||||
|
<artifactId>http4k-bom</artifactId>
|
||||||
|
<version>${http4k.version}</version>
|
||||||
|
<type>pom</type>
|
||||||
|
<scope>import</scope>
|
||||||
|
</dependency>
|
||||||
|
</dependencies>
|
||||||
|
</dependencyManagement>
|
||||||
|
|
||||||
|
<build>
|
||||||
|
<plugins>
|
||||||
|
<plugin>
|
||||||
|
<groupId>org.apache.maven.plugins</groupId>
|
||||||
|
<artifactId>maven-shade-plugin</artifactId>
|
||||||
|
<executions>
|
||||||
|
<execution>
|
||||||
|
<phase>package</phase>
|
||||||
|
<goals>
|
||||||
|
<goal>shade</goal>
|
||||||
|
</goals>
|
||||||
|
<configuration>
|
||||||
|
<minimizeJar>true</minimizeJar>
|
||||||
|
<transformers>
|
||||||
|
<transformer
|
||||||
|
implementation="org.apache.maven.plugins.shade.resource.ManifestResourceTransformer">
|
||||||
|
<mainClass>be.simplenotes.app.SimpleNotesKt</mainClass>
|
||||||
|
</transformer>
|
||||||
|
</transformers>
|
||||||
|
<filters>
|
||||||
|
<filter>
|
||||||
|
<artifact>com.h2database:h2</artifact>
|
||||||
|
<includes>
|
||||||
|
<include>**</include>
|
||||||
|
</includes>
|
||||||
|
</filter>
|
||||||
|
<filter>
|
||||||
|
<artifact>org.mariadb.jdbc:mariadb-java-client</artifact>
|
||||||
|
<includes>
|
||||||
|
<include>**</include>
|
||||||
|
</includes>
|
||||||
|
</filter>
|
||||||
|
<filter>
|
||||||
|
<artifact>org.jetbrains.kotlin:kotlin-reflect</artifact>
|
||||||
|
<includes>
|
||||||
|
<include>**</include>
|
||||||
|
</includes>
|
||||||
|
</filter>
|
||||||
|
<filter>
|
||||||
|
<artifact>org.eclipse.jetty:*</artifact>
|
||||||
|
<includes>
|
||||||
|
<include>**</include>
|
||||||
|
</includes>
|
||||||
|
</filter>
|
||||||
|
<filter>
|
||||||
|
<artifact>org.apache.lucene:*</artifact>
|
||||||
|
<includes>
|
||||||
|
<include>**</include>
|
||||||
|
</includes>
|
||||||
|
</filter>
|
||||||
|
<filter>
|
||||||
|
<artifact>org.ocpsoft.prettytime:prettytime</artifact>
|
||||||
|
<includes>
|
||||||
|
<include>**</include>
|
||||||
|
</includes>
|
||||||
|
</filter>
|
||||||
|
<filter>
|
||||||
|
<artifact>*:*</artifact>
|
||||||
|
<excludes>
|
||||||
|
<exclude>META-INF/maven/**</exclude>
|
||||||
|
<exclude>META-INF/proguard/**</exclude>
|
||||||
|
<exclude>META-INF/*.kotlin_module</exclude>
|
||||||
|
<exclude>META-INF/DEPENDENCIES*</exclude>
|
||||||
|
<exclude>META-INF/NOTICE*</exclude>
|
||||||
|
<exclude>META-INF/LICENSE*</exclude>
|
||||||
|
<exclude>LICENSE*</exclude>
|
||||||
|
<exclude>META-INF/README*</exclude>
|
||||||
|
<exclude>META-INF/native-image/**</exclude>
|
||||||
|
</excludes>
|
||||||
|
</filter>
|
||||||
|
</filters>
|
||||||
|
</configuration>
|
||||||
|
</execution>
|
||||||
|
</executions>
|
||||||
|
</plugin>
|
||||||
|
</plugins>
|
||||||
|
</build>
|
||||||
|
|
||||||
|
</project>
|
||||||
@@ -1,9 +1,12 @@
|
|||||||
package be.simplenotes.config
|
package be.simplenotes.app
|
||||||
|
|
||||||
|
import be.simplenotes.shared.config.DataSourceConfig
|
||||||
|
import be.simplenotes.shared.config.JwtConfig
|
||||||
|
import be.simplenotes.shared.config.ServerConfig
|
||||||
import java.util.*
|
import java.util.*
|
||||||
import java.util.concurrent.TimeUnit
|
import java.util.concurrent.TimeUnit
|
||||||
|
|
||||||
class ConfigLoader {
|
class Config {
|
||||||
//region Config loading
|
//region Config loading
|
||||||
private val properties: Properties = javaClass
|
private val properties: Properties = javaClass
|
||||||
.getResource("/application.properties")
|
.getResource("/application.properties")
|
||||||
@@ -2,7 +2,7 @@ package be.simplenotes.app
|
|||||||
|
|
||||||
import org.http4k.server.Http4kServer
|
import org.http4k.server.Http4kServer
|
||||||
import org.slf4j.LoggerFactory
|
import org.slf4j.LoggerFactory
|
||||||
import be.simplenotes.config.ServerConfig as SimpleNotesServerConfig
|
import be.simplenotes.shared.config.ServerConfig as SimpleNotesServerConfig
|
||||||
|
|
||||||
class Server(
|
class Server(
|
||||||
private val config: SimpleNotesServerConfig,
|
private val config: SimpleNotesServerConfig,
|
||||||
@@ -2,12 +2,10 @@ package be.simplenotes.app
|
|||||||
|
|
||||||
import be.simplenotes.app.extensions.addShutdownHook
|
import be.simplenotes.app.extensions.addShutdownHook
|
||||||
import be.simplenotes.app.modules.*
|
import be.simplenotes.app.modules.*
|
||||||
import be.simplenotes.config.configModule
|
|
||||||
import be.simplenotes.domain.domainModule
|
import be.simplenotes.domain.domainModule
|
||||||
import be.simplenotes.persistance.migrationModule
|
import be.simplenotes.persistance.migrationModule
|
||||||
import be.simplenotes.persistance.persistanceModule
|
import be.simplenotes.persistance.persistanceModule
|
||||||
import be.simplenotes.search.searchModule
|
import be.simplenotes.search.searchModule
|
||||||
import be.simplenotes.views.viewModule
|
|
||||||
import org.koin.core.context.startKoin
|
import org.koin.core.context.startKoin
|
||||||
import org.koin.core.context.unloadKoinModules
|
import org.koin.core.context.unloadKoinModules
|
||||||
|
|
||||||
@@ -18,8 +16,10 @@ fun main() {
|
|||||||
persistanceModule,
|
persistanceModule,
|
||||||
migrationModule,
|
migrationModule,
|
||||||
configModule,
|
configModule,
|
||||||
viewModule,
|
baseModule,
|
||||||
controllerModule,
|
userModule,
|
||||||
|
noteModule,
|
||||||
|
settingsModule,
|
||||||
domainModule,
|
domainModule,
|
||||||
searchModule,
|
searchModule,
|
||||||
apiModule,
|
apiModule,
|
||||||
@@ -0,0 +1,77 @@
|
|||||||
|
package be.simplenotes.app.api
|
||||||
|
|
||||||
|
import be.simplenotes.app.extensions.json
|
||||||
|
import be.simplenotes.app.utils.parseSearchTerms
|
||||||
|
import be.simplenotes.domain.model.PersistedNote
|
||||||
|
import be.simplenotes.domain.model.PersistedNoteMetadata
|
||||||
|
import be.simplenotes.domain.security.JwtPayload
|
||||||
|
import be.simplenotes.domain.usecases.NoteService
|
||||||
|
import kotlinx.serialization.Contextual
|
||||||
|
import kotlinx.serialization.Serializable
|
||||||
|
import kotlinx.serialization.builtins.ListSerializer
|
||||||
|
import kotlinx.serialization.json.Json
|
||||||
|
import org.http4k.core.Request
|
||||||
|
import org.http4k.core.Response
|
||||||
|
import org.http4k.core.Status.Companion.BAD_REQUEST
|
||||||
|
import org.http4k.core.Status.Companion.NOT_FOUND
|
||||||
|
import org.http4k.core.Status.Companion.OK
|
||||||
|
import org.http4k.routing.path
|
||||||
|
import java.util.*
|
||||||
|
|
||||||
|
class ApiNoteController(private val noteService: NoteService, private val json: Json) {
|
||||||
|
|
||||||
|
fun createNote(request: Request, jwtPayload: JwtPayload): Response {
|
||||||
|
val content = json.decodeFromString(NoteContent.serializer(), request.bodyString()).content
|
||||||
|
return noteService.create(jwtPayload.userId, content).fold(
|
||||||
|
{
|
||||||
|
Response(BAD_REQUEST)
|
||||||
|
},
|
||||||
|
{
|
||||||
|
Response(OK).json(json.encodeToString(UuidContent.serializer(), UuidContent(it.uuid)))
|
||||||
|
}
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
fun notes(request: Request, jwtPayload: JwtPayload): Response {
|
||||||
|
val notes = noteService.paginatedNotes(jwtPayload.userId, page = 1).notes
|
||||||
|
val json = json.encodeToString(ListSerializer(PersistedNoteMetadata.serializer()), notes)
|
||||||
|
return Response(OK).json(json)
|
||||||
|
}
|
||||||
|
|
||||||
|
fun note(request: Request, jwtPayload: JwtPayload): Response {
|
||||||
|
val uuid = request.path("uuid")!!
|
||||||
|
|
||||||
|
return noteService.find(jwtPayload.userId, UUID.fromString(uuid))
|
||||||
|
?.let { Response(OK).json(json.encodeToString(PersistedNote.serializer(), it)) }
|
||||||
|
?: Response(NOT_FOUND)
|
||||||
|
}
|
||||||
|
|
||||||
|
fun update(request: Request, jwtPayload: JwtPayload): Response {
|
||||||
|
val uuid = UUID.fromString(request.path("uuid")!!)
|
||||||
|
val content = json.decodeFromString(NoteContent.serializer(), request.bodyString()).content
|
||||||
|
return noteService.update(jwtPayload.userId, uuid, content).fold({
|
||||||
|
Response(BAD_REQUEST)
|
||||||
|
}, {
|
||||||
|
if (it == null) Response(NOT_FOUND)
|
||||||
|
else Response(OK)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
fun search(request: Request, jwtPayload: JwtPayload): Response {
|
||||||
|
val query = json.decodeFromString(SearchContent.serializer(), request.bodyString()).query
|
||||||
|
val terms = parseSearchTerms(query)
|
||||||
|
val notes = noteService.search(jwtPayload.userId, terms)
|
||||||
|
val json = json.encodeToString(ListSerializer(PersistedNoteMetadata.serializer()), notes)
|
||||||
|
return Response(OK).json(json)
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
@Serializable
|
||||||
|
data class NoteContent(val content: String)
|
||||||
|
|
||||||
|
@Serializable
|
||||||
|
data class UuidContent(@Contextual val uuid: UUID)
|
||||||
|
|
||||||
|
@Serializable
|
||||||
|
data class SearchContent(@Contextual val query: String)
|
||||||
@@ -0,0 +1,26 @@
|
|||||||
|
package be.simplenotes.app.api
|
||||||
|
|
||||||
|
import be.simplenotes.app.extensions.json
|
||||||
|
import be.simplenotes.domain.usecases.UserService
|
||||||
|
import be.simplenotes.domain.usecases.users.login.LoginForm
|
||||||
|
import kotlinx.serialization.Serializable
|
||||||
|
import kotlinx.serialization.json.Json
|
||||||
|
import org.http4k.core.Request
|
||||||
|
import org.http4k.core.Response
|
||||||
|
import org.http4k.core.Status
|
||||||
|
|
||||||
|
class ApiUserController(private val userService: UserService, private val json: Json) {
|
||||||
|
|
||||||
|
fun login(request: Request): Response {
|
||||||
|
val form = json.decodeFromString(LoginForm.serializer(), request.bodyString())
|
||||||
|
val result = userService.login(form)
|
||||||
|
return result.fold({
|
||||||
|
Response(Status.BAD_REQUEST)
|
||||||
|
}, {
|
||||||
|
Response(Status.OK).json(json.encodeToString(Token.serializer(), Token(it)))
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Serializable
|
||||||
|
data class Token(val token: String)
|
||||||
@@ -1,13 +1,13 @@
|
|||||||
package be.simplenotes.app.controllers
|
package be.simplenotes.app.controllers
|
||||||
|
|
||||||
import be.simplenotes.app.extensions.html
|
import be.simplenotes.app.extensions.html
|
||||||
import be.simplenotes.types.LoggedInUser
|
import be.simplenotes.app.views.BaseView
|
||||||
import be.simplenotes.views.BaseView
|
import be.simplenotes.domain.security.JwtPayload
|
||||||
import org.http4k.core.Request
|
import org.http4k.core.Request
|
||||||
import org.http4k.core.Response
|
import org.http4k.core.Response
|
||||||
import org.http4k.core.Status.Companion.OK
|
import org.http4k.core.Status.Companion.OK
|
||||||
|
|
||||||
class BaseController(private val view: BaseView) {
|
class BaseController(private val view: BaseView) {
|
||||||
fun index(@Suppress("UNUSED_PARAMETER") request: Request, loggedInUser: LoggedInUser?) =
|
fun index(@Suppress("UNUSED_PARAMETER") request: Request, jwtPayload: JwtPayload?) =
|
||||||
Response(OK).html(view.renderHome(loggedInUser))
|
Response(OK).html(view.renderHome(jwtPayload))
|
||||||
}
|
}
|
||||||
@@ -7,6 +7,6 @@ import org.http4k.core.Status.Companion.OK
|
|||||||
import org.http4k.core.Status.Companion.SERVICE_UNAVAILABLE
|
import org.http4k.core.Status.Companion.SERVICE_UNAVAILABLE
|
||||||
|
|
||||||
class HealthCheckController(private val dbHealthCheck: DbHealthCheck) {
|
class HealthCheckController(private val dbHealthCheck: DbHealthCheck) {
|
||||||
fun healthCheck(@Suppress("UNUSED_PARAMETER") request: Request) =
|
fun healthCheck(request: Request) =
|
||||||
if (dbHealthCheck.isOk()) Response(OK) else Response(SERVICE_UNAVAILABLE)
|
if (dbHealthCheck.isOk()) Response(OK) else Response(SERVICE_UNAVAILABLE)
|
||||||
}
|
}
|
||||||
@@ -3,12 +3,12 @@ package be.simplenotes.app.controllers
|
|||||||
import be.simplenotes.app.extensions.html
|
import be.simplenotes.app.extensions.html
|
||||||
import be.simplenotes.app.extensions.redirect
|
import be.simplenotes.app.extensions.redirect
|
||||||
import be.simplenotes.app.utils.parseSearchTerms
|
import be.simplenotes.app.utils.parseSearchTerms
|
||||||
|
import be.simplenotes.app.views.NoteView
|
||||||
|
import be.simplenotes.domain.security.JwtPayload
|
||||||
import be.simplenotes.domain.usecases.NoteService
|
import be.simplenotes.domain.usecases.NoteService
|
||||||
import be.simplenotes.domain.usecases.markdown.InvalidMeta
|
import be.simplenotes.domain.usecases.markdown.InvalidMeta
|
||||||
import be.simplenotes.domain.usecases.markdown.MissingMeta
|
import be.simplenotes.domain.usecases.markdown.MissingMeta
|
||||||
import be.simplenotes.domain.usecases.markdown.ValidationError
|
import be.simplenotes.domain.usecases.markdown.ValidationError
|
||||||
import be.simplenotes.types.LoggedInUser
|
|
||||||
import be.simplenotes.views.NoteView
|
|
||||||
import org.http4k.core.Method
|
import org.http4k.core.Method
|
||||||
import org.http4k.core.Request
|
import org.http4k.core.Request
|
||||||
import org.http4k.core.Response
|
import org.http4k.core.Response
|
||||||
@@ -25,26 +25,18 @@ class NoteController(
|
|||||||
private val noteService: NoteService,
|
private val noteService: NoteService,
|
||||||
) {
|
) {
|
||||||
|
|
||||||
fun new(request: Request, loggedInUser: LoggedInUser): Response {
|
fun new(request: Request, jwtPayload: JwtPayload): Response {
|
||||||
if (request.method == Method.GET) return Response(OK).html(view.noteEditor(loggedInUser))
|
if (request.method == Method.GET) return Response(OK).html(view.noteEditor(jwtPayload))
|
||||||
|
|
||||||
val markdownForm = request.form("markdown") ?: ""
|
val markdownForm = request.form("markdown") ?: ""
|
||||||
|
|
||||||
return noteService.create(loggedInUser.userId, markdownForm).fold(
|
return noteService.create(jwtPayload.userId, markdownForm).fold(
|
||||||
{
|
{
|
||||||
val html = when (it) {
|
val html = when (it) {
|
||||||
MissingMeta -> view.noteEditor(
|
MissingMeta -> view.noteEditor(jwtPayload, error = "Missing note metadata", textarea = markdownForm)
|
||||||
loggedInUser,
|
InvalidMeta -> view.noteEditor(jwtPayload, error = "Invalid note metadata", textarea = markdownForm)
|
||||||
error = "Missing note metadata",
|
|
||||||
textarea = markdownForm
|
|
||||||
)
|
|
||||||
InvalidMeta -> view.noteEditor(
|
|
||||||
loggedInUser,
|
|
||||||
error = "Invalid note metadata",
|
|
||||||
textarea = markdownForm
|
|
||||||
)
|
|
||||||
is ValidationError -> view.noteEditor(
|
is ValidationError -> view.noteEditor(
|
||||||
loggedInUser,
|
jwtPayload,
|
||||||
validationErrors = it.validationErrors,
|
validationErrors = it.validationErrors,
|
||||||
textarea = markdownForm
|
textarea = markdownForm
|
||||||
)
|
)
|
||||||
@@ -57,74 +49,66 @@ class NoteController(
|
|||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
fun list(request: Request, loggedInUser: LoggedInUser): Response {
|
fun list(request: Request, jwtPayload: JwtPayload): Response {
|
||||||
val currentPage = request.query("page")?.toIntOrNull()?.let(::abs) ?: 1
|
val currentPage = request.query("page")?.toIntOrNull()?.let(::abs) ?: 1
|
||||||
val tag = request.query("tag")
|
val tag = request.query("tag")
|
||||||
val (pages, notes) = noteService.paginatedNotes(loggedInUser.userId, currentPage, tag = tag)
|
val (pages, notes) = noteService.paginatedNotes(jwtPayload.userId, currentPage, tag = tag)
|
||||||
val deletedCount = noteService.countDeleted(loggedInUser.userId)
|
val deletedCount = noteService.countDeleted(jwtPayload.userId)
|
||||||
return Response(OK).html(view.notes(loggedInUser, notes, currentPage, pages, deletedCount, tag = tag))
|
return Response(OK).html(view.notes(jwtPayload, notes, currentPage, pages, deletedCount, tag = tag))
|
||||||
}
|
}
|
||||||
|
|
||||||
fun search(request: Request, loggedInUser: LoggedInUser): Response {
|
fun search(request: Request, jwtPayload: JwtPayload): Response {
|
||||||
val query = request.form("search") ?: ""
|
val query = request.form("search") ?: ""
|
||||||
val terms = parseSearchTerms(query)
|
val terms = parseSearchTerms(query)
|
||||||
val notes = noteService.search(loggedInUser.userId, terms)
|
val notes = noteService.search(jwtPayload.userId, terms)
|
||||||
val deletedCount = noteService.countDeleted(loggedInUser.userId)
|
val deletedCount = noteService.countDeleted(jwtPayload.userId)
|
||||||
return Response(OK).html(view.search(loggedInUser, notes, query, deletedCount))
|
return Response(OK).html(view.search(jwtPayload, notes, query, deletedCount))
|
||||||
}
|
}
|
||||||
|
|
||||||
fun note(request: Request, loggedInUser: LoggedInUser): Response {
|
fun note(request: Request, jwtPayload: JwtPayload): Response {
|
||||||
val noteUuid = request.uuidPath() ?: return Response(NOT_FOUND)
|
val noteUuid = request.uuidPath() ?: return Response(NOT_FOUND)
|
||||||
|
|
||||||
if (request.method == Method.POST) {
|
if (request.method == Method.POST) {
|
||||||
if (request.form("delete") != null) {
|
if (request.form("delete") != null) {
|
||||||
return if (noteService.trash(loggedInUser.userId, noteUuid))
|
return if (noteService.trash(jwtPayload.userId, noteUuid))
|
||||||
Response.redirect("/notes") // TODO: flash cookie to show success ?
|
Response.redirect("/notes") // TODO: flash cookie to show success ?
|
||||||
else
|
else
|
||||||
Response(NOT_FOUND) // TODO: show an error
|
Response(NOT_FOUND) // TODO: show an error
|
||||||
}
|
}
|
||||||
if (request.form("public") != null) {
|
if (request.form("public") != null) {
|
||||||
if (!noteService.makePublic(loggedInUser.userId, noteUuid)) return Response(NOT_FOUND)
|
if (!noteService.makePublic(jwtPayload.userId, noteUuid)) return Response(NOT_FOUND)
|
||||||
} else if (request.form("private") != null) {
|
} else if (request.form("private") != null) {
|
||||||
if (!noteService.makePrivate(loggedInUser.userId, noteUuid)) return Response(NOT_FOUND)
|
if (!noteService.makePrivate(jwtPayload.userId, noteUuid)) return Response(NOT_FOUND)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
val note = noteService.find(loggedInUser.userId, noteUuid) ?: return Response(NOT_FOUND)
|
val note = noteService.find(jwtPayload.userId, noteUuid) ?: return Response(NOT_FOUND)
|
||||||
return Response(OK).html(view.renderedNote(loggedInUser, note, shared = false))
|
return Response(OK).html(view.renderedNote(jwtPayload, note, shared = false))
|
||||||
}
|
}
|
||||||
|
|
||||||
fun public(request: Request, loggedInUser: LoggedInUser?): Response {
|
fun public(request: Request, jwtPayload: JwtPayload?): Response {
|
||||||
val noteUuid = request.uuidPath() ?: return Response(NOT_FOUND)
|
val noteUuid = request.uuidPath() ?: return Response(NOT_FOUND)
|
||||||
val note = noteService.findPublic(noteUuid) ?: return Response(NOT_FOUND)
|
val note = noteService.findPublic(noteUuid) ?: return Response(NOT_FOUND)
|
||||||
return Response(OK).html(view.renderedNote(loggedInUser, note, shared = true))
|
return Response(OK).html(view.renderedNote(jwtPayload, note, shared = true))
|
||||||
}
|
}
|
||||||
|
|
||||||
fun edit(request: Request, loggedInUser: LoggedInUser): Response {
|
fun edit(request: Request, jwtPayload: JwtPayload): Response {
|
||||||
val noteUuid = request.uuidPath() ?: return Response(NOT_FOUND)
|
val noteUuid = request.uuidPath() ?: return Response(NOT_FOUND)
|
||||||
val note = noteService.find(loggedInUser.userId, noteUuid) ?: return Response(NOT_FOUND)
|
val note = noteService.find(jwtPayload.userId, noteUuid) ?: return Response(NOT_FOUND)
|
||||||
|
|
||||||
if (request.method == Method.GET) {
|
if (request.method == Method.GET) {
|
||||||
return Response(OK).html(view.noteEditor(loggedInUser, textarea = note.markdown))
|
return Response(OK).html(view.noteEditor(jwtPayload, textarea = note.markdown))
|
||||||
}
|
}
|
||||||
|
|
||||||
val markdownForm = request.form("markdown") ?: ""
|
val markdownForm = request.form("markdown") ?: ""
|
||||||
|
|
||||||
return noteService.update(loggedInUser.userId, note.uuid, markdownForm).fold(
|
return noteService.update(jwtPayload.userId, note.uuid, markdownForm).fold(
|
||||||
{
|
{
|
||||||
val html = when (it) {
|
val html = when (it) {
|
||||||
MissingMeta -> view.noteEditor(
|
MissingMeta -> view.noteEditor(jwtPayload, error = "Missing note metadata", textarea = markdownForm)
|
||||||
loggedInUser,
|
InvalidMeta -> view.noteEditor(jwtPayload, error = "Invalid note metadata", textarea = markdownForm)
|
||||||
error = "Missing note metadata",
|
|
||||||
textarea = markdownForm
|
|
||||||
)
|
|
||||||
InvalidMeta -> view.noteEditor(
|
|
||||||
loggedInUser,
|
|
||||||
error = "Invalid note metadata",
|
|
||||||
textarea = markdownForm
|
|
||||||
)
|
|
||||||
is ValidationError -> view.noteEditor(
|
is ValidationError -> view.noteEditor(
|
||||||
loggedInUser,
|
jwtPayload,
|
||||||
validationErrors = it.validationErrors,
|
validationErrors = it.validationErrors,
|
||||||
textarea = markdownForm
|
textarea = markdownForm
|
||||||
)
|
)
|
||||||
@@ -137,21 +121,21 @@ class NoteController(
|
|||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
fun trash(request: Request, loggedInUser: LoggedInUser): Response {
|
fun trash(request: Request, jwtPayload: JwtPayload): Response {
|
||||||
val currentPage = request.query("page")?.toIntOrNull()?.let(::abs) ?: 1
|
val currentPage = request.query("page")?.toIntOrNull()?.let(::abs) ?: 1
|
||||||
val tag = request.query("tag")
|
val tag = request.query("tag")
|
||||||
val (pages, notes) = noteService.paginatedNotes(loggedInUser.userId, currentPage, tag = tag, deleted = true)
|
val (pages, notes) = noteService.paginatedNotes(jwtPayload.userId, currentPage, tag = tag, deleted = true)
|
||||||
return Response(OK).html(view.trash(loggedInUser, notes, currentPage, pages))
|
return Response(OK).html(view.trash(jwtPayload, notes, currentPage, pages))
|
||||||
}
|
}
|
||||||
|
|
||||||
fun deleted(request: Request, loggedInUser: LoggedInUser): Response {
|
fun deleted(request: Request, jwtPayload: JwtPayload): Response {
|
||||||
val uuid = request.uuidPath() ?: return Response(NOT_FOUND)
|
val uuid = request.uuidPath() ?: return Response(NOT_FOUND)
|
||||||
return if (request.form("delete") != null)
|
return if (request.form("delete") != null)
|
||||||
if (noteService.delete(loggedInUser.userId, uuid))
|
if (noteService.delete(jwtPayload.userId, uuid))
|
||||||
Response.redirect("/notes/trash")
|
Response.redirect("/notes/trash")
|
||||||
else
|
else
|
||||||
Response(NOT_FOUND)
|
Response(NOT_FOUND)
|
||||||
else if (noteService.restore(loggedInUser.userId, uuid))
|
else if (noteService.restore(jwtPayload.userId, uuid))
|
||||||
Response.redirect("/notes/$uuid")
|
Response.redirect("/notes/$uuid")
|
||||||
else
|
else
|
||||||
Response(NOT_FOUND)
|
Response(NOT_FOUND)
|
||||||
@@ -2,11 +2,11 @@ package be.simplenotes.app.controllers
|
|||||||
|
|
||||||
import be.simplenotes.app.extensions.html
|
import be.simplenotes.app.extensions.html
|
||||||
import be.simplenotes.app.extensions.redirect
|
import be.simplenotes.app.extensions.redirect
|
||||||
|
import be.simplenotes.app.views.SettingView
|
||||||
|
import be.simplenotes.domain.security.JwtPayload
|
||||||
import be.simplenotes.domain.usecases.UserService
|
import be.simplenotes.domain.usecases.UserService
|
||||||
import be.simplenotes.domain.usecases.users.delete.DeleteError
|
import be.simplenotes.domain.usecases.users.delete.DeleteError
|
||||||
import be.simplenotes.domain.usecases.users.delete.DeleteForm
|
import be.simplenotes.domain.usecases.users.delete.DeleteForm
|
||||||
import be.simplenotes.types.LoggedInUser
|
|
||||||
import be.simplenotes.views.SettingView
|
|
||||||
import org.http4k.core.*
|
import org.http4k.core.*
|
||||||
import org.http4k.core.body.form
|
import org.http4k.core.body.form
|
||||||
import org.http4k.core.cookie.invalidateCookie
|
import org.http4k.core.cookie.invalidateCookie
|
||||||
@@ -15,11 +15,11 @@ class SettingsController(
|
|||||||
private val userService: UserService,
|
private val userService: UserService,
|
||||||
private val settingView: SettingView,
|
private val settingView: SettingView,
|
||||||
) {
|
) {
|
||||||
fun settings(request: Request, loggedInUser: LoggedInUser): Response {
|
fun settings(request: Request, jwtPayload: JwtPayload): Response {
|
||||||
if (request.method == Method.GET)
|
if (request.method == Method.GET)
|
||||||
return Response(Status.OK).html(settingView.settings(loggedInUser))
|
return Response(Status.OK).html(settingView.settings(jwtPayload))
|
||||||
|
|
||||||
val deleteForm = request.deleteForm(loggedInUser)
|
val deleteForm = request.deleteForm(jwtPayload)
|
||||||
val result = userService.delete(deleteForm)
|
val result = userService.delete(deleteForm)
|
||||||
|
|
||||||
return result.fold(
|
return result.fold(
|
||||||
@@ -28,13 +28,13 @@ class SettingsController(
|
|||||||
DeleteError.Unregistered -> Response.redirect("/").invalidateCookie("Bearer")
|
DeleteError.Unregistered -> Response.redirect("/").invalidateCookie("Bearer")
|
||||||
DeleteError.WrongPassword -> Response(Status.OK).html(
|
DeleteError.WrongPassword -> Response(Status.OK).html(
|
||||||
settingView.settings(
|
settingView.settings(
|
||||||
loggedInUser,
|
jwtPayload,
|
||||||
error = "Wrong password"
|
error = "Wrong password"
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
is DeleteError.InvalidForm -> Response(Status.OK).html(
|
is DeleteError.InvalidForm -> Response(Status.OK).html(
|
||||||
settingView.settings(
|
settingView.settings(
|
||||||
loggedInUser,
|
jwtPayload,
|
||||||
validationErrors = it.validationErrors
|
validationErrors = it.validationErrors
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
@@ -53,26 +53,23 @@ class SettingsController(
|
|||||||
.header("Content-Type", contentType)
|
.header("Content-Type", contentType)
|
||||||
}
|
}
|
||||||
|
|
||||||
fun export(request: Request, loggedInUser: LoggedInUser): Response {
|
fun export(request: Request, jwtPayload: JwtPayload): Response {
|
||||||
val isDownload = request.form("download") != null
|
val isDownload = request.form("download") != null
|
||||||
|
|
||||||
return if (isDownload) {
|
return if (isDownload) {
|
||||||
val filename = "simplenotes-export-${loggedInUser.username}"
|
val filename = "simplenotes-export-${jwtPayload.username}"
|
||||||
if (request.form("format") == "zip") {
|
if (request.form("format") == "zip") {
|
||||||
val zip = userService.exportAsZip(loggedInUser.userId)
|
val zip = userService.exportAsZip(jwtPayload.userId)
|
||||||
Response(Status.OK)
|
Response(Status.OK)
|
||||||
.with(attachment("$filename.zip", "application/zip"))
|
.with(attachment("$filename.zip", "application/zip"))
|
||||||
.body(zip)
|
.body(zip)
|
||||||
} else
|
} else
|
||||||
Response(Status.OK)
|
Response(Status.OK)
|
||||||
.with(attachment("$filename.json", "application/json"))
|
.with(attachment("$filename.json", "application/json"))
|
||||||
.body(userService.exportAsJson(loggedInUser.userId))
|
.body(userService.exportAsJson(jwtPayload.userId))
|
||||||
} else Response(Status.OK).body(userService.exportAsJson(loggedInUser.userId)).header(
|
} else Response(Status.OK).body(userService.exportAsJson(jwtPayload.userId)).header("Content-Type", "application/json")
|
||||||
"Content-Type",
|
|
||||||
"application/json"
|
|
||||||
)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private fun Request.deleteForm(loggedInUser: LoggedInUser) =
|
private fun Request.deleteForm(jwtPayload: JwtPayload) =
|
||||||
DeleteForm(loggedInUser.username, form("password"), form("checked") != null)
|
DeleteForm(jwtPayload.username, form("password"), form("checked") != null)
|
||||||
}
|
}
|
||||||
@@ -3,14 +3,14 @@ package be.simplenotes.app.controllers
|
|||||||
import be.simplenotes.app.extensions.html
|
import be.simplenotes.app.extensions.html
|
||||||
import be.simplenotes.app.extensions.isSecure
|
import be.simplenotes.app.extensions.isSecure
|
||||||
import be.simplenotes.app.extensions.redirect
|
import be.simplenotes.app.extensions.redirect
|
||||||
import be.simplenotes.config.JwtConfig
|
import be.simplenotes.app.views.UserView
|
||||||
|
import be.simplenotes.domain.security.JwtPayload
|
||||||
import be.simplenotes.domain.usecases.UserService
|
import be.simplenotes.domain.usecases.UserService
|
||||||
import be.simplenotes.domain.usecases.users.login.*
|
import be.simplenotes.domain.usecases.users.login.*
|
||||||
import be.simplenotes.domain.usecases.users.register.InvalidRegisterForm
|
import be.simplenotes.domain.usecases.users.register.InvalidRegisterForm
|
||||||
import be.simplenotes.domain.usecases.users.register.RegisterForm
|
import be.simplenotes.domain.usecases.users.register.RegisterForm
|
||||||
import be.simplenotes.domain.usecases.users.register.UserExists
|
import be.simplenotes.domain.usecases.users.register.UserExists
|
||||||
import be.simplenotes.types.LoggedInUser
|
import be.simplenotes.shared.config.JwtConfig
|
||||||
import be.simplenotes.views.UserView
|
|
||||||
import org.http4k.core.Method.GET
|
import org.http4k.core.Method.GET
|
||||||
import org.http4k.core.Request
|
import org.http4k.core.Request
|
||||||
import org.http4k.core.Response
|
import org.http4k.core.Response
|
||||||
@@ -27,9 +27,9 @@ class UserController(
|
|||||||
private val userView: UserView,
|
private val userView: UserView,
|
||||||
private val jwtConfig: JwtConfig,
|
private val jwtConfig: JwtConfig,
|
||||||
) {
|
) {
|
||||||
fun register(request: Request, loggedInUser: LoggedInUser?): Response {
|
fun register(request: Request, jwtPayload: JwtPayload?): Response {
|
||||||
if (request.method == GET) return Response(OK).html(
|
if (request.method == GET) return Response(OK).html(
|
||||||
userView.register(loggedInUser)
|
userView.register(jwtPayload)
|
||||||
)
|
)
|
||||||
|
|
||||||
val result = userService.register(request.registerForm())
|
val result = userService.register(request.registerForm())
|
||||||
@@ -38,12 +38,12 @@ class UserController(
|
|||||||
{
|
{
|
||||||
val html = when (it) {
|
val html = when (it) {
|
||||||
UserExists -> userView.register(
|
UserExists -> userView.register(
|
||||||
loggedInUser,
|
jwtPayload,
|
||||||
error = "User already exists"
|
error = "User already exists"
|
||||||
)
|
)
|
||||||
is InvalidRegisterForm ->
|
is InvalidRegisterForm ->
|
||||||
userView.register(
|
userView.register(
|
||||||
loggedInUser,
|
jwtPayload,
|
||||||
validationErrors = it.validationErrors
|
validationErrors = it.validationErrors
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
@@ -58,9 +58,9 @@ class UserController(
|
|||||||
private fun Request.registerForm() = RegisterForm(form("username"), form("password"))
|
private fun Request.registerForm() = RegisterForm(form("username"), form("password"))
|
||||||
private fun Request.loginForm(): LoginForm = registerForm()
|
private fun Request.loginForm(): LoginForm = registerForm()
|
||||||
|
|
||||||
fun login(request: Request, loggedInUser: LoggedInUser?): Response {
|
fun login(request: Request, jwtPayload: JwtPayload?): Response {
|
||||||
if (request.method == GET) return Response(OK).html(
|
if (request.method == GET) return Response(OK).html(
|
||||||
userView.login(loggedInUser)
|
userView.login(jwtPayload)
|
||||||
)
|
)
|
||||||
|
|
||||||
val result = userService.login(request.loginForm())
|
val result = userService.login(request.loginForm())
|
||||||
@@ -70,17 +70,17 @@ class UserController(
|
|||||||
val html = when (it) {
|
val html = when (it) {
|
||||||
Unregistered ->
|
Unregistered ->
|
||||||
userView.login(
|
userView.login(
|
||||||
loggedInUser,
|
jwtPayload,
|
||||||
error = "User does not exist"
|
error = "User does not exist"
|
||||||
)
|
)
|
||||||
WrongPassword ->
|
WrongPassword ->
|
||||||
userView.login(
|
userView.login(
|
||||||
loggedInUser,
|
jwtPayload,
|
||||||
error = "Wrong password"
|
error = "Wrong password"
|
||||||
)
|
)
|
||||||
is InvalidLoginForm ->
|
is InvalidLoginForm ->
|
||||||
userView.login(
|
userView.login(
|
||||||
loggedInUser,
|
jwtPayload,
|
||||||
validationErrors = it.validationErrors
|
validationErrors = it.validationErrors
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
@@ -0,0 +1,17 @@
|
|||||||
|
package be.simplenotes.app.extensions
|
||||||
|
|
||||||
|
import org.http4k.core.Request
|
||||||
|
import org.http4k.core.Response
|
||||||
|
import org.http4k.core.Status.Companion.FOUND
|
||||||
|
import org.http4k.core.Status.Companion.MOVED_PERMANENTLY
|
||||||
|
|
||||||
|
fun Response.html(html: String) = body(html)
|
||||||
|
.header("Content-Type", "text/html; charset=utf-8")
|
||||||
|
.header("Cache-Control", "no-cache")
|
||||||
|
|
||||||
|
fun Response.json(json: String) = body(json).header("Content-Type", "application/json")
|
||||||
|
|
||||||
|
fun Response.Companion.redirect(url: String, permanent: Boolean = false) =
|
||||||
|
Response(if (permanent) MOVED_PERMANENTLY else FOUND).header("Location", url)
|
||||||
|
|
||||||
|
fun Request.isSecure() = header("X-Forwarded-Proto")?.contains("https") ?: false
|
||||||
@@ -0,0 +1,15 @@
|
|||||||
|
package be.simplenotes.app.extensions
|
||||||
|
|
||||||
|
import kotlinx.html.*
|
||||||
|
|
||||||
|
class SUMMARY(consumer: TagConsumer<*>) :
|
||||||
|
HTMLTag(
|
||||||
|
"summary", consumer, emptyMap(),
|
||||||
|
inlineTag = true,
|
||||||
|
emptyTag = false
|
||||||
|
),
|
||||||
|
HtmlInlineTag
|
||||||
|
|
||||||
|
fun DETAILS.summary(block: SUMMARY.() -> Unit = {}) {
|
||||||
|
SUMMARY(consumer).visit(block)
|
||||||
|
}
|
||||||
@@ -1,8 +1,8 @@
|
|||||||
package be.simplenotes.app.filters
|
package be.simplenotes.app.filters
|
||||||
|
|
||||||
import be.simplenotes.app.extensions.redirect
|
import be.simplenotes.app.extensions.redirect
|
||||||
|
import be.simplenotes.domain.security.JwtPayload
|
||||||
import be.simplenotes.domain.security.JwtPayloadExtractor
|
import be.simplenotes.domain.security.JwtPayloadExtractor
|
||||||
import be.simplenotes.types.LoggedInUser
|
|
||||||
import org.http4k.core.*
|
import org.http4k.core.*
|
||||||
import org.http4k.core.Status.Companion.UNAUTHORIZED
|
import org.http4k.core.Status.Companion.UNAUTHORIZED
|
||||||
import org.http4k.core.cookie.cookie
|
import org.http4k.core.cookie.cookie
|
||||||
@@ -25,7 +25,7 @@ class AuthFilter(
|
|||||||
JwtSource.Header -> it.bearerTokenHeader()
|
JwtSource.Header -> it.bearerTokenHeader()
|
||||||
JwtSource.Cookie -> it.bearerTokenCookie()
|
JwtSource.Cookie -> it.bearerTokenCookie()
|
||||||
}
|
}
|
||||||
val jwtPayload = token?.let { extractor(token) }
|
val jwtPayload = token?.let { token -> extractor(token) }
|
||||||
when {
|
when {
|
||||||
jwtPayload != null -> {
|
jwtPayload != null -> {
|
||||||
ctx[it][authKey] = jwtPayload
|
ctx[it][authKey] = jwtPayload
|
||||||
@@ -40,7 +40,7 @@ class AuthFilter(
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fun Request.jwtPayload(ctx: RequestContexts): LoggedInUser? = ctx[this][authKey]
|
fun Request.jwtPayload(ctx: RequestContexts): JwtPayload? = ctx[this][authKey]
|
||||||
|
|
||||||
enum class JwtSource {
|
enum class JwtSource {
|
||||||
Header, Cookie
|
Header, Cookie
|
||||||
@@ -1,8 +1,8 @@
|
|||||||
package be.simplenotes.app.filters
|
package be.simplenotes.app.filters
|
||||||
|
|
||||||
import be.simplenotes.app.extensions.html
|
import be.simplenotes.app.extensions.html
|
||||||
import be.simplenotes.views.ErrorView
|
import be.simplenotes.app.views.ErrorView
|
||||||
import be.simplenotes.views.ErrorView.Type.*
|
import be.simplenotes.app.views.ErrorView.Type.*
|
||||||
import org.http4k.core.*
|
import org.http4k.core.*
|
||||||
import org.http4k.core.Status.Companion.INTERNAL_SERVER_ERROR
|
import org.http4k.core.Status.Companion.INTERNAL_SERVER_ERROR
|
||||||
import org.http4k.core.Status.Companion.NOT_FOUND
|
import org.http4k.core.Status.Companion.NOT_FOUND
|
||||||
@@ -0,0 +1,11 @@
|
|||||||
|
package be.simplenotes.app.modules
|
||||||
|
|
||||||
|
import be.simplenotes.app.Config
|
||||||
|
import org.koin.dsl.module
|
||||||
|
|
||||||
|
val configModule = module {
|
||||||
|
single { Config() }
|
||||||
|
single { get<Config>().dataSourceConfig }
|
||||||
|
single { get<Config>().jwtConfig }
|
||||||
|
single { get<Config>().serverConfig }
|
||||||
|
}
|
||||||
@@ -0,0 +1,29 @@
|
|||||||
|
package be.simplenotes.app.modules
|
||||||
|
|
||||||
|
import be.simplenotes.app.controllers.*
|
||||||
|
import be.simplenotes.app.views.BaseView
|
||||||
|
import be.simplenotes.app.views.NoteView
|
||||||
|
import be.simplenotes.app.views.SettingView
|
||||||
|
import be.simplenotes.app.views.UserView
|
||||||
|
import org.koin.dsl.module
|
||||||
|
|
||||||
|
val userModule = module {
|
||||||
|
single { UserController(get(), get(), get()) }
|
||||||
|
single { UserView(get()) }
|
||||||
|
}
|
||||||
|
|
||||||
|
val baseModule = module {
|
||||||
|
single { HealthCheckController(get()) }
|
||||||
|
single { BaseController(get()) }
|
||||||
|
single { BaseView(get()) }
|
||||||
|
}
|
||||||
|
|
||||||
|
val noteModule = module {
|
||||||
|
single { NoteController(get(), get()) }
|
||||||
|
single { NoteView(get()) }
|
||||||
|
}
|
||||||
|
|
||||||
|
val settingsModule = module {
|
||||||
|
single { SettingsController(get(), get()) }
|
||||||
|
single { SettingView(get()) }
|
||||||
|
}
|
||||||
@@ -5,16 +5,17 @@ import be.simplenotes.app.filters.AuthFilter
|
|||||||
import be.simplenotes.app.filters.AuthType
|
import be.simplenotes.app.filters.AuthType
|
||||||
import be.simplenotes.app.filters.ErrorFilter
|
import be.simplenotes.app.filters.ErrorFilter
|
||||||
import be.simplenotes.app.filters.TransactionFilter
|
import be.simplenotes.app.filters.TransactionFilter
|
||||||
import be.simplenotes.app.jetty.ConnectorBuilder
|
|
||||||
import be.simplenotes.app.jetty.Jetty
|
|
||||||
import be.simplenotes.app.routes.Router
|
import be.simplenotes.app.routes.Router
|
||||||
import be.simplenotes.app.utils.StaticFileResolver
|
import be.simplenotes.app.utils.StaticFileResolver
|
||||||
import be.simplenotes.app.utils.StaticFileResolverImpl
|
import be.simplenotes.app.utils.StaticFileResolverImpl
|
||||||
import be.simplenotes.config.ServerConfig
|
import be.simplenotes.app.views.ErrorView
|
||||||
|
import be.simplenotes.shared.config.ServerConfig
|
||||||
import org.eclipse.jetty.server.ServerConnector
|
import org.eclipse.jetty.server.ServerConnector
|
||||||
import org.http4k.core.Filter
|
import org.http4k.core.Filter
|
||||||
import org.http4k.core.RequestContexts
|
import org.http4k.core.RequestContexts
|
||||||
import org.http4k.routing.RoutingHttpHandler
|
import org.http4k.routing.RoutingHttpHandler
|
||||||
|
import org.http4k.server.ConnectorBuilder
|
||||||
|
import org.http4k.server.Jetty
|
||||||
import org.http4k.server.asServer
|
import org.http4k.server.asServer
|
||||||
import org.koin.core.qualifier.named
|
import org.koin.core.qualifier.named
|
||||||
import org.koin.core.qualifier.qualifier
|
import org.koin.core.qualifier.qualifier
|
||||||
@@ -58,5 +59,5 @@ val serverModule = module {
|
|||||||
single<Filter>(AuthType.Required.qualifier) { AuthFilter(get(), AuthType.Required, get()) }
|
single<Filter>(AuthType.Required.qualifier) { AuthFilter(get(), AuthType.Required, get()) }
|
||||||
single { ErrorFilter(get()) }
|
single { ErrorFilter(get()) }
|
||||||
single { TransactionFilter(get()) }
|
single { TransactionFilter(get()) }
|
||||||
single(named("styles")) { get<StaticFileResolver>().resolve("styles.css") }
|
single { ErrorView(get()) }
|
||||||
}
|
}
|
||||||
@@ -4,7 +4,7 @@ import be.simplenotes.app.api.ApiNoteController
|
|||||||
import be.simplenotes.app.api.ApiUserController
|
import be.simplenotes.app.api.ApiUserController
|
||||||
import be.simplenotes.app.controllers.*
|
import be.simplenotes.app.controllers.*
|
||||||
import be.simplenotes.app.filters.*
|
import be.simplenotes.app.filters.*
|
||||||
import be.simplenotes.types.LoggedInUser
|
import be.simplenotes.domain.security.JwtPayload
|
||||||
import org.http4k.core.*
|
import org.http4k.core.*
|
||||||
import org.http4k.core.Method.*
|
import org.http4k.core.Method.*
|
||||||
import org.http4k.filter.ResponseFilters.GZip
|
import org.http4k.filter.ResponseFilters.GZip
|
||||||
@@ -102,5 +102,5 @@ class Router(
|
|||||||
this to transactionFilter.then { handler(it, it.jwtPayload(contexts)) }
|
this to transactionFilter.then { handler(it, it.jwtPayload(contexts)) }
|
||||||
}
|
}
|
||||||
|
|
||||||
private typealias PublicHandler = (Request, LoggedInUser?) -> Response
|
private typealias PublicHandler = (Request, JwtPayload?) -> Response
|
||||||
private typealias ProtectedHandler = (Request, LoggedInUser) -> Response
|
private typealias ProtectedHandler = (Request, JwtPayload) -> Response
|
||||||
@@ -10,7 +10,7 @@ import java.util.*
|
|||||||
|
|
||||||
internal class UuidSerializer : KSerializer<UUID> {
|
internal class UuidSerializer : KSerializer<UUID> {
|
||||||
override val descriptor: SerialDescriptor
|
override val descriptor: SerialDescriptor
|
||||||
get() = PrimitiveSerialDescriptor("UUID", PrimitiveKind.STRING)
|
get() = PrimitiveSerialDescriptor("LocalDateTime", PrimitiveKind.STRING)
|
||||||
|
|
||||||
override fun serialize(encoder: Encoder, value: UUID) {
|
override fun serialize(encoder: Encoder, value: UUID) {
|
||||||
encoder.encodeString(value.toString())
|
encoder.encodeString(value.toString())
|
||||||
@@ -0,0 +1,10 @@
|
|||||||
|
package be.simplenotes.app.utils
|
||||||
|
|
||||||
|
import org.ocpsoft.prettytime.PrettyTime
|
||||||
|
import java.time.LocalDateTime
|
||||||
|
import java.time.ZoneId
|
||||||
|
import java.util.*
|
||||||
|
|
||||||
|
private val prettyTime = PrettyTime()
|
||||||
|
|
||||||
|
fun LocalDateTime.toTimeAgo(): String = prettyTime.format(Date.from(atZone(ZoneId.systemDefault()).toInstant()))
|
||||||
@@ -1,6 +1,6 @@
|
|||||||
package be.simplenotes.app.utils
|
package be.simplenotes.app.utils
|
||||||
|
|
||||||
import be.simplenotes.search.SearchTerms
|
import be.simplenotes.domain.usecases.search.SearchTerms
|
||||||
|
|
||||||
private fun innerRegex(name: String) =
|
private fun innerRegex(name: String) =
|
||||||
"""$name:['"](.*?)['"]""".toRegex()
|
"""$name:['"](.*?)['"]""".toRegex()
|
||||||
@@ -1,18 +1,19 @@
|
|||||||
package be.simplenotes.views
|
package be.simplenotes.app.views
|
||||||
|
|
||||||
import be.simplenotes.types.LoggedInUser
|
import be.simplenotes.app.utils.StaticFileResolver
|
||||||
|
import be.simplenotes.domain.security.JwtPayload
|
||||||
import kotlinx.html.*
|
import kotlinx.html.*
|
||||||
import kotlinx.html.ThScope.col
|
import kotlinx.html.ThScope.col
|
||||||
|
|
||||||
class BaseView(styles: String) : View(styles) {
|
class BaseView(staticFileResolver: StaticFileResolver) : View(staticFileResolver) {
|
||||||
fun renderHome(loggedInUser: LoggedInUser?) = renderPage(
|
fun renderHome(jwtPayload: JwtPayload?) = renderPage(
|
||||||
title = "Home",
|
title = "Home",
|
||||||
description = "A fast and simple note taking website",
|
description = "A fast and simple note taking website",
|
||||||
loggedInUser = loggedInUser
|
jwtPayload = jwtPayload
|
||||||
) {
|
) {
|
||||||
section("text-center my-2 p-2") {
|
section("text-center my-2 p-2") {
|
||||||
h1("text-5xl casual") {
|
h1("text-5xl casual") {
|
||||||
span("text-teal-300") { +"SimpleNotes " }
|
span("text-teal-300") { +"Simplenotes " }
|
||||||
+"- access your notes anywhere"
|
+"- access your notes anywhere"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -1,11 +1,12 @@
|
|||||||
package be.simplenotes.views
|
package be.simplenotes.app.views
|
||||||
|
|
||||||
import be.simplenotes.views.components.Alert
|
import be.simplenotes.app.utils.StaticFileResolver
|
||||||
import be.simplenotes.views.components.alert
|
import be.simplenotes.app.views.components.Alert
|
||||||
|
import be.simplenotes.app.views.components.alert
|
||||||
import kotlinx.html.a
|
import kotlinx.html.a
|
||||||
import kotlinx.html.div
|
import kotlinx.html.div
|
||||||
|
|
||||||
class ErrorView(styles: String) : View(styles) {
|
class ErrorView(staticFileResolver: StaticFileResolver) : View(staticFileResolver) {
|
||||||
|
|
||||||
enum class Type(val title: String) {
|
enum class Type(val title: String) {
|
||||||
SqlTransientError("Database unavailable"),
|
SqlTransientError("Database unavailable"),
|
||||||
@@ -13,7 +14,7 @@ class ErrorView(styles: String) : View(styles) {
|
|||||||
Other("Error"),
|
Other("Error"),
|
||||||
}
|
}
|
||||||
|
|
||||||
fun error(errorType: Type) = renderPage(errorType.title, loggedInUser = null) {
|
fun error(errorType: Type) = renderPage(errorType.title, jwtPayload = null) {
|
||||||
div("container mx-auto p-4") {
|
div("container mx-auto p-4") {
|
||||||
when (errorType) {
|
when (errorType) {
|
||||||
Type.SqlTransientError -> alert(
|
Type.SqlTransientError -> alert(
|
||||||
@@ -1,20 +1,21 @@
|
|||||||
package be.simplenotes.views
|
package be.simplenotes.app.views
|
||||||
|
|
||||||
import be.simplenotes.types.LoggedInUser
|
import be.simplenotes.app.utils.StaticFileResolver
|
||||||
import be.simplenotes.types.PersistedNote
|
import be.simplenotes.app.views.components.*
|
||||||
import be.simplenotes.types.PersistedNoteMetadata
|
import be.simplenotes.domain.model.PersistedNote
|
||||||
import be.simplenotes.views.components.*
|
import be.simplenotes.domain.model.PersistedNoteMetadata
|
||||||
|
import be.simplenotes.domain.security.JwtPayload
|
||||||
import io.konform.validation.ValidationError
|
import io.konform.validation.ValidationError
|
||||||
import kotlinx.html.*
|
import kotlinx.html.*
|
||||||
|
|
||||||
class NoteView(styles: String) : View(styles) {
|
class NoteView(staticFileResolver: StaticFileResolver) : View(staticFileResolver) {
|
||||||
|
|
||||||
fun noteEditor(
|
fun noteEditor(
|
||||||
loggedInUser: LoggedInUser,
|
jwtPayload: JwtPayload,
|
||||||
error: String? = null,
|
error: String? = null,
|
||||||
textarea: String? = null,
|
textarea: String? = null,
|
||||||
validationErrors: List<ValidationError> = emptyList(),
|
validationErrors: List<ValidationError> = emptyList(),
|
||||||
) = renderPage(title = "New note", loggedInUser = loggedInUser) {
|
) = renderPage(title = "New note", jwtPayload = jwtPayload) {
|
||||||
div("container mx-auto p-4") {
|
div("container mx-auto p-4") {
|
||||||
error?.let { alert(Alert.Warning, error) }
|
error?.let { alert(Alert.Warning, error) }
|
||||||
validationErrors.forEach {
|
validationErrors.forEach {
|
||||||
@@ -37,9 +38,7 @@ class NoteView(styles: String) : View(styles) {
|
|||||||
|tags: []
|
|tags: []
|
||||||
|---
|
|---
|
||||||
|
|
|
|
||||||
""".trimMargin(
|
""".trimMargin("|")
|
||||||
"|"
|
|
||||||
)
|
|
||||||
}
|
}
|
||||||
submitButton("Save")
|
submitButton("Save")
|
||||||
}
|
}
|
||||||
@@ -47,13 +46,13 @@ class NoteView(styles: String) : View(styles) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
fun notes(
|
fun notes(
|
||||||
loggedInUser: LoggedInUser,
|
jwtPayload: JwtPayload,
|
||||||
notes: List<PersistedNoteMetadata>,
|
notes: List<PersistedNoteMetadata>,
|
||||||
currentPage: Int,
|
currentPage: Int,
|
||||||
numberOfPages: Int,
|
numberOfPages: Int,
|
||||||
numberOfDeletedNotes: Int,
|
numberOfDeletedNotes: Int,
|
||||||
tag: String?,
|
tag: String?,
|
||||||
) = renderPage(title = "Notes", loggedInUser = loggedInUser) {
|
) = renderPage(title = "Notes", jwtPayload = jwtPayload) {
|
||||||
div("container mx-auto p-4") {
|
div("container mx-auto p-4") {
|
||||||
noteListHeader(numberOfDeletedNotes)
|
noteListHeader(numberOfDeletedNotes)
|
||||||
if (notes.isNotEmpty())
|
if (notes.isNotEmpty())
|
||||||
@@ -69,11 +68,11 @@ class NoteView(styles: String) : View(styles) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
fun search(
|
fun search(
|
||||||
loggedInUser: LoggedInUser,
|
jwtPayload: JwtPayload,
|
||||||
notes: List<PersistedNoteMetadata>,
|
notes: List<PersistedNoteMetadata>,
|
||||||
query: String,
|
query: String,
|
||||||
numberOfDeletedNotes: Int,
|
numberOfDeletedNotes: Int,
|
||||||
) = renderPage("Notes", loggedInUser = loggedInUser) {
|
) = renderPage("Notes", jwtPayload = jwtPayload) {
|
||||||
div("container mx-auto p-4") {
|
div("container mx-auto p-4") {
|
||||||
noteListHeader(numberOfDeletedNotes, query)
|
noteListHeader(numberOfDeletedNotes, query)
|
||||||
noteTable(notes)
|
noteTable(notes)
|
||||||
@@ -81,11 +80,11 @@ class NoteView(styles: String) : View(styles) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
fun trash(
|
fun trash(
|
||||||
loggedInUser: LoggedInUser,
|
jwtPayload: JwtPayload,
|
||||||
notes: List<PersistedNoteMetadata>,
|
notes: List<PersistedNoteMetadata>,
|
||||||
currentPage: Int,
|
currentPage: Int,
|
||||||
numberOfPages: Int,
|
numberOfPages: Int,
|
||||||
) = renderPage(title = "Notes", loggedInUser = loggedInUser) {
|
) = renderPage(title = "Notes", jwtPayload = jwtPayload) {
|
||||||
div("container mx-auto p-4") {
|
div("container mx-auto p-4") {
|
||||||
div("flex justify-between mb-4") {
|
div("flex justify-between mb-4") {
|
||||||
h1("text-2xl underline") { +"Deleted notes" }
|
h1("text-2xl underline") { +"Deleted notes" }
|
||||||
@@ -117,9 +116,9 @@ class NoteView(styles: String) : View(styles) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fun renderedNote(loggedInUser: LoggedInUser?, note: PersistedNote, shared: Boolean) = renderPage(
|
fun renderedNote(jwtPayload: JwtPayload?, note: PersistedNote, shared: Boolean) = renderPage(
|
||||||
note.meta.title,
|
note.meta.title,
|
||||||
loggedInUser = loggedInUser,
|
jwtPayload = jwtPayload,
|
||||||
scripts = listOf("/highlight.10.1.2.js", "/init-highlight.0.0.1.js")
|
scripts = listOf("/highlight.10.1.2.js", "/init-highlight.0.0.1.js")
|
||||||
) {
|
) {
|
||||||
div("container mx-auto p-4") {
|
div("container mx-auto p-4") {
|
||||||
@@ -144,6 +143,7 @@ class NoteView(styles: String) : View(styles) {
|
|||||||
}
|
}
|
||||||
if (!shared) {
|
if (!shared) {
|
||||||
noteActionForm(note)
|
noteActionForm(note)
|
||||||
|
publicPrivateForm(note)
|
||||||
|
|
||||||
if (note.public) {
|
if (note.public) {
|
||||||
p("my-4") {
|
p("my-4") {
|
||||||
@@ -166,30 +166,12 @@ class NoteView(styles: String) : View(styles) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
private fun DIV.noteActionForm(note: PersistedNote) {
|
private fun DIV.noteActionForm(note: PersistedNote) {
|
||||||
form(method = FormMethod.post, classes = "inline flex space-x-2 justify-end mb-4") {
|
span("flex space-x-2 justify-end mb-4") {
|
||||||
a(
|
a(
|
||||||
href = "/notes/${note.uuid}/edit",
|
href = "/notes/${note.uuid}/edit",
|
||||||
classes = "btn btn-green"
|
classes = "btn btn-teal"
|
||||||
) { +"Edit" }
|
) { +"Edit" }
|
||||||
span {
|
form(method = FormMethod.post, classes = "inline") {
|
||||||
button(
|
|
||||||
type = ButtonType.submit,
|
|
||||||
name = if (note.public) "private" else "public",
|
|
||||||
classes = "font-semibold border-b-4 ${if (note.public) "border-teal-200" else "border-green-500"}" +
|
|
||||||
" p-2 rounded-l bg-teal-200 text-gray-800"
|
|
||||||
) {
|
|
||||||
+"Private"
|
|
||||||
}
|
|
||||||
button(
|
|
||||||
type = ButtonType.submit,
|
|
||||||
name = if (note.public) "private" else "public",
|
|
||||||
classes = "font-semibold border-b-4 " +
|
|
||||||
(if (!note.public) "border-teal-200" else "border-green-500") +
|
|
||||||
" p-2 rounded-r bg-teal-200 text-gray-800"
|
|
||||||
) {
|
|
||||||
+"Public"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
button(
|
button(
|
||||||
type = ButtonType.submit,
|
type = ButtonType.submit,
|
||||||
name = "delete",
|
name = "delete",
|
||||||
@@ -198,3 +180,22 @@ class NoteView(styles: String) : View(styles) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private fun DIV.publicPrivateForm(note: PersistedNote) {
|
||||||
|
span("flex space-x-2 justify-end mb-4") {
|
||||||
|
|
||||||
|
form(method = FormMethod.post, classes = "ml-auto ") {
|
||||||
|
button(
|
||||||
|
type = ButtonType.submit,
|
||||||
|
name = if (note.public) "private" else "public",
|
||||||
|
classes = "btn btn-teal"
|
||||||
|
) {
|
||||||
|
if (note.public)
|
||||||
|
+"This note is public, do you want to make it private ?"
|
||||||
|
else
|
||||||
|
+"This note is private, do you want to make it public ?"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,49 +1,47 @@
|
|||||||
package be.simplenotes.views
|
package be.simplenotes.app.views
|
||||||
|
|
||||||
import be.simplenotes.types.LoggedInUser
|
import be.simplenotes.app.extensions.summary
|
||||||
import be.simplenotes.views.components.Alert
|
import be.simplenotes.app.utils.StaticFileResolver
|
||||||
import be.simplenotes.views.components.alert
|
import be.simplenotes.app.views.components.Alert
|
||||||
import be.simplenotes.views.components.input
|
import be.simplenotes.app.views.components.alert
|
||||||
import be.simplenotes.views.extensions.summary
|
import be.simplenotes.app.views.components.input
|
||||||
|
import be.simplenotes.domain.security.JwtPayload
|
||||||
import io.konform.validation.ValidationError
|
import io.konform.validation.ValidationError
|
||||||
import kotlinx.html.*
|
import kotlinx.html.*
|
||||||
import kotlinx.html.ButtonType.submit
|
import kotlinx.html.ButtonType.submit
|
||||||
|
|
||||||
class SettingView(styles: String) : View(styles) {
|
class SettingView(staticFileResolver: StaticFileResolver) : View(staticFileResolver) {
|
||||||
|
|
||||||
fun settings(
|
fun settings(
|
||||||
loggedInUser: LoggedInUser,
|
jwtPayload: JwtPayload,
|
||||||
error: String? = null,
|
error: String? = null,
|
||||||
validationErrors: List<ValidationError> = emptyList(),
|
validationErrors: List<ValidationError> = emptyList(),
|
||||||
) = renderPage("Settings", loggedInUser = loggedInUser) {
|
) = renderPage("Settings", jwtPayload = jwtPayload) {
|
||||||
div("container mx-auto") {
|
div("container mx-auto") {
|
||||||
|
|
||||||
section("m-4 p-4 bg-gray-800 rounded") {
|
section("m-4 p-4 bg-gray-800 rounded") {
|
||||||
h1("text-xl") {
|
h1("text-xl") {
|
||||||
+"Welcome "
|
+"Welcome "
|
||||||
span("text-teal-200 font-semibold") { +loggedInUser.username }
|
span("text-teal-200 font-semibold") { +jwtPayload.username }
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
section("m-4 p-2 bg-gray-800 rounded flex flex-wrap justify-around items-end") {
|
section("m-4 p-4 bg-gray-800 rounded flex justify-around") {
|
||||||
|
|
||||||
form(classes = "m-2", method = FormMethod.post, action = "/export") {
|
form(method = FormMethod.post, action = "/export") {
|
||||||
button(
|
button(name = "display",
|
||||||
name = "display",
|
|
||||||
classes = "inline btn btn-teal block",
|
classes = "inline btn btn-teal block",
|
||||||
type = submit
|
type = submit) { +"Display my data" }
|
||||||
) { +"Display my data" }
|
|
||||||
}
|
}
|
||||||
|
|
||||||
form(classes = "m-2", method = FormMethod.post, action = "/export") {
|
form(method = FormMethod.post, action = "/export") {
|
||||||
|
|
||||||
div {
|
|
||||||
listOf("json", "zip").forEach { format ->
|
listOf("json", "zip").forEach { format ->
|
||||||
|
div {
|
||||||
radioInput(name = "format") {
|
radioInput(name = "format") {
|
||||||
id = format
|
id = format
|
||||||
attributes["value"] = format
|
attributes["value"] = format
|
||||||
if(format == "json") attributes["checked"] = ""
|
if(format == "json") attributes["checked"] = ""
|
||||||
else attributes["class"] = "ml-4"
|
|
||||||
}
|
}
|
||||||
label(classes = "ml-2") {
|
label(classes = "ml-2") {
|
||||||
attributes["for"] = format
|
attributes["for"] = format
|
||||||
@@ -1,22 +1,23 @@
|
|||||||
package be.simplenotes.views
|
package be.simplenotes.app.views
|
||||||
|
|
||||||
import be.simplenotes.types.LoggedInUser
|
import be.simplenotes.app.utils.StaticFileResolver
|
||||||
import be.simplenotes.views.components.Alert
|
import be.simplenotes.app.views.components.Alert
|
||||||
import be.simplenotes.views.components.alert
|
import be.simplenotes.app.views.components.alert
|
||||||
import be.simplenotes.views.components.input
|
import be.simplenotes.app.views.components.input
|
||||||
import be.simplenotes.views.components.submitButton
|
import be.simplenotes.app.views.components.submitButton
|
||||||
|
import be.simplenotes.domain.security.JwtPayload
|
||||||
import io.konform.validation.ValidationError
|
import io.konform.validation.ValidationError
|
||||||
import kotlinx.html.*
|
import kotlinx.html.*
|
||||||
|
|
||||||
class UserView(styles: String) : View(styles) {
|
class UserView(staticFileResolver: StaticFileResolver) : View(staticFileResolver) {
|
||||||
fun register(
|
fun register(
|
||||||
loggedInUser: LoggedInUser?,
|
jwtPayload: JwtPayload?,
|
||||||
error: String? = null,
|
error: String? = null,
|
||||||
validationErrors: List<ValidationError> = emptyList(),
|
validationErrors: List<ValidationError> = emptyList(),
|
||||||
) = accountForm(
|
) = accountForm(
|
||||||
"Register",
|
"Register",
|
||||||
"Registration page",
|
"Registration page",
|
||||||
loggedInUser,
|
jwtPayload,
|
||||||
error,
|
error,
|
||||||
validationErrors,
|
validationErrors,
|
||||||
"Create an account",
|
"Create an account",
|
||||||
@@ -27,11 +28,11 @@ class UserView(styles: String) : View(styles) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
fun login(
|
fun login(
|
||||||
loggedInUser: LoggedInUser?,
|
jwtPayload: JwtPayload?,
|
||||||
error: String? = null,
|
error: String? = null,
|
||||||
validationErrors: List<ValidationError> = emptyList(),
|
validationErrors: List<ValidationError> = emptyList(),
|
||||||
new: Boolean = false,
|
new: Boolean = false,
|
||||||
) = accountForm("Login", "Login page", loggedInUser, error, validationErrors, "Sign In", "Sign In", new) {
|
) = accountForm("Login", "Login page", jwtPayload, error, validationErrors, "Sign In", "Sign In", new) {
|
||||||
+"Don't have an account yet? "
|
+"Don't have an account yet? "
|
||||||
a(href = "/register", classes = "no-underline text-blue-500 hover:text-blue-400 font-bold") {
|
a(href = "/register", classes = "no-underline text-blue-500 hover:text-blue-400 font-bold") {
|
||||||
+"Create an account"
|
+"Create an account"
|
||||||
@@ -41,14 +42,14 @@ class UserView(styles: String) : View(styles) {
|
|||||||
private fun accountForm(
|
private fun accountForm(
|
||||||
title: String,
|
title: String,
|
||||||
description: String,
|
description: String,
|
||||||
loggedInUser: LoggedInUser?,
|
jwtPayload: JwtPayload?,
|
||||||
error: String? = null,
|
error: String? = null,
|
||||||
validationErrors: List<ValidationError> = emptyList(),
|
validationErrors: List<ValidationError> = emptyList(),
|
||||||
h1: String,
|
h1: String,
|
||||||
submit: String,
|
submit: String,
|
||||||
new: Boolean = false,
|
new: Boolean = false,
|
||||||
footer: FlowContent.() -> Unit,
|
footer: FlowContent.() -> Unit,
|
||||||
) = renderPage(title = title, description, loggedInUser = loggedInUser) {
|
) = renderPage(title = title, description, jwtPayload = jwtPayload) {
|
||||||
div("centered container mx-auto flex justify-center items-center") {
|
div("centered container mx-auto flex justify-center items-center") {
|
||||||
div("w-full md:w-1/2 lg:w-1/3 m-4") {
|
div("w-full md:w-1/2 lg:w-1/3 m-4") {
|
||||||
div("p-8 mb-6") {
|
div("p-8 mb-6") {
|
||||||
@@ -1,16 +1,19 @@
|
|||||||
package be.simplenotes.views
|
package be.simplenotes.app.views
|
||||||
|
|
||||||
import be.simplenotes.types.LoggedInUser
|
import be.simplenotes.app.utils.StaticFileResolver
|
||||||
import be.simplenotes.views.components.navbar
|
import be.simplenotes.app.views.components.navbar
|
||||||
|
import be.simplenotes.domain.security.JwtPayload
|
||||||
import kotlinx.html.*
|
import kotlinx.html.*
|
||||||
import kotlinx.html.stream.appendHTML
|
import kotlinx.html.stream.appendHTML
|
||||||
|
|
||||||
abstract class View(private val styles: String) {
|
abstract class View(staticFileResolver: StaticFileResolver) {
|
||||||
|
|
||||||
|
private val styles = staticFileResolver.resolve("styles.css")!!
|
||||||
|
|
||||||
fun renderPage(
|
fun renderPage(
|
||||||
title: String,
|
title: String,
|
||||||
description: String? = null,
|
description: String? = null,
|
||||||
loggedInUser: LoggedInUser?,
|
jwtPayload: JwtPayload?,
|
||||||
scripts: List<String> = emptyList(),
|
scripts: List<String> = emptyList(),
|
||||||
body: MAIN.() -> Unit = {},
|
body: MAIN.() -> Unit = {},
|
||||||
) = buildString {
|
) = buildString {
|
||||||
@@ -34,7 +37,7 @@ abstract class View(private val styles: String) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
body("bg-gray-900 text-white") {
|
body("bg-gray-900 text-white") {
|
||||||
navbar(loggedInUser)
|
navbar(jwtPayload)
|
||||||
main { body() }
|
main { body() }
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -1,8 +1,8 @@
|
|||||||
package be.simplenotes.views.components
|
package be.simplenotes.app.views.components
|
||||||
|
|
||||||
import kotlinx.html.*
|
import kotlinx.html.*
|
||||||
|
|
||||||
internal fun FlowContent.alert(type: Alert, title: String, details: String? = null, multiline: Boolean = false) {
|
fun FlowContent.alert(type: Alert, title: String, details: String? = null, multiline: Boolean = false) {
|
||||||
val colors = when (type) {
|
val colors = when (type) {
|
||||||
Alert.Success -> "bg-green-500 border border-green-400 text-gray-800"
|
Alert.Success -> "bg-green-500 border border-green-400 text-gray-800"
|
||||||
Alert.Warning -> "bg-red-500 border border-red-400 text-red-200"
|
Alert.Warning -> "bg-red-500 border border-red-400 text-red-200"
|
||||||
@@ -17,6 +17,6 @@ internal fun FlowContent.alert(type: Alert, title: String, details: String? = nu
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
internal enum class Alert {
|
enum class Alert {
|
||||||
Success, Warning
|
Success, Warning
|
||||||
}
|
}
|
||||||
@@ -1,13 +1,13 @@
|
|||||||
package be.simplenotes.views.components
|
package be.simplenotes.app.views.components
|
||||||
|
|
||||||
import be.simplenotes.types.PersistedNoteMetadata
|
import be.simplenotes.app.utils.toTimeAgo
|
||||||
import be.simplenotes.views.utils.toTimeAgo
|
import be.simplenotes.domain.model.PersistedNoteMetadata
|
||||||
import kotlinx.html.*
|
import kotlinx.html.*
|
||||||
import kotlinx.html.ButtonType.submit
|
import kotlinx.html.ButtonType.submit
|
||||||
import kotlinx.html.FormMethod.post
|
import kotlinx.html.FormMethod.post
|
||||||
import kotlinx.html.ThScope.col
|
import kotlinx.html.ThScope.col
|
||||||
|
|
||||||
internal fun FlowContent.deletedNoteTable(notes: List<PersistedNoteMetadata>) = div("overflow-x-auto") {
|
fun FlowContent.deletedNoteTable(notes: List<PersistedNoteMetadata>) = div("overflow-x-auto") {
|
||||||
table {
|
table {
|
||||||
id = "notes"
|
id = "notes"
|
||||||
thead {
|
thead {
|
||||||
@@ -25,8 +25,8 @@ internal fun FlowContent.deletedNoteTable(notes: List<PersistedNoteMetadata>) =
|
|||||||
td("text-center") { +updatedAt.toTimeAgo() }
|
td("text-center") { +updatedAt.toTimeAgo() }
|
||||||
td { tags(tags) }
|
td { tags(tags) }
|
||||||
td("text-center") {
|
td("text-center") {
|
||||||
form(method = post, action = "/notes/deleted/$uuid") {
|
form(classes = "inline", method = post, action = "/notes/deleted/$uuid") {
|
||||||
button(classes = "btn btn-red mb-2", type = submit, name = "delete") {
|
button(classes = "btn btn-red", type = submit, name = "delete") {
|
||||||
+"Delete permanently"
|
+"Delete permanently"
|
||||||
}
|
}
|
||||||
button(classes = "ml-2 btn btn-green", type = submit, name = "restore") {
|
button(classes = "ml-2 btn btn-green", type = submit, name = "restore") {
|
||||||
@@ -1,9 +1,9 @@
|
|||||||
package be.simplenotes.views.components
|
package be.simplenotes.app.views.components
|
||||||
|
|
||||||
import kotlinx.html.*
|
import kotlinx.html.*
|
||||||
import kotlinx.html.ButtonType.submit
|
import kotlinx.html.ButtonType.submit
|
||||||
|
|
||||||
internal fun FlowContent.input(
|
fun FlowContent.input(
|
||||||
type: InputType = InputType.text,
|
type: InputType = InputType.text,
|
||||||
placeholder: String,
|
placeholder: String,
|
||||||
id: String,
|
id: String,
|
||||||
@@ -26,7 +26,7 @@ internal fun FlowContent.input(
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
internal fun FlowContent.submitButton(text: String) {
|
fun FlowContent.submitButton(text: String) {
|
||||||
div("flex items-center mt-6") {
|
div("flex items-center mt-6") {
|
||||||
button(
|
button(
|
||||||
type = submit,
|
type = submit,
|
||||||
@@ -1,9 +1,9 @@
|
|||||||
package be.simplenotes.views.components
|
package be.simplenotes.app.views.components
|
||||||
|
|
||||||
import be.simplenotes.types.LoggedInUser
|
import be.simplenotes.domain.security.JwtPayload
|
||||||
import kotlinx.html.*
|
import kotlinx.html.*
|
||||||
|
|
||||||
internal fun BODY.navbar(loggedInUser: LoggedInUser?) {
|
fun BODY.navbar(jwtPayload: JwtPayload?) {
|
||||||
nav {
|
nav {
|
||||||
id = "navbar"
|
id = "navbar"
|
||||||
a("/") {
|
a("/") {
|
||||||
@@ -12,7 +12,7 @@ internal fun BODY.navbar(loggedInUser: LoggedInUser?) {
|
|||||||
}
|
}
|
||||||
ul("space-x-2") {
|
ul("space-x-2") {
|
||||||
id = "navigation"
|
id = "navigation"
|
||||||
if (loggedInUser != null) {
|
if (jwtPayload != null) {
|
||||||
val links = listOf(
|
val links = listOf(
|
||||||
"/notes" to "Notes",
|
"/notes" to "Notes",
|
||||||
"/settings" to "Settings",
|
"/settings" to "Settings",
|
||||||
@@ -1,10 +1,10 @@
|
|||||||
package be.simplenotes.views.components
|
package be.simplenotes.app.views.components
|
||||||
|
|
||||||
import kotlinx.html.*
|
import kotlinx.html.*
|
||||||
import kotlinx.html.ButtonType.submit
|
import kotlinx.html.ButtonType.submit
|
||||||
import kotlinx.html.FormMethod.post
|
import kotlinx.html.FormMethod.post
|
||||||
|
|
||||||
internal fun DIV.noteListHeader(numberOfDeletedNotes: Int, query: String = "") {
|
fun DIV.noteListHeader(numberOfDeletedNotes: Int, query: String = "") {
|
||||||
div("flex justify-between mb-4") {
|
div("flex justify-between mb-4") {
|
||||||
h1("text-2xl underline") { +"Notes" }
|
h1("text-2xl underline") { +"Notes" }
|
||||||
span {
|
span {
|
||||||
@@ -1,11 +1,11 @@
|
|||||||
package be.simplenotes.views.components
|
package be.simplenotes.app.views.components
|
||||||
|
|
||||||
import be.simplenotes.types.PersistedNoteMetadata
|
import be.simplenotes.app.utils.toTimeAgo
|
||||||
import be.simplenotes.views.utils.toTimeAgo
|
import be.simplenotes.domain.model.PersistedNoteMetadata
|
||||||
import kotlinx.html.*
|
import kotlinx.html.*
|
||||||
import kotlinx.html.ThScope.col
|
import kotlinx.html.ThScope.col
|
||||||
|
|
||||||
internal fun FlowContent.noteTable(notes: List<PersistedNoteMetadata>) = div("overflow-x-auto") {
|
fun FlowContent.noteTable(notes: List<PersistedNoteMetadata>) = div("overflow-x-auto") {
|
||||||
table {
|
table {
|
||||||
id = "notes"
|
id = "notes"
|
||||||
thead {
|
thead {
|
||||||
|
Before Width: | Height: | Size: 7.8 KiB After Width: | Height: | Size: 7.8 KiB |
|
Before Width: | Height: | Size: 22 KiB After Width: | Height: | Size: 22 KiB |
|
Before Width: | Height: | Size: 6.2 KiB After Width: | Height: | Size: 6.2 KiB |
|
Before Width: | Height: | Size: 814 B After Width: | Height: | Size: 814 B |
|
Before Width: | Height: | Size: 1.1 KiB After Width: | Height: | Size: 1.1 KiB |
|
Before Width: | Height: | Size: 15 KiB After Width: | Height: | Size: 15 KiB |
|
Before Width: | Height: | Size: 2.1 KiB After Width: | Height: | Size: 2.1 KiB |
|
Before Width: | Height: | Size: 2.0 KiB After Width: | Height: | Size: 2.0 KiB |
|
Before Width: | Height: | Size: 2.2 KiB After Width: | Height: | Size: 2.2 KiB |
|
Before Width: | Height: | Size: 4.3 KiB After Width: | Height: | Size: 4.3 KiB |
|
Before Width: | Height: | Size: 1.5 KiB After Width: | Height: | Size: 1.5 KiB |
|
Before Width: | Height: | Size: 1.9 KiB After Width: | Height: | Size: 1.9 KiB |
@@ -0,0 +1 @@
|
|||||||
|
package be.simplenotes.app
|
||||||
@@ -1,9 +1,9 @@
|
|||||||
package be.simplenotes.app.filters
|
package be.simplenotes.app.filters
|
||||||
|
|
||||||
import be.simplenotes.config.JwtConfig
|
import be.simplenotes.domain.security.JwtPayload
|
||||||
import be.simplenotes.domain.security.JwtPayloadExtractor
|
import be.simplenotes.domain.security.JwtPayloadExtractor
|
||||||
import be.simplenotes.domain.security.SimpleJwt
|
import be.simplenotes.domain.security.SimpleJwt
|
||||||
import be.simplenotes.types.LoggedInUser
|
import be.simplenotes.shared.config.JwtConfig
|
||||||
import com.natpryce.hamkrest.assertion.assertThat
|
import com.natpryce.hamkrest.assertion.assertThat
|
||||||
import org.http4k.core.*
|
import org.http4k.core.*
|
||||||
import org.http4k.core.Method.GET
|
import org.http4k.core.Method.GET
|
||||||
@@ -58,7 +58,7 @@ internal class AuthFilterTest {
|
|||||||
|
|
||||||
@Test
|
@Test
|
||||||
fun `it should allow a valid token`() {
|
fun `it should allow a valid token`() {
|
||||||
val jwtPayload = LoggedInUser(1, "user")
|
val jwtPayload = JwtPayload(1, "user")
|
||||||
val token = simpleJwt.sign(jwtPayload)
|
val token = simpleJwt.sign(jwtPayload)
|
||||||
val response = app(Request(GET, "/optional").cookie("Bearer", token))
|
val response = app(Request(GET, "/optional").cookie("Bearer", token))
|
||||||
assertThat(response, hasStatus(OK))
|
assertThat(response, hasStatus(OK))
|
||||||
@@ -84,7 +84,7 @@ internal class AuthFilterTest {
|
|||||||
|
|
||||||
@Test
|
@Test
|
||||||
fun `it should allow a valid token"`() {
|
fun `it should allow a valid token"`() {
|
||||||
val jwtPayload = LoggedInUser(1, "user")
|
val jwtPayload = JwtPayload(1, "user")
|
||||||
val token = simpleJwt.sign(jwtPayload)
|
val token = simpleJwt.sign(jwtPayload)
|
||||||
val response = app(Request(GET, "/protected").cookie("Bearer", token))
|
val response = app(Request(GET, "/protected").cookie("Bearer", token))
|
||||||
assertThat(response, hasStatus(OK))
|
assertThat(response, hasStatus(OK))
|
||||||
@@ -1,6 +1,6 @@
|
|||||||
package be.simplenotes.app.utils
|
package be.simplenotes.app.utils
|
||||||
|
|
||||||
import be.simplenotes.search.SearchTerms
|
import be.simplenotes.domain.usecases.search.SearchTerms
|
||||||
import org.assertj.core.api.Assertions.assertThat
|
import org.assertj.core.api.Assertions.assertThat
|
||||||
import org.junit.jupiter.params.ParameterizedTest
|
import org.junit.jupiter.params.ParameterizedTest
|
||||||
import org.junit.jupiter.params.provider.MethodSource
|
import org.junit.jupiter.params.provider.MethodSource
|
||||||
@@ -30,9 +30,7 @@ internal class SearchTermsParserKtTest {
|
|||||||
createResult("tag:'example' title:'other' end", title = "other", tag = "example", all = "end"),
|
createResult("tag:'example' title:'other' end", title = "other", tag = "example", all = "end"),
|
||||||
createResult(
|
createResult(
|
||||||
"tag:'example abc' title:'other with words' this is the end ",
|
"tag:'example abc' title:'other with words' this is the end ",
|
||||||
title = "other with words",
|
title = "other with words", tag = "example abc", all = "this is the end"
|
||||||
tag = "example abc",
|
|
||||||
all = "this is the end"
|
|
||||||
),
|
),
|
||||||
)
|
)
|
||||||
|
|
||||||
@@ -1 +0,0 @@
|
|||||||
org.gradle.caching=true
|
|
||||||
@@ -1,18 +0,0 @@
|
|||||||
plugins {
|
|
||||||
`kotlin-dsl`
|
|
||||||
kotlin("jvm") version "1.4.10"
|
|
||||||
id("com.github.johnrengelman.shadow") version "6.1.0" apply false
|
|
||||||
kotlin("plugin.serialization") version "1.4.10"
|
|
||||||
}
|
|
||||||
|
|
||||||
repositories {
|
|
||||||
gradlePluginPortal()
|
|
||||||
maven { setUrl("https://kotlin.bintray.com/kotlinx") }
|
|
||||||
}
|
|
||||||
|
|
||||||
dependencies {
|
|
||||||
implementation("org.jetbrains.kotlin:kotlin-gradle-plugin:1.4.10")
|
|
||||||
implementation("org.jetbrains.kotlin:kotlin-serialization:1.4.10")
|
|
||||||
implementation("com.github.jengelman.gradle.plugins:shadow:6.1.0")
|
|
||||||
implementation("org.jlleitschuh.gradle:ktlint-gradle:9.4.1")
|
|
||||||
}
|
|
||||||
@@ -1,77 +0,0 @@
|
|||||||
package be.simplenotes
|
|
||||||
|
|
||||||
import org.gradle.api.DefaultTask
|
|
||||||
import org.gradle.api.GradleException
|
|
||||||
import org.gradle.api.tasks.*
|
|
||||||
import org.gradle.kotlin.dsl.getByType
|
|
||||||
import java.io.File
|
|
||||||
import java.lang.ProcessBuilder.Redirect.PIPE
|
|
||||||
import java.util.concurrent.TimeUnit
|
|
||||||
import kotlin.concurrent.thread
|
|
||||||
|
|
||||||
open class CssTask : DefaultTask() {
|
|
||||||
|
|
||||||
private val root = project.parent!!.rootDir
|
|
||||||
|
|
||||||
private val viewsProject = project
|
|
||||||
.parent
|
|
||||||
?.project(":simplenotes-views")
|
|
||||||
?: error("Missing :simplenotes-views")
|
|
||||||
|
|
||||||
@get:InputDirectory
|
|
||||||
val templatesDir = viewsProject.extensions
|
|
||||||
.getByType<SourceSetContainer>()
|
|
||||||
.asMap.getOrElse("main") { error("main sources not found") }
|
|
||||||
.allSource.srcDirs
|
|
||||||
.find { it.endsWith("kotlin") }
|
|
||||||
?: error("kotlin sources not found")
|
|
||||||
|
|
||||||
private val yarnRoot = File(project.rootDir, "css")
|
|
||||||
|
|
||||||
@get:InputDirectory
|
|
||||||
val postCssDir = File(project.rootDir, "css/src")
|
|
||||||
|
|
||||||
@get:InputFiles
|
|
||||||
val postCssConfig = listOf(
|
|
||||||
"tailwind.config.js",
|
|
||||||
"postcss.config.js",
|
|
||||||
"package.json"
|
|
||||||
).map { File(yarnRoot, it) }
|
|
||||||
|
|
||||||
@get:OutputDirectory
|
|
||||||
val outputRootDir = File(project.buildDir, "generated-resources/css")
|
|
||||||
|
|
||||||
private val cssIndex = File(postCssDir, "styles.pcss")
|
|
||||||
|
|
||||||
private val cssOutput = File(outputRootDir, "static/styles.css")
|
|
||||||
private val manifestOutput = File(outputRootDir, "css-manifest.json")
|
|
||||||
|
|
||||||
private val purgeGlob = "$templatesDir/**/*.kt"
|
|
||||||
|
|
||||||
@TaskAction
|
|
||||||
fun generateCss() {
|
|
||||||
// TODO: auto yarn install ?
|
|
||||||
|
|
||||||
outputRootDir.deleteRecursively()
|
|
||||||
|
|
||||||
ProcessBuilder("yarn", "run", "postcss", "build", "$cssIndex", "--output", "$cssOutput")
|
|
||||||
.apply {
|
|
||||||
environment().let {
|
|
||||||
it["MANIFEST"] = "$manifestOutput"
|
|
||||||
it["NODE_ENV"] = "production"
|
|
||||||
it["PURGE"] = purgeGlob
|
|
||||||
}
|
|
||||||
}
|
|
||||||
.redirectOutput(PIPE)
|
|
||||||
.redirectError(PIPE)
|
|
||||||
.directory(yarnRoot)
|
|
||||||
.start()
|
|
||||||
.apply {
|
|
||||||
thread { inputStream.use { it.copyTo(System.out) } }
|
|
||||||
thread { errorStream.use { it.copyTo(System.out) } }
|
|
||||||
waitFor(30, TimeUnit.SECONDS)
|
|
||||||
if (exitValue() != 0) throw GradleException(":/")
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
@@ -1,42 +0,0 @@
|
|||||||
@file:Suppress("SpellCheckingInspection")
|
|
||||||
|
|
||||||
package be.simplenotes
|
|
||||||
|
|
||||||
object Libs {
|
|
||||||
const val arrowCoreData = "io.arrow-kt:arrow-core-data:0.11.0"
|
|
||||||
const val commonsCompress = "org.apache.commons:commons-compress:1.20"
|
|
||||||
const val flexmark = "com.vladsch.flexmark:flexmark:0.62.2"
|
|
||||||
const val flexmarkGfmTasklist = "com.vladsch.flexmark:flexmark-ext-gfm-tasklist:0.62.2"
|
|
||||||
const val flywayCore = "org.flywaydb:flyway-core:6.5.4"
|
|
||||||
const val h2 = "com.h2database:h2:1.4.200"
|
|
||||||
const val hikariCP = "com.zaxxer:HikariCP:3.4.3"
|
|
||||||
const val http4kCore = "org.http4k:http4k-core:3.268.0"
|
|
||||||
const val javaJwt = "com.auth0:java-jwt:3.10.3"
|
|
||||||
const val javaxServlet = "javax.servlet:javax.servlet-api:4.0.1"
|
|
||||||
const val jbcrypt = "org.mindrot:jbcrypt:0.4"
|
|
||||||
const val jettyServer = "org.eclipse.jetty:jetty-server:9.4.32.v20200930"
|
|
||||||
const val jettyServlet = "org.eclipse.jetty:jetty-servlet:9.4.32.v20200930"
|
|
||||||
const val koinCore = "org.koin:koin-core:2.1.6"
|
|
||||||
const val konform = "io.konform:konform-jvm:0.2.0"
|
|
||||||
const val kotlinxHtml = "org.jetbrains.kotlinx:kotlinx-html-jvm:0.7.1"
|
|
||||||
const val kotlinxSerializationJson = "org.jetbrains.kotlinx:kotlinx-serialization-json-jvm:1.0.0"
|
|
||||||
const val ktormCore = "me.liuwj.ktorm:ktorm-core:3.0.0"
|
|
||||||
const val ktormMysql = "me.liuwj.ktorm:ktorm-support-mysql:3.0.0"
|
|
||||||
const val logbackClassic = "ch.qos.logback:logback-classic:1.2.3"
|
|
||||||
const val luceneAnalyzersCommon = "org.apache.lucene:lucene-analyzers-common:8.6.1"
|
|
||||||
const val luceneCore = "org.apache.lucene:lucene-core:8.6.1"
|
|
||||||
const val luceneQueryParser = "org.apache.lucene:lucene-queryparser:8.6.1"
|
|
||||||
const val mapstruct = "org.mapstruct:mapstruct:1.4.1.Final"
|
|
||||||
const val mapstructProcessor = "org.mapstruct:mapstruct-processor:1.4.1.Final"
|
|
||||||
const val mariadbClient = "org.mariadb.jdbc:mariadb-java-client:2.6.2"
|
|
||||||
const val owaspHtmlSanitizer = "com.googlecode.owasp-java-html-sanitizer:owasp-java-html-sanitizer:20200713.1"
|
|
||||||
const val prettytime ="org.ocpsoft.prettytime:prettytime:4.0.5.Final"
|
|
||||||
const val slf4jApi = "org.slf4j:slf4j-api:1.7.25"
|
|
||||||
const val snakeyaml = "org.yaml:snakeyaml:1.26"
|
|
||||||
|
|
||||||
const val assertJ = "org.assertj:assertj-core:3.16.1"
|
|
||||||
const val hamkrest = "com.natpryce:hamkrest:1.7.0.3"
|
|
||||||
const val http4kTestingHamkrest = "org.http4k:http4k-testing-hamkrest:3.268.0"
|
|
||||||
const val junit = "org.junit.jupiter:junit-jupiter:5.6.2"
|
|
||||||
const val mockk = "io.mockk:mockk:1.10.0"
|
|
||||||
}
|
|
||||||
@@ -1,15 +0,0 @@
|
|||||||
package be.simplenotes
|
|
||||||
|
|
||||||
import org.gradle.kotlin.dsl.register
|
|
||||||
|
|
||||||
plugins {
|
|
||||||
java apply false
|
|
||||||
}
|
|
||||||
|
|
||||||
tasks.register<CssTask>("css")
|
|
||||||
|
|
||||||
sourceSets {
|
|
||||||
val main by getting
|
|
||||||
val root = file("$buildDir/generated-resources/css")
|
|
||||||
main.resources.srcDir(root)
|
|
||||||
}
|
|
||||||
@@ -1,39 +0,0 @@
|
|||||||
package be.simplenotes
|
|
||||||
|
|
||||||
import java.util.concurrent.TimeUnit.MINUTES
|
|
||||||
import kotlin.concurrent.thread
|
|
||||||
|
|
||||||
fun runCommand(vararg args: String, onError: () -> Unit) {
|
|
||||||
logging.captureStandardOutput(LogLevel.INFO)
|
|
||||||
ProcessBuilder(*args)
|
|
||||||
.redirectOutput(ProcessBuilder.Redirect.PIPE)
|
|
||||||
.redirectError(ProcessBuilder.Redirect.PIPE)
|
|
||||||
.directory(rootProject.projectDir)
|
|
||||||
.start()
|
|
||||||
.apply {
|
|
||||||
thread { inputStream.use { it.copyTo(System.out) } }
|
|
||||||
thread { errorStream.use { it.copyTo(System.out) } }
|
|
||||||
waitFor(2, MINUTES)
|
|
||||||
if (exitValue() != 0) onError()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
tasks.create("dockerBuild") {
|
|
||||||
dependsOn("package")
|
|
||||||
|
|
||||||
doLast {
|
|
||||||
runCommand("docker", "build", "-t", "hubv/simplenotes:latest", ".") {
|
|
||||||
throw GradleException("Docker build failed")
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
tasks.create("dockerPush") {
|
|
||||||
dependsOn("dockerBuild")
|
|
||||||
|
|
||||||
doLast {
|
|
||||||
runCommand("docker", "push", "hubv/simplenotes:latest") {
|
|
||||||
throw GradleException("Docker Push failed")
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,47 +0,0 @@
|
|||||||
package be.simplenotes
|
|
||||||
|
|
||||||
import com.github.jengelman.gradle.plugins.shadow.tasks.ShadowJar
|
|
||||||
|
|
||||||
plugins {
|
|
||||||
id("com.github.johnrengelman.shadow")
|
|
||||||
}
|
|
||||||
|
|
||||||
tasks.withType<ShadowJar> {
|
|
||||||
|
|
||||||
archiveBaseName.set("app")
|
|
||||||
manifest.attributes["Main-Class"] = "be.simplenotes.app.SimpleNotesKt"
|
|
||||||
|
|
||||||
mergeServiceFiles()
|
|
||||||
|
|
||||||
// minimize()
|
|
||||||
|
|
||||||
// include("org.mariadb.jdbc:mariadb-java-client")
|
|
||||||
// include("com.h2database:h2")
|
|
||||||
// include("org.jetbrains.kotlin:kotlin-reflect")
|
|
||||||
// include("org.eclipse.jetty:*")
|
|
||||||
// include("org.apache.lucene:*")
|
|
||||||
// include("org.ocpsoft.prettytime:prettytime")
|
|
||||||
|
|
||||||
File(rootProject.projectDir, "buildSrc/src/main/resources/exclusions")
|
|
||||||
.listFiles()!!
|
|
||||||
.flatMap {
|
|
||||||
it.readLines()
|
|
||||||
.asSequence()
|
|
||||||
.map { it.trim() }
|
|
||||||
.filterNot { it.isBlank() }
|
|
||||||
.filterNot { it.startsWith("#") }
|
|
||||||
.asIterable()
|
|
||||||
}.forEach { exclude(it) }
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
tasks.create("package") {
|
|
||||||
rootProject.subprojects.forEach { dependsOn(":${it.name}:test") }
|
|
||||||
|
|
||||||
dependsOn("shadowJar")
|
|
||||||
dependsOn("css")
|
|
||||||
|
|
||||||
doLast {
|
|
||||||
println("SimpleNotes Packaged !")
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,57 +0,0 @@
|
|||||||
package be.simplenotes
|
|
||||||
|
|
||||||
import org.jetbrains.kotlin.gradle.tasks.KotlinCompile
|
|
||||||
|
|
||||||
plugins {
|
|
||||||
java
|
|
||||||
kotlin("jvm")
|
|
||||||
`java-library`
|
|
||||||
id("org.jlleitschuh.gradle.ktlint")
|
|
||||||
}
|
|
||||||
|
|
||||||
repositories {
|
|
||||||
mavenLocal()
|
|
||||||
mavenCentral()
|
|
||||||
jcenter()
|
|
||||||
maven { url = uri("https://dl.bintray.com/arrow-kt/arrow-kt/") }
|
|
||||||
maven { url = uri("https://kotlin.bintray.com/kotlinx") }
|
|
||||||
}
|
|
||||||
|
|
||||||
group = "be.simplenotes"
|
|
||||||
version = "1.0-SNAPSHOT"
|
|
||||||
|
|
||||||
dependencies {
|
|
||||||
implementation(kotlin("stdlib-jdk8"))
|
|
||||||
implementation(platform("org.jetbrains.kotlin:kotlin-bom:1.4.10"))
|
|
||||||
}
|
|
||||||
|
|
||||||
tasks.withType<Test> {
|
|
||||||
useJUnitPlatform()
|
|
||||||
}
|
|
||||||
|
|
||||||
java {
|
|
||||||
sourceCompatibility = JavaVersion.VERSION_14
|
|
||||||
targetCompatibility = JavaVersion.VERSION_14
|
|
||||||
}
|
|
||||||
|
|
||||||
sourceSets {
|
|
||||||
val test by getting
|
|
||||||
test.resources.srcDir("${rootProject.projectDir}/simplenotes-test-resources/src/test/resources")
|
|
||||||
}
|
|
||||||
|
|
||||||
tasks.withType<KotlinCompile> {
|
|
||||||
kotlinOptions {
|
|
||||||
jvmTarget = "14"
|
|
||||||
javaParameters = true
|
|
||||||
freeCompilerArgs = listOf(
|
|
||||||
"-Xinline-classes",
|
|
||||||
"-Xno-param-assertions",
|
|
||||||
"-Xno-call-assertions",
|
|
||||||
"-Xno-receiver-assertions"
|
|
||||||
)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
tasks.withType<JavaCompile> {
|
|
||||||
options.encoding = "UTF-8"
|
|
||||||
}
|
|
||||||
@@ -1,6 +0,0 @@
|
|||||||
package be.simplenotes
|
|
||||||
|
|
||||||
plugins {
|
|
||||||
kotlin("jvm") apply false
|
|
||||||
kotlin("plugin.serialization")
|
|
||||||
}
|
|
||||||
@@ -1,14 +0,0 @@
|
|||||||
META-INF/maven/**
|
|
||||||
META-INF/proguard/**
|
|
||||||
META-INF/*.kotlin_module
|
|
||||||
META-INF/DEPENDENCIES*
|
|
||||||
META-INF/NOTICE*
|
|
||||||
META-INF/LICENSE*
|
|
||||||
LICENSE*
|
|
||||||
META-INF/README*
|
|
||||||
META-INF/native-image/**
|
|
||||||
|
|
||||||
# Jetty
|
|
||||||
about.html
|
|
||||||
jetty-dir.css
|
|
||||||
server-ssl-cert.pem
|
|
||||||
@@ -1,261 +0,0 @@
|
|||||||
com/google/common/util/**
|
|
||||||
com/google/common/eventbus/**
|
|
||||||
com/google/common/reflect/**
|
|
||||||
com/google/common/escape/**
|
|
||||||
com/google/common/graph/**
|
|
||||||
com/google/common/html/**
|
|
||||||
com/google/common/hash/**
|
|
||||||
com/google/common/xml/**
|
|
||||||
com/google/common/io/**
|
|
||||||
com/google/common/cache/**
|
|
||||||
com/google/common/net/**
|
|
||||||
|
|
||||||
# Collections
|
|
||||||
com/google/common/collect/AbstractBiMap$1.class
|
|
||||||
com/google/common/collect/AbstractBiMap$BiMapEntry.class
|
|
||||||
com/google/common/collect/AbstractBiMap$EntrySet.class
|
|
||||||
com/google/common/collect/AbstractBiMap$Inverse.class
|
|
||||||
com/google/common/collect/AbstractBiMap$KeySet.class
|
|
||||||
com/google/common/collect/AbstractBiMap$ValueSet.class
|
|
||||||
com/google/common/collect/AbstractBiMap.class
|
|
||||||
com/google/common/collect/AbstractSortedKeySortedSetMultimap.class
|
|
||||||
com/google/common/collect/AbstractSortedMultiset$1DescendingMultisetImpl.class
|
|
||||||
com/google/common/collect/AbstractSortedMultiset.class
|
|
||||||
com/google/common/collect/AbstractTable$1.class
|
|
||||||
com/google/common/collect/AbstractTable$CellSet.class
|
|
||||||
com/google/common/collect/AbstractTable$Values.class
|
|
||||||
com/google/common/collect/AbstractTable.class
|
|
||||||
com/google/common/collect/ArrayListMultimap.class
|
|
||||||
com/google/common/collect/ArrayListMultimapGwtSerializationDependencies.class
|
|
||||||
com/google/common/collect/ArrayTable$1.class
|
|
||||||
com/google/common/collect/ArrayTable$2.class
|
|
||||||
com/google/common/collect/ArrayTable$3.class
|
|
||||||
com/google/common/collect/ArrayTable$ArrayMap$1.class
|
|
||||||
com/google/common/collect/ArrayTable$ArrayMap$2.class
|
|
||||||
com/google/common/collect/ArrayTable$ArrayMap.class
|
|
||||||
com/google/common/collect/ArrayTable$Column.class
|
|
||||||
com/google/common/collect/ArrayTable$ColumnMap.class
|
|
||||||
com/google/common/collect/ArrayTable$Row.class
|
|
||||||
com/google/common/collect/ArrayTable$RowMap.class
|
|
||||||
com/google/common/collect/ArrayTable.class
|
|
||||||
com/google/common/collect/ClassToInstanceMap.class
|
|
||||||
com/google/common/collect/CompactHashMap$1.class
|
|
||||||
com/google/common/collect/CompactHashMap$2.class
|
|
||||||
com/google/common/collect/CompactHashMap$3.class
|
|
||||||
com/google/common/collect/CompactHashMap$EntrySetView.class
|
|
||||||
com/google/common/collect/CompactHashMap$Itr.class
|
|
||||||
com/google/common/collect/CompactHashMap$KeySetView.class
|
|
||||||
com/google/common/collect/CompactHashMap$MapEntry.class
|
|
||||||
com/google/common/collect/CompactHashMap$ValuesView.class
|
|
||||||
com/google/common/collect/CompactHashMap.class
|
|
||||||
com/google/common/collect/CompactHashSet$1.class
|
|
||||||
com/google/common/collect/CompactHashSet.class
|
|
||||||
com/google/common/collect/CompactLinkedHashMap$1EntrySetImpl.class
|
|
||||||
com/google/common/collect/CompactLinkedHashMap$1KeySetImpl.class
|
|
||||||
com/google/common/collect/CompactLinkedHashMap$1ValuesImpl.class
|
|
||||||
com/google/common/collect/CompactLinkedHashMap.class
|
|
||||||
com/google/common/collect/CompactLinkedHashSet.class
|
|
||||||
com/google/common/collect/Comparators.class
|
|
||||||
com/google/common/collect/ComputationException.class
|
|
||||||
com/google/common/collect/ConcurrentHashMultiset$1.class
|
|
||||||
com/google/common/collect/ConcurrentHashMultiset$2.class
|
|
||||||
com/google/common/collect/ConcurrentHashMultiset$3.class
|
|
||||||
com/google/common/collect/ConcurrentHashMultiset$EntrySet.class
|
|
||||||
com/google/common/collect/ConcurrentHashMultiset$FieldSettersHolder.class
|
|
||||||
com/google/common/collect/ConcurrentHashMultiset.class
|
|
||||||
com/google/common/collect/DenseImmutableTable$1.class
|
|
||||||
com/google/common/collect/DenseImmutableTable$Column.class
|
|
||||||
com/google/common/collect/DenseImmutableTable$ColumnMap.class
|
|
||||||
com/google/common/collect/DenseImmutableTable$ImmutableArrayMap$1.class
|
|
||||||
com/google/common/collect/DenseImmutableTable$ImmutableArrayMap.class
|
|
||||||
com/google/common/collect/DenseImmutableTable$Row.class
|
|
||||||
com/google/common/collect/DenseImmutableTable$RowMap.class
|
|
||||||
com/google/common/collect/DenseImmutableTable.class
|
|
||||||
com/google/common/collect/DescendingImmutableSortedMultiset.class
|
|
||||||
com/google/common/collect/DescendingMultiset$1EntrySetImpl.class
|
|
||||||
com/google/common/collect/DescendingMultiset.class
|
|
||||||
com/google/common/collect/EnumBiMap.class
|
|
||||||
com/google/common/collect/EnumHashBiMap.class
|
|
||||||
com/google/common/collect/EnumMultiset$1.class
|
|
||||||
com/google/common/collect/EnumMultiset$2$1.class
|
|
||||||
com/google/common/collect/EnumMultiset$2.class
|
|
||||||
com/google/common/collect/EnumMultiset$Itr.class
|
|
||||||
com/google/common/collect/EnumMultiset.class
|
|
||||||
com/google/common/collect/EvictingQueue.class
|
|
||||||
com/google/common/collect/ForwardingBlockingDeque.class
|
|
||||||
com/google/common/collect/ForwardingDeque.class
|
|
||||||
com/google/common/collect/ForwardingImmutableCollection.class
|
|
||||||
com/google/common/collect/ForwardingImmutableList.class
|
|
||||||
com/google/common/collect/ForwardingImmutableMap.class
|
|
||||||
com/google/common/collect/ForwardingImmutableSet.class
|
|
||||||
com/google/common/collect/ForwardingIterator.class
|
|
||||||
com/google/common/collect/ForwardingListIterator.class
|
|
||||||
com/google/common/collect/ForwardingListMultimap.class
|
|
||||||
com/google/common/collect/ForwardingNavigableMap$StandardDescendingMap$1.class
|
|
||||||
com/google/common/collect/ForwardingNavigableMap$StandardDescendingMap.class
|
|
||||||
com/google/common/collect/ForwardingNavigableMap$StandardNavigableKeySet.class
|
|
||||||
com/google/common/collect/ForwardingNavigableMap.class
|
|
||||||
com/google/common/collect/ForwardingQueue.class
|
|
||||||
com/google/common/collect/ForwardingSetMultimap.class
|
|
||||||
com/google/common/collect/ForwardingSortedMultiset$StandardDescendingMultiset.class
|
|
||||||
com/google/common/collect/ForwardingSortedMultiset$StandardElementSet.class
|
|
||||||
com/google/common/collect/ForwardingSortedMultiset.class
|
|
||||||
com/google/common/collect/ForwardingSortedSetMultimap.class
|
|
||||||
com/google/common/collect/ForwardingTable.class
|
|
||||||
com/google/common/collect/GeneralRange.class
|
|
||||||
com/google/common/collect/GwtTransient.class
|
|
||||||
com/google/common/collect/HashBasedTable$Factory.class
|
|
||||||
com/google/common/collect/HashBasedTable.class
|
|
||||||
com/google/common/collect/HashBiMap$1$MapEntry.class
|
|
||||||
com/google/common/collect/HashBiMap$1.class
|
|
||||||
com/google/common/collect/HashBiMap$BiEntry.class
|
|
||||||
com/google/common/collect/HashBiMap$Inverse$1$InverseEntry.class
|
|
||||||
com/google/common/collect/HashBiMap$Inverse$1.class
|
|
||||||
com/google/common/collect/HashBiMap$Inverse$InverseKeySet$1.class
|
|
||||||
com/google/common/collect/HashBiMap$Inverse$InverseKeySet.class
|
|
||||||
com/google/common/collect/HashBiMap$Inverse.class
|
|
||||||
com/google/common/collect/HashBiMap$InverseSerializedForm.class
|
|
||||||
com/google/common/collect/HashBiMap$Itr.class
|
|
||||||
com/google/common/collect/HashBiMap$KeySet$1.class
|
|
||||||
com/google/common/collect/HashBiMap$KeySet.class
|
|
||||||
com/google/common/collect/HashBiMap.class
|
|
||||||
com/google/common/collect/HashMultimap.class
|
|
||||||
com/google/common/collect/HashMultimapGwtSerializationDependencies.class
|
|
||||||
com/google/common/collect/ImmutableClassToInstanceMap$1.class
|
|
||||||
com/google/common/collect/ImmutableClassToInstanceMap$Builder.class
|
|
||||||
com/google/common/collect/ImmutableClassToInstanceMap.class
|
|
||||||
com/google/common/collect/ImmutableSortedMultiset$Builder.class
|
|
||||||
com/google/common/collect/ImmutableSortedMultiset$SerializedForm.class
|
|
||||||
com/google/common/collect/ImmutableSortedMultiset.class
|
|
||||||
com/google/common/collect/ImmutableSortedMultisetFauxverideShim.class
|
|
||||||
com/google/common/collect/ImmutableTable$1.class
|
|
||||||
com/google/common/collect/ImmutableTable$Builder.class
|
|
||||||
com/google/common/collect/ImmutableTable$CollectorState.class
|
|
||||||
com/google/common/collect/ImmutableTable$MutableCell.class
|
|
||||||
com/google/common/collect/ImmutableTable$SerializedForm.class
|
|
||||||
com/google/common/collect/ImmutableTable.class
|
|
||||||
com/google/common/collect/Interner.class
|
|
||||||
com/google/common/collect/Interners$1.class
|
|
||||||
com/google/common/collect/Interners$InternerBuilder.class
|
|
||||||
com/google/common/collect/Interners$InternerFunction.class
|
|
||||||
com/google/common/collect/Interners$InternerImpl.class
|
|
||||||
com/google/common/collect/Interners.class
|
|
||||||
com/google/common/collect/LinkedHashMultimap$1.class
|
|
||||||
com/google/common/collect/LinkedHashMultimap$ValueEntry.class
|
|
||||||
com/google/common/collect/LinkedHashMultimap$ValueSet$1.class
|
|
||||||
com/google/common/collect/LinkedHashMultimap$ValueSet.class
|
|
||||||
com/google/common/collect/LinkedHashMultimap$ValueSetLink.class
|
|
||||||
com/google/common/collect/LinkedHashMultimap.class
|
|
||||||
com/google/common/collect/LinkedHashMultimapGwtSerializationDependencies.class
|
|
||||||
com/google/common/collect/LinkedListMultimap$1.class
|
|
||||||
com/google/common/collect/LinkedListMultimap$1EntriesImpl.class
|
|
||||||
com/google/common/collect/LinkedListMultimap$1KeySetImpl.class
|
|
||||||
com/google/common/collect/LinkedListMultimap$1ValuesImpl$1.class
|
|
||||||
com/google/common/collect/LinkedListMultimap$1ValuesImpl.class
|
|
||||||
com/google/common/collect/LinkedListMultimap$DistinctKeyIterator.class
|
|
||||||
com/google/common/collect/LinkedListMultimap$KeyList.class
|
|
||||||
com/google/common/collect/LinkedListMultimap$Node.class
|
|
||||||
com/google/common/collect/LinkedListMultimap$NodeIterator.class
|
|
||||||
com/google/common/collect/LinkedListMultimap$ValueForKeyIterator.class
|
|
||||||
com/google/common/collect/LinkedListMultimap.class
|
|
||||||
com/google/common/collect/MinMaxPriorityQueue$1.class
|
|
||||||
com/google/common/collect/MinMaxPriorityQueue$Builder.class
|
|
||||||
com/google/common/collect/MinMaxPriorityQueue$Heap.class
|
|
||||||
com/google/common/collect/MinMaxPriorityQueue$MoveDesc.class
|
|
||||||
com/google/common/collect/MinMaxPriorityQueue$QueueIterator.class
|
|
||||||
com/google/common/collect/MinMaxPriorityQueue.class
|
|
||||||
com/google/common/collect/MoreCollectors$ToOptionalState.class
|
|
||||||
com/google/common/collect/MoreCollectors.class
|
|
||||||
com/google/common/collect/MutableClassToInstanceMap$1.class
|
|
||||||
com/google/common/collect/MutableClassToInstanceMap$2$1.class
|
|
||||||
com/google/common/collect/MutableClassToInstanceMap$2.class
|
|
||||||
com/google/common/collect/MutableClassToInstanceMap$SerializedForm.class
|
|
||||||
com/google/common/collect/MutableClassToInstanceMap.class
|
|
||||||
com/google/common/collect/Queues.class
|
|
||||||
com/google/common/collect/RegularImmutableSortedMultiset.class
|
|
||||||
com/google/common/collect/RegularImmutableTable$1.class
|
|
||||||
com/google/common/collect/RegularImmutableTable$CellSet.class
|
|
||||||
com/google/common/collect/RegularImmutableTable$Values.class
|
|
||||||
com/google/common/collect/RegularImmutableTable.class
|
|
||||||
com/google/common/collect/RowSortedTable.class
|
|
||||||
com/google/common/collect/SingletonImmutableTable.class
|
|
||||||
com/google/common/collect/SortedMultisets$ElementSet.class
|
|
||||||
com/google/common/collect/SortedMultisets$NavigableElementSet.class
|
|
||||||
com/google/common/collect/SortedMultisets.class
|
|
||||||
com/google/common/collect/SparseImmutableTable.class
|
|
||||||
com/google/common/collect/StandardRowSortedTable$1.class
|
|
||||||
com/google/common/collect/StandardRowSortedTable$RowSortedMap.class
|
|
||||||
com/google/common/collect/StandardRowSortedTable.class
|
|
||||||
com/google/common/collect/StandardTable$1.class
|
|
||||||
com/google/common/collect/StandardTable$CellIterator.class
|
|
||||||
com/google/common/collect/StandardTable$Column$EntrySet.class
|
|
||||||
com/google/common/collect/StandardTable$Column$EntrySetIterator$1EntryImpl.class
|
|
||||||
com/google/common/collect/StandardTable$Column$EntrySetIterator.class
|
|
||||||
com/google/common/collect/StandardTable$Column$KeySet.class
|
|
||||||
com/google/common/collect/StandardTable$Column$Values.class
|
|
||||||
com/google/common/collect/StandardTable$Column.class
|
|
||||||
com/google/common/collect/StandardTable$ColumnKeyIterator.class
|
|
||||||
com/google/common/collect/StandardTable$ColumnKeySet.class
|
|
||||||
com/google/common/collect/StandardTable$ColumnMap$ColumnMapEntrySet$1.class
|
|
||||||
com/google/common/collect/StandardTable$ColumnMap$ColumnMapEntrySet.class
|
|
||||||
com/google/common/collect/StandardTable$ColumnMap$ColumnMapValues.class
|
|
||||||
com/google/common/collect/StandardTable$ColumnMap.class
|
|
||||||
com/google/common/collect/StandardTable$Row$1.class
|
|
||||||
com/google/common/collect/StandardTable$Row$2.class
|
|
||||||
com/google/common/collect/StandardTable$Row.class
|
|
||||||
com/google/common/collect/StandardTable$RowMap$EntrySet$1.class
|
|
||||||
com/google/common/collect/StandardTable$RowMap$EntrySet.class
|
|
||||||
com/google/common/collect/StandardTable$RowMap.class
|
|
||||||
com/google/common/collect/StandardTable$TableSet.class
|
|
||||||
com/google/common/collect/StandardTable.class
|
|
||||||
com/google/common/collect/Tables$1.class
|
|
||||||
com/google/common/collect/Tables$AbstractCell.class
|
|
||||||
com/google/common/collect/Tables$ImmutableCell.class
|
|
||||||
com/google/common/collect/Tables$TransformedTable$1.class
|
|
||||||
com/google/common/collect/Tables$TransformedTable$2.class
|
|
||||||
com/google/common/collect/Tables$TransformedTable$3.class
|
|
||||||
com/google/common/collect/Tables$TransformedTable.class
|
|
||||||
com/google/common/collect/Tables$TransposeTable$1.class
|
|
||||||
com/google/common/collect/Tables$TransposeTable.class
|
|
||||||
com/google/common/collect/Tables$UnmodifiableRowSortedMap.class
|
|
||||||
com/google/common/collect/Tables$UnmodifiableTable.class
|
|
||||||
com/google/common/collect/Tables.class
|
|
||||||
com/google/common/collect/TreeBasedTable$1.class
|
|
||||||
com/google/common/collect/TreeBasedTable$2.class
|
|
||||||
com/google/common/collect/TreeBasedTable$Factory.class
|
|
||||||
com/google/common/collect/TreeBasedTable$TreeRow.class
|
|
||||||
com/google/common/collect/TreeBasedTable.class
|
|
||||||
com/google/common/collect/TreeMultimap.class
|
|
||||||
com/google/common/collect/TreeMultiset$1.class
|
|
||||||
com/google/common/collect/TreeMultiset$2.class
|
|
||||||
com/google/common/collect/TreeMultiset$3.class
|
|
||||||
com/google/common/collect/TreeMultiset$4.class
|
|
||||||
com/google/common/collect/TreeMultiset$Aggregate$1.class
|
|
||||||
com/google/common/collect/TreeMultiset$Aggregate$2.class
|
|
||||||
com/google/common/collect/TreeMultiset$Aggregate.class
|
|
||||||
com/google/common/collect/TreeMultiset$AvlNode.class
|
|
||||||
com/google/common/collect/TreeMultiset$Reference.class
|
|
||||||
com/google/common/collect/TreeMultiset.class
|
|
||||||
com/google/common/collect/TreeRangeMap$1.class
|
|
||||||
com/google/common/collect/TreeRangeMap$AsMapOfRanges.class
|
|
||||||
com/google/common/collect/TreeRangeMap$RangeMapEntry.class
|
|
||||||
com/google/common/collect/TreeRangeMap$SubRangeMap$1$1.class
|
|
||||||
com/google/common/collect/TreeRangeMap$SubRangeMap$1.class
|
|
||||||
com/google/common/collect/TreeRangeMap$SubRangeMap$SubRangeMapAsMap$1.class
|
|
||||||
com/google/common/collect/TreeRangeMap$SubRangeMap$SubRangeMapAsMap$2.class
|
|
||||||
com/google/common/collect/TreeRangeMap$SubRangeMap$SubRangeMapAsMap$3.class
|
|
||||||
com/google/common/collect/TreeRangeMap$SubRangeMap$SubRangeMapAsMap$4.class
|
|
||||||
com/google/common/collect/TreeRangeMap$SubRangeMap$SubRangeMapAsMap.class
|
|
||||||
com/google/common/collect/TreeRangeMap$SubRangeMap.class
|
|
||||||
com/google/common/collect/TreeRangeMap.class
|
|
||||||
com/google/common/collect/TreeTraverser$1.class
|
|
||||||
com/google/common/collect/TreeTraverser$2$1.class
|
|
||||||
com/google/common/collect/TreeTraverser$2.class
|
|
||||||
com/google/common/collect/TreeTraverser$3$1.class
|
|
||||||
com/google/common/collect/TreeTraverser$3.class
|
|
||||||
com/google/common/collect/TreeTraverser$4.class
|
|
||||||
com/google/common/collect/TreeTraverser$BreadthFirstIterator.class
|
|
||||||
com/google/common/collect/TreeTraverser$PostOrderIterator.class
|
|
||||||
com/google/common/collect/TreeTraverser$PostOrderNode.class
|
|
||||||
com/google/common/collect/TreeTraverser$PreOrderIterator.class
|
|
||||||
com/google/common/collect/TreeTraverser.class
|
|
||||||
@@ -1,2 +0,0 @@
|
|||||||
ch/qos/logback/core/db/**
|
|
||||||
ch/qos/logback/classic/db/**
|
|
||||||
@@ -1,5 +0,0 @@
|
|||||||
org/checkerframework/**
|
|
||||||
org/intellij/**
|
|
||||||
com/google/errorprone/**
|
|
||||||
com/google/thirdparty/**
|
|
||||||
com/google/j2objc/**
|
|
||||||
@@ -2,7 +2,8 @@
|
|||||||
"name": "css",
|
"name": "css",
|
||||||
"version": "1.0.0",
|
"version": "1.0.0",
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"//": "`gradle css`"
|
"css": "NODE_ENV=dev MANIFEST=../app/src/main/resources/css-manifest.json postcss build src/styles.pcss --output ../app/src/main/resources/static/styles.css",
|
||||||
|
"css-purge": "NODE_ENV=production MANIFEST=../app/src/main/resources/css-manifest.json postcss build src/styles.pcss --output ../app/src/main/resources/static/styles.css"
|
||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"autoprefixer": "^9.8.6",
|
"autoprefixer": "^9.8.6",
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
module.exports = {
|
module.exports = {
|
||||||
purge: {
|
purge: {
|
||||||
content: [
|
content: [
|
||||||
process.env.PURGE
|
'../app/src/main/kotlin/views/**/*.kt'
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
theme: {
|
theme: {
|
||||||
|
|||||||
@@ -0,0 +1,8 @@
|
|||||||
|
#!/bin/sh
|
||||||
|
|
||||||
|
rm app/src/main/resources/css-manifest.json
|
||||||
|
rm app/src/main/resources/static/styles*
|
||||||
|
|
||||||
|
yarn --cwd css run css-purge \
|
||||||
|
&& docker build -t hubv/simplenotes:latest . \
|
||||||
|
&& docker push hubv/simplenotes:latest
|
||||||
@@ -0,0 +1,97 @@
|
|||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
|
||||||
|
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
|
||||||
|
<parent>
|
||||||
|
<artifactId>parent</artifactId>
|
||||||
|
<groupId>be.simplenotes</groupId>
|
||||||
|
<version>1.0-SNAPSHOT</version>
|
||||||
|
</parent>
|
||||||
|
<modelVersion>4.0.0</modelVersion>
|
||||||
|
|
||||||
|
<artifactId>domain</artifactId>
|
||||||
|
|
||||||
|
<dependencies>
|
||||||
|
<dependency>
|
||||||
|
<groupId>be.simplenotes</groupId>
|
||||||
|
<artifactId>shared</artifactId>
|
||||||
|
<version>1.0-SNAPSHOT</version>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>be.simplenotes</groupId>
|
||||||
|
<artifactId>shared</artifactId>
|
||||||
|
<version>1.0-SNAPSHOT</version>
|
||||||
|
<type>test-jar</type>
|
||||||
|
<scope>test</scope>
|
||||||
|
</dependency>
|
||||||
|
|
||||||
|
<dependency>
|
||||||
|
<groupId>io.arrow-kt</groupId>
|
||||||
|
<artifactId>arrow-core</artifactId>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>org.koin</groupId>
|
||||||
|
<artifactId>koin-core</artifactId>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>com.natpryce</groupId>
|
||||||
|
<artifactId>hamkrest</artifactId>
|
||||||
|
<scope>test</scope>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>org.junit.jupiter</groupId>
|
||||||
|
<artifactId>junit-jupiter</artifactId>
|
||||||
|
<scope>test</scope>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>io.mockk</groupId>
|
||||||
|
<artifactId>mockk</artifactId>
|
||||||
|
<scope>test</scope>
|
||||||
|
</dependency>
|
||||||
|
|
||||||
|
<dependency>
|
||||||
|
<groupId>io.konform</groupId>
|
||||||
|
<artifactId>konform-jvm</artifactId>
|
||||||
|
<version>0.2.0</version>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>org.mindrot</groupId>
|
||||||
|
<artifactId>jbcrypt</artifactId>
|
||||||
|
<version>0.4</version>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>com.auth0</groupId>
|
||||||
|
<artifactId>java-jwt</artifactId>
|
||||||
|
<version>3.10.3</version>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>com.vladsch.flexmark</groupId>
|
||||||
|
<artifactId>flexmark</artifactId>
|
||||||
|
<version>0.62.2</version>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>com.vladsch.flexmark</groupId>
|
||||||
|
<artifactId>flexmark-ext-gfm-tasklist</artifactId>
|
||||||
|
<version>0.62.2</version>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>org.yaml</groupId>
|
||||||
|
<artifactId>snakeyaml</artifactId>
|
||||||
|
<version>1.26</version>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>com.googlecode.owasp-java-html-sanitizer</groupId>
|
||||||
|
<artifactId>owasp-java-html-sanitizer</artifactId>
|
||||||
|
<version>20200713.1</version>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>org.jetbrains.kotlinx</groupId>
|
||||||
|
<artifactId>kotlinx-serialization-json-jvm</artifactId>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>org.apache.commons</groupId>
|
||||||
|
<artifactId>commons-compress</artifactId>
|
||||||
|
<version>1.20</version>
|
||||||
|
</dependency>
|
||||||
|
</dependencies>
|
||||||
|
|
||||||
|
</project>
|
||||||
@@ -1,4 +1,4 @@
|
|||||||
package be.simplenotes.types
|
package be.simplenotes.domain.model
|
||||||
|
|
||||||
import kotlinx.serialization.Contextual
|
import kotlinx.serialization.Contextual
|
||||||
import kotlinx.serialization.Serializable
|
import kotlinx.serialization.Serializable
|
||||||
@@ -0,0 +1,4 @@
|
|||||||
|
package be.simplenotes.domain.model
|
||||||
|
|
||||||
|
data class User(val username: String, val password: String)
|
||||||
|
data class PersistedUser(val username: String, val password: String, val id: Int)
|
||||||
@@ -2,7 +2,7 @@ package be.simplenotes.domain.security
|
|||||||
|
|
||||||
import org.owasp.html.HtmlPolicyBuilder
|
import org.owasp.html.HtmlPolicyBuilder
|
||||||
|
|
||||||
internal object HtmlSanitizer {
|
object HtmlSanitizer {
|
||||||
private val htmlPolicy = HtmlPolicyBuilder()
|
private val htmlPolicy = HtmlPolicyBuilder()
|
||||||
.allowElements("a")
|
.allowElements("a")
|
||||||
.allowCommonBlockElements()
|
.allowCommonBlockElements()
|
||||||
@@ -1,14 +1,18 @@
|
|||||||
package be.simplenotes.domain.security
|
package be.simplenotes.domain.security
|
||||||
|
|
||||||
import be.simplenotes.types.LoggedInUser
|
import be.simplenotes.domain.model.PersistedUser
|
||||||
import com.auth0.jwt.exceptions.JWTVerificationException
|
import com.auth0.jwt.exceptions.JWTVerificationException
|
||||||
|
|
||||||
|
data class JwtPayload(val userId: Int, val username: String) {
|
||||||
|
constructor(user: PersistedUser) : this(user.id, user.username)
|
||||||
|
}
|
||||||
|
|
||||||
class JwtPayloadExtractor(private val jwt: SimpleJwt) {
|
class JwtPayloadExtractor(private val jwt: SimpleJwt) {
|
||||||
operator fun invoke(token: String): LoggedInUser? = try {
|
operator fun invoke(token: String): JwtPayload? = try {
|
||||||
val decodedJWT = jwt.verifier.verify(token)
|
val decodedJWT = jwt.verifier.verify(token)
|
||||||
val id = decodedJWT.getClaim(userIdField).asInt() ?: null
|
val id = decodedJWT.getClaim(userIdField).asInt() ?: null
|
||||||
val username = decodedJWT.getClaim(usernameField).asString() ?: null
|
val username = decodedJWT.getClaim(usernameField).asString() ?: null
|
||||||
id?.let { username?.let { LoggedInUser(id, username) } }
|
id?.let { username?.let { JwtPayload(id, username) } }
|
||||||
} catch (e: JWTVerificationException) {
|
} catch (e: JWTVerificationException) {
|
||||||
null
|
null
|
||||||
} catch (e: IllegalArgumentException) {
|
} catch (e: IllegalArgumentException) {
|
||||||
@@ -1,7 +1,6 @@
|
|||||||
package be.simplenotes.domain.security
|
package be.simplenotes.domain.security
|
||||||
|
|
||||||
import be.simplenotes.config.JwtConfig
|
import be.simplenotes.shared.config.JwtConfig
|
||||||
import be.simplenotes.types.LoggedInUser
|
|
||||||
import com.auth0.jwt.JWT
|
import com.auth0.jwt.JWT
|
||||||
import com.auth0.jwt.JWTVerifier
|
import com.auth0.jwt.JWTVerifier
|
||||||
import com.auth0.jwt.algorithms.Algorithm
|
import com.auth0.jwt.algorithms.Algorithm
|
||||||
@@ -16,9 +15,9 @@ class SimpleJwt(jwtConfig: JwtConfig) {
|
|||||||
private val algorithm = Algorithm.HMAC256(jwtConfig.secret)
|
private val algorithm = Algorithm.HMAC256(jwtConfig.secret)
|
||||||
|
|
||||||
val verifier: JWTVerifier = JWT.require(algorithm).build()
|
val verifier: JWTVerifier = JWT.require(algorithm).build()
|
||||||
fun sign(loggedInUser: LoggedInUser): String = JWT.create()
|
fun sign(jwtPayload: JwtPayload): String = JWT.create()
|
||||||
.withClaim(userIdField, loggedInUser.userId)
|
.withClaim(userIdField, jwtPayload.userId)
|
||||||
.withClaim(usernameField, loggedInUser.username)
|
.withClaim(usernameField, jwtPayload.username)
|
||||||
.withExpiresAt(getExpiration())
|
.withExpiresAt(getExpiration())
|
||||||
.sign(algorithm)
|
.sign(algorithm)
|
||||||
|
|
||||||
@@ -1,16 +1,17 @@
|
|||||||
package be.simplenotes.domain.usecases
|
package be.simplenotes.domain.usecases
|
||||||
|
|
||||||
import arrow.core.computations.either
|
import arrow.core.Either
|
||||||
|
import arrow.core.extensions.fx
|
||||||
|
import be.simplenotes.domain.model.Note
|
||||||
|
import be.simplenotes.domain.model.PersistedNote
|
||||||
|
import be.simplenotes.domain.model.PersistedNoteMetadata
|
||||||
import be.simplenotes.domain.security.HtmlSanitizer
|
import be.simplenotes.domain.security.HtmlSanitizer
|
||||||
import be.simplenotes.domain.usecases.markdown.MarkdownConverter
|
import be.simplenotes.domain.usecases.markdown.MarkdownConverter
|
||||||
import be.simplenotes.domain.usecases.markdown.MarkdownParsingError
|
import be.simplenotes.domain.usecases.markdown.MarkdownParsingError
|
||||||
import be.simplenotes.persistance.repositories.NoteRepository
|
import be.simplenotes.domain.usecases.repositories.NoteRepository
|
||||||
import be.simplenotes.persistance.repositories.UserRepository
|
import be.simplenotes.domain.usecases.repositories.UserRepository
|
||||||
import be.simplenotes.search.NoteSearcher
|
import be.simplenotes.domain.usecases.search.NoteSearcher
|
||||||
import be.simplenotes.search.SearchTerms
|
import be.simplenotes.domain.usecases.search.SearchTerms
|
||||||
import be.simplenotes.types.Note
|
|
||||||
import be.simplenotes.types.PersistedNote
|
|
||||||
import be.simplenotes.types.PersistedNoteMetadata
|
|
||||||
import java.util.*
|
import java.util.*
|
||||||
|
|
||||||
class NoteService(
|
class NoteService(
|
||||||
@@ -20,7 +21,7 @@ class NoteService(
|
|||||||
private val searcher: NoteSearcher,
|
private val searcher: NoteSearcher,
|
||||||
) {
|
) {
|
||||||
|
|
||||||
fun create(userId: Int, markdownText: String) = either.eager<MarkdownParsingError, PersistedNote> {
|
fun create(userId: Int, markdownText: String) = Either.fx<MarkdownParsingError, PersistedNote> {
|
||||||
val persistedNote = !markdownConverter.renderDocument(markdownText)
|
val persistedNote = !markdownConverter.renderDocument(markdownText)
|
||||||
.map { it.copy(html = HtmlSanitizer.sanitize(it.html)) }
|
.map { it.copy(html = HtmlSanitizer.sanitize(it.html)) }
|
||||||
.map { Note(it.metadata, markdown = markdownText, html = it.html) }
|
.map { Note(it.metadata, markdown = markdownText, html = it.html) }
|
||||||
@@ -30,7 +31,7 @@ class NoteService(
|
|||||||
persistedNote
|
persistedNote
|
||||||
}
|
}
|
||||||
|
|
||||||
fun update(userId: Int, uuid: UUID, markdownText: String) = either.eager<MarkdownParsingError, PersistedNote?> {
|
fun update(userId: Int, uuid: UUID, markdownText: String) = Either.fx<MarkdownParsingError, PersistedNote?> {
|
||||||
val persistedNote = !markdownConverter.renderDocument(markdownText)
|
val persistedNote = !markdownConverter.renderDocument(markdownText)
|
||||||
.map { it.copy(html = HtmlSanitizer.sanitize(it.html)) }
|
.map { it.copy(html = HtmlSanitizer.sanitize(it.html)) }
|
||||||
.map { Note(it.metadata, markdown = markdownText, html = it.html) }
|
.map { Note(it.metadata, markdown = markdownText, html = it.html) }
|
||||||
@@ -1,7 +1,7 @@
|
|||||||
package be.simplenotes.domain.usecases.export
|
package be.simplenotes.domain.usecases.export
|
||||||
|
|
||||||
import be.simplenotes.persistance.repositories.NoteRepository
|
import be.simplenotes.domain.model.ExportedNote
|
||||||
import be.simplenotes.types.ExportedNote
|
import be.simplenotes.domain.usecases.repositories.NoteRepository
|
||||||
import kotlinx.serialization.builtins.ListSerializer
|
import kotlinx.serialization.builtins.ListSerializer
|
||||||
import kotlinx.serialization.json.Json
|
import kotlinx.serialization.json.Json
|
||||||
import org.apache.commons.compress.archivers.zip.ZipArchiveEntry
|
import org.apache.commons.compress.archivers.zip.ZipArchiveEntry
|
||||||
@@ -31,6 +31,7 @@ internal class ExportUseCaseImpl(private val noteRepository: NoteRepository, pri
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
class ZipOutput : AutoCloseable {
|
class ZipOutput : AutoCloseable {
|
||||||
val outputStream = ByteArrayOutputStream()
|
val outputStream = ByteArrayOutputStream()
|
||||||
private val zipOutputStream = ZipArchiveOutputStream(outputStream)
|
private val zipOutputStream = ZipArchiveOutputStream(outputStream)
|
||||||
@@ -1,11 +1,11 @@
|
|||||||
package be.simplenotes.domain.usecases.markdown
|
package be.simplenotes.domain.usecases.markdown
|
||||||
|
|
||||||
import arrow.core.Either
|
import arrow.core.Either
|
||||||
import arrow.core.computations.either
|
import arrow.core.extensions.fx
|
||||||
import arrow.core.left
|
import arrow.core.left
|
||||||
import arrow.core.right
|
import arrow.core.right
|
||||||
|
import be.simplenotes.domain.model.NoteMetadata
|
||||||
import be.simplenotes.domain.validation.NoteValidations
|
import be.simplenotes.domain.validation.NoteValidations
|
||||||
import be.simplenotes.types.NoteMetadata
|
|
||||||
import com.vladsch.flexmark.ext.gfm.tasklist.TaskListExtension
|
import com.vladsch.flexmark.ext.gfm.tasklist.TaskListExtension
|
||||||
import com.vladsch.flexmark.html.HtmlRenderer
|
import com.vladsch.flexmark.html.HtmlRenderer
|
||||||
import com.vladsch.flexmark.parser.Parser
|
import com.vladsch.flexmark.parser.Parser
|
||||||
@@ -73,10 +73,10 @@ internal class MarkdownConverterImpl : MarkdownConverter {
|
|||||||
|
|
||||||
private fun renderMarkdown(markdown: String) = parser.parse(markdown).run(renderer::render)
|
private fun renderMarkdown(markdown: String) = parser.parse(markdown).run(renderer::render)
|
||||||
|
|
||||||
override fun renderDocument(input: String) = either.eager<MarkdownParsingError, Document> {
|
override fun renderDocument(input: String) = Either.fx<MarkdownParsingError, Document> {
|
||||||
val (meta, md) = !splitMetaFromDocument(input)
|
val (meta, md) = !splitMetaFromDocument(input)
|
||||||
val parsedMeta = !parseMeta(meta)
|
val parsedMeta = !parseMeta(meta)
|
||||||
!Either.fromNullable(NoteValidations.validateMetadata(parsedMeta)).swap()
|
!NoteValidations.validateMetadata(parsedMeta).toEither { }.swap()
|
||||||
val html = renderMarkdown(md)
|
val html = renderMarkdown(md)
|
||||||
Document(parsedMeta, html)
|
Document(parsedMeta, html)
|
||||||
}
|
}
|
||||||
@@ -1,9 +1,9 @@
|
|||||||
package be.simplenotes.persistance.repositories
|
package be.simplenotes.domain.usecases.repositories
|
||||||
|
|
||||||
import be.simplenotes.types.ExportedNote
|
import be.simplenotes.domain.model.ExportedNote
|
||||||
import be.simplenotes.types.Note
|
import be.simplenotes.domain.model.Note
|
||||||
import be.simplenotes.types.PersistedNote
|
import be.simplenotes.domain.model.PersistedNote
|
||||||
import be.simplenotes.types.PersistedNoteMetadata
|
import be.simplenotes.domain.model.PersistedNoteMetadata
|
||||||
import java.util.*
|
import java.util.*
|
||||||
|
|
||||||
interface NoteRepository {
|
interface NoteRepository {
|
||||||
@@ -1,7 +1,7 @@
|
|||||||
package be.simplenotes.persistance.repositories
|
package be.simplenotes.domain.usecases.repositories
|
||||||
|
|
||||||
import be.simplenotes.types.PersistedUser
|
import be.simplenotes.domain.model.PersistedUser
|
||||||
import be.simplenotes.types.User
|
import be.simplenotes.domain.model.User
|
||||||
|
|
||||||
interface UserRepository {
|
interface UserRepository {
|
||||||
fun create(user: User): PersistedUser?
|
fun create(user: User): PersistedUser?
|
||||||
@@ -1,7 +1,7 @@
|
|||||||
package be.simplenotes.search
|
package be.simplenotes.domain.usecases.search
|
||||||
|
|
||||||
import be.simplenotes.types.PersistedNote
|
import be.simplenotes.domain.model.PersistedNote
|
||||||
import be.simplenotes.types.PersistedNoteMetadata
|
import be.simplenotes.domain.model.PersistedNoteMetadata
|
||||||
import java.util.*
|
import java.util.*
|
||||||
|
|
||||||
data class SearchTerms(val title: String?, val tag: String?, val content: String?, val all: String?)
|
data class SearchTerms(val title: String?, val tag: String?, val content: String?, val all: String?)
|
||||||
@@ -1,27 +1,27 @@
|
|||||||
package be.simplenotes.domain.usecases.users.delete
|
package be.simplenotes.domain.usecases.users.delete
|
||||||
|
|
||||||
import arrow.core.Either
|
import arrow.core.Either
|
||||||
import arrow.core.computations.either
|
import arrow.core.extensions.fx
|
||||||
import arrow.core.rightIfNotNull
|
import arrow.core.rightIfNotNull
|
||||||
import be.simplenotes.domain.security.PasswordHash
|
import be.simplenotes.domain.security.PasswordHash
|
||||||
|
import be.simplenotes.domain.usecases.repositories.UserRepository
|
||||||
|
import be.simplenotes.domain.usecases.search.NoteSearcher
|
||||||
import be.simplenotes.domain.validation.UserValidations
|
import be.simplenotes.domain.validation.UserValidations
|
||||||
import be.simplenotes.persistance.repositories.UserRepository
|
|
||||||
import be.simplenotes.search.NoteSearcher
|
|
||||||
|
|
||||||
internal class DeleteUseCaseImpl(
|
internal class DeleteUseCaseImpl(
|
||||||
private val userRepository: UserRepository,
|
private val userRepository: UserRepository,
|
||||||
private val passwordHash: PasswordHash,
|
private val passwordHash: PasswordHash,
|
||||||
private val searcher: NoteSearcher,
|
private val searcher: NoteSearcher,
|
||||||
) : DeleteUseCase {
|
) : DeleteUseCase {
|
||||||
override fun delete(form: DeleteForm) = either.eager<DeleteError, Unit> {
|
override fun delete(form: DeleteForm) = Either.fx<DeleteError, Unit> {
|
||||||
val user = !UserValidations.validateDelete(form)
|
val user = !UserValidations.validateDelete(form)
|
||||||
val persistedUser = !userRepository.find(user.username).rightIfNotNull { DeleteError.Unregistered }
|
val persistedUser = !userRepository.find(user.username).rightIfNotNull { DeleteError.Unregistered }
|
||||||
!Either.conditionally(
|
!Either.cond(
|
||||||
passwordHash.verify(user.password, persistedUser.password),
|
passwordHash.verify(user.password, persistedUser.password),
|
||||||
{ DeleteError.WrongPassword },
|
{ Unit },
|
||||||
{ Unit }
|
{ DeleteError.WrongPassword }
|
||||||
)
|
)
|
||||||
!Either.conditionally(userRepository.delete(persistedUser.id), { DeleteError.Unregistered }, { Unit })
|
!Either.cond(userRepository.delete(persistedUser.id), { Unit }, { DeleteError.Unregistered })
|
||||||
searcher.dropIndex(persistedUser.id)
|
searcher.dropIndex(persistedUser.id)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||