614 lines
20 KiB
Kotlin
614 lines
20 KiB
Kotlin
package com.andr3h3nriqu3s.applications
|
|
|
|
import java.sql.ResultSet
|
|
import java.util.UUID
|
|
import kotlin.collections.emptyList
|
|
import kotlin.collections.setOf
|
|
import org.springframework.http.HttpStatus
|
|
import org.springframework.http.MediaType
|
|
import org.springframework.jdbc.core.JdbcTemplate
|
|
import org.springframework.jdbc.core.RowMapper
|
|
import org.springframework.stereotype.Service
|
|
import org.springframework.web.bind.annotation.ControllerAdvice
|
|
import org.springframework.web.bind.annotation.DeleteMapping
|
|
import org.springframework.web.bind.annotation.GetMapping
|
|
import org.springframework.web.bind.annotation.PathVariable
|
|
import org.springframework.web.bind.annotation.PostMapping
|
|
import org.springframework.web.bind.annotation.PutMapping
|
|
import org.springframework.web.bind.annotation.RequestBody
|
|
import org.springframework.web.bind.annotation.RequestHeader
|
|
import org.springframework.web.bind.annotation.RequestMapping
|
|
import org.springframework.web.bind.annotation.RestController
|
|
import org.springframework.web.server.ResponseStatusException
|
|
|
|
data class ApplicationUrl(
|
|
var application_id: String,
|
|
var url: String,
|
|
) {
|
|
companion object : RowMapper<ApplicationUrl> {
|
|
override public fun mapRow(rs: ResultSet, rowNum: Int): ApplicationUrl {
|
|
return ApplicationUrl(
|
|
rs.getString("application_id"),
|
|
rs.getString("url"),
|
|
)
|
|
}
|
|
}
|
|
}
|
|
|
|
data class Application(
|
|
var id: String,
|
|
var url: String,
|
|
var title: String,
|
|
var user_id: String,
|
|
var extra_data: String,
|
|
var payrange: String,
|
|
var status_id: String?,
|
|
var company: String,
|
|
var recruiter: String,
|
|
var agency: Boolean,
|
|
var message: String,
|
|
var create_time: String,
|
|
var simple_url: String,
|
|
var flairs: List<Flair>,
|
|
var events: List<Event>,
|
|
) {
|
|
companion object : RowMapper<Application> {
|
|
override public fun mapRow(rs: ResultSet, rowNum: Int): Application {
|
|
return Application(
|
|
rs.getString("id"),
|
|
rs.getString("url"),
|
|
rs.getString("title"),
|
|
rs.getString("user_id"),
|
|
rs.getString("extra_data"),
|
|
rs.getString("payrange"),
|
|
rs.getString("status_id"),
|
|
rs.getString("company"),
|
|
rs.getString("recruiter"),
|
|
rs.getBoolean("agency"),
|
|
rs.getString("message"),
|
|
rs.getString("create_time"),
|
|
rs.getString("simple_url"),
|
|
emptyList(),
|
|
emptyList(),
|
|
)
|
|
}
|
|
}
|
|
}
|
|
|
|
data class SubmitRequest(val text: String)
|
|
|
|
data class StatusRequest(val id: String, val status_id: String?)
|
|
|
|
data class FlairRequest(val id: String, val text: String)
|
|
|
|
data class UpdateUrl(val id: String, val url: String)
|
|
|
|
data class CVData(
|
|
val company: String,
|
|
val recruiter: String,
|
|
val message: String,
|
|
val agency: Boolean,
|
|
val flairs: List<SimpleFlair>
|
|
)
|
|
|
|
@RestController
|
|
@ControllerAdvice
|
|
@RequestMapping("/api/application")
|
|
class ApplicationsController(
|
|
val sessionService: SessionService,
|
|
val applicationService: ApplicationService,
|
|
val flairService: FlairService,
|
|
val eventService: EventService,
|
|
) {
|
|
|
|
@GetMapping(path = ["/cv/{id}"], produces = [MediaType.APPLICATION_JSON_VALUE])
|
|
public fun getCV(@PathVariable id: String, @RequestHeader("token") token: String?): CVData? {
|
|
val user = sessionService.verifyToken(token)
|
|
|
|
val application = applicationService.findApplicationByIdNoUser(id)
|
|
|
|
if (application == null) return null
|
|
|
|
if (user == null) {
|
|
eventService.create(application.id, EventType.View)
|
|
}
|
|
|
|
val flairs = application.flairs.map { it.toFlairSimple() }
|
|
|
|
return CVData(
|
|
application.company,
|
|
application.recruiter,
|
|
application.message,
|
|
application.agency,
|
|
flairs
|
|
)
|
|
}
|
|
|
|
/** Create a new application from the link */
|
|
@PostMapping(path = ["/link"], produces = [MediaType.APPLICATION_JSON_VALUE])
|
|
public fun submitLink(
|
|
@RequestBody submit: SubmitRequest,
|
|
@RequestHeader("token") token: String
|
|
): Application {
|
|
val user = sessionService.verifyTokenThrow(token)
|
|
|
|
var application =
|
|
Application(
|
|
UUID.randomUUID().toString(),
|
|
submit.text,
|
|
"New Application",
|
|
user.id,
|
|
"",
|
|
"",
|
|
null,
|
|
"",
|
|
"",
|
|
false,
|
|
"",
|
|
"",
|
|
"",
|
|
emptyList(),
|
|
emptyList(),
|
|
)
|
|
|
|
if (!applicationService.createApplication(user, application)) {
|
|
throw ResponseStatusException(HttpStatus.CONFLICT, "Application already exists", null)
|
|
}
|
|
|
|
print("Created: ")
|
|
println(application)
|
|
|
|
return application
|
|
}
|
|
|
|
@PostMapping(path = ["/text/flair"], produces = [MediaType.APPLICATION_JSON_VALUE])
|
|
public fun textFlair(
|
|
@RequestBody info: FlairRequest,
|
|
@RequestHeader("token") token: String
|
|
): Int {
|
|
val user = sessionService.verifyTokenThrow(token)
|
|
val application = applicationService.findApplicationById(user, info.id)
|
|
if (application == null) {
|
|
throw ResponseStatusException(HttpStatus.NOT_FOUND, "Application not found", null)
|
|
}
|
|
|
|
val flairs = flairService.listUser(user)
|
|
|
|
var count = 0
|
|
|
|
for (flair: Flair in flairs) {
|
|
val regex =
|
|
Regex(
|
|
".*" + flair.expr + ".*",
|
|
setOf(RegexOption.IGNORE_CASE, RegexOption.DOT_MATCHES_ALL)
|
|
)
|
|
|
|
if (regex.matches(info.text)) {
|
|
count += 1
|
|
flairService.linkFlair(application, flair)
|
|
}
|
|
}
|
|
|
|
return count
|
|
}
|
|
|
|
@PostMapping(path = ["/text"], produces = [MediaType.APPLICATION_JSON_VALUE])
|
|
public fun submitText(
|
|
@RequestBody submit: SubmitRequest,
|
|
@RequestHeader("token") token: String
|
|
): List<Application> {
|
|
val user = sessionService.verifyTokenThrow(token)
|
|
|
|
var text = submit.text.replace("=\n", "")
|
|
|
|
var urls: List<String> = emptyList()
|
|
|
|
while (true) {
|
|
var index = text.indexOf("href")
|
|
|
|
if (index == -1) {
|
|
break
|
|
}
|
|
|
|
var new_url = StringBuilder()
|
|
|
|
var found_start = false
|
|
|
|
while (true) {
|
|
if (found_start) {
|
|
if (text[index] == '"') {
|
|
break
|
|
}
|
|
new_url.append(text[index])
|
|
} else if (text[index] == '"') {
|
|
found_start = true
|
|
}
|
|
index++
|
|
}
|
|
|
|
text = text.substring(index)
|
|
|
|
urls = urls.plus(new_url.toString().replace("&", "&").replace("=3D", "="))
|
|
}
|
|
|
|
print("found: ")
|
|
print(urls.size)
|
|
print(" links\n")
|
|
|
|
// jobListing is for glassdoor urls
|
|
// jobs/view is for linkeding urls
|
|
urls =
|
|
urls.filter { predicate ->
|
|
print("url: ")
|
|
print(predicate)
|
|
print("\n")
|
|
predicate.contains("jobListing") || predicate.contains("jobs/view")
|
|
}
|
|
|
|
print("found fileted: ")
|
|
print(urls.size)
|
|
print(" links\n")
|
|
|
|
urls = urls.toSet().toList()
|
|
|
|
print("removed duplicates: ")
|
|
print(urls.size)
|
|
print(" links\n")
|
|
|
|
var applications =
|
|
urls.map { elm ->
|
|
Application(
|
|
UUID.randomUUID().toString(),
|
|
if (elm.contains("linkedin")) elm.split("?")[0] else elm,
|
|
"New Application",
|
|
user.id,
|
|
"",
|
|
"",
|
|
null,
|
|
"",
|
|
"",
|
|
false,
|
|
"",
|
|
"",
|
|
if (elm.contains("linkedin")) elm.split("?")[0] else "",
|
|
emptyList(),
|
|
emptyList(),
|
|
)
|
|
}
|
|
|
|
applications =
|
|
applications.filter { elm -> applicationService.createApplication(user, elm) }
|
|
|
|
print("created new: ")
|
|
print(applications.size)
|
|
print(" links\n")
|
|
|
|
return applications
|
|
}
|
|
|
|
@GetMapping(path = ["/list"], produces = [MediaType.APPLICATION_JSON_VALUE])
|
|
public fun list(@RequestHeader("token") token: String): List<Application> {
|
|
val user = sessionService.verifyTokenThrow(token)
|
|
return applicationService.findAll(user)
|
|
}
|
|
|
|
@GetMapping(path = ["/{id}"], produces = [MediaType.APPLICATION_JSON_VALUE])
|
|
public fun get(@PathVariable id: String, @RequestHeader("token") token: String): Application? {
|
|
val user = sessionService.verifyTokenThrow(token)
|
|
val app = applicationService.findApplicationById(user, id)
|
|
if (app == null) {
|
|
throw ResponseStatusException(HttpStatus.NOT_FOUND, "Application not found", null)
|
|
}
|
|
return app
|
|
}
|
|
|
|
@PostMapping(
|
|
path = ["/link/application/{toLink}/{surviving}"],
|
|
produces = [MediaType.APPLICATION_JSON_VALUE]
|
|
)
|
|
public fun get(
|
|
@PathVariable toLink: String,
|
|
@PathVariable surviving: String,
|
|
@RequestHeader("token") token: String
|
|
): Application {
|
|
val user = sessionService.verifyTokenThrow(token)
|
|
val toLinkApp = applicationService.findApplicationById(user, toLink)
|
|
val app = applicationService.findApplicationById(user, surviving)
|
|
|
|
if (app == null || toLinkApp == null) {
|
|
throw ResponseStatusException(HttpStatus.NOT_FOUND, "Application not found", null)
|
|
}
|
|
|
|
applicationService.linkApplications(toLinkApp, app)
|
|
applicationService.delete(toLinkApp)
|
|
|
|
return app
|
|
}
|
|
|
|
@PutMapping(path = ["/status"], produces = [MediaType.APPLICATION_JSON_VALUE])
|
|
public fun status(
|
|
@RequestBody info: StatusRequest,
|
|
@RequestHeader("token") token: String,
|
|
): Application {
|
|
val user = sessionService.verifyTokenThrow(token)
|
|
var application = applicationService.findApplicationById(user, info.id)
|
|
if (application == null) {
|
|
throw NotFound()
|
|
}
|
|
|
|
if (application.status_id == info.status_id) {
|
|
return application
|
|
}
|
|
|
|
application.status_id = info.status_id
|
|
|
|
applicationService.updateStatus(application)
|
|
|
|
return application
|
|
}
|
|
|
|
@PutMapping(path = ["/update"], produces = [MediaType.APPLICATION_JSON_VALUE])
|
|
public fun update(
|
|
@RequestBody info: Application,
|
|
@RequestHeader("token") token: String
|
|
): Application {
|
|
val user = sessionService.verifyTokenThrow(token)
|
|
var application = applicationService.findApplicationById(user, info.id)
|
|
if (application == null) {
|
|
throw NotFound()
|
|
}
|
|
applicationService.update(info)
|
|
return info
|
|
}
|
|
|
|
@PostMapping(path = ["/update/url"], produces = [MediaType.APPLICATION_JSON_VALUE])
|
|
public fun updateUrl(
|
|
@RequestBody info: UpdateUrl,
|
|
@RequestHeader("token") token: String
|
|
): Application {
|
|
val user = sessionService.verifyTokenThrow(token)
|
|
var application = applicationService.findApplicationById(user, info.id)
|
|
|
|
if (application == null) {
|
|
throw ResponseStatusException(HttpStatus.NOT_FOUND, "Application not found", null)
|
|
}
|
|
|
|
application.url = info.url
|
|
application.simple_url = info.url.split("?")[0]
|
|
|
|
var maybe_exists =
|
|
applicationService.findApplicationByUrl(user, application.url)
|
|
?: applicationService.findApplicationByUrl(user, application.simple_url)
|
|
|
|
if (maybe_exists != null && maybe_exists.id != application.id) {
|
|
applicationService.delete(application)
|
|
|
|
maybe_exists.flairs = flairService.listFromLinkApplicationId(maybe_exists.id)
|
|
maybe_exists.events = eventService.listFromApplicationId(maybe_exists.id).toList()
|
|
|
|
return maybe_exists
|
|
}
|
|
|
|
applicationService.addUrl(application.id, info.url)
|
|
applicationService.addUrl(application.id, info.url.split("?")[0])
|
|
applicationService.update(application)
|
|
|
|
application.flairs = flairService.listFromLinkApplicationId(application.id)
|
|
application.events = eventService.listFromApplicationId(application.id).toList()
|
|
|
|
return application
|
|
}
|
|
|
|
@DeleteMapping(path = ["/flair/{id}/{flairid}"], produces = [MediaType.APPLICATION_JSON_VALUE])
|
|
public fun delete(
|
|
@PathVariable id: String,
|
|
@PathVariable flairid: String,
|
|
@RequestHeader("token") token: String
|
|
): Application {
|
|
val user = sessionService.verifyTokenThrow(token)
|
|
|
|
val application = applicationService.findApplicationById(user, id)
|
|
if (application == null) {
|
|
throw NotFound()
|
|
}
|
|
|
|
flairService.unlinkFlair(id, flairid)
|
|
return applicationService.findApplicationById(user, id)!!
|
|
}
|
|
|
|
@DeleteMapping(path = ["/{id}"], produces = [MediaType.APPLICATION_JSON_VALUE])
|
|
public fun delete(
|
|
@PathVariable id: String,
|
|
@RequestHeader("token") token: String
|
|
): Application {
|
|
val user = sessionService.verifyTokenThrow(token)
|
|
|
|
val application = applicationService.findApplicationById(user, id)
|
|
if (application == null) {
|
|
throw NotFound()
|
|
}
|
|
|
|
applicationService.delete(application)
|
|
return application
|
|
}
|
|
}
|
|
|
|
@Service
|
|
class ApplicationService(
|
|
val db: JdbcTemplate,
|
|
val flairService: FlairService,
|
|
val eventService: EventService,
|
|
) {
|
|
|
|
public fun findApplicationByUrl(user: UserDb, url: String): Application? {
|
|
val applications =
|
|
db.query(
|
|
"select * from applications as app inner join applications_urls as app_url on app_url.application_id=app.id where app_url.url=? and user_id=?",
|
|
arrayOf(url, user.id),
|
|
Application
|
|
)
|
|
.toList()
|
|
|
|
if (applications.size == 0) {
|
|
return null
|
|
}
|
|
|
|
return applications[0]
|
|
}
|
|
|
|
public fun findApplicationById(user: UserDb, id: String): Application? {
|
|
var applications =
|
|
db.query(
|
|
"select * from applications where id=? and user_id=?",
|
|
arrayOf(id, user.id),
|
|
Application
|
|
)
|
|
.toList()
|
|
|
|
if (applications.size == 0) {
|
|
return null
|
|
}
|
|
|
|
var application = applications[0]
|
|
|
|
application.flairs = flairService.listFromLinkApplicationId(application.id)
|
|
application.events = eventService.listFromApplicationId(application.id).toList()
|
|
|
|
return application
|
|
}
|
|
|
|
public fun findApplicationByIdNoUser(id: String): Application? {
|
|
var applications =
|
|
db.query("select * from applications where id=?", arrayOf(id), Application).toList()
|
|
|
|
if (applications.size == 0) {
|
|
return null
|
|
}
|
|
|
|
var application = applications[0]
|
|
|
|
// Views / Events are not needed for this request
|
|
application.flairs = flairService.listFromLinkApplicationId(application.id)
|
|
|
|
return application
|
|
}
|
|
|
|
public fun addUrl(id: String, url: String) {
|
|
val applications =
|
|
db.query(
|
|
"select * from applications_urls as app_url where app_url.url=? and app_url.application_id=?",
|
|
arrayOf(url, id),
|
|
ApplicationUrl
|
|
)
|
|
.toList()
|
|
|
|
if (applications.size > 0) {
|
|
return
|
|
}
|
|
|
|
db.update("insert into applications_urls (application_id, url) values (?, ?);", id, url)
|
|
}
|
|
|
|
public fun createApplication(user: UserDb, application: Application): Boolean {
|
|
if (this.findApplicationByUrl(user, application.url) != null) {
|
|
return false
|
|
}
|
|
|
|
// Create time is auto created by the database
|
|
db.update(
|
|
"insert into applications (id, url, title, user_id, extra_data, payrange, status_id, company, recruiter, message, agency, simple_url) values (?,?,?,?,?,?,?,?,?,?,?,?);",
|
|
application.id,
|
|
application.url,
|
|
application.title,
|
|
application.user_id,
|
|
application.extra_data,
|
|
application.payrange,
|
|
application.status_id,
|
|
application.company,
|
|
application.recruiter,
|
|
application.message,
|
|
application.agency,
|
|
application.simple_url,
|
|
)
|
|
|
|
eventService.create(application.id, EventType.Creation)
|
|
addUrl(application.id, application.url)
|
|
|
|
return true
|
|
}
|
|
|
|
private fun internalFindAll(user: UserDb): Iterable<Application> {
|
|
return db.query(
|
|
"select * from applications where user_id=? order by title asc;",
|
|
arrayOf(user.id),
|
|
Application
|
|
)
|
|
}
|
|
|
|
public fun findAll(user: UserDb): List<Application> {
|
|
var iter = internalFindAll(user)
|
|
return iter.toList()
|
|
}
|
|
|
|
public fun findAllByUserStatusId(statusId: String, user: UserDb): List<Application> {
|
|
return db.query(
|
|
"select * from applications where status_id=? and user_id=? order by title asc;",
|
|
arrayOf(statusId, user.id),
|
|
Application
|
|
)
|
|
}
|
|
|
|
// Update the stauts on the application object before giving it to this function
|
|
// TODO how status history works
|
|
public fun updateStatus(application: Application): Application {
|
|
|
|
eventService.create(application.id, EventType.StatusUpdate, application.status_id)
|
|
|
|
db.update(
|
|
"update applications set status_id=? where id=?",
|
|
application.status_id,
|
|
application.id,
|
|
)
|
|
return application
|
|
}
|
|
|
|
// Note this does not update status
|
|
public fun update(application: Application): Application {
|
|
// I don't want ot update create_time
|
|
db.update(
|
|
"update applications set url=?, title=?, user_id=?, extra_data=?, payrange=?, company=?, recruiter=?, message=?, agency=?, simple_url=? where id=?",
|
|
application.url,
|
|
application.title,
|
|
application.user_id,
|
|
application.extra_data,
|
|
application.payrange,
|
|
application.company,
|
|
application.recruiter,
|
|
application.message,
|
|
application.agency,
|
|
application.simple_url,
|
|
application.id,
|
|
)
|
|
return application
|
|
}
|
|
|
|
public fun linkApplications(toLink: Application, surviving: Application) {
|
|
db.update(
|
|
"update applications_urls set application_id=? where application_id=?;",
|
|
surviving.id,
|
|
toLink.id,
|
|
)
|
|
}
|
|
|
|
public fun delete(application: Application) {
|
|
db.update(
|
|
"delete from applications where id=?",
|
|
application.id,
|
|
)
|
|
db.update(
|
|
"delete from applications_urls where application_id=?",
|
|
application.id,
|
|
)
|
|
}
|
|
}
|