defmodule Livebook.MixProject do use Mix.Project @elixir_requirement "~> 1.14" @version "0.7.2" @description "Interactive and collaborative code notebooks - made with Phoenix LiveView" @app_elixir_version "1.14.0" @app_rebar3_version "3.19.0" def project do [ app: :livebook, version: @version, elixir: @elixir_requirement, name: "Livebook", description: @description, elixirc_paths: elixirc_paths(Mix.env()), start_permanent: Mix.env() == :prod, aliases: aliases(), deps: with_lock(target_deps(Mix.target()) ++ deps()), escript: escript(), package: package(), default_release: :livebook, releases: releases() ] end def application do [ mod: {Livebook.Application, []}, extra_applications: [:logger, :runtime_tools, :os_mon, :inets, :ssl, :xmerl] ++ extra_applications(Mix.target()), env: Application.get_all_env(:livebook) ] end defp extra_applications(:app), do: [:wx] defp extra_applications(_), do: [] defp elixirc_paths(:test), do: elixirc_paths(:dev) ++ ["test/support"] defp elixirc_paths(_), do: ["lib", "proto/lib"] defp package do [ licenses: ["Apache-2.0"], links: %{ "GitHub" => "https://github.com/livebook-dev/livebook" }, files: ~w(lib static config mix.exs mix.lock README.md LICENSE CHANGELOG.md iframe/priv/static/iframe proto/lib) ] end defp aliases do [ "dev.setup": ["deps.get", "cmd npm install --prefix assets"], "dev.build": ["cmd npm run deploy --prefix ./assets"], "format.all": ["format", "cmd npm run format --prefix ./assets"], "protobuf.generate": ["cmd --cd proto mix protobuf.generate"] ] end defp escript do [ main_module: LivebookCLI, app: nil ] end ## Dependencies # Although we use requirements here, the with_lock() function # below ensures we only use the locked versions. This is important # for two reasons: # # 1. because we bundle assets from phoenix, phoenix_live_view, # and phoenix_html, we want to make sure we have those exact # versions # # 2. we don't want users to potentially get a new dependency # when installing from git or as an escript # # Therefore, to update any dependency, you must call before: # # mix deps.unlock foo bar baz # defp deps do [ {:phoenix, "~> 1.5"}, {:phoenix_html, "~> 3.0"}, {:phoenix_live_view, "~> 0.18.1"}, {:phoenix_live_dashboard, "~> 0.7.0"}, {:telemetry_metrics, "~> 0.4"}, {:telemetry_poller, "~> 1.0"}, {:jason, "~> 1.0"}, {:plug_cowboy, "~> 2.0"}, {:earmark_parser, "~> 1.4"}, {:castore, "~> 0.1.0"}, {:aws_signature, "~> 0.3.0"}, {:ecto, "~> 3.9.0"}, {:phoenix_ecto, "~> 4.4.0"}, {:mint_web_socket, "~> 1.0.0"}, {:protobuf, "~> 0.8.0"}, {:phoenix_live_reload, "~> 1.2", only: :dev}, {:floki, ">= 0.27.0", only: :test}, {:bypass, "~> 2.1", only: :test} ] end defp target_deps(:app), do: [{:app_bundler, path: "app_bundler"}] defp target_deps(_), do: [] @lock (with {:ok, contents} <- File.read("mix.lock"), {:ok, quoted} <- Code.string_to_quoted(contents, warn_on_unnecessary_quotes: false), {%{} = lock, _binding} <- Code.eval_quoted(quoted, []) do for {dep, hex} when elem(hex, 0) == :hex <- lock, do: {dep, elem(hex, 2)}, into: %{} else _ -> %{} end) defp with_lock(deps) do for dep <- deps do name = elem(dep, 0) put_elem(dep, 1, @lock[name] || elem(dep, 1)) end end ## Releases defp releases do macos_notarization = macos_notarization() additional_paths = [ "rel/vendor/otp/erts-#{:erlang.system_info(:version)}/bin", "rel/vendor/otp/bin", "rel/vendor/elixir/bin" ] [ livebook: [ include_executables_for: [:unix], include_erts: false, rel_templates_path: "rel/server", steps: [:assemble, &remove_cookie/1] ], app: [ include_erts: false, rel_templates_path: "rel/app", steps: [ :assemble, &remove_cookie/1, &standalone_erlang_elixir/1, &AppBundler.bundle/1 ], app: [ name: "Livebook", url_schemes: ["livebook"], document_types: [ [ name: "LiveMarkdown", extensions: ["livemd"], macos: [ icon_path: "rel/app/icon.png", role: "Editor" ], windows: [ icon_path: "rel/app/icon.ico" ] ] ], macos: [ app_type: :agent, icon_path: "rel/app/icon-macos.png", build_dmg: macos_notarization != nil, notarization: macos_notarization, additional_paths: additional_paths ++ ["/usr/local/bin"] ], windows: [ icon_path: "rel/app/icon.ico", build_installer: true, additional_paths: additional_paths ] ] ] ] end defp macos_notarization do identity = System.get_env("NOTARIZE_IDENTITY") team_id = System.get_env("NOTARIZE_TEAM_ID") apple_id = System.get_env("NOTARIZE_APPLE_ID") password = System.get_env("NOTARIZE_PASSWORD") if identity && team_id && apple_id && password do [identity: identity, team_id: team_id, apple_id: apple_id, password: password] end end defp remove_cookie(release) do File.rm!(Path.join(release.path, "releases/COOKIE")) release end defp standalone_erlang_elixir(release) do Code.require_file("rel/app/standalone.exs") release |> Standalone.copy_otp() |> Standalone.copy_elixir(@app_elixir_version) |> Standalone.copy_hex() |> Standalone.copy_rebar3(@app_rebar3_version) end end