├── test ├── test_helper.exs └── code_reloader_test.exs ├── lib ├── code_reloader.ex └── code_reloader │ ├── proxy.ex │ ├── server.ex │ └── plug.ex ├── mix.lock ├── README.md ├── .gitignore ├── mix.exs ├── LICENSE └── config └── config.exs /test/test_helper.exs: -------------------------------------------------------------------------------- 1 | ExUnit.start() 2 | -------------------------------------------------------------------------------- /lib/code_reloader.ex: -------------------------------------------------------------------------------- 1 | defmodule CodeReloader do 2 | end 3 | -------------------------------------------------------------------------------- /test/code_reloader_test.exs: -------------------------------------------------------------------------------- 1 | defmodule CodeReloaderTest do 2 | use ExUnit.Case 3 | doctest CodeReloader 4 | 5 | test "the truth" do 6 | assert 1 + 1 == 2 7 | end 8 | end 9 | -------------------------------------------------------------------------------- /mix.lock: -------------------------------------------------------------------------------- 1 | %{"mime": {:hex, :mime, "1.1.0", "01c1d6f4083d8aa5c7b8c246ade95139620ef8effb009edde934e0ec3b28090a", [:mix], []}, 2 | "plug": {:hex, :plug, "1.3.5", "7503bfcd7091df2a9761ef8cecea666d1f2cc454cbbaf0afa0b6e259203b7031", [:mix], [{:cowboy, "~> 1.0.1 or ~> 1.1", [hex: :cowboy, optional: true]}, {:mime, "~> 1.0", [hex: :mime, optional: false]}]}} 3 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # CodeReloader 2 | 3 | Code reloader Plug extracted from [Phoenix](https://github.com/phoenixframework/phoenix/) and adapted to be a generic Plug. 4 | 5 | So far it's just a proof of concept to understand if having a generic code reload Plug makes sense or not. 6 | 7 | ## Why 8 | 9 | If you have an Elixir web app using only Plug without Phoenix, you need to restart it everytime you update the code. 10 | 11 | ## Usage 12 | 13 | The the example app at [https://github.com/pilu/code-reload-example](https://github.com/pilu/code-reload-example) 14 | -------------------------------------------------------------------------------- /.gitignore: -------------------------------------------------------------------------------- 1 | # The directory Mix will write compiled artifacts to. 2 | /_build 3 | 4 | # If you run "mix test --cover", coverage assets end up here. 5 | /cover 6 | 7 | # The directory Mix downloads your dependencies sources to. 8 | /deps 9 | 10 | # Where 3rd-party dependencies like ExDoc output generated docs. 11 | /doc 12 | 13 | # Ignore .fetch files in case you like to edit your project deps locally. 14 | /.fetch 15 | 16 | # If the VM crashes, it generates a dump, let's ignore it too. 17 | erl_crash.dump 18 | 19 | # Also ignore archive artifacts (built via "mix archive.build"). 20 | *.ez 21 | -------------------------------------------------------------------------------- /mix.exs: -------------------------------------------------------------------------------- 1 | defmodule CodeReloader.Mixfile do 2 | use Mix.Project 3 | 4 | def project do 5 | [app: :code_reloader, 6 | version: "0.1.0", 7 | elixir: "~> 1.4", 8 | build_embedded: Mix.env == :prod, 9 | start_permanent: Mix.env == :prod, 10 | deps: deps()] 11 | end 12 | 13 | # Configuration for the OTP application 14 | # 15 | # Type "mix help compile.app" for more information 16 | def application do 17 | # Specify extra applications you'll use from Erlang/Elixir 18 | [extra_applications: [:logger]] 19 | end 20 | 21 | # Dependencies can be Hex packages: 22 | # 23 | # {:my_dep, "~> 0.3.0"} 24 | # 25 | # Or git/path repositories: 26 | # 27 | # {:my_dep, git: "https://github.com/elixir-lang/my_dep.git", tag: "0.1.0"} 28 | # 29 | # Type "mix help deps" for more examples and options 30 | defp deps do 31 | [{:plug, "~> 1.0"}] 32 | end 33 | end 34 | -------------------------------------------------------------------------------- /LICENSE: -------------------------------------------------------------------------------- 1 | MIT License 2 | 3 | Copyright (c) 2017 Andrea Franz 4 | 5 | Permission is hereby granted, free of charge, to any person obtaining a copy 6 | of this software and associated documentation files (the "Software"), to deal 7 | in the Software without restriction, including without limitation the rights 8 | to use, copy, modify, merge, publish, distribute, sublicense, and/or sell 9 | copies of the Software, and to permit persons to whom the Software is 10 | furnished to do so, subject to the following conditions: 11 | 12 | The above copyright notice and this permission notice shall be included in all 13 | copies or substantial portions of the Software. 14 | 15 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR 16 | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, 17 | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE 18 | AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER 19 | LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, 20 | OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE 21 | SOFTWARE. 22 | -------------------------------------------------------------------------------- /config/config.exs: -------------------------------------------------------------------------------- 1 | # This file is responsible for configuring your application 2 | # and its dependencies with the aid of the Mix.Config module. 3 | use Mix.Config 4 | 5 | # This configuration is loaded before any dependency and is restricted 6 | # to this project. If another project depends on this project, this 7 | # file won't be loaded nor affect the parent project. For this reason, 8 | # if you want to provide default values for your application for 9 | # 3rd-party users, it should be done in your "mix.exs" file. 10 | 11 | # You can configure for your application as: 12 | # 13 | # config :code_reloader, key: :value 14 | # 15 | # And access this configuration in your application as: 16 | # 17 | # Application.get_env(:code_reloader, :key) 18 | # 19 | # Or configure a 3rd-party app: 20 | # 21 | # config :logger, level: :info 22 | # 23 | 24 | # It is also possible to import configuration files, relative to this 25 | # directory. For example, you can emulate configuration per environment 26 | # by uncommenting the line below and defining dev.exs, test.exs and such. 27 | # Configuration from the imported file will override the ones defined 28 | # here (which is why it is important to import them last). 29 | # 30 | # import_config "#{Mix.env}.exs" 31 | -------------------------------------------------------------------------------- /lib/code_reloader/proxy.ex: -------------------------------------------------------------------------------- 1 | # A tiny proxy that stores all output sent to the group leader 2 | # while forwarding all requests to it. 3 | defmodule CodeReloader.Proxy do 4 | @moduledoc false 5 | use GenServer 6 | 7 | def start() do 8 | GenServer.start(__MODULE__, "") 9 | end 10 | 11 | def stop(proxy) do 12 | GenServer.call(proxy, :stop) 13 | end 14 | 15 | ## Callbacks 16 | 17 | def handle_call(:stop, _from, output) do 18 | {:stop, :normal, output, output} 19 | end 20 | 21 | def handle_info(msg, output) do 22 | case msg do 23 | {:io_request, from, reply, {:put_chars, chars}} -> 24 | put_chars(from, reply, chars, output) 25 | 26 | {:io_request, from, reply, {:put_chars, m, f, as}} -> 27 | put_chars(from, reply, apply(m, f, as), output) 28 | 29 | {:io_request, from, reply, {:put_chars, _encoding, chars}} -> 30 | put_chars(from, reply, chars, output) 31 | 32 | {:io_request, from, reply, {:put_chars, _encoding, m, f, as}} -> 33 | put_chars(from, reply, apply(m, f, as), output) 34 | 35 | {:io_request, _from, _reply, _request} = msg -> 36 | send(Process.group_leader, msg) 37 | {:noreply, output} 38 | 39 | _ -> 40 | {:noreply, output} 41 | end 42 | end 43 | 44 | defp put_chars(from, reply, chars, output) do 45 | send(Process.group_leader, {:io_request, from, reply, {:put_chars, chars}}) 46 | {:noreply, output <> IO.chardata_to_string(chars)} 47 | end 48 | end 49 | -------------------------------------------------------------------------------- /lib/code_reloader/server.ex: -------------------------------------------------------------------------------- 1 | defmodule CodeReloader.Server do 2 | @moduledoc false 3 | use GenServer 4 | 5 | require Logger 6 | alias CodeReloader.Proxy 7 | 8 | def start_link() do 9 | GenServer.start_link(__MODULE__, false, name: __MODULE__) 10 | end 11 | 12 | def check_symlinks do 13 | GenServer.call(__MODULE__, :check_symlinks, :infinity) 14 | end 15 | 16 | def reload!(endpoint) do 17 | GenServer.call(__MODULE__, {:reload!, endpoint}, :infinity) 18 | end 19 | 20 | ## Callbacks 21 | 22 | def init(false) do 23 | {:ok, false} 24 | end 25 | 26 | def handle_call(:check_symlinks, _from, checked?) do 27 | if not checked? and Code.ensure_loaded?(Mix.Project) do 28 | build_path = Mix.Project.build_path() 29 | symlink = Path.join(Path.dirname(build_path), "__phoenix__") 30 | 31 | case File.ln_s(build_path, symlink) do 32 | :ok -> 33 | File.rm(symlink) 34 | {:error, :eexist} -> 35 | File.rm(symlink) 36 | {:error, _} -> 37 | Logger.warn "Phoenix is unable to create symlinks. Phoenix' code reloader will run " <> 38 | "considerably faster if symlinks are allowed." <> os_symlink(:os.type) 39 | end 40 | end 41 | 42 | {:reply, :ok, true} 43 | end 44 | 45 | def handle_call({:reload!, endpoint}, from, state) do 46 | compilers = endpoint.config(:reloadable_compilers) 47 | backup = load_backup(endpoint) 48 | froms = all_waiting([from], endpoint) 49 | 50 | {res, out} = 51 | proxy_io(fn -> 52 | try do 53 | mix_compile(Code.ensure_loaded(Mix.Task), compilers) 54 | catch 55 | :exit, {:shutdown, 1} -> 56 | :error 57 | kind, reason -> 58 | IO.puts Exception.format(kind, reason, System.stacktrace) 59 | :error 60 | end 61 | end) 62 | 63 | reply = 64 | case res do 65 | :ok -> 66 | :ok 67 | :error -> 68 | write_backup(backup) 69 | {:error, out} 70 | end 71 | 72 | Enum.each(froms, &GenServer.reply(&1, reply)) 73 | {:noreply, state} 74 | end 75 | 76 | def handle_info(_, state) do 77 | {:noreply, state} 78 | end 79 | 80 | defp os_symlink({:win32, _}), 81 | do: " On Windows, such can be done by starting the shell with \"Run as Administrator\"." 82 | defp os_symlink(_), 83 | do: "" 84 | 85 | defp load_backup(mod) do 86 | mod 87 | |> :code.which() 88 | |> read_backup() 89 | end 90 | defp read_backup(path) when is_list(path) do 91 | case File.read(path) do 92 | {:ok, binary} -> {:ok, path, binary} 93 | _ -> :error 94 | end 95 | end 96 | defp read_backup(_path), do: :error 97 | 98 | defp write_backup({:ok, path, file}), do: File.write!(path, file) 99 | defp write_backup(:error), do: :ok 100 | 101 | defp all_waiting(acc, endpoint) do 102 | receive do 103 | {:"$gen_call", from, {:reload!, ^endpoint}} -> all_waiting([from | acc], endpoint) 104 | after 105 | 0 -> acc 106 | end 107 | end 108 | 109 | # TODO: Remove the function_exported call after 1.3 support is removed 110 | # and just use loaded. apply/3 is used to prevent a compilation 111 | # warning. 112 | defp mix_compile({:module, Mix.Task}, compilers) do 113 | if Mix.Project.umbrella? do 114 | deps = 115 | if function_exported?(Mix.Dep.Umbrella, :cached, 0) do 116 | apply(Mix.Dep.Umbrella, :cached, []) 117 | else 118 | Mix.Dep.Umbrella.loaded 119 | end 120 | Enum.each deps, fn dep -> 121 | Mix.Dep.in_dependency(dep, fn _ -> 122 | mix_compile_unless_stale_config(compilers) 123 | end) 124 | end 125 | else 126 | mix_compile_unless_stale_config(compilers) 127 | :ok 128 | end 129 | end 130 | defp mix_compile({:error, _reason}, _) do 131 | raise "the Code Reloader is enabled but Mix is not available. If you want to " <> 132 | "use the Code Reloader in production or inside an escript, you must add " <> 133 | ":mix to your applications list. Otherwise, you must disable code reloading " <> 134 | "in such environments" 135 | end 136 | 137 | defp mix_compile_unless_stale_config(compilers) do 138 | manifests = Mix.Tasks.Compile.Elixir.manifests 139 | configs = Mix.Project.config_files 140 | 141 | case Mix.Utils.extract_stale(configs, manifests) do 142 | [] -> 143 | mix_compile(compilers) 144 | files -> 145 | raise """ 146 | could not compile application: #{Mix.Project.config[:app]}. 147 | 148 | You must restart your server after changing the following config or lib files: 149 | 150 | * #{Enum.map_join(files, "\n * ", &Path.relative_to_cwd/1)} 151 | """ 152 | end 153 | end 154 | 155 | defp mix_compile(compilers) do 156 | all = Mix.Project.config[:compilers] || Mix.compilers 157 | 158 | compilers = 159 | for compiler <- compilers, compiler in all do 160 | Mix.Task.reenable("compile.#{compiler}") 161 | compiler 162 | end 163 | 164 | # We call build_structure mostly for Windows so new 165 | # assets in priv are copied to the build directory. 166 | Mix.Project.build_structure 167 | res = Enum.map(compilers, &Mix.Task.run("compile.#{&1}", [])) 168 | 169 | if :ok in res && consolidate_protocols?() do 170 | Mix.Task.reenable("compile.protocols") 171 | Mix.Task.run("compile.protocols", []) 172 | end 173 | 174 | res 175 | end 176 | 177 | defp consolidate_protocols? do 178 | Mix.Project.config[:consolidate_protocols] 179 | end 180 | 181 | defp proxy_io(fun) do 182 | original_gl = Process.group_leader 183 | {:ok, proxy_gl} = Proxy.start() 184 | Process.group_leader(self(), proxy_gl) 185 | 186 | try do 187 | {fun.(), Proxy.stop(proxy_gl)} 188 | after 189 | Process.group_leader(self(), original_gl) 190 | Process.exit(proxy_gl, :kill) 191 | end 192 | end 193 | end 194 | -------------------------------------------------------------------------------- /lib/code_reloader/plug.ex: -------------------------------------------------------------------------------- 1 | defmodule CodeReloader.Plug do 2 | @moduledoc """ 3 | A plug and module to handle automatic code reloading. 4 | 5 | For each request, Phoenix checks if any of the modules previously 6 | compiled requires recompilation via `__phoenix_recompile__?/0` and then 7 | calls `mix compile` for sources exclusive to the `web` directory. 8 | 9 | To avoid race conditions, all code reloads are funneled through a 10 | sequential call operation. 11 | """ 12 | 13 | ## Server delegation 14 | 15 | @doc """ 16 | Reloads code for the current Mix project by invoking the 17 | `:reloadable_compilers`. 18 | 19 | This is configured in your application environment like: 20 | 21 | config :your_app, YourApp.Endpoint, 22 | reloadable_compilers: [:gettext, :phoenix, :elixir] 23 | 24 | Keep in mind `:reloadable_compilers` must be a subset of the 25 | `:compilers` specified in `project/0` in your `mix.exs`. 26 | """ 27 | @spec reload!(module) :: :ok | {:error, binary()} 28 | defdelegate reload!(endpoint), to: CodeReloader.Server 29 | 30 | ## Plug 31 | 32 | @behaviour Plug 33 | import Plug.Conn 34 | import Logger 35 | 36 | @style %{ 37 | primary: "#EB532D", 38 | accent: "#a0b0c0", 39 | text_color: "304050", 40 | logo: "", 41 | monospace_font: "menlo, consolas, monospace" 42 | } 43 | 44 | @doc """ 45 | API used by Plug to start the code reloader. 46 | """ 47 | def init(opts), do: Keyword.put_new(opts, :reloader, &CodeReloader.Plug.reload!/1) 48 | 49 | @doc """ 50 | API used by Plug to invoke the code reloader on every request. 51 | """ 52 | def call(conn, opts) do 53 | reloader = Keyword.get(opts, :reloader) 54 | endpoint = Keyword.get(opts, :endpoint) 55 | do_call(conn, reloader, endpoint) 56 | end 57 | 58 | defp do_call(conn, _reloader, endpoint) when is_nil(endpoint) do 59 | Logger.error("CodeReloader: couldn't reload. opts[:endpoint] must be specified when using CodeReloader.Plug.") 60 | conn 61 | end 62 | defp do_call(conn, reloader, endpoint) do 63 | Logger.info("CodeReloader: reloading") 64 | case reloader.(endpoint) do 65 | :ok -> 66 | conn 67 | {:error, output} -> 68 | conn 69 | |> put_resp_content_type("text/html") 70 | |> send_resp(500, template(output)) 71 | |> halt() 72 | end 73 | end 74 | 75 | defp template(output) do 76 | {error, headline} = get_error_details(output) 77 | 78 | """ 79 | 80 | 81 |
82 | 83 |#{format_output(output)}
259 |