From 3e0eee9461eb2af7b09d552cadfca040b9f86f4d Mon Sep 17 00:00:00 2001 From: Ellis Pritchard Date: Wed, 17 Jan 2018 18:55:01 +0000 Subject: [PATCH 1/5] Standardize on start_link/1 so works out of the box with 1.5 sty;e `Supervisor.start_link/2` specs --- lib/code_reloader/server.ex | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/lib/code_reloader/server.ex b/lib/code_reloader/server.ex index 0cd6d52..0064bea 100644 --- a/lib/code_reloader/server.ex +++ b/lib/code_reloader/server.ex @@ -5,7 +5,7 @@ defmodule CodeReloader.Server do require Logger alias CodeReloader.Proxy - def start_link() do + def start_link(_) do GenServer.start_link(__MODULE__, false, name: __MODULE__) end From 176330b7715cfb9354106bcfe1ae7c5fd5446008 Mon Sep 17 00:00:00 2001 From: Ellis Pritchard Date: Tue, 23 Jan 2018 10:47:36 +0000 Subject: [PATCH 2/5] Enhancements (#1) * reloadable compiler config moved to GenServer init args * re-loads files compiled outside of the VM running CodeReloader * `mix format` --- lib/code_reloader/plug.ex | 52 ++++++------ lib/code_reloader/proxy.ex | 8 +- lib/code_reloader/server.ex | 156 ++++++++++++++++++++++++++---------- mix.exs | 14 ++-- 4 files changed, 155 insertions(+), 75 deletions(-) diff --git a/lib/code_reloader/plug.ex b/lib/code_reloader/plug.ex index 35250e1..5c76a86 100644 --- a/lib/code_reloader/plug.ex +++ b/lib/code_reloader/plug.ex @@ -2,29 +2,25 @@ defmodule CodeReloader.Plug do @moduledoc """ A plug and module to handle automatic code reloading. - For each request, Phoenix checks if any of the modules previously - compiled requires recompilation via `__phoenix_recompile__?/0` and then - calls `mix compile` for sources exclusive to the `web` directory. + For each request, the Plug checks and recompiles any of the modules in + the project using `CodeReloader.Server.reload!/1`. - To avoid race conditions, all code reloads are funneled through a - sequential call operation. - """ - - ## Server delegation - - @doc """ - Reloads code for the current Mix project by invoking the - `:reloadable_compilers`. - - This is configured in your application environment like: + ``` + defmodule MyRouter do + use Plug.Router - config :your_app, YourApp.Endpoint, - reloadable_compilers: [:gettext, :phoenix, :elixir] + plug CodeReloader.Plug, endpoint: __MODULE__ + + ...etc... + end + ``` - Keep in mind `:reloadable_compilers` must be a subset of the - `:compilers` specified in `project/0` in your `mix.exs`. + Every request through the router will attempt to kick-off a recomplile + of the current project, and report failures by rendering an error page + for the web browser. """ - @spec reload!(module) :: :ok | {:error, binary()} + + @spec reload!(module) :: {:ok, binary()} | {:error, binary()} defdelegate reload!(endpoint), to: CodeReloader.Server ## Plug @@ -35,9 +31,11 @@ defmodule CodeReloader.Plug do @style %{ primary: "#EB532D", + xx: "0", accent: "#a0b0c0", text_color: "304050", - logo: "", + logo: + "", monospace_font: "menlo, consolas, monospace" } @@ -56,14 +54,18 @@ defmodule CodeReloader.Plug do end defp do_call(conn, _reloader, endpoint) when is_nil(endpoint) do - Logger.error("CodeReloader: couldn't reload. opts[:endpoint] must be specified when using CodeReloader.Plug.") + Logger.error(fn -> + "CodeReloader: couldn't reload. opts[:endpoint] must be specified when using CodeReloader.Plug." + end) + conn end + defp do_call(conn, reloader, endpoint) do - Logger.info("CodeReloader: reloading") case reloader.(endpoint) do - :ok -> + {:ok, _output} -> conn + {:error, output} -> conn |> put_resp_content_type("text/html") @@ -264,8 +266,8 @@ defmodule CodeReloader.Plug do defp format_output(output) do output - |> String.trim - |> Plug.HTML.html_escape + |> String.trim() + |> Plug.HTML.html_escape() end defp get_error_details(output) do diff --git a/lib/code_reloader/proxy.ex b/lib/code_reloader/proxy.ex index 9c8091b..38e8a8d 100644 --- a/lib/code_reloader/proxy.ex +++ b/lib/code_reloader/proxy.ex @@ -14,6 +14,10 @@ defmodule CodeReloader.Proxy do ## Callbacks + def init(args) do + {:ok, args} + end + def handle_call(:stop, _from, output) do {:stop, :normal, output, output} end @@ -33,7 +37,7 @@ defmodule CodeReloader.Proxy do put_chars(from, reply, apply(m, f, as), output) {:io_request, _from, _reply, _request} = msg -> - send(Process.group_leader, msg) + send(Process.group_leader(), msg) {:noreply, output} _ -> @@ -42,7 +46,7 @@ defmodule CodeReloader.Proxy do end defp put_chars(from, reply, chars, output) do - send(Process.group_leader, {:io_request, from, reply, {:put_chars, chars}}) + send(Process.group_leader(), {:io_request, from, reply, {:put_chars, chars}}) {:noreply, output <> IO.chardata_to_string(chars)} end end diff --git a/lib/code_reloader/server.ex b/lib/code_reloader/server.ex index 0064bea..2edc905 100644 --- a/lib/code_reloader/server.ex +++ b/lib/code_reloader/server.ex @@ -1,12 +1,49 @@ defmodule CodeReloader.Server do - @moduledoc false + @moduledoc """ + Recompiles modified files in the current mix project by invoking configured reloadable compilers. + + Specify the compilers that should be run for reloading when starting the server, e.g.: + + ``` + children = [{CodeReloader.Server, [:elixir, :erlang]}] + Supervisor.start_link(children, [strategy: :one_for_one]) + ``` + + Code can then be reloaded by calling: + + ``` + CodeReloader.Server.reload!(mod) + ``` + + where `mod` will normally be a `Plug.Router` module containing the `CodeReloader.Plug` + used to instigate a code reload on every web-server call (it could potentially + be any another module being used to kick-off the reload). + + The `mod` argument is used for two purposes: + + * To avoid race conditions from multiple calls: all code reloads from the same + module are funneled through a sequential call operation. + * To back-up the module's `.beam` file so if compilation of the module itself fails, + it can be restored to working order, otherwise code reload through that + module would no-longer be available, which would kill an endpoint. + + We also keep track of the last time that we compiled the code, so that if the code changes + outside of the VM, e.g. an external tool recompiles the code, we notice that the manifest + is newer than when we compiled, and explicitly reload all modified modules (see `:code.modified_modules/0`) + since compiling will potentially be a no-op. + + This code is based on that in the [Pheonix Project](https://github.com/phoenixframework/phoenix), + without the Phoenix dependencies, and modified to deal with the edge-case of projects recompiled + outside of the `CodeReloader.Server` (the original only copes with modified source code). + + """ use GenServer require Logger alias CodeReloader.Proxy - def start_link(_) do - GenServer.start_link(__MODULE__, false, name: __MODULE__) + def start_link(reloadable_compilers) do + GenServer.start_link(__MODULE__, reloadable_compilers, name: __MODULE__) end def check_symlinks do @@ -19,43 +56,47 @@ defmodule CodeReloader.Server do ## Callbacks - def init(false) do - {:ok, false} + def init(reloadable_compilers) do + {:ok, {false, reloadable_compilers, System.os_time(:seconds)}} end - def handle_call(:check_symlinks, _from, checked?) do + def handle_call(:check_symlinks, _from, {checked?, reloadable_compilers, last_compile_time}) do if not checked? and Code.ensure_loaded?(Mix.Project) do build_path = Mix.Project.build_path() - symlink = Path.join(Path.dirname(build_path), "__phoenix__") + symlink = Path.join(Path.dirname(build_path), "#{__MODULE__}") case File.ln_s(build_path, symlink) do :ok -> File.rm(symlink) + {:error, :eexist} -> File.rm(symlink) + {:error, _} -> - Logger.warn "Phoenix is unable to create symlinks. Phoenix' code reloader will run " <> - "considerably faster if symlinks are allowed." <> os_symlink(:os.type) + Logger.warn( + "App is unable to create symlinks. CodeReloader will run " <> + "considerably faster if symlinks are allowed." <> os_symlink(:os.type()) + ) end end - {:reply, :ok, true} + {:reply, :ok, {true, reloadable_compilers, last_compile_time}} end - def handle_call({:reload!, endpoint}, from, state) do - compilers = endpoint.config(:reloadable_compilers) + def handle_call({:reload!, endpoint}, from, {checked?, compilers, last_compile_time}) do backup = load_backup(endpoint) - froms = all_waiting([from], endpoint) + froms = all_waiting([from], endpoint) {res, out} = proxy_io(fn -> try do - mix_compile(Code.ensure_loaded(Mix.Task), compilers) + mix_compile(Code.ensure_loaded(Mix.Task), compilers, last_compile_time) catch :exit, {:shutdown, 1} -> :error + kind, reason -> - IO.puts Exception.format(kind, reason, System.stacktrace) + IO.puts(Exception.format(kind, reason, System.stacktrace())) :error end end) @@ -63,14 +104,15 @@ defmodule CodeReloader.Server do reply = case res do :ok -> - :ok + {:ok, out} + :error -> write_backup(backup) {:error, out} end Enum.each(froms, &GenServer.reply(&1, reply)) - {:noreply, state} + {:noreply, {checked?, compilers, System.os_time(:seconds)}} end def handle_info(_, state) do @@ -79,20 +121,22 @@ defmodule CodeReloader.Server do defp os_symlink({:win32, _}), do: " On Windows, such can be done by starting the shell with \"Run as Administrator\"." - defp os_symlink(_), - do: "" + + defp os_symlink(_), do: "" defp load_backup(mod) do mod |> :code.which() |> read_backup() end + defp read_backup(path) when is_list(path) do case File.read(path) do {:ok, binary} -> {:ok, path, binary} _ -> :error end end + defp read_backup(_path), do: :error defp write_backup({:ok, path, file}), do: File.write!(path, file) @@ -109,51 +153,60 @@ defmodule CodeReloader.Server do # TODO: Remove the function_exported call after 1.3 support is removed # and just use loaded. apply/3 is used to prevent a compilation # warning. - defp mix_compile({:module, Mix.Task}, compilers) do - if Mix.Project.umbrella? do + defp mix_compile({:module, Mix.Task}, compilers, last_compile_time) do + if Mix.Project.umbrella?() do deps = if function_exported?(Mix.Dep.Umbrella, :cached, 0) do apply(Mix.Dep.Umbrella, :cached, []) else - Mix.Dep.Umbrella.loaded + Mix.Dep.Umbrella.loaded() end - Enum.each deps, fn dep -> + + Enum.each(deps, fn dep -> Mix.Dep.in_dependency(dep, fn _ -> - mix_compile_unless_stale_config(compilers) + mix_compile_unless_stale_config(compilers, last_compile_time) end) - end + end) else - mix_compile_unless_stale_config(compilers) + mix_compile_unless_stale_config(compilers, last_compile_time) :ok end end - defp mix_compile({:error, _reason}, _) do + + defp mix_compile({:error, _reason}, _, _) do raise "the Code Reloader is enabled but Mix is not available. If you want to " <> - "use the Code Reloader in production or inside an escript, you must add " <> - ":mix to your applications list. Otherwise, you must disable code reloading " <> - "in such environments" + "use the Code Reloader in production or inside an escript, you must add " <> + ":mix to your applications list. Otherwise, you must disable code reloading " <> + "in such environments" end - defp mix_compile_unless_stale_config(compilers) do - manifests = Mix.Tasks.Compile.Elixir.manifests - configs = Mix.Project.config_files + defp mix_compile_unless_stale_config(compilers, last_compile_time) do + manifests = Mix.Tasks.Compile.Elixir.manifests() + configs = Mix.Project.config_files() + + # did the manifest change outside of us compiling the project? + manifests_last_updated = + Enum.map(manifests, &File.stat!(&1, time: :posix).mtime) |> Enum.max() + + out_of_date? = manifests_last_updated > last_compile_time case Mix.Utils.extract_stale(configs, manifests) do [] -> - mix_compile(compilers) + do_mix_compile(compilers, out_of_date?) + files -> raise """ - could not compile application: #{Mix.Project.config[:app]}. + could not compile application: #{Mix.Project.config()[:app]}. You must restart your server after changing the following config or lib files: * #{Enum.map_join(files, "\n * ", &Path.relative_to_cwd/1)} """ - end - end + end + end - defp mix_compile(compilers) do - all = Mix.Project.config[:compilers] || Mix.compilers + defp do_mix_compile(compilers, out_of_date?) do + all = Mix.Project.config()[:compilers] || Mix.compilers() compilers = for compiler <- compilers, compiler in all do @@ -163,7 +216,7 @@ defmodule CodeReloader.Server do # We call build_structure mostly for Windows so new # assets in priv are copied to the build directory. - Mix.Project.build_structure + Mix.Project.build_structure() res = Enum.map(compilers, &Mix.Task.run("compile.#{&1}", [])) if :ok in res && consolidate_protocols?() do @@ -171,15 +224,34 @@ defmodule CodeReloader.Server do Mix.Task.run("compile.protocols", []) end + if(out_of_date?, do: reload_modules()) + res end defp consolidate_protocols? do - Mix.Project.config[:consolidate_protocols] + Mix.Project.config()[:consolidate_protocols] + end + + defp reload_modules() do + :code.modified_modules() + |> Enum.each(fn mod -> + IO.puts("Reloading #{inspect(mod)}\n") + + case :code.soft_purge(mod) do + true -> + :code.load_file(mod) + + false -> + Process.sleep(500) + :code.purge(mod) + :code.load_file(mod) + end + end) end defp proxy_io(fun) do - original_gl = Process.group_leader + original_gl = Process.group_leader() {:ok, proxy_gl} = Proxy.start() Process.group_leader(self(), proxy_gl) diff --git a/mix.exs b/mix.exs index 14bc171..4320cb1 100644 --- a/mix.exs +++ b/mix.exs @@ -2,12 +2,14 @@ defmodule CodeReloader.Mixfile do use Mix.Project def project do - [app: :code_reloader, - version: "0.1.0", - elixir: "~> 1.4", - build_embedded: Mix.env == :prod, - start_permanent: Mix.env == :prod, - deps: deps()] + [ + app: :code_reloader, + version: "0.1.0", + elixir: "~> 1.4", + build_embedded: Mix.env() == :prod, + start_permanent: Mix.env() == :prod, + deps: deps() + ] end # Configuration for the OTP application From d217b75c191e23101849111e2e496a7cc8247d37 Mon Sep 17 00:00:00 2001 From: Ellis Pritchard Date: Tue, 23 Jan 2018 11:01:33 +0000 Subject: [PATCH 3/5] Update README.md --- README.md | 33 +++++++++++++++++++++++++++++++-- 1 file changed, 31 insertions(+), 2 deletions(-) diff --git a/README.md b/README.md index a6ab42d..056945f 100644 --- a/README.md +++ b/README.md @@ -2,7 +2,7 @@ Code reloader Plug extracted from [Phoenix](https://github.com/phoenixframework/phoenix/) and adapted to be a generic Plug. -So far it's just a proof of concept to understand if having a generic code reload Plug makes sense or not. +If code changes outside of the VM e.g. you call `mix compile` in another shell, or your editor recompiles files automatically, any modules that changed on disk will be re-loaded: the Pheonix version doesn't currently handle this use case. ## Why @@ -10,4 +10,33 @@ If you have an Elixir web app using only Plug without Phoenix, you need to resta ## Usage -The the example app at [https://github.com/pilu/code-reload-example](https://github.com/pilu/code-reload-example) +Add the `CodeReloader.Server` to your supervision tree, e.g. in your `Application.start/2`: + +``` +children = [ + ..., {CodeReloader.Server, [:elixir]} +] +Supervisor.start_link(children, strategy: :one_for_one) +``` + +The list argument configures the compilers we run when when `CodeReloader.Server.reload!/1` is called: each is called (effectively) as `mix compile.`. + + +For re-loading code on every HTTP request, add the plug to your `Plug.Router`: + +``` +plug CodeReloader.Plug, endpoint: __MODULE__ +``` + +The endpoint is used to serialize requests and prevent a compile failure in the router from stopping subsequent re-loading. + +If your code has compile errors, an error page will be rendered to the browser; errors and other compile messages also appear on the standard output. + + +### Using outside of Plug + +You can also call `CodeReloader.Server.reload!/1` directly, e.g. for reloading on another event. + +``` +CodeReloader.Server.reload!(__MODULE__) +``` From 41b728b4606a6509f9638a5f653dc14c1358979b Mon Sep 17 00:00:00 2001 From: Ellis Pritchard Date: Tue, 23 Jan 2018 11:04:45 +0000 Subject: [PATCH 4/5] Update README.md --- README.md | 6 ++++-- 1 file changed, 4 insertions(+), 2 deletions(-) diff --git a/README.md b/README.md index 056945f..95c2c61 100644 --- a/README.md +++ b/README.md @@ -2,12 +2,14 @@ Code reloader Plug extracted from [Phoenix](https://github.com/phoenixframework/phoenix/) and adapted to be a generic Plug. -If code changes outside of the VM e.g. you call `mix compile` in another shell, or your editor recompiles files automatically, any modules that changed on disk will be re-loaded: the Pheonix version doesn't currently handle this use case. - ## Why If you have an Elixir web app using only Plug without Phoenix, you need to restart it everytime you update the code. +This module uses `Mix` to compile any files changed on disk every time an HTTP request is received, allowing your server to keep running while you make changes. + +If code changes outside of the VM e.g. you call `mix compile` in another shell, or your editor recompiles files automatically, any modules that changed on disk will be re-loaded (the Pheonix version doesn't currently handle this use case). + ## Usage Add the `CodeReloader.Server` to your supervision tree, e.g. in your `Application.start/2`: From 2bc7465c0d45d554210cc8dc2b7e4505d30fd96f Mon Sep 17 00:00:00 2001 From: Rhys Evans Date: Tue, 2 Feb 2021 12:35:01 +0000 Subject: [PATCH 5/5] [Skip CI] Automatically creating CODEOWNERS file --- .github/CODEOWNERS | 2 ++ 1 file changed, 2 insertions(+) create mode 100644 .github/CODEOWNERS diff --git a/.github/CODEOWNERS b/.github/CODEOWNERS new file mode 100644 index 0000000..8d11d77 --- /dev/null +++ b/.github/CODEOWNERS @@ -0,0 +1,2 @@ +# Guessed from commit history +* @Financial-Times/membership