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:
Griffin Smith 2021-12-13 21:28:25 -05:00
parent 479e9ea279
commit c3cb7b0df8
35 changed files with 2549 additions and 0 deletions

View file

@ -0,0 +1 @@
{:lint-as {garden.def/defstyles clojure.core/def}}

1
users/grfn/bbbg/.envrc Normal file
View file

@ -0,0 +1 @@
eval "$(lorri direnv)"

9
users/grfn/bbbg/.gitignore vendored Normal file
View file

@ -0,0 +1,9 @@
/target
/classes
*.jar
*.class
/.nrepl-port
/.cpcache
/.clojure
/result
/.clj-kondo/.cache

2
users/grfn/bbbg/Makefile Normal file
View file

@ -0,0 +1,2 @@
deps.nix: deps.edn
clj2nix ./deps.edn ./deps.nix '-A:uberjar' '-A:clj-test'

View file

@ -0,0 +1,15 @@
{ ... }:
{
services = {
postgres.service = {
image = "postgres:12";
environment = {
POSTGRES_DB = "bbbg";
POSTGRES_USER = "bbbg";
POSTGRES_PASSWORD = "password";
};
ports = [ "5432:5432" ];
};
};
}

View file

@ -0,0 +1,2 @@
let depot = import ../../.. {};
in depot.third_party.nixpkgs

View 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
View 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

File diff suppressed because it is too large Load diff

View file

@ -0,0 +1,3 @@
(ns bbbg.env)
(def environment :env/dev)

15
users/grfn/bbbg/env/dev/logback.xml vendored Normal file
View 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>

View file

@ -0,0 +1,3 @@
(ns bbbg.env)
(def environment :env/prod)

31
users/grfn/bbbg/env/prod/logback.xml vendored Normal file
View 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>

View file

@ -0,0 +1,3 @@
(ns bbbg.env)
(def environment :env/test)

11
users/grfn/bbbg/env/test/logback.xml vendored Normal file
View 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
View 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>

View 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();
}
};
});
};

View file

@ -0,0 +1,14 @@
drop table "public"."user";
-- ;;
drop table "public"."event_attendee";
-- ;;
drop table "public"."event";
-- ;;
drop table "public"."attendee";

View file

@ -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
View 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";
}

View file

@ -0,0 +1,4 @@
(ns bbbg.attendee
(:require [clojure.spec.alpha :as s]))
(s/def ::id uuid?)

View 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)
)

View 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)
)

View 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")
)

View 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]))))
)

View file

@ -0,0 +1,4 @@
(ns bbbg.event
(:require [clojure.spec.alpha :as s]))
(s/def ::id uuid?)

View file

@ -0,0 +1,4 @@
(ns bbbg.event-attendee
(:require [clojure.spec.alpha :as s]))
(s/def ::attended? boolean?)

View 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")))))

View 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"])
)

View 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"))))))

View 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)))))

View 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)))))

View 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))

View 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 #{})))

View 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]))