mkdir atemoia
cd atemoia
echo {} > deps.edn
mkdir dev src test
git init .
git add deps.edn
git commit -am"atemoia"
Add your deps in deps.edn
{:paths ["src"]
:deps {com.github.seancorfield/next.jdbc {:mvn/version "1.3.883"}
hiccup/hiccup {:mvn/version "2.0.0-RC1"}
io.pedestal/pedestal.jetty {:mvn/version "0.6.0"}
io.pedestal/pedestal.service {:mvn/version "0.6.0"}
org.clojure/clojure {:mvn/version "1.11.1"}
org.postgresql/postgresql {:mvn/version "42.6.0"}
org.slf4j/slf4j-simple {:mvn/version "2.0.7"}}
:aliases {:test-runner {:extra-deps {io.github.cognitect-labs/test-runner {:git/url "https://github.com/cognitect-labs/test-runner.git"
:git/sha "dfb30dd6605cb6c0efc275e1df1736f6e90d4d73"}}
:main-opts ["-m" "cognitect.test-runner"]}
:dev {:extra-paths ["dev" "test"]
:jvm-opts ["-XX:-OmitStackTraceInFastThrow"
"-Dclojure.core.async.go-checking=true"
"-Dclojure.main.report=stderr"]
:extra-deps {com.google.guava/guava {:mvn/version "32.1.1-jre"}
io.github.clojure/tools.build {:mvn/version "0.9.4"}
com.h2database/h2 {:mvn/version "2.2.220"}
reagent/reagent {:mvn/version "1.2.0"}
thheller/shadow-cljs {:mvn/version "2.24.1"}}}}}
git commit -am"Add deps in deps.edn file"
start the REPL:
clj -M:dev
mkdir src/atemoia
echo '(ns atemoia.server)' > src/atemoia/server.clj
git add src/atemoia/server.clj
git commit -am"Creating server namespace"
Let's add an minimum http server in server.clj
(ns atemoia.server
(:require [io.pedestal.http :as http]))
(defn index
[_]
{:body "Hello"
:status 200})
(def routes
`#{["/" :get index]})
(defonce *state
(atom nil))
(defn -main
[& _]
(swap! state
(fn [st]
(some-> st http/stop)
(-> {::http/port 8080
::http/type :jetty
::http/routes routes
::http/host "0.0.0.0"
::http/join? false}
http/default-interceptors
http/dev-interceptors
http/create-server
http/start))))
(comment
(-main))
git commit -am"Minimal HTTP server"
load and start the HTTP server in the REPL
(require 'atemoia.server :reload)
(atemoia.server/-main)
connect to localhost:8080
Add hiccup
server.clj
(ns atemoia.server
(:require [io.pedestal.http :as http]
[hiccup2.core :as h]))
(defn index
[_]
(let [html [:html
{:lang "en"}
[:head
[:meta {:charset "UTF-8"}]
[:title "atemoia"]]
[:body
[:div "hello"]]]]
{:body (->> html
(h/html {:mode :html})
(str "<!DOCTYPE html>\n"))
:headers {"Content-Security-Policy" ""
"Content-Type" "text/html"}
:status 200}))
run in the REPL
(require 'atemoia.server :reload)
(atemoia.server/-main)
connect to localhost:8080
echo {} > package.json
echo {} > shadow-cljs.edn
echo '(ns atemoia.client)' > src/atemoia/client.cljs
npm install --save react react-dom --save-dev [email protected]
## git add package.json shadow-cljs.edn package-lock.json src/atemoia/client.cljs
shadow-cljs.edn
{:deps {:aliases [:dev]}
:builds {:atemoia {:target :browser
:output-dir "target/classes/public/atemoia"
:asset-path "/atemoia"
:modules {:main {:init-fn atemoia.client/start}}
:devtools {:after-load atemoia.client/after-load}}}}
client.cljs
(ns atemoia.client)
(defn start
[]
(prn :start))
(defn after-load
[]
(prn :after-load))
start shadow-cljs
server.clj
(defn dev-main
[& _]
(-> `shadow.cljs.devtools.server/start!
requiring-resolve
(apply []))
(-> `shadow.cljs.devtools.api/watch
requiring-resolve
(apply [:atemoia]))
(-main))
run in the REPL
(require 'atemoia.server :reload)
(atemoia.server/dev-main)
run and see the output
target/
└── classes
└── public
└── atemoia
├── cljs-runtime
│ ├── atemoia.client.js
│ ...
├── main.js
└── manifest.edn
- Ok, we need to serve the
target/classes/public
folder
server.cljs
;; inside -main, add `::http/file-path` key
(-> {::http/port 8080
::http/type :jetty
::http/routes routes
::http/file-path "target/classes/public"
::http/join? false}
http/default-interceptors
http/dev-interceptors
http/create-server
http/start)
- Now we can require our cljs
;; inside index
[:body
[:div
{:id "atemoia"}
"hello"]
[:script
{:src "/atemoia/main.js"}]]
this will require /atemoia/main.js
from the browser the server will receive /atemoia/main.js
, append with
the file-path
and search for target/classes/public/atemoia/main.js
if it find, it will return. if not, will return 404.
Minimal reagent UI
(ns atemoia.client
(:require [reagent.core :as r]
[reagent.dom :as rd]))
(defonce *state
(r/atom {:n 0}))
(defn ui-root
[]
[:button {:on-click (swap! *state :n inc)} (str @*state)])
(defn start
[]
(some->> (js/document.getElementById "atemoia")
(rd/render [ui-root])))
(defn after-load
[]
(some->> (js/document.getElementById "atemoia")
(rd/render [ui-root])))
A todo list in UI-only
Let's edit our atemoia.client
file
Here a basic todo-app
(ns atemoia.client
(:require [reagent.core :as r]
[reagent.dom :as rd]))
(defonce *state
(r/atom {:todos []}))
(defn ui-root
[]
(let [{:keys [todos]} @*state]
[:div
[:form
{:on-submit (fn [^js evt]
(.preventDefault evt)
(let [el (-> evt
.-target
.-elements
.-note)]
(swap! *state update :todos (fn [todos]
(conj todos
{:todo/id (count todos)
:todo/note (.-value el)})))
(set! (.-value el) "")))}
[:label
"note: " [:input {:name "note"}]]]
[:ul
(for [{:todo/keys [id note]} todos]
[:li {:key id}
note])]]))
(defn start
[]
(some->> (js/document.getElementById "atemoia")
(rd/render [ui-root])))
(defn after-load
[]
(some->> (js/document.getElementById "atemoia")
(rd/render [ui-root])))
My first endpoint
Connect to JDBC
All endpoints
Back to UI
Build script.
Dockerizing
Heroku
We are done