【问题标题】:Clojure Hot ReloadClojure 热重载
【发布时间】:2020-01-28 09:46:42
【问题描述】:

在开发中使用 docker-compose 和 Hot Reload 运行 Clojure 的最佳选择是什么? 我正在调查 Clojure repl 并试图在 Docker 容器中运行但不成功。

目前我们在没有 Docker 的情况下运行 Clojure 应用程序,并且在每次更改时我们手动重新启动 Clojure 应用程序。

我也尝试过 inotify-tools,观察 clojure 目录的变化,但是当我运行 Docker 容器时,出现以下错误“请安装 rlwrap 以进行命令编辑或改用“clojure”。”容器停止。

我的 Dockerfile 是:

FROM clojure:openjdk-11-tools-deps

WORKDIR /usr/src/app
RUN apt-get update && apt-get install -y \
    inotify-tools 

COPY deps.edn .

RUN clojure -e "(clojure-version)"

COPY ./src /usr/src/app/src

COPY ./data /usr/src/app/data

COPY .lein-env .

RUN  cd /usr/src/app/data/questions && ls -la

RUN ls -la


COPY test test/
COPY test.sh .
COPY watch.sh .
COPY run.sh .

EXPOSE 8800 8888

#CMD ["clojure", "-m", "bloom.server.core"]
# Run CLojure Service
RUN ["chmod", "+x", "./watch.sh"]

观看脚本

    #!/bin/bash

# Inotify script to trigger a command on file changes.
#
# The script triggers the command as soon as a file event occurs. Events
# occurring during command execution are aggregated and trigger a single command
# execution only.
#
# Usage example: Trigger rsync for synchronizing file changes.
# ./watch.sh rsync -Cra --out-format='[%t]--%n' --delete SOURCE TARGET


######### Configuration #########

EVENTS="CREATE,CLOSE_WRITE,DELETE,MODIFY,MOVED_FROM,MOVED_TO"
COMMAND="$@"
WATCHDIR=/src

## Exclude Git and temporary files from PHPstorm from watching.
EXCLUDE='(\.git|___jb_|sites/default/dev)'

## Whether to enable verbosity. If enabled, change events are output.
VERBOSE=0

##################################

if [ -z "$1" ]; then
 echo "Usage: $0 Command"
 exit 1;
fi

##
## Setup pipes. For usage with read we need to assign them to file descriptors.
##
RUN=$(mktemp -u)
mkfifo "$RUN"
exec 3<>$RUN

RESULT=$(mktemp -u)
mkfifo "$RESULT"
exec 4<>$RESULT

clean_up () {
  ## Cleanup pipes.
  rm $RUN
  rm $RESULT
}

## Execute "clean_up" on exit.
trap "clean_up" EXIT


##
## Run inotifywait in a loop that is not blocked on command execution and ignore
## irrelevant events.
##
inotifywait -m -q -r -e $EVENTS --exclude $EXCLUDE --format '%w%f' $WATCHDIR | \
  while read FILE
  do
    if [ $VERBOSE -ne 0 ]; then
      echo [CHANGE] $FILE
    fi

    ## Clear $PID if the last command has finished.
    if [ ! -z "$PID" ] && ( ! ps -p $PID > /dev/null ); then
      PID=""
    fi

    ## If no command is being executed, execute one.
    ## Else, wait for the command to finish and then execute again.
    if [ -z "$PID" ]; then
      ## Execute the following as background process.
      ## It runs the command once and repeats if we tell him so.
      ($COMMAND; while read -t0.001 -u3 LINE; do
        echo running >&4
        $COMMAND
      done)&

      PID=$!
      WAITING=0
    else
      ## If a previous waiting command has been executed, reset the variable.
      if [ $WAITING -eq 1 ] && read -t0.001 -u4; then
        WAITING=0
      fi

      ## Tell the subprocess to execute the command again if it is not waiting
      ## for repeated execution already.
      if [ $WAITING -eq 0 ]; then
        echo "run" >&3
        WAITING=1
      fi

      ## If we are already waiting, there is nothing todo.
    fi
done

system.clj

(ns testapp.server.system
  (:require
   [com.stuartsierra.component :as component]
   [testapp.server.internal.service :as internal-service]
   [testapp.server.internal.rest :as internal-rest]
   [testapp.server.internal.html :as internal-html]
   [testapp.server.app.service :as app-service]
   [testapp.server.app.rest :as app-rest]
   [testapp.server.settings :as settings]))

(defn new-system []
  (merge (component/system-map)
         (settings/new-settings-provider)
         (internal-service/new-server)
         (internal-rest/new-provider)
         (internal-html/new-provider)
         (app-service/new-server)
         (app-rest/new-provider)
         ))

core.clj

(ns testapp.server.core
  (:require
   [com.stuartsierra.component :as component]
   [testapp.server.system :as system])
  (:gen-class))

(defonce server (atom nil))

(defn stop-server []
  (component/stop-system @server)
  (reset! server nil))

(defn start-server [system]
  (reset! server (component/start-system system)))

(defn -main [& args]
  (start-server (system/new-system))
  (if @server
    (println (str "Services started on the following ports:\n"
                  "\n- Internal API: "
                  (get-in @server [:settings-provider :settings :internal-api-port])
                  "\n- App API: "
                  (get-in @server [:settings-provider :settings :app-api-port])
                  "\nAll running in the "
                  (get-in @server [:settings-provider :settings :run-env])
                  " environment."
                  ))
    (println "Starting the server failed somehow ¯\\_(ツ)_/¯")))

settings.clj

    (ns testapp.server.settings
  (:require
   [com.stuartsierra.component :as component]
   [environ.core :refer [env]]))

(defrecord Settings [settings]
  component/Lifecycle
  (start [this]
    (assoc this :settings
           {

            :internal-api-port
            (or (some->> (env :internal-api-port) str (re-matches #"\d+") Integer.)
                8800)

            :app-api-port
            (or (some->> (env :app-api-port) str (re-matches #"\d+") Integer.)
                8888)

            :run-env
            (or (some #{(env :run-env)} ["local" "dev" "staging" "production"])
                "local")

            :google-auth-client-id (env :google-auth-client-id)
            :google-auth-client-secret (env :google-auth-client-secret)
            :google-auth-callback-host (env :google-auth-callback-host)



            }))
  (stop [this] (assoc this :settings nil)))

(defn new-settings-provider []
  {:settings-provider (map->Settings {})})

谢谢。

【问题讨论】:

  • 也许可以添加一些关于您的工作流程的示例?另外,我不确定 docker-compose 是如何参与的。
  • 嗨@AlanThompson。我更新了我的帖子。你能检查一下吗。我会非常感谢你。
  • 如错误消息所述:您很可能在某处运行clj(基本上只是对rlwrap clojure "$@" 的调用。在您的容器中安装rlwarp 或直接使用clojure。跨度>
  • 谢谢@cfrick。我要试试。如何在本地开发中使用 clojure。你的热重载方式是什么?
  • 另外,为什么不走“clojurey”路线并使用“重新加载”工作流程呢?例如。在你的情况下github.com/stuartsierra/component#reloading 有意义

标签: docker clojure docker-compose dockerfile clojurescript


【解决方案1】:

在 REPL 中,您可以重用来自 duct 框架的代码。它只需要 hawk 文件观察器作为开发环境中的依赖项。

下面是它的样子:

(defn- clojure-file? [_ {:keys [file]}]
  (re-matches #"[^.].*(\.clj|\.edn)$" (.getName file)))

(defn- auto-reset-handler [ctx event]
  (binding [*ns* *ns*]
    (clojure.tools.namespace.repl/refresh :after 'function.to.restart.system/restart)
    ctx))

(defn auto-reset
  "Automatically reset the system when a Clojure or edn file is changed in
  `src` or `resources`."
  []
  (hawk.core/watch! [{:paths ["src/"]
                      :filter clojure-file?
                      :handler auto-reset-handler}]))

【讨论】:

    猜你喜欢
    • 1970-01-01
    • 2016-10-10
    • 2023-03-15
    • 2017-01-20
    • 2021-03-19
    • 2018-07-04
    • 2018-09-26
    • 2017-02-26
    相关资源
    最近更新 更多