feat(grfn/bbbg): Init
This will eventually become a signup sheet + no-show tracker for my local board game meetup group Change-Id: Id8d1d80d95d1e2fda5041275cff2fecfd6fa43f1
This commit is contained in:
parent
479e9ea279
commit
c3cb7b0df8
35 changed files with 2549 additions and 0 deletions
1
users/grfn/bbbg/.clj-kondo/config.edn
Normal file
1
users/grfn/bbbg/.clj-kondo/config.edn
Normal file
|
@ -0,0 +1 @@
|
|||
{:lint-as {garden.def/defstyles clojure.core/def}}
|
1
users/grfn/bbbg/.envrc
Normal file
1
users/grfn/bbbg/.envrc
Normal file
|
@ -0,0 +1 @@
|
|||
eval "$(lorri direnv)"
|
9
users/grfn/bbbg/.gitignore
vendored
Normal file
9
users/grfn/bbbg/.gitignore
vendored
Normal file
|
@ -0,0 +1,9 @@
|
|||
/target
|
||||
/classes
|
||||
*.jar
|
||||
*.class
|
||||
/.nrepl-port
|
||||
/.cpcache
|
||||
/.clojure
|
||||
/result
|
||||
/.clj-kondo/.cache
|
2
users/grfn/bbbg/Makefile
Normal file
2
users/grfn/bbbg/Makefile
Normal file
|
@ -0,0 +1,2 @@
|
|||
deps.nix: deps.edn
|
||||
clj2nix ./deps.edn ./deps.nix '-A:uberjar' '-A:clj-test'
|
15
users/grfn/bbbg/arion-compose.nix
Normal file
15
users/grfn/bbbg/arion-compose.nix
Normal file
|
@ -0,0 +1,15 @@
|
|||
{ ... }:
|
||||
|
||||
{
|
||||
services = {
|
||||
postgres.service = {
|
||||
image = "postgres:12";
|
||||
environment = {
|
||||
POSTGRES_DB = "bbbg";
|
||||
POSTGRES_USER = "bbbg";
|
||||
POSTGRES_PASSWORD = "password";
|
||||
};
|
||||
ports = [ "5432:5432" ];
|
||||
};
|
||||
};
|
||||
}
|
2
users/grfn/bbbg/arion-pkgs.nix
Normal file
2
users/grfn/bbbg/arion-pkgs.nix
Normal file
|
@ -0,0 +1,2 @@
|
|||
let depot = import ../../.. {};
|
||||
in depot.third_party.nixpkgs
|
78
users/grfn/bbbg/default.nix
Normal file
78
users/grfn/bbbg/default.nix
Normal file
|
@ -0,0 +1,78 @@
|
|||
{ depot, pkgs, ... }:
|
||||
|
||||
with pkgs.lib;
|
||||
|
||||
let
|
||||
inherit (depot.third_party) gitignoreSource;
|
||||
|
||||
deps = import ./deps.nix {
|
||||
inherit (pkgs) fetchMavenArtifact fetchgit lib;
|
||||
};
|
||||
in rec {
|
||||
meta.targets = [
|
||||
"db-util"
|
||||
"server"
|
||||
];
|
||||
|
||||
depsPaths = deps.makePaths {};
|
||||
|
||||
resources = builtins.filterSource (_: type: type != "symlink") ./resources;
|
||||
|
||||
classpath.dev = concatStringsSep ":" (
|
||||
(map gitignoreSource [./src ./test ./env/dev]) ++ [resources] ++ depsPaths
|
||||
);
|
||||
|
||||
classpath.test = concatStringsSep ":" (
|
||||
(map gitignoreSource [./src ./test ./env/test]) ++ [resources] ++ depsPaths
|
||||
);
|
||||
|
||||
classpath.prod = concatStringsSep ":" (
|
||||
(map gitignoreSource [./src ./env/prod]) ++ [resources] ++ depsPaths
|
||||
);
|
||||
|
||||
testClojure = pkgs.writeShellScript "test-clojure" ''
|
||||
export HOME=$(pwd)
|
||||
${pkgs.clojure}/bin/clojure -Scp ${depsPaths}
|
||||
'';
|
||||
|
||||
mkJar = name: opts:
|
||||
with pkgs;
|
||||
assert (hasSuffix ".jar" name);
|
||||
stdenv.mkDerivation rec {
|
||||
inherit name;
|
||||
dontUnpack = true;
|
||||
buildPhase = ''
|
||||
export HOME=$(pwd)
|
||||
cp ${./pom.xml} pom.xml
|
||||
cp ${./deps.edn} deps.edn
|
||||
${clojure}/bin/clojure \
|
||||
-Scp ${classpath.prod} \
|
||||
-A:uberjar \
|
||||
${name} \
|
||||
-C ${opts}
|
||||
'';
|
||||
|
||||
doCheck = true;
|
||||
|
||||
checkPhase = ''
|
||||
echo "checking for existence of ${name}"
|
||||
[ -f ${name} ]
|
||||
'';
|
||||
|
||||
installPhase = ''
|
||||
cp ${name} $out
|
||||
'';
|
||||
};
|
||||
|
||||
db-util-jar = mkJar "bbbg-db-util.jar" "-m bbbg.db";
|
||||
|
||||
db-util = pkgs.writeShellScriptBin "bbbg-db-util" ''
|
||||
exec ${pkgs.openjdk17_headless}/bin/java -jar ${db-util-jar} "$@"
|
||||
'';
|
||||
|
||||
server-jar = mkJar "bbbg-server.jar" "-m bbbg.core";
|
||||
|
||||
server = pkgs.writeShellScriptBin "bbbg-server" ''
|
||||
exec ${pkgs.openjdk17_headless}/bin/java -jar ${server-jar} "$@"
|
||||
'';
|
||||
}
|
63
users/grfn/bbbg/deps.edn
Normal file
63
users/grfn/bbbg/deps.edn
Normal file
|
@ -0,0 +1,63 @@
|
|||
{:deps
|
||||
{org.clojure/clojure {:mvn/version "1.11.0-alpha3"}
|
||||
|
||||
;; DB
|
||||
com.github.seancorfield/next.jdbc {:mvn/version "1.2.753"}
|
||||
com.impossibl.pgjdbc-ng/pgjdbc-ng {:mvn/version "0.8.4"}
|
||||
com.zaxxer/HikariCP {:mvn/version "5.0.0"}
|
||||
migratus/migratus {:mvn/version "1.3.5"}
|
||||
com.github.seancorfield/honeysql {:mvn/version "2.1.833"}
|
||||
nilenso/honeysql-postgres {:mvn/version "0.4.112"}
|
||||
|
||||
;; HTTP
|
||||
http-kit/http-kit {:mvn/version "2.5.3"}
|
||||
ring/ring {:mvn/version "1.9.4"}
|
||||
compojure/compojure {:mvn/version "1.6.2"}
|
||||
javax.servlet/servlet-api {:mvn/version "2.5"}
|
||||
|
||||
;; Web
|
||||
hiccup/hiccup {:mvn/version "1.0.5"}
|
||||
garden/garden {:mvn/version "1.3.10"}
|
||||
|
||||
;; Utils
|
||||
com.stuartsierra/component {:mvn/version "1.0.0"}
|
||||
|
||||
;; Logging + Observability
|
||||
ch.qos.logback/logback-classic {:mvn/version "1.2.3"
|
||||
:exclusions [org.slf4j/slf4j-api]}
|
||||
org.slf4j/jul-to-slf4j {:mvn/version "1.7.30"}
|
||||
org.slf4j/jcl-over-slf4j {:mvn/version "1.7.30"}
|
||||
org.slf4j/log4j-over-slf4j {:mvn/version "1.7.30"}
|
||||
cambium/cambium.core {:mvn/version "0.9.3"}
|
||||
cambium/cambium.codec-cheshire {:mvn/version "0.9.3"}
|
||||
cambium/cambium.logback.core {:mvn/version "0.4.3"}
|
||||
cambium/cambium.logback.json {:mvn/version "0.4.3"}
|
||||
clj-commons/iapetos {:mvn/version "0.1.12"}
|
||||
|
||||
;; Utilities
|
||||
yogthos/config {:mvn/version "1.1.8"}
|
||||
clojure.java-time/clojure.java-time {:mvn/version "0.3.3"}
|
||||
cheshire/cheshire {:mvn/version "5.10.1"}
|
||||
|
||||
;; Spec
|
||||
org.clojure/spec.alpha {:mvn/version "0.3.214"}
|
||||
org.clojure/core.specs.alpha {:mvn/version "0.2.62"}
|
||||
expound/expound {:mvn/version "0.8.10"}}
|
||||
|
||||
:paths
|
||||
["src"
|
||||
"test"
|
||||
"resources"
|
||||
"target/classes"]
|
||||
:aliases
|
||||
{:dev {:extra-paths ["env/dev"]
|
||||
:jvm-opts ["-XX:-OmitStackTraceInFastThrow"]}
|
||||
:clj-test {:extra-paths ["test" "env/test"]
|
||||
:extra-deps {io.github.cognitect-labs/test-runner
|
||||
{:git/url "https://github.com/cognitect-labs/test-runner"
|
||||
:sha "cc75980b43011773162b485f46f939dc5fba91e4"}}
|
||||
:main-opts ["-m" "cognitect.test-runner"
|
||||
"-d" "test"]}
|
||||
:uberjar {:extra-deps {seancorfield/depstar {:mvn/version "1.0.94"}}
|
||||
:extra-paths ["env/prod"]
|
||||
:main-opts ["-m" "hf.depstar.uberjar"]}}}
|
1255
users/grfn/bbbg/deps.nix
Normal file
1255
users/grfn/bbbg/deps.nix
Normal file
File diff suppressed because it is too large
Load diff
3
users/grfn/bbbg/env/dev/bbbg-signup/env.clj
vendored
Normal file
3
users/grfn/bbbg/env/dev/bbbg-signup/env.clj
vendored
Normal file
|
@ -0,0 +1,3 @@
|
|||
(ns bbbg.env)
|
||||
|
||||
(def environment :env/dev)
|
15
users/grfn/bbbg/env/dev/logback.xml
vendored
Normal file
15
users/grfn/bbbg/env/dev/logback.xml
vendored
Normal file
|
@ -0,0 +1,15 @@
|
|||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<configuration>
|
||||
<appender name="STDOUT" class="ch.qos.logback.core.ConsoleAppender">
|
||||
<encoder>
|
||||
<pattern>%d{HH:mm:ss.SSS} [%thread] %-5level %logger{36} - %msg { %mdc }%n</pattern>
|
||||
</encoder>
|
||||
</appender>
|
||||
|
||||
<root level="INFO">
|
||||
<appender-ref ref="STDOUT" />
|
||||
</root>
|
||||
|
||||
<logger name="user" level="ALL" />
|
||||
<logger name="ci.windtunnel" level="ALL" />
|
||||
</configuration>
|
3
users/grfn/bbbg/env/prod/bbbg-signup/env.clj
vendored
Normal file
3
users/grfn/bbbg/env/prod/bbbg-signup/env.clj
vendored
Normal file
|
@ -0,0 +1,3 @@
|
|||
(ns bbbg.env)
|
||||
|
||||
(def environment :env/prod)
|
31
users/grfn/bbbg/env/prod/logback.xml
vendored
Normal file
31
users/grfn/bbbg/env/prod/logback.xml
vendored
Normal file
|
@ -0,0 +1,31 @@
|
|||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<configuration>
|
||||
<!-- Silence Logback's own status messages about config parsing -->
|
||||
<statusListener class="ch.qos.logback.core.status.NopStatusListener" />
|
||||
|
||||
<!-- Console output -->
|
||||
<appender name="STDOUT" class="ch.qos.logback.core.ConsoleAppender">
|
||||
<!-- Only log level INFO and above -->
|
||||
<filter class="ch.qos.logback.classic.filter.ThresholdFilter">
|
||||
<level>INFO</level>
|
||||
</filter>
|
||||
<encoder class="ch.qos.logback.core.encoder.LayoutWrappingEncoder">
|
||||
<layout class="cambium.logback.json.FlatJsonLayout">
|
||||
<jsonFormatter class="ch.qos.logback.contrib.jackson.JacksonJsonFormatter">
|
||||
<prettyPrint>false</prettyPrint>
|
||||
</jsonFormatter>
|
||||
<!-- <context>api</context> -->
|
||||
<timestampFormat>yyyy-MM-dd'T'HH:mm:ss.SSS'Z'</timestampFormat>
|
||||
<timestampFormatTimezoneId>UTC</timestampFormatTimezoneId>
|
||||
<appendLineSeparator>true</appendLineSeparator>
|
||||
</layout>
|
||||
</encoder>
|
||||
</appender>
|
||||
|
||||
|
||||
<root level="INFO">
|
||||
<appender-ref ref="STDOUT" />
|
||||
</root>
|
||||
|
||||
<logger name="user" level="ALL" />
|
||||
</configuration>
|
3
users/grfn/bbbg/env/test/bbbg-signup/env.clj
vendored
Normal file
3
users/grfn/bbbg/env/test/bbbg-signup/env.clj
vendored
Normal file
|
@ -0,0 +1,3 @@
|
|||
(ns bbbg.env)
|
||||
|
||||
(def environment :env/test)
|
11
users/grfn/bbbg/env/test/logback.xml
vendored
Normal file
11
users/grfn/bbbg/env/test/logback.xml
vendored
Normal file
|
@ -0,0 +1,11 @@
|
|||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<configuration>
|
||||
<appender name="CONSOLE" class="ch.qos.logback.core.ConsoleAppender">
|
||||
<encoder class="ch.qos.logback.classic.encoder.PatternLayoutEncoder">
|
||||
<pattern>%msg%n</pattern>
|
||||
</encoder>
|
||||
</appender>
|
||||
<root level="OFF">
|
||||
<appender-ref ref="CONSOLE"/>
|
||||
</root>
|
||||
</configuration>
|
42
users/grfn/bbbg/pom.xml
Normal file
42
users/grfn/bbbg/pom.xml
Normal file
|
@ -0,0 +1,42 @@
|
|||
<?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">
|
||||
<modelVersion>4.0.0</modelVersion>
|
||||
<groupId>fyi.gws</groupId>
|
||||
<artifactId>bbbg</artifactId>
|
||||
<version>0.1.0-SNAPSHOT</version>
|
||||
<name>fyi.gws/bbbg</name>
|
||||
<description>webhook listener for per-branch deploys</description>
|
||||
<url>https://bbbg.gws.fyi</url>
|
||||
<developers>
|
||||
<developer>
|
||||
<name>Griffin Smith</name>
|
||||
</developer>
|
||||
</developers>
|
||||
<dependencies>
|
||||
<dependency>
|
||||
<groupId>org.clojure</groupId>
|
||||
<artifactId>clojure</artifactId>
|
||||
<version>1.11.0-alpha3</version>
|
||||
</dependency>
|
||||
</dependencies>
|
||||
<build>
|
||||
<sourceDirectory>src</sourceDirectory>
|
||||
</build>
|
||||
<repositories>
|
||||
<repository>
|
||||
<id>clojars</id>
|
||||
<url>https://repo.clojars.org/</url>
|
||||
</repository>
|
||||
<repository>
|
||||
<id>sonatype</id>
|
||||
<url>https://oss.sonatype.org/content/repositories/snapshots/</url>
|
||||
</repository>
|
||||
</repositories>
|
||||
<distributionManagement>
|
||||
<repository>
|
||||
<id>clojars</id>
|
||||
<name>Clojars repository</name>
|
||||
<url>https://clojars.org/repo</url>
|
||||
</repository>
|
||||
</distributionManagement>
|
||||
</project>
|
49
users/grfn/bbbg/resources/main.js
Normal file
49
users/grfn/bbbg/resources/main.js
Normal file
|
@ -0,0 +1,49 @@
|
|||
window.onload = () => {
|
||||
console.log("loaded");
|
||||
const input = document.getElementById("name-autocomplete");
|
||||
if (input != null) {
|
||||
const eventID = document.getElementById("event-id").value;
|
||||
|
||||
const autocomplete = new autoComplete({
|
||||
selector: "#name-autocomplete",
|
||||
placeHolder: "Enter your name",
|
||||
data: {
|
||||
src: async (query) => {
|
||||
const resp = await fetch(
|
||||
`/attendees.json?q=${query}&event_id=${eventID}&attended=false`
|
||||
);
|
||||
console.log("got resp");
|
||||
const { results } = await resp.json();
|
||||
return results;
|
||||
},
|
||||
keys: ["bbbg.attendee/meetup-name"],
|
||||
},
|
||||
resultItem: {
|
||||
highlight: {
|
||||
render: true,
|
||||
},
|
||||
},
|
||||
});
|
||||
|
||||
input.addEventListener("selection", function (event) {
|
||||
const attendee = event.detail.selection.value;
|
||||
event.target.value = attendee["bbbg.attendee/meetup-name"];
|
||||
|
||||
const attendeeID = attendee["bbbg.attendee/id"];
|
||||
document.getElementById("attendee-id").value = attendeeID;
|
||||
document.getElementById("signup-form").removeAttribute("disabled");
|
||||
document
|
||||
.getElementById("signup-form")
|
||||
.querySelector('input[type="submit"]')
|
||||
.removeAttribute("disabled");
|
||||
});
|
||||
}
|
||||
|
||||
document.querySelectorAll("form").forEach((form) => {
|
||||
form.onsubmit = (e) => {
|
||||
if (e.target.attributes.disabled) {
|
||||
e.preventDefault();
|
||||
}
|
||||
};
|
||||
});
|
||||
};
|
|
@ -0,0 +1,14 @@
|
|||
drop table "public"."user";
|
||||
|
||||
-- ;;
|
||||
|
||||
drop table "public"."event_attendee";
|
||||
|
||||
|
||||
-- ;;
|
||||
|
||||
drop table "public"."event";
|
||||
|
||||
-- ;;
|
||||
|
||||
drop table "public"."attendee";
|
|
@ -0,0 +1,31 @@
|
|||
CREATE EXTENSION IF NOT EXISTS "uuid-ossp";
|
||||
-- ;;
|
||||
CREATE TABLE "attendee" (
|
||||
"id" UUID PRIMARY KEY NOT NULL DEFAULT uuid_generate_v4(),
|
||||
"meetup_name" TEXT NOT NULL,
|
||||
"discord_name" TEXT,
|
||||
"meetup_user_id" TEXT,
|
||||
"organizer_notes" TEXT NOT NULL DEFAULT '',
|
||||
"created_at" TIMESTAMP WITHOUT TIME ZONE NOT NULL DEFAULT now()
|
||||
);
|
||||
-- ;;
|
||||
CREATE TABLE "event" (
|
||||
"id" UUID PRIMARY KEY NOT NULL DEFAULT uuid_generate_v4(),
|
||||
"date" DATE NOT NULL,
|
||||
"created_at" TIMESTAMP WITHOUT TIME ZONE NOT NULL DEFAULT now()
|
||||
);
|
||||
-- ;;
|
||||
CREATE TABLE "event_attendee" (
|
||||
"event_id" UUID NOT NULL REFERENCES "event" ("id"),
|
||||
"attendee_id" UUID NOT NULL REFERENCES "attendee" ("id"),
|
||||
"rsvpd_attending" BOOL,
|
||||
"attended" BOOL,
|
||||
"created_at" TIMESTAMP WITHOUT TIME ZONE NOT NULL DEFAULT now(),
|
||||
PRIMARY KEY ("event_id", "attendee_id")
|
||||
);
|
||||
-- ;;
|
||||
CREATE TABLE "user" (
|
||||
"id" UUID PRIMARY KEY NOT NULL DEFAULT uuid_generate_v4(),
|
||||
"discord_user_id" TEXT NOT NULL,
|
||||
"created_at" TIMESTAMP WITHOUT TIME ZONE NOT NULL DEFAULT now()
|
||||
);
|
20
users/grfn/bbbg/shell.nix
Normal file
20
users/grfn/bbbg/shell.nix
Normal file
|
@ -0,0 +1,20 @@
|
|||
let
|
||||
depot = import ../../.. {};
|
||||
in
|
||||
with depot.third_party.nixpkgs;
|
||||
|
||||
mkShell {
|
||||
buildInputs = [
|
||||
arion
|
||||
depot.third_party.clj2nix
|
||||
clojure
|
||||
openjdk11_headless
|
||||
postgresql_12
|
||||
nix-prefetch-git
|
||||
];
|
||||
|
||||
PGHOST = "localhost";
|
||||
PGUSER = "bbbg";
|
||||
PGDATABASE = "bbbg";
|
||||
PGPASSWORD = "password";
|
||||
}
|
4
users/grfn/bbbg/src/bbbg/attendee.clj
Normal file
4
users/grfn/bbbg/src/bbbg/attendee.clj
Normal file
|
@ -0,0 +1,4 @@
|
|||
(ns bbbg.attendee
|
||||
(:require [clojure.spec.alpha :as s]))
|
||||
|
||||
(s/def ::id uuid?)
|
58
users/grfn/bbbg/src/bbbg/core.clj
Normal file
58
users/grfn/bbbg/src/bbbg/core.clj
Normal file
|
@ -0,0 +1,58 @@
|
|||
(ns bbbg.core
|
||||
(:gen-class)
|
||||
(:require
|
||||
[bbbg.db :as db]
|
||||
[bbbg.web :as web]
|
||||
[clojure.spec.alpha :as s]
|
||||
[clojure.spec.test.alpha :as stest]
|
||||
[com.stuartsierra.component :as component]
|
||||
[expound.alpha :as exp]))
|
||||
|
||||
(s/def ::config
|
||||
(s/merge
|
||||
::db/config
|
||||
::web/config))
|
||||
|
||||
(defn make-system [config]
|
||||
(component/system-map
|
||||
:db (db/make-database config)
|
||||
:web (web/make-server config)))
|
||||
|
||||
(defn env->config []
|
||||
(s/assert
|
||||
::config
|
||||
(merge
|
||||
(db/env->config)
|
||||
(web/env->config))))
|
||||
|
||||
(defn dev-config []
|
||||
(s/assert
|
||||
::config
|
||||
(merge
|
||||
(db/dev-config)
|
||||
(web/dev-config))))
|
||||
|
||||
(defonce system nil)
|
||||
|
||||
(defn init-dev []
|
||||
(s/check-asserts true)
|
||||
(set! s/*explain-out* exp/printer)
|
||||
(stest/instrument))
|
||||
|
||||
(defn run-dev []
|
||||
(init-dev)
|
||||
(alter-var-root
|
||||
#'system
|
||||
(fn [sys]
|
||||
(when sys
|
||||
(component/start sys))
|
||||
(component/start (make-system (dev-config))))))
|
||||
|
||||
(defn -main [& _args]
|
||||
(alter-var-root
|
||||
#'system
|
||||
(constantly (component/start (make-system (env->config))))))
|
||||
|
||||
(comment
|
||||
(run-dev)
|
||||
)
|
357
users/grfn/bbbg/src/bbbg/db.clj
Normal file
357
users/grfn/bbbg/src/bbbg/db.clj
Normal file
|
@ -0,0 +1,357 @@
|
|||
(ns bbbg.db
|
||||
(:gen-class)
|
||||
(:refer-clojure :exclude [get list])
|
||||
(:require [camel-snake-kebab.core :as csk :refer [->kebab-case ->snake_case]]
|
||||
[bbbg.util.core :as u]
|
||||
[clojure.set :as set]
|
||||
[clojure.spec.alpha :as s]
|
||||
[clojure.string :as str]
|
||||
[com.stuartsierra.component :as component]
|
||||
[config.core :refer [env]]
|
||||
[honeysql.format :as hformat]
|
||||
[migratus.core :as migratus]
|
||||
[next.jdbc :as jdbc]
|
||||
[next.jdbc.connection :as jdbc.conn]
|
||||
next.jdbc.date-time
|
||||
[next.jdbc.optional :as jdbc.opt]
|
||||
[next.jdbc.result-set :as rs]
|
||||
[next.jdbc.sql :as sql])
|
||||
(:import [com.impossibl.postgres.jdbc PGSQLSimpleException]
|
||||
com.zaxxer.hikari.HikariDataSource
|
||||
[java.sql Connection ResultSet Types]
|
||||
javax.sql.DataSource))
|
||||
|
||||
(s/def ::host string?)
|
||||
(s/def ::database string?)
|
||||
(s/def ::user string?)
|
||||
(s/def ::password string?)
|
||||
|
||||
(s/def ::config
|
||||
(s/keys :opt [::host
|
||||
::database
|
||||
::user
|
||||
::password]))
|
||||
|
||||
(s/fdef make-database
|
||||
:args
|
||||
(s/cat :config (s/keys :opt [::config])))
|
||||
|
||||
(s/fdef env->config :ret ::config)
|
||||
|
||||
(s/def ::db any?)
|
||||
|
||||
;;;
|
||||
|
||||
(def default-config
|
||||
(s/assert
|
||||
::config
|
||||
{::host "localhost"
|
||||
::database "bbbg"
|
||||
::user "bbbg"
|
||||
::password "password"}))
|
||||
|
||||
(defn dev-config [] default-config)
|
||||
|
||||
(defn env->config []
|
||||
(->>
|
||||
{::host (:pghost env)
|
||||
::database (:pgdatabase env)
|
||||
::user (:pguser env)
|
||||
::password (:pgpassword env)}
|
||||
u/remove-nils
|
||||
(s/assert ::config)))
|
||||
|
||||
(defn ->db-spec [config]
|
||||
(-> default-config
|
||||
(merge config)
|
||||
(set/rename-keys
|
||||
{::host :host
|
||||
::database :dbname
|
||||
::user :username
|
||||
::password :password})
|
||||
(assoc :dbtype "pgsql")))
|
||||
|
||||
(defn connection
|
||||
"Make a one-off connection from the given `::config` map, or the environment
|
||||
if not provided"
|
||||
([] (connection (env->config)))
|
||||
([config]
|
||||
(-> config
|
||||
->db-spec
|
||||
(set/rename-keys {:username :user})
|
||||
jdbc/get-datasource
|
||||
jdbc/get-connection)))
|
||||
|
||||
(defrecord Database [config]
|
||||
component/Lifecycle
|
||||
(start [this]
|
||||
(assoc this :pool (jdbc.conn/->pool HikariDataSource (->db-spec config))))
|
||||
(stop [this]
|
||||
(some-> this :pool .close)
|
||||
(dissoc this :pool))
|
||||
|
||||
clojure.lang.IFn
|
||||
(invoke [this] (:pool this)))
|
||||
|
||||
(defn make-database [config]
|
||||
(map->Database {:config config}))
|
||||
|
||||
;;;
|
||||
;;; Migrations
|
||||
;;;
|
||||
|
||||
(defn migratus-config
|
||||
[db]
|
||||
{:store :database
|
||||
:migration-dir "migrations/"
|
||||
:migration-table-name "__migrations__"
|
||||
:db
|
||||
(let [db (if (ifn? db) (db) db)]
|
||||
(cond
|
||||
(.isInstance Connection db)
|
||||
{:connection db}
|
||||
(.isInstance DataSource db)
|
||||
{:datasource db}
|
||||
:else (throw
|
||||
(ex-info "migratus-config called with value of unrecognized type"
|
||||
{:value db}))))})
|
||||
|
||||
(defn generate-migration
|
||||
([db name] (generate-migration db name :sql))
|
||||
([db name type] (migratus/create (migratus-config db) name type)))
|
||||
|
||||
(defn migrate!
|
||||
[db] (migratus/migrate (migratus-config db)))
|
||||
|
||||
(defn rollback!
|
||||
[db] (migratus/rollback (migratus-config db)))
|
||||
|
||||
;;;
|
||||
;;; Database interaction
|
||||
;;;
|
||||
|
||||
(defn ->key-ns [tn]
|
||||
(let [tn (name tn)
|
||||
tn (if (str/starts-with? tn "public.")
|
||||
(second (str/split tn #"\." 2))
|
||||
tn)]
|
||||
(str "bbbg." (->kebab-case tn))))
|
||||
|
||||
(defn ->table-name [kns]
|
||||
(let [kns (name kns)]
|
||||
(->snake_case
|
||||
(if (str/starts-with? kns "public.")
|
||||
kns
|
||||
(str "public." (last (str/split kns #"\.")))))))
|
||||
|
||||
(defn ->column
|
||||
([col] (->column nil col))
|
||||
([table col]
|
||||
(let [col-table (some-> col namespace ->table-name)
|
||||
snake-col (-> col name ->snake_case (str/replace #"\?$" ""))]
|
||||
(if (or (not (namespace col))
|
||||
(not table)
|
||||
(= (->table-name table) col-table))
|
||||
snake-col
|
||||
;; different table, assume fk
|
||||
(str
|
||||
(str/replace-first col-table "public." "")
|
||||
"_"
|
||||
snake-col)))))
|
||||
|
||||
(defn ->value [v]
|
||||
(if (keyword? v)
|
||||
(-> v name csk/->snake_case_string)
|
||||
v))
|
||||
|
||||
(defn process-key-map [table key-map]
|
||||
(into {}
|
||||
(map (fn [[k v]] [(->column table k)
|
||||
(->value v)]))
|
||||
key-map))
|
||||
|
||||
(defn fkize [col]
|
||||
(if (str/ends-with? col "-id")
|
||||
(let [table (str/join "-" (butlast (str/split (name col) #"-")))]
|
||||
(keyword (->key-ns table) "id"))
|
||||
col))
|
||||
|
||||
(def ^:private enum-members-cache (atom {}))
|
||||
(defn- enum-members
|
||||
"Returns a set of enum members as strings for the enum with the given name"
|
||||
[db name]
|
||||
(if-let [e (find @enum-members-cache name)]
|
||||
(val e)
|
||||
(let [r (try
|
||||
(-> (jdbc/execute-one!
|
||||
(db)
|
||||
[(format "select enum_range(null::%s) as members" name)])
|
||||
:members
|
||||
.getArray
|
||||
set)
|
||||
(catch PGSQLSimpleException _
|
||||
nil))]
|
||||
(swap! enum-members-cache assoc name r)
|
||||
r)))
|
||||
|
||||
(def ^{:private true
|
||||
:dynamic true}
|
||||
*meta-db*
|
||||
"Database connection to use to query metadata"
|
||||
nil)
|
||||
|
||||
(extend-protocol rs/ReadableColumn
|
||||
String
|
||||
(read-column-by-label [x _] x)
|
||||
(read-column-by-index [x rsmeta idx]
|
||||
(if-not *meta-db*
|
||||
x
|
||||
(let [typ (.getColumnTypeName rsmeta idx)]
|
||||
;; TODO: Is there a better way to figure out if a type is an enum?
|
||||
(if (enum-members *meta-db* typ)
|
||||
(keyword (csk/->kebab-case-string typ)
|
||||
(csk/->kebab-case-string x))
|
||||
x)))))
|
||||
|
||||
(comment
|
||||
(->key-ns :public.user)
|
||||
(->key-ns :public.api-token)
|
||||
(->key-ns :api-token)
|
||||
(->table-name :api-token)
|
||||
(->table-name :public.user)
|
||||
(->table-name :bbbg.user)
|
||||
)
|
||||
|
||||
(defn as-fq-maps [^ResultSet rs _opts]
|
||||
(let [qualify #(when (seq %) (str "bbbg." (->kebab-case %)))
|
||||
rsmeta (.getMetaData rs)
|
||||
cols (mapv
|
||||
(fn [^Integer i]
|
||||
(let [ty (.getColumnType rsmeta i)
|
||||
lab (.getColumnLabel rsmeta i)
|
||||
n (str (->kebab-case lab)
|
||||
(when (= ty Types/BOOLEAN) "?"))]
|
||||
(fkize
|
||||
(if-let [q (some-> rsmeta (.getTableName i) qualify not-empty)]
|
||||
(keyword q n)
|
||||
(keyword n)))))
|
||||
(range 1 (inc (.getColumnCount rsmeta))))]
|
||||
(jdbc.opt/->MapResultSetOptionalBuilder rs rsmeta cols)))
|
||||
|
||||
(def jdbc-opts
|
||||
{:builder-fn as-fq-maps
|
||||
:column-fn ->snake_case
|
||||
:table-fn ->snake_case})
|
||||
|
||||
(defmethod hformat/fn-handler "count-distinct" [_ field]
|
||||
(str "count(distinct " (hformat/to-sql field) ")"))
|
||||
|
||||
(defn fetch
|
||||
"Fetch a single row from the db matching the given `sql-map` or query"
|
||||
[db sql-map & [opts]]
|
||||
(s/assert
|
||||
(s/nilable (s/keys))
|
||||
(binding [*meta-db* db]
|
||||
(jdbc/execute-one!
|
||||
(db)
|
||||
(if (map? sql-map)
|
||||
(hformat/format sql-map)
|
||||
sql-map)
|
||||
(merge jdbc-opts opts)))))
|
||||
|
||||
(defn get
|
||||
"Retrieve a single record from the given table by ID"
|
||||
[db table id & [opts]]
|
||||
(when id
|
||||
(fetch
|
||||
db
|
||||
{:select [:*]
|
||||
:from [table]
|
||||
:where [:= :id id]}
|
||||
opts)))
|
||||
|
||||
(defn list
|
||||
"Returns a list of rows from the db matching the given sql-map, table or
|
||||
query"
|
||||
[db sql-map-or-table & [opts]]
|
||||
(s/assert
|
||||
(s/coll-of (s/keys))
|
||||
(binding [*meta-db* db]
|
||||
(jdbc/execute!
|
||||
(db)
|
||||
(cond
|
||||
(map? sql-map-or-table)
|
||||
(hformat/format sql-map-or-table)
|
||||
(keyword? sql-map-or-table)
|
||||
(hformat/format {:select [:*] :from [sql-map-or-table]})
|
||||
:else
|
||||
sql-map-or-table)
|
||||
(merge jdbc-opts opts)))))
|
||||
|
||||
(defn exists?
|
||||
"Returns true if the given sql query-map would return any results"
|
||||
[db sql-map]
|
||||
(binding [*meta-db* db]
|
||||
(pos?
|
||||
(:count
|
||||
(fetch db {:select [[:%count.* :count]], :from [[sql-map :sq]]})))))
|
||||
|
||||
(defn execute!
|
||||
"Given a database and a honeysql query map, perform an operation on the
|
||||
database and discard the results"
|
||||
[db sql-map & [opts]]
|
||||
(jdbc/execute!
|
||||
(db)
|
||||
(hformat/format sql-map)
|
||||
(merge jdbc-opts opts)))
|
||||
|
||||
(defn insert!
|
||||
"Given a database, a table name, and a data hash map, inserts the
|
||||
data as a single row in the database and attempts to return a map of generated
|
||||
keys."
|
||||
[db table key-map & [opts]]
|
||||
(binding [*meta-db* db]
|
||||
(sql/insert!
|
||||
(db)
|
||||
table
|
||||
(process-key-map table key-map)
|
||||
(merge jdbc-opts opts))))
|
||||
|
||||
(defn update!
|
||||
"Given a database, a table name, a hash map of columns and values
|
||||
to set, and a honeysql predicate, perform an update on the table.
|
||||
Will "
|
||||
[db table key-map where-params & [opts]]
|
||||
(binding [*meta-db* db]
|
||||
(execute! db
|
||||
{:update table
|
||||
:set (u/map-keys keyword (process-key-map table key-map))
|
||||
:where where-params
|
||||
:returning [:id]}
|
||||
opts)))
|
||||
|
||||
(defn delete!
|
||||
"Delete all rows from the given table matching the given where clause"
|
||||
[db table where-clause]
|
||||
(binding [*meta-db* db]
|
||||
(sql/delete! (db) table (hformat/format-predicate where-clause))))
|
||||
|
||||
(defmacro with-transaction [[sym db opts] & body]
|
||||
`(jdbc/with-transaction
|
||||
[tx# (~db) ~opts]
|
||||
(let [~sym (constantly tx#)]
|
||||
~@body)))
|
||||
|
||||
(defn -main [& args]
|
||||
(let [db (component/start (make-database {::config (env->config)}))]
|
||||
(case (first args)
|
||||
"migrate" (migrate! db)
|
||||
"rollback" (rollback! db))))
|
||||
|
||||
(comment
|
||||
(def db (:db bbbg.core/system))
|
||||
(generate-migration db "init-schema")
|
||||
(migrate! db)
|
||||
|
||||
|
||||
)
|
29
users/grfn/bbbg/src/bbbg/db/attendee.clj
Normal file
29
users/grfn/bbbg/src/bbbg/db/attendee.clj
Normal file
|
@ -0,0 +1,29 @@
|
|||
(ns bbbg.db.attendee
|
||||
(:require
|
||||
[bbbg.db :as db]
|
||||
honeysql-postgres.helpers
|
||||
[honeysql.helpers :refer [merge-join merge-where]]))
|
||||
|
||||
(defn search
|
||||
([query]
|
||||
(cond->
|
||||
{:select [:attendee.*]
|
||||
:from [:attendee]}
|
||||
query
|
||||
(assoc
|
||||
:where [:or
|
||||
[:ilike :meetup_name (str "%" query "%")]
|
||||
[:ilike :discord_name (str "%" query "%")]])))
|
||||
([db query]
|
||||
(db/list db (search query))))
|
||||
|
||||
(defn for-event
|
||||
([query event-id]
|
||||
(-> query
|
||||
(merge-join :event_attendee [:= :attendee.id :event_attendee.attendee_id])
|
||||
(merge-where [:= :event_attendee.event_id event-id]))))
|
||||
|
||||
(comment
|
||||
(def db (:db bbbg.core/system))
|
||||
(search db "gri")
|
||||
)
|
50
users/grfn/bbbg/src/bbbg/db/event.clj
Normal file
50
users/grfn/bbbg/src/bbbg/db/event.clj
Normal file
|
@ -0,0 +1,50 @@
|
|||
(ns bbbg.db.event
|
||||
(:require
|
||||
[bbbg.attendee :as attendee]
|
||||
[bbbg.db :as db]
|
||||
[bbbg.event :as event]
|
||||
[honeysql.helpers :refer [merge-group-by merge-join merge-select]]
|
||||
[java-time :refer [local-date]]))
|
||||
|
||||
(defn create! [db event]
|
||||
(db/insert! db :event (select-keys event [::event/date])))
|
||||
|
||||
(defn attended!
|
||||
[db params]
|
||||
(db/execute!
|
||||
db
|
||||
{:insert-into :event-attendee
|
||||
:values [{:event_id (::event/id params)
|
||||
:attendee_id (::attendee/id params)
|
||||
:attended true}]
|
||||
:upsert {:on-conflict [:event-id :attendee-id]
|
||||
:do-update-set! {:attended true}}}))
|
||||
|
||||
(defn on-day
|
||||
([day] {:select [:event.*]
|
||||
:from [:event]
|
||||
:where [:= :date (str day)]})
|
||||
([db day]
|
||||
(db/list db (on-day day))))
|
||||
|
||||
(defn today
|
||||
([] (on-day (local-date)))
|
||||
([db] (db/list db (today))))
|
||||
|
||||
(defn with-attendee-counts
|
||||
[query]
|
||||
(-> query
|
||||
(merge-join :event_attendee [:= :event.id :event_attendee.event-id])
|
||||
(merge-select :%count.event_attendee.attendee_id)
|
||||
(merge-group-by :event.id :event_attendee.event-id)))
|
||||
|
||||
(comment
|
||||
(def db (:db bbbg.core/system))
|
||||
(db/list db (-> (today) (with-attendee-counts)))
|
||||
|
||||
(honeysql.format/format
|
||||
(honeysql-postgres.helpers/upsert {:insert-into :foo
|
||||
:values {:bar 1}}
|
||||
(-> (honeysql-postgres.helpers/on-conflict :did)
|
||||
(honeysql-postgres.helpers/do-update-set! [:did true]))))
|
||||
)
|
4
users/grfn/bbbg/src/bbbg/event.clj
Normal file
4
users/grfn/bbbg/src/bbbg/event.clj
Normal file
|
@ -0,0 +1,4 @@
|
|||
(ns bbbg.event
|
||||
(:require [clojure.spec.alpha :as s]))
|
||||
|
||||
(s/def ::id uuid?)
|
4
users/grfn/bbbg/src/bbbg/event_attendee.clj
Normal file
4
users/grfn/bbbg/src/bbbg/event_attendee.clj
Normal file
|
@ -0,0 +1,4 @@
|
|||
(ns bbbg.event-attendee
|
||||
(:require [clojure.spec.alpha :as s]))
|
||||
|
||||
(s/def ::attended? boolean?)
|
40
users/grfn/bbbg/src/bbbg/handlers/attendees.clj
Normal file
40
users/grfn/bbbg/src/bbbg/handlers/attendees.clj
Normal file
|
@ -0,0 +1,40 @@
|
|||
(ns bbbg.handlers.attendees
|
||||
(:require
|
||||
[bbbg.attendee :as attendee]
|
||||
[bbbg.db :as db]
|
||||
[bbbg.db.attendee :as db.attendee]
|
||||
[bbbg.db.event :as db.event]
|
||||
[bbbg.event :as event]
|
||||
[cheshire.core :as json]
|
||||
[compojure.core :refer [GET POST routes]]
|
||||
[honeysql.helpers :refer [merge-where]]
|
||||
[ring.util.response :refer [content-type redirect response]]))
|
||||
|
||||
(defn attendees-routes [{:keys [db]}]
|
||||
(routes
|
||||
(GET "/attendees.json" [q event_id attended]
|
||||
(let [results
|
||||
(db/list
|
||||
db
|
||||
(cond->
|
||||
(if q
|
||||
(db.attendee/search q)
|
||||
{:select [:attendee.*] :from [:attendee]})
|
||||
event_id (db.attendee/for-event event_id)
|
||||
(some? attended) (merge-where [:= :attended (case attended
|
||||
"true" true
|
||||
"false" false)])))]
|
||||
(-> {:results results}
|
||||
json/generate-string
|
||||
response
|
||||
(content-type "application/json"))))
|
||||
|
||||
(POST "/event_attendees" [event_id attendee_id]
|
||||
(if (and (db/exists? db {:select [:id] :from [:event] :where [:= :id event_id]})
|
||||
(db/exists? db {:select [:id] :from [:attendee] :where [:= :id attendee_id]}))
|
||||
(do
|
||||
(db.event/attended! db {::event/id event_id
|
||||
::attendee/id attendee_id})
|
||||
(-> (redirect (str "/signup-forms/" event_id))
|
||||
(assoc :flash "Thank you for signing in! Enjoy the event.")))
|
||||
(response "Something went wrong")))))
|
34
users/grfn/bbbg/src/bbbg/handlers/core.clj
Normal file
34
users/grfn/bbbg/src/bbbg/handlers/core.clj
Normal file
|
@ -0,0 +1,34 @@
|
|||
(ns bbbg.handlers.core
|
||||
(:require
|
||||
[hiccup.core :refer [html]]
|
||||
[ring.util.response :refer [content-type response]]))
|
||||
|
||||
(defn render-page [opts & body]
|
||||
(let [[{:keys [title]} body]
|
||||
(if (map? opts)
|
||||
[opts body]
|
||||
[{} (into [opts] body)])]
|
||||
(html
|
||||
[:html {:lang "en"}
|
||||
[:head
|
||||
[:meta {:charset "UTF-8"}]
|
||||
[:title (if title
|
||||
(str title " - BBBG")
|
||||
"BBBG")]
|
||||
[:link {:rel "stylesheet"
|
||||
:type "text/css"
|
||||
:href "/main.css"}]]
|
||||
[:body
|
||||
(into [:div.content] body)
|
||||
[:script {:src "https://cdnjs.cloudflare.com/ajax/libs/tarekraafat-autocomplete.js/10.2.6/autoComplete.js"}]
|
||||
[:script {:src "/main.js"}]]])))
|
||||
|
||||
(defn page-response [& render-page-args]
|
||||
(-> (apply render-page render-page-args)
|
||||
response
|
||||
(content-type "text/html")))
|
||||
|
||||
(comment
|
||||
(render-page
|
||||
[:h1 "hi"])
|
||||
)
|
44
users/grfn/bbbg/src/bbbg/handlers/events.clj
Normal file
44
users/grfn/bbbg/src/bbbg/handlers/events.clj
Normal file
|
@ -0,0 +1,44 @@
|
|||
(ns bbbg.handlers.events
|
||||
(:require
|
||||
[bbbg.db :as db]
|
||||
[bbbg.db.event :as db.event]
|
||||
[bbbg.event :as event]
|
||||
[bbbg.handlers.core :refer [page-response]]
|
||||
[compojure.core :refer [context GET POST]]
|
||||
[ring.util.response :refer [redirect]]))
|
||||
|
||||
(defn events-index [events]
|
||||
[:ul.events-list
|
||||
(for [event events]
|
||||
[:li (::event/date event)])])
|
||||
|
||||
(defn event-form
|
||||
([] (event-form {}))
|
||||
([event]
|
||||
[:form {:method "POST" :action "/events"}
|
||||
[:div.form-group
|
||||
[:label "Date"
|
||||
[:input {:type "date"
|
||||
:id "date"
|
||||
:name "date"
|
||||
:value (str (::event/date event))}]]]
|
||||
[:div.form-group
|
||||
[:input {:type "submit"
|
||||
:value "Create Event"}]]]))
|
||||
|
||||
(defn events-routes [{:keys [db]}]
|
||||
(context "/events" []
|
||||
(GET "/" []
|
||||
(let [events (db/list db :event)]
|
||||
(events-index events)))
|
||||
|
||||
(GET "/new" [date]
|
||||
(page-response
|
||||
{:title "New Event"}
|
||||
(event-form {::event/date date})))
|
||||
|
||||
(POST "/" [date]
|
||||
(let [event (db.event/create! db {::event/date date})]
|
||||
(-> (str "/signup-forms/" (::event/id event))
|
||||
redirect
|
||||
(assoc-in [:flash :message] "Event Created"))))))
|
17
users/grfn/bbbg/src/bbbg/handlers/home.clj
Normal file
17
users/grfn/bbbg/src/bbbg/handlers/home.clj
Normal file
|
@ -0,0 +1,17 @@
|
|||
(ns bbbg.handlers.home
|
||||
(:require
|
||||
[bbbg.handlers.core :refer [page-response]]
|
||||
[compojure.core :refer [GET routes]]))
|
||||
|
||||
(defn- home-page []
|
||||
[:nav.home-nav
|
||||
[:ul
|
||||
[:li [:a {:href "/signup-forms"}
|
||||
"Event Signup Form"]]
|
||||
[:li [:a {:href "/login"}
|
||||
"Sign In"]]]])
|
||||
|
||||
(defn home-routes [_env]
|
||||
(routes
|
||||
(GET "/" []
|
||||
(page-response (home-page)))))
|
57
users/grfn/bbbg/src/bbbg/handlers/signup_form.clj
Normal file
57
users/grfn/bbbg/src/bbbg/handlers/signup_form.clj
Normal file
|
@ -0,0 +1,57 @@
|
|||
(ns bbbg.handlers.signup-form
|
||||
(:require
|
||||
[bbbg.db :as db]
|
||||
[bbbg.db.event :as db.event]
|
||||
[bbbg.event :as event]
|
||||
[bbbg.handlers.core :refer [page-response]]
|
||||
[compojure.core :refer [GET context]]
|
||||
[java-time :refer [local-date]]
|
||||
[ring.util.response :refer [redirect]]))
|
||||
|
||||
(defn no-events-page []
|
||||
[:div.no-events
|
||||
[:p
|
||||
"There are no events for today"]
|
||||
[:p
|
||||
[:a {:href (str "/events/new?date=" (str (local-date)))} "Create Event"]
|
||||
[:a {:href "/events"} "All Events"]]])
|
||||
|
||||
(defn signup-page [event]
|
||||
[:div.signup-page
|
||||
[:form#signup-form
|
||||
{:method "POST"
|
||||
:action "/event_attendees"
|
||||
:disabled "disabled"}
|
||||
[:input#event-id {:type "hidden" :name "event_id" :value (::event/id event)}]
|
||||
[:input#attendee-id {:type "hidden" :name "attendee_id"}]
|
||||
[:label "Name"
|
||||
[:input#name-autocomplete
|
||||
{:type "search"
|
||||
:name "name"
|
||||
:spellcheck "false"
|
||||
:autocorrect "off"
|
||||
:autocomplete "off"
|
||||
:autocapitalize "off"
|
||||
:maxlength "2048"}]]
|
||||
[:input {:type "submit"
|
||||
:value "Sign In"
|
||||
:disabled "disabled"}]]])
|
||||
|
||||
(defn event-not-found []
|
||||
[:div.event-not-found
|
||||
[:p "Event not found"]
|
||||
[:p [:a {:href (str "/events/new")} "Create a new event"]]])
|
||||
|
||||
;;;
|
||||
|
||||
(defn signup-form-routes [{:keys [db]}]
|
||||
(context "/signup-forms" []
|
||||
(GET "/" []
|
||||
(if-let [event (db/fetch db (db.event/today))]
|
||||
(redirect (str "/signup-forms/" (::event/id event)))
|
||||
(page-response (no-events-page))))
|
||||
|
||||
(GET "/:event-id" [event-id]
|
||||
(if-let [event (db/get db :event event-id)]
|
||||
(page-response (signup-page event))
|
||||
(event-not-found)))))
|
9
users/grfn/bbbg/src/bbbg/styles.clj
Normal file
9
users/grfn/bbbg/src/bbbg/styles.clj
Normal file
|
@ -0,0 +1,9 @@
|
|||
(ns bbbg.styles
|
||||
(:require [garden.def :refer [defstyles]]
|
||||
[garden.compiler :refer [compile-css]]))
|
||||
|
||||
(defstyles styles
|
||||
)
|
||||
|
||||
(def stylesheet
|
||||
(compile-css styles))
|
117
users/grfn/bbbg/src/bbbg/util/core.clj
Normal file
117
users/grfn/bbbg/src/bbbg/util/core.clj
Normal file
|
@ -0,0 +1,117 @@
|
|||
(ns bbbg.util.core
|
||||
(:import java.util.UUID))
|
||||
|
||||
(defn remove-nils
|
||||
"Remove all keys with nil values from m"
|
||||
[m]
|
||||
(let [!m (transient m)]
|
||||
(doseq [[k v] m]
|
||||
(when (nil? v)
|
||||
(dissoc! !m k)))
|
||||
(persistent! !m)))
|
||||
|
||||
|
||||
(defn alongside
|
||||
"Apply a pair of functions to the first and second element of a two element
|
||||
vector, respectively. The two argument form partially applies, such that:
|
||||
|
||||
((alongside f g) xy) ≡ (alongside f g xy)
|
||||
|
||||
This is equivalent to (***) in haskell's Control.Arrow"
|
||||
([f g] (partial alongside f g))
|
||||
([f g [x y]] [(f x) (g y)]))
|
||||
|
||||
(defn map-kv
|
||||
"Map a pair of functions over the keys and values of a map, respectively.
|
||||
Preserves metadata on the incoming map.
|
||||
The two argument form returns a transducer that yields map-entries.
|
||||
|
||||
(partial map-kv identity identity) ≡ identity"
|
||||
([kf vf]
|
||||
(map (fn [[k v]]
|
||||
;; important to return a map-entry here so that callers down the road
|
||||
;; can use `key` or `val`
|
||||
(first {(kf k) (vf v)}))))
|
||||
([kf vf m]
|
||||
(into (empty m) (map-kv kf vf) m)))
|
||||
|
||||
(defn filter-kv
|
||||
"Returns a map containing the elements of m for which (f k v) returns logical
|
||||
true. The one-argument form returns a transducer that yields map entries"
|
||||
([f] (filter (partial apply f)))
|
||||
([f m]
|
||||
(into (empty m) (filter-kv f) m)))
|
||||
|
||||
(defn map-keys
|
||||
"Map f over the keys of m. Preserves metadata on the incoming map. The
|
||||
one-argument form returns a transducer that yields map-entries."
|
||||
([f] (map-kv f identity))
|
||||
([f m] (map-kv f identity m)))
|
||||
|
||||
(defn map-vals
|
||||
"Map f over the values of m. Preserves metadata on the incoming map. The
|
||||
one-argument form returns a transducer that yields map-entries."
|
||||
([f] (map-kv identity f))
|
||||
([f m] (map-kv identity f m)))
|
||||
|
||||
(defn map-keys-recursive [f x]
|
||||
(cond
|
||||
(map? x) (map-kv f (partial map-keys-recursive f) x)
|
||||
(sequential? x) (map (partial map-keys-recursive f) x)
|
||||
:else x))
|
||||
|
||||
(defn denamespace [x]
|
||||
(if (keyword? x)
|
||||
(keyword (name x))
|
||||
(map-keys-recursive denamespace x)))
|
||||
|
||||
(defn reverse-merge
|
||||
"Like `clojure.core/merge`, except duplicate keys from maps earlier in the
|
||||
argument list take precedence
|
||||
|
||||
=> (merge {:x 1} {:x 2})
|
||||
{:x 2}
|
||||
|
||||
=> (sut/reverse-merge {:x 1} {:x 2})
|
||||
{:x 1}"
|
||||
[& ms]
|
||||
(apply merge (reverse ms)))
|
||||
|
||||
(defn invert-map
|
||||
"Invert the keys and vals of m. Behavior with duplicate vals is undefined.
|
||||
|
||||
=> (sut/invert-map {:x 1 :y 2})
|
||||
{1 :x 2 :y}"
|
||||
[m]
|
||||
(into {} (map (comp vec reverse)) m))
|
||||
|
||||
(defn ->uuid
|
||||
"Converts x to uuid, returning nil if x is nil or empty"
|
||||
[x]
|
||||
(cond
|
||||
(not x) nil
|
||||
(uuid? x) x
|
||||
(and (string? x) (seq x))
|
||||
(UUID/fromString x)))
|
||||
|
||||
(defn key-by
|
||||
"Create a map from a seq obtaining keys via f
|
||||
|
||||
=> (sut/key-by :x [{:x 1} {:x 2 :y 3}])
|
||||
{1 {:x 1}, 2 {:x 2 :y 3}}"
|
||||
[f l]
|
||||
(into {} (map (juxt f identity)) l))
|
||||
|
||||
(defn distinct-by
|
||||
"Like clojure.core/distinct, but can take a function f by which
|
||||
distinctiveness is calculated"
|
||||
[distinction-fn coll]
|
||||
(let [step (fn step [xs seen]
|
||||
(lazy-seq
|
||||
((fn [[f :as xs] seen]
|
||||
(when-let [s (seq xs)]
|
||||
(if (contains? seen (distinction-fn f))
|
||||
(recur (rest s) seen)
|
||||
(cons f (step (rest s) (conj seen (distinction-fn f)))))))
|
||||
xs seen)))]
|
||||
(step coll #{})))
|
77
users/grfn/bbbg/src/bbbg/web.clj
Normal file
77
users/grfn/bbbg/src/bbbg/web.clj
Normal file
|
@ -0,0 +1,77 @@
|
|||
(ns bbbg.web
|
||||
(:require
|
||||
[bbbg.handlers.attendees :as attendees]
|
||||
[bbbg.handlers.events :as events]
|
||||
[bbbg.handlers.home :as home]
|
||||
[bbbg.handlers.signup-form :as signup-form]
|
||||
[bbbg.styles :refer [stylesheet]]
|
||||
[clojure.spec.alpha :as s]
|
||||
[com.stuartsierra.component :as component]
|
||||
[compojure.core :refer [GET routes]]
|
||||
[config.core :refer [env]]
|
||||
[org.httpkit.server :as http-kit]
|
||||
[ring.middleware.flash :refer [wrap-flash]]
|
||||
[ring.middleware.keyword-params :refer [wrap-keyword-params]]
|
||||
[ring.middleware.params :refer [wrap-params]]
|
||||
[ring.util.response :refer [content-type response resource-response]]))
|
||||
|
||||
(s/def ::port pos-int?)
|
||||
|
||||
(s/def ::config
|
||||
(s/keys :req [::port]))
|
||||
|
||||
(s/fdef make-server
|
||||
:args (s/cat :config ::config))
|
||||
|
||||
(defn env->config []
|
||||
(s/assert
|
||||
::config
|
||||
{::port (:port env 8888)}))
|
||||
|
||||
(defn dev-config []
|
||||
(s/assert ::config {::port 8888}))
|
||||
|
||||
;;;
|
||||
|
||||
(defn app-routes [env]
|
||||
(routes
|
||||
(GET "/main.css" []
|
||||
(-> (response stylesheet)
|
||||
(content-type "text/css")))
|
||||
(GET "/main.js" []
|
||||
(-> (resource-response "main.js")
|
||||
(content-type "text/javascript")))
|
||||
|
||||
(attendees/attendees-routes env)
|
||||
(signup-form/signup-form-routes env)
|
||||
(events/events-routes env)
|
||||
(home/home-routes env)))
|
||||
|
||||
(defn middleware [app]
|
||||
(-> app
|
||||
wrap-keyword-params
|
||||
wrap-params
|
||||
wrap-flash))
|
||||
|
||||
(defn handler [this]
|
||||
(middleware
|
||||
(app-routes this)))
|
||||
|
||||
(defrecord WebServer [port db]
|
||||
component/Lifecycle
|
||||
(start [this]
|
||||
(assoc this
|
||||
::shutdown-fn
|
||||
(http-kit/run-server
|
||||
(fn [r] ((handler this) r))
|
||||
{:port port})))
|
||||
(stop [this]
|
||||
(if-let [shutdown-fn (::shutdown-fn this)]
|
||||
(do (shutdown-fn :timeout 100)
|
||||
(dissoc this ::shutdown-fn))
|
||||
this)))
|
||||
|
||||
(defn make-server [{::keys [port]}]
|
||||
(component/using
|
||||
(map->WebServer {:port port})
|
||||
[:db]))
|
Loading…
Reference in a new issue