commit c39ae78b86691acd5d780b846c37ff04e168de5e Author: Mikko Ahlroth Date: Fri Jan 8 21:32:19 2021 +0200 Initial commit diff --git a/.formatter.exs b/.formatter.exs new file mode 100644 index 0000000..4761678 --- /dev/null +++ b/.formatter.exs @@ -0,0 +1,4 @@ +[ + import_deps: [:phoenix], + inputs: ["*.{ex,exs}", "{config,lib,test}/**/*.{ex,exs}"] +] diff --git a/.gitignore b/.gitignore new file mode 100644 index 0000000..6961c14 --- /dev/null +++ b/.gitignore @@ -0,0 +1,29 @@ +# The directory Mix will write compiled artifacts to. +/_build/ + +# If you run "mix test --cover", coverage assets end up here. +/cover/ + +# The directory Mix downloads your dependencies sources to. +/deps/ + +# Where 3rd-party dependencies like ExDoc output generated docs. +/doc/ + +# Ignore .fetch files in case you like to edit your project deps locally. +/.fetch + +# If the VM crashes, it generates a dump, let's ignore it too. +erl_crash.dump + +# Also ignore archive artifacts (built via "mix archive.build"). +*.ez + +# Ignore package tarball (built via "mix hex.build"). +tracker_app-*.tar + +# Default db directory for dev +/db/ + +# Secret configs +config/*.secret.exs diff --git a/.tool-versions b/.tool-versions new file mode 100644 index 0000000..866f7f4 --- /dev/null +++ b/.tool-versions @@ -0,0 +1,2 @@ +elixir 1.11.0-otp-23 +erlang 23.1.1 diff --git a/README.md b/README.md new file mode 100644 index 0000000..4e6f083 --- /dev/null +++ b/README.md @@ -0,0 +1,18 @@ +# TrackerApp + +To start your Phoenix server: + + * Install dependencies with `mix deps.get` + * Start Phoenix endpoint with `mix phx.server` + +Now you can visit [`localhost:55864`](http://localhost:55864) from your browser. + +Ready to run in production? Please [check our deployment guides](https://hexdocs.pm/phoenix/deployment.html). + +## Learn more + + * Official website: https://www.phoenixframework.org/ + * Guides: https://hexdocs.pm/phoenix/overview.html + * Docs: https://hexdocs.pm/phoenix + * Forum: https://elixirforum.com/c/phoenix-forum + * Source: https://github.com/phoenixframework/phoenix diff --git a/config/config.exs b/config/config.exs new file mode 100644 index 0000000..e0a519a --- /dev/null +++ b/config/config.exs @@ -0,0 +1,28 @@ +# This file is responsible for configuring your application +# and its dependencies with the aid of the Mix.Config module. +# +# This configuration file is loaded before any dependency and +# is restricted to this project. + +# General application configuration +use Mix.Config + +# Configures the endpoint +config :tracker_app, TrackerAppWeb.Endpoint, + url: [host: "localhost"], + secret_key_base: "fTB7icMy0XnADRAB8P/j/bhSx2bmbGBaeVIj/TCA2gBJKDqvkBzaLCTKqM+3524M", + render_errors: [view: TrackerAppWeb.ErrorView, accepts: ~w(html json), layout: false], + pubsub_server: TrackerApp.PubSub, + live_view: [signing_salt: "h081us1G"] + +# Configures Elixir's Logger +config :logger, :console, + format: "$time $metadata[$level] $message\n", + metadata: [:request_id] + +# Use Jason for JSON parsing in Phoenix +config :phoenix, :json_library, Jason + +# Import environment specific config. This must remain at the bottom +# of this file so it overrides the configuration defined above. +import_config "#{Mix.env()}.exs" diff --git a/config/dev.exs b/config/dev.exs new file mode 100644 index 0000000..9354bf8 --- /dev/null +++ b/config/dev.exs @@ -0,0 +1,59 @@ +use Mix.Config + +# For development, we disable any cache and enable +# debugging and code reloading. +# +# The watchers configuration can be used to run external +# watchers to your application. For example, we use it +# with webpack to recompile .js and .css sources. +config :tracker_app, TrackerAppWeb.Endpoint, + http: [port: 55864], + debug_errors: true, + code_reloader: true, + check_origin: false, + watchers: [] + +# ## SSL Support +# +# In order to use HTTPS in development, a self-signed +# certificate can be generated by running the following +# Mix task: +# +# mix phx.gen.cert +# +# Note that this task requires Erlang/OTP 20 or later. +# Run `mix help phx.gen.cert` for more information. +# +# The `http:` config above can be replaced with: +# +# https: [ +# port: 4001, +# cipher_suite: :strong, +# keyfile: "priv/cert/selfsigned_key.pem", +# certfile: "priv/cert/selfsigned.pem" +# ], +# +# If desired, both `http:` and `https:` keys can be +# configured to run both http and https servers on +# different ports. + +# Watch static and templates for browser reloading. +config :tracker_app, TrackerAppWeb.Endpoint, + live_reload: [ + patterns: [ + ~r"priv/static/.*(js|css|png|jpeg|jpg|gif|svg)$", + ~r"priv/gettext/.*(po)$", + ~r"lib/tracker_app_web/(live|views)/.*(ex)$", + ~r"lib/tracker_app_web/templates/.*(eex)$" + ] + ] + +# Do not include metadata nor timestamps in development logs +config :logger, :console, format: "[$level] $message\n" + +# Set a higher stacktrace during development. Avoid configuring such +# in production as building large stacktraces may be expensive. +config :phoenix, :stacktrace_depth, 20 + +# Initialize plugs at runtime for faster development compilation +config :phoenix, :plug_init_mode, :runtime diff --git a/config/prod.exs b/config/prod.exs new file mode 100644 index 0000000..c667dd8 --- /dev/null +++ b/config/prod.exs @@ -0,0 +1,55 @@ +use Mix.Config + +# For production, don't forget to configure the url host +# to something meaningful, Phoenix uses this information +# when generating URLs. +# +# Note we also include the path to a cache manifest +# containing the digested version of static files. This +# manifest is generated by the `mix phx.digest` task, +# which you should run after static files are built and +# before starting your production server. +config :tracker_app, TrackerAppWeb.Endpoint, + url: [host: "example.com", port: 80], + cache_static_manifest: "priv/static/cache_manifest.json" + +# Do not print debug messages in production +config :logger, level: :info + +# ## SSL Support +# +# To get SSL working, you will need to add the `https` key +# to the previous section and set your `:url` port to 443: +# +# config :tracker_app, TrackerAppWeb.Endpoint, +# ... +# url: [host: "example.com", port: 443], +# https: [ +# port: 443, +# cipher_suite: :strong, +# keyfile: System.get_env("SOME_APP_SSL_KEY_PATH"), +# certfile: System.get_env("SOME_APP_SSL_CERT_PATH"), +# transport_options: [socket_opts: [:inet6]] +# ] +# +# The `cipher_suite` is set to `:strong` to support only the +# latest and more secure SSL ciphers. This means old browsers +# and clients may not be supported. You can set it to +# `:compatible` for wider support. +# +# `:keyfile` and `:certfile` expect an absolute path to the key +# and cert in disk or a relative path inside priv, for example +# "priv/ssl/server.key". For all supported SSL configuration +# options, see https://hexdocs.pm/plug/Plug.SSL.html#configure/1 +# +# We also recommend setting `force_ssl` in your endpoint, ensuring +# no data is ever sent via http, always redirecting to https: +# +# config :tracker_app, TrackerAppWeb.Endpoint, +# force_ssl: [hsts: true] +# +# Check `Plug.SSL` for all available options in `force_ssl`. + +# Finally import the config/prod.secret.exs which loads secrets +# and configuration from environment variables. +import_config "prod.secret.exs" diff --git a/config/runtime.exs b/config/runtime.exs new file mode 100644 index 0000000..a6873b1 --- /dev/null +++ b/config/runtime.exs @@ -0,0 +1,3 @@ +import Config + +config :tracker_app, file_db_path: "db" diff --git a/config/test.exs b/config/test.exs new file mode 100644 index 0000000..6ab8698 --- /dev/null +++ b/config/test.exs @@ -0,0 +1,10 @@ +use Mix.Config + +# We don't run a server during test. If one is required, +# you can enable the server option below. +config :tracker_app, TrackerAppWeb.Endpoint, + http: [port: 4002], + server: false + +# Print only warnings and errors during test +config :logger, level: :warn diff --git a/lib/file_db/file_db.ex b/lib/file_db/file_db.ex new file mode 100644 index 0000000..5393a3e --- /dev/null +++ b/lib/file_db/file_db.ex @@ -0,0 +1,54 @@ +defmodule TrackerApp.FileDB do + @moduledoc """ + Flat file key-value store where each key represents a filename and values are stored in those + files using `:erlang.term_to_binary/2`. + """ + + @file_prefix "filedb_" + @file_extension "dat" + + @typedoc "ID (key) for data, only use `[a-zA-Z0-9]` for safety." + @type id :: String.t() + + @spec has_id?(id()) :: boolean() + def has_id?(id) do + id |> path_to() |> File.exists?() + end + + @spec read(id()) :: {:ok, term()} | {:error, File.posix()} + def read(id) do + with {:ok, data} <- File.read(path_to(id)), do: {:ok, from_file_format(data)} + end + + @spec write(id(), term()) :: :ok | {:error, File.posix()} + def write(id, data) do + File.write(path_to(id), to_file_format(data)) + end + + @spec delete(id()) :: :ok | {:error, File.posix()} + def delete(id) do + id |> path_to() |> File.rm() + end + + @spec to_file_format(term()) :: binary() + defp to_file_format(data) do + :erlang.term_to_binary(data, minor_version: 2, compressed: db_compression()) + end + + @spec from_file_format(binary()) :: term() + defp from_file_format(data) do + :erlang.binary_to_term(data) + end + + @spec path_to(id()) :: String.t() + defp path_to(<> = id) do + Path.join([db_path(), prefix, "#{@file_prefix}#{id}.#{@file_extension}"]) + end + + @spec db_path() :: String.t() + defp db_path(), + do: Application.get_env(:tracker_app, :file_db_path) || raise("Missing FileDB path!") + + @spec db_compression() :: 0..9 + defp db_compression(), do: Application.get_env(:tracker_app, :file_db_compression, 0) +end diff --git a/lib/item_db/item_db.ex b/lib/item_db/item_db.ex new file mode 100644 index 0000000..2e905a9 --- /dev/null +++ b/lib/item_db/item_db.ex @@ -0,0 +1,48 @@ +defmodule TrackerApp.ItemDB do + @moduledoc """ + Database for account and item data, uses FileDB as a backend. + """ + + alias TrackerApp.FileDB + alias TrackerApp.ItemDB.Schemas.Account + + @doc """ + Check if account exists with given id. + """ + @spec account_exists?(Account.id()) :: boolean() + def account_exists?(id) do + FileDB.has_id?(id) + end + + @doc """ + Create new empty account. Does not store the account, just returns it. + """ + @spec create_account() :: Account.t() + def create_account() do + Account.new() + end + + @spec get_account(Account.id()) :: {:ok, Account.t()} | {:error, term()} + def get_account(id) do + case FileDB.read(id) do + {:ok, data} -> {:ok, Account.from_storage(data)} + {:error, posix} -> {:error, "Unable to read file due to: #{inspect(posix)}"} + end + end + + @spec update_account(Account.t()) :: :ok | {:error, term()} + def update_account(account) do + case FileDB.write(account.id, Account.to_storage(account)) do + :ok -> :ok + {:error, posix} -> {:error, "Unable to write file due to: #{inspect(posix)}"} + end + end + + @spec delete_account(Account.id()) :: :ok | {:error, String.t()} + def delete_account(id) do + case FileDB.delete(id) do + :ok -> :ok + {:error, posix} -> {:error, "Unable to delete file due to: #{inspect(posix)}"} + end + end +end diff --git a/lib/item_db/schemas/account.ex b/lib/item_db/schemas/account.ex new file mode 100644 index 0000000..f162182 --- /dev/null +++ b/lib/item_db/schemas/account.ex @@ -0,0 +1,58 @@ +defmodule TrackerApp.ItemDB.Schemas.Account do + import TrackerApp.Utils.TypedStruct + + alias TrackerApp.ItemDB.Schemas.{Item, Types} + + @current_version 1 + + deftypedstruct(%{ + id: Types.id(), + items: {[Item.t()], []}, + paid_until: {Date.t() | nil, nil}, + version: {1, @current_version} + }) + + @doc """ + Create new account structure. + """ + @spec new() :: t() + def new() do + %__MODULE__{ + id: UUID.uuid4() + } + end + + @spec to_storage(t()) :: map() + def to_storage(account) do + items = Enum.map(account.items, &Item.to_storage/1) + + %{ + id: account.id, + items: items, + paid_until: account.paid_until, + version: @current_version + } + end + + @spec from_storage(map()) :: Account.t() + def from_storage(data), do: from_storage(data, data.version) + + @spec from_storage(map(), Types.version()) :: Account.t() + def from_storage(data, version) + + def from_storage(data, 1) do + items = Map.get(data, :items) |> Enum.map(&Item.from_storage(&1, 1)) + + %__MODULE__{ + id: data.id, + items: items, + paid_until: data.paid_until, + version: 1 + } + end + + @spec find_item_by_id(t(), Types.id()) :: Item.t() | nil + def find_item_by_id(account, id) do + Enum.find(account.items, &(&1.id == id)) + end +end diff --git a/lib/item_db/schemas/item.ex b/lib/item_db/schemas/item.ex new file mode 100644 index 0000000..2665901 --- /dev/null +++ b/lib/item_db/schemas/item.ex @@ -0,0 +1,37 @@ +defmodule TrackerApp.ItemDB.Schemas.Item do + import TrackerApp.Utils.TypedStruct + + deftypedstruct(%{ + id: TrackerApp.ItemDB.Schemas.Types.id(), + name: {String.t(), ""}, + expiry_date: {Date.t() | nil, nil}, + amount: {pos_integer(), 0}, + amount_unit: {String.t() | nil, nil} + }) + + @doc """ + Create new item structure. + """ + @spec new() :: t() + def new() do + %__MODULE__{ + id: UUID.uuid4() + } + end + + @spec to_storage(t()) :: map() + def to_storage(item), do: Map.from_struct(item) + + @spec from_storage(map(), TrackerApp.ItemDB.Schemas.Types.version()) :: t() + def from_storage(data, version) + + def from_storage(data, 1) do + %__MODULE__{ + id: data.id, + name: data.name, + expiry_date: data.expiry_date, + amount: data.amount, + amount_unit: data.amount_unit + } + end +end diff --git a/lib/item_db/schemas/types.ex b/lib/item_db/schemas/types.ex new file mode 100644 index 0000000..b9d3dfd --- /dev/null +++ b/lib/item_db/schemas/types.ex @@ -0,0 +1,7 @@ +defmodule TrackerApp.ItemDB.Schemas.Types do + @typedoc "Version of data in storage." + @type version :: 1 + + @typedoc "UUIDv4 used as object identifier" + @type id :: String.t() +end diff --git a/lib/item_db/schemas/utils.ex b/lib/item_db/schemas/utils.ex new file mode 100644 index 0000000..25a5f20 --- /dev/null +++ b/lib/item_db/schemas/utils.ex @@ -0,0 +1,5 @@ +defmodule TrackerApp.ItemDB.Schemas.Utils do + @id_re ~R/^[0-9a-f]{8}-[0-9a-f]{4}-4[0-9a-f]{3}-[89ab][0-9a-f]{3}-[0-9a-f]{12}$/ + + def id_re(), do: @id_re +end diff --git a/lib/tracker_app/account_manager.ex b/lib/tracker_app/account_manager.ex new file mode 100644 index 0000000..9e336df --- /dev/null +++ b/lib/tracker_app/account_manager.ex @@ -0,0 +1,121 @@ +defmodule TrackerApp.AccountManager do + @moduledoc """ + Account manager is responsible for loading account information from files and saving it back, + plus updating listening processes about the changes in information. + """ + + use GenServer, restart: :transient + + require Logger + + import TrackerApp.Utils.TypedStruct + + alias TrackerApp.ItemDB + alias TrackerApp.ItemDB.Schemas.{Account, Types} + + @alive_check_interval 10 * 1_000 + + defmodule Options do + deftypedstruct(%{ + id: Types.id() + }) + end + + defmodule State do + deftypedstruct(%{ + data: Account.t(), + update_ref: {reference() | nil, nil} + }) + end + + @spec start_link(Options.t()) :: GenServer.on_start() + def start_link(%Options{} = options) do + GenServer.start_link(__MODULE__, %{id: options.id}, + name: {:via, Registry, {__MODULE__.NameRegistry, id_to_name(options.id)}} + ) + end + + ### SERVER API + + @impl true + @spec init(%{id: Types.id()}) :: {:ok, State.t()} | {:stop, :account_not_found} + def init(%{id: id}) do + Logger.debug("Starting AccountManger for #{id}...") + + case ItemDB.get_account(id) do + {:ok, account} -> + schedule_check() + {:ok, %State{data: account}} + + {:error, _} -> + {:stop, :account_not_found} + end + end + + @impl true + def handle_call(msg, from, state) + + def handle_call(:get_data, _from, %State{} = state) do + {:reply, state.data, state} + end + + def handle_call({:update_data, %Account{} = account}, _from, %State{} = state) do + case ItemDB.update_account(account) do + :ok -> + Logger.debug("Account data updated for #{state.data.id}, updating listeners...") + listeners = Registry.lookup(__MODULE__.ClientRegistry, id_to_name(state.data.id)) + Enum.each(listeners, &send(&1 |> elem(0), {__MODULE__, :data_updated, account})) + {:reply, :ok, %State{state | data: account}} + + {:error, term} -> + {:reply, {:error, term}, state} + end + end + + @impl true + def handle_info(msg, state) + + def handle_info(:check_alive, %State{} = state) do + with results when results != [] <- + Registry.lookup(__MODULE__.ClientRegistry, id_to_name(state.data.id)), + alive when alive != [] <- Enum.filter(results, &(&1 |> elem(0) |> Process.alive?())) do + schedule_check() + {:noreply, state} + else + _ -> + Logger.debug("No more listeners, AccountManager for #{state.data.id} closing...") + :ok = ItemDB.update_account(state.data) + {:stop, :normal, state} + end + end + + ### CLIENT API + + @spec start(Options.t()) :: DynamicSupervisor.on_start_child() + def start(%Options{} = options) do + DynamicSupervisor.start_child(__MODULE__.Supervisor, {__MODULE__, options}) + end + + @spec get_data(GenServer.name()) :: Account.t() + def get_data(server) do + GenServer.call(server, :get_data) + end + + @spec update_data(GenServer.name(), Account.t()) :: :ok + def update_data(server, %Account{} = account) do + GenServer.call(server, {:update_data, account}) + end + + @spec register(GenServer.name()) :: Account.t() + def register(server) do + account = get_data(server) + {:ok, _} = Registry.register(__MODULE__.ClientRegistry, id_to_name(account.id), nil) + account + end + + @spec schedule_check() :: reference() + defp schedule_check(), do: Process.send_after(self(), :check_alive, @alive_check_interval) + + @spec id_to_name(Types.id()) :: String.t() + defp id_to_name(id), do: "account:#{id}" +end diff --git a/lib/tracker_app/application.ex b/lib/tracker_app/application.ex new file mode 100644 index 0000000..b56e857 --- /dev/null +++ b/lib/tracker_app/application.ex @@ -0,0 +1,35 @@ +defmodule TrackerApp.Application do + # See https://hexdocs.pm/elixir/Application.html + # for more information on OTP Applications + @moduledoc false + + use Application + + def start(_type, _args) do + children = [ + # Start the Telemetry supervisor + TrackerAppWeb.Telemetry, + # Start the PubSub system + {Phoenix.PubSub, name: TrackerApp.PubSub}, + # Start the Endpoint (http/https) + TrackerAppWeb.Endpoint, + # Start a worker by calling: TrackerApp.Worker.start_link(arg) + # {TrackerApp.Worker, arg} + {Registry, keys: :unique, name: TrackerApp.AccountManager.NameRegistry}, + {Registry, keys: :duplicate, name: TrackerApp.AccountManager.ClientRegistry}, + {DynamicSupervisor, strategy: :one_for_one, name: TrackerApp.AccountManager.Supervisor} + ] + + # See https://hexdocs.pm/elixir/Supervisor.html + # for other strategies and supported options + opts = [strategy: :one_for_one, name: TrackerApp.Supervisor] + Supervisor.start_link(children, opts) + end + + # Tell Phoenix to update the endpoint configuration + # whenever the application is updated. + def config_change(changed, _new, removed) do + TrackerAppWeb.Endpoint.config_change(changed, removed) + :ok + end +end diff --git a/lib/tracker_app/utils/typed_struct.ex b/lib/tracker_app/utils/typed_struct.ex new file mode 100644 index 0000000..73932d4 --- /dev/null +++ b/lib/tracker_app/utils/typed_struct.ex @@ -0,0 +1,71 @@ +defmodule TrackerApp.Utils.TypedStruct do + @doc """ + Create typed struct with a type, default values, and enforced keys. + + Input should be a map where the key names are names of the struct keys and values are the + field information. The value can be a typespec, in which case the field will be enforced, or + a 2-tuple of `{typespec, default_value}`, making the field unenforced. + + To prevent ambiguity, a value of `{typespec, :ts_enforced}` will be interpreted as enforced, + this will allow you to typespec a 2-tuple. + + NOTE: Due to the ambiguity removal technique above, `:ts_enforced` is not allowed as a default + value. + + Example: + + ```elixir + deftypedstruct(%{ + # Enforced with simple type + foo: integer(), + + # Enforced 2-tuple typed field, written like this to remove ambiguity + bar: {{String.t(), integer()}, :ts_enforced}, + + # Non-enforced field with default value + baz: {any(), ""} + }) + ``` + """ + defmacro deftypedstruct(fields) do + fields_list = + case fields do + {:%{}, _, flist} -> flist + _ -> raise ArgumentError, "Fields must be a map!" + end + + enforced_list = + fields_list + |> Enum.filter(fn + {_, {_, :ts_enforced}} -> true + {_, {_, _}} -> false + {_, _} -> true + end) + |> Enum.map(&elem(&1, 0)) + + field_specs = + Enum.map(fields_list, fn + {field, {typespec, :ts_enforced}} -> + {field, typespec} + + {field, {typespec, _}} -> + {field, typespec} + + {field, typespec} -> + {field, typespec} + end) + + field_vals = + Enum.map(fields_list, fn + {field, {_, :ts_enforced}} -> field + {field, {_, default}} -> {field, default} + {field, _} -> field + end) + + quote do + @type t :: %__MODULE__{unquote_splicing(field_specs)} + @enforce_keys unquote(enforced_list) + defstruct unquote(field_vals) + end + end +end diff --git a/lib/tracker_app_web.ex b/lib/tracker_app_web.ex new file mode 100644 index 0000000..022ddfe --- /dev/null +++ b/lib/tracker_app_web.ex @@ -0,0 +1,102 @@ +defmodule TrackerAppWeb do + @moduledoc """ + The entrypoint for defining your web interface, such + as controllers, views, channels and so on. + + This can be used in your application as: + + use TrackerAppWeb, :controller + use TrackerAppWeb, :view + + The definitions below will be executed for every view, + controller, etc, so keep them short and clean, focused + on imports, uses and aliases. + + Do NOT define functions inside the quoted expressions + below. Instead, define any helper function in modules + and import those modules here. + """ + + def controller do + quote do + use Phoenix.Controller, namespace: TrackerAppWeb + + import Plug.Conn + import TrackerAppWeb.Gettext + alias TrackerAppWeb.Router.Helpers, as: Routes + end + end + + def view do + quote do + use Phoenix.View, + root: "lib/tracker_app_web/templates", + namespace: TrackerAppWeb + + # Import convenience functions from controllers + import Phoenix.Controller, + only: [get_flash: 1, get_flash: 2, view_module: 1, view_template: 1] + + # Include shared imports and aliases for views + unquote(view_helpers()) + end + end + + def live_view do + quote do + use Phoenix.LiveView, + layout: {TrackerAppWeb.LayoutView, "live.html"} + + unquote(view_helpers()) + end + end + + def live_component do + quote do + use Phoenix.LiveComponent + + unquote(view_helpers()) + end + end + + def router do + quote do + use Phoenix.Router + + import Plug.Conn + import Phoenix.Controller + import Phoenix.LiveView.Router + end + end + + def channel do + quote do + use Phoenix.Channel + import TrackerAppWeb.Gettext + end + end + + defp view_helpers do + quote do + # Use all HTML functionality (forms, tags, etc) + use Phoenix.HTML + + # Import LiveView helpers (live_render, live_component, live_patch, etc) + import Phoenix.LiveView.Helpers + + # Import basic rendering functionality (render, render_layout, etc) + import Phoenix.View + + import TrackerAppWeb.ErrorHelpers + import TrackerAppWeb.Gettext + alias TrackerAppWeb.Router.Helpers, as: Routes + end + end + + @doc """ + When used, dispatch to the appropriate controller/view/etc. + """ + defmacro __using__(which) when is_atom(which) do + apply(__MODULE__, which, []) + end +end diff --git a/lib/tracker_app_web/endpoint.ex b/lib/tracker_app_web/endpoint.ex new file mode 100644 index 0000000..aa9b39a --- /dev/null +++ b/lib/tracker_app_web/endpoint.ex @@ -0,0 +1,49 @@ +defmodule TrackerAppWeb.Endpoint do + use Phoenix.Endpoint, otp_app: :tracker_app + + # The session will be stored in the cookie and signed, + # this means its contents can be read but not tampered with. + # Set :encryption_salt if you would also like to encrypt it. + @session_options [ + store: :cookie, + key: "_tracker_app_key", + signing_salt: "JYoZEADW" + ] + + socket "/live", Phoenix.LiveView.Socket, websocket: [connect_info: [session: @session_options]] + + # Serve at "/" the static files from "priv/static" directory. + # + # You should set gzip to true if you are running phx.digest + # when deploying your static files in production. + plug Plug.Static, + at: "/", + from: :tracker_app, + gzip: false, + only: ~w(css fonts images js favicon.ico robots.txt) + + # Code reloading can be explicitly enabled under the + # :code_reloader configuration of your endpoint. + if code_reloading? do + socket "/phoenix/live_reload/socket", Phoenix.LiveReloader.Socket + plug Phoenix.LiveReloader + plug Phoenix.CodeReloader + end + + plug Phoenix.LiveDashboard.RequestLogger, + param_key: "request_logger", + cookie_key: "request_logger" + + plug Plug.RequestId + plug Plug.Telemetry, event_prefix: [:phoenix, :endpoint] + + plug Plug.Parsers, + parsers: [:urlencoded, :multipart, :json], + pass: ["*/*"], + json_decoder: Phoenix.json_library() + + plug Plug.MethodOverride + plug Plug.Head + plug Plug.Session, @session_options + plug TrackerAppWeb.Router +end diff --git a/lib/tracker_app_web/gettext.ex b/lib/tracker_app_web/gettext.ex new file mode 100644 index 0000000..888ed7d --- /dev/null +++ b/lib/tracker_app_web/gettext.ex @@ -0,0 +1,24 @@ +defmodule TrackerAppWeb.Gettext do + @moduledoc """ + A module providing Internationalization with a gettext-based API. + + By using [Gettext](https://hexdocs.pm/gettext), + your module gains a set of macros for translations, for example: + + import TrackerAppWeb.Gettext + + # Simple translation + gettext("Here is the string to translate") + + # Plural translation + ngettext("Here is the string to translate", + "Here are the strings to translate", + 3) + + # Domain-based translation + dgettext("errors", "Here is the error message to translate") + + See the [Gettext Docs](https://hexdocs.pm/gettext) for detailed usage. + """ + use Gettext, otp_app: :tracker_app +end diff --git a/lib/tracker_app_web/live/account_live.ex b/lib/tracker_app_web/live/account_live.ex new file mode 100644 index 0000000..ffdf7c6 --- /dev/null +++ b/lib/tracker_app_web/live/account_live.ex @@ -0,0 +1,78 @@ +defmodule TrackerAppWeb.AccountLive do + use TrackerAppWeb, :live_view + + alias TrackerApp.ItemDB.Schemas.Account + alias TrackerApp.AccountManager + + @impl true + def mount(%{"accountId" => id}, _session, socket) do + manager = + case AccountManager.start(%AccountManager.Options{id: id}) do + {:ok, pid} -> pid + {:error, {:already_started, pid}} -> pid + {:error, :account_not_found} -> :account_not_found + end + + if manager != :account_not_found do + account = + if connected?(socket) do + AccountManager.register(manager) + else + AccountManager.get_data(manager) + end + + socket = common_assigns(socket, account) + {:ok, assign(socket, account: account, manager: manager)} + else + {:ok, assign(socket, :account, :not_found)} + end + end + + @impl true + def handle_event(event, params, socket) + + def handle_event("add-item", _params, socket) do + {:noreply, + push_redirect(socket, + to: + Routes.item_path( + socket, + socket.assigns.live_action, + socket.assigns.account.id + ) + )} + end + + def handle_event("edit-item", params, socket) do + item = Account.find_item_by_id(socket.assigns.account, params["id"]) + + {:noreply, + push_redirect(socket, + to: + Routes.item_path(socket, socket.assigns.live_action, socket.assigns.account.id, item.id) + )} + end + + @impl true + def handle_info(msg, socket) + + def handle_info({AccountManager, :data_updated, %Account{} = account}, socket) do + socket = common_assigns(socket, account) + {:noreply, assign(socket, account: account)} + end + + @spec common_assigns(Phoenix.LiveView.Socket.t(), Account.t()) :: Phoenix.LiveView.Socket.t() + defp common_assigns(socket, %Account{} = account) do + # TODO: UTC + today = Date.utc_today() + + {expired, nonexpired} = + Enum.split_with(account.items, &(Date.compare(&1.expiry_date, today) == :lt)) + + assign( + socket, + expired: expired, + nonexpired: nonexpired + ) + end +end diff --git a/lib/tracker_app_web/live/account_live.html.leex b/lib/tracker_app_web/live/account_live.html.leex new file mode 100644 index 0000000..e350515 --- /dev/null +++ b/lib/tracker_app_web/live/account_live.html.leex @@ -0,0 +1,23 @@ +<%= if @account == :not_found do %> + <%= render(TrackerAppWeb.UtilsView, "account_not_found.html", []) %> +<% else %> +
+ <%= for item <- @nonexpired do %> +
+ <%= item.name %> <%= item.amount %> <%= item.amount_unit %>, expires <%= Date.to_iso8601(item.expiry_date) %> +
+ <% end %> + + + + <%= if @expired != [] do %> +
+ + <%= for item <- @expired do %> +
+ <%= item.name %> <%= item.amount %> <%= item.amount_unit %>, expires <%= Date.to_iso8601(item.expiry_date) %> +
+ <% end %> + <% end %> +
+<% end %> diff --git a/lib/tracker_app_web/live/components/item_editor.ex b/lib/tracker_app_web/live/components/item_editor.ex new file mode 100644 index 0000000..6ad62f0 --- /dev/null +++ b/lib/tracker_app_web/live/components/item_editor.ex @@ -0,0 +1,15 @@ +defmodule TrackerApp.Live.Components.ItemEditor do + use Phoenix.LiveComponent + + @impl true + def update(assigns, socket) do + assigns = Map.put(assigns, :invalid, false) + {:ok, %{socket | assigns: Map.merge(socket.assigns, assigns)}} + end + + @impl true + def handle_event("save-item", params, socket) do + IO.inspect(socket |> Map.to_list()) + {:noreply, socket} + end +end diff --git a/lib/tracker_app_web/live/components/item_editor.html.leex b/lib/tracker_app_web/live/components/item_editor.html.leex new file mode 100644 index 0000000..a7c16de --- /dev/null +++ b/lib/tracker_app_web/live/components/item_editor.html.leex @@ -0,0 +1,70 @@ +
+
+ <%= if @invalid do %> +
+ There was an error with the input data, please check the values. +
+ <% end %> + + + + + +
+ + + + + +
+ +
+ <% ed = if not is_nil(@item.expiry_date), do: Date.to_iso8601(@item.expiry_date) , else: "" %> + +
+ + + +
+
diff --git a/lib/tracker_app_web/live/item_live.ex b/lib/tracker_app_web/live/item_live.ex new file mode 100644 index 0000000..dd0a827 --- /dev/null +++ b/lib/tracker_app_web/live/item_live.ex @@ -0,0 +1,105 @@ +defmodule TrackerAppWeb.ItemLive do + use TrackerAppWeb, :live_view + + alias TrackerApp.ItemDB.Schemas.{Account, Item, Types} + alias TrackerApp.AccountManager + + @impl true + def mount(%{"accountId" => id} = params, _session, socket) do + socket = assign(socket, invalid: false) + + manager = + case AccountManager.start(%AccountManager.Options{id: id}) do + {:ok, pid} -> pid + {:error, {:already_started, pid}} -> pid + {:error, :account_not_found} -> :account_not_found + end + + if manager != :account_not_found do + account = + if connected?(socket) do + AccountManager.register(manager) + else + AccountManager.get_data(manager) + end + + item_id = Map.get(params, "itemId") + + {is_new?, item} = + if is_nil(item_id) do + {true, Item.new()} + else + {false, find_item_by_id(account.items, item_id) || :not_found} + end + + {:ok, assign(socket, account: account, item: item, is_new?: is_new?, manager: manager)} + else + {:ok, assign(socket, account: :not_found)} + end + end + + @impl true + def handle_event(event, params, socket) + + def handle_event("save-item", params, socket) do + with true <- Map.has_key?(params, "name"), + true <- Map.has_key?(params, "amount_unit"), + {:ok, expiry_date} <- Date.from_iso8601(params["expiry"]), + {amount, _} when amount >= 0 <- Integer.parse(params["amount"]), + true <- String.length(params["name"]) > 0 do + existing_item = find_item_by_id(socket.assigns.account.items, socket.assigns.item.id) + + item = %Item{ + socket.assigns.item + | name: params["name"], + amount: amount, + expiry_date: expiry_date, + amount_unit: params["amount_unit"] + } + + items = + if is_nil(existing_item) do + [item | socket.assigns.account.items] + else + [item | Enum.filter(socket.assigns.account.items, &(&1.id != item.id))] + end + + account = %Account{socket.assigns.account | items: items} + :ok = AccountManager.update_data(socket.assigns.manager, account) + + {:noreply, push_redirect(socket, to: Routes.account_path(socket, :paid, account.id))} + else + _ -> + {:noreply, assign(socket, invalid: true)} + end + end + + def handle_event("delete-item", _params, socket) do + existing_item = find_item_by_id(socket.assigns.account.items, socket.assigns.item.id) + + items = + if not is_nil(existing_item) do + Enum.filter(socket.assigns.account.items, &(&1.id != socket.assigns.item.id)) + else + # Item had already been deleted, do nothing + socket.assigns.account.items + end + + account = %Account{socket.assigns.account | items: items} + :ok = AccountManager.update_data(socket.assigns.manager, account) + + {:noreply, push_redirect(socket, to: Routes.account_path(socket, :paid, account.id))} + end + + @impl true + def handle_info(msg, socket) + + def handle_info({AccountManager, :data_updated, %Account{} = account}, socket) do + {:noreply, assign(socket, account: account)} + end + + @spec find_item_by_id([Item.t()], Types.id()) :: Item.t() | nil + defp find_item_by_id(items, id) do + Enum.find(items, &(&1.id == id)) + end +end diff --git a/lib/tracker_app_web/live/item_live.html.leex b/lib/tracker_app_web/live/item_live.html.leex new file mode 100644 index 0000000..cc0d96b --- /dev/null +++ b/lib/tracker_app_web/live/item_live.html.leex @@ -0,0 +1,88 @@ +<%= if @account == :not_found do %> + <%= render(TrackerAppWeb.UtilsView, "account_not_found.html", []) %> +<% else %> +
+ <%= if @item == :not_found do %> +
+

This item was not found.

+ + <%= live_redirect(to: Routes.account_path(@socket, :paid, @account.id)) do %> + Go back to main view. + <% end %> +
+ <% else %> +
+ <%= if @invalid do %> +
+ There was an error with the input data, please check the values. +
+ <% end %> + + + + + +
+ + + + + +
+ +
+ <% ed = if not is_nil(@item.expiry_date), do: Date.to_iso8601(@item.expiry_date) , else: "" %> + +
+ + <%= if not @is_new? do %> + + <% end %> + + +
+ <% end %> +
+<% end %> diff --git a/lib/tracker_app_web/live/page_live.ex b/lib/tracker_app_web/live/page_live.ex new file mode 100644 index 0000000..d5cd433 --- /dev/null +++ b/lib/tracker_app_web/live/page_live.ex @@ -0,0 +1,8 @@ +defmodule TrackerAppWeb.PageLive do + use TrackerAppWeb, :live_view + + @impl true + def mount(_params, _session, socket) do + {:ok, socket} + end +end diff --git a/lib/tracker_app_web/live/page_live.html.leex b/lib/tracker_app_web/live/page_live.html.leex new file mode 100644 index 0000000..e84d301 --- /dev/null +++ b/lib/tracker_app_web/live/page_live.html.leex @@ -0,0 +1,14 @@ +
+

TrackerApp

+ +

Welcome to TrackerApp.

+ +
+ <%= live_redirect(to: Routes.register_path(@socket, :free)) do %> + Free mode + <% end %> + <%= live_redirect(to: Routes.register_path(@socket, :paid)) do %> + Paid mode + <% end %> +
+
diff --git a/lib/tracker_app_web/live/register_live.ex b/lib/tracker_app_web/live/register_live.ex new file mode 100644 index 0000000..40859f4 --- /dev/null +++ b/lib/tracker_app_web/live/register_live.ex @@ -0,0 +1,20 @@ +defmodule TrackerAppWeb.RegisterLive do + use TrackerAppWeb, :live_view + + alias TrackerApp.ItemDB + + @impl true + def mount(_params, _session, socket) do + {:ok, socket} + end + + @impl true + def handle_event(evt, params, socket) + + def handle_event("create", _, socket) do + with account <- ItemDB.create_account(), + :ok <- ItemDB.update_account(account) do + {:noreply, push_redirect(socket, to: Routes.account_path(socket, :paid, account.id))} + end + end +end diff --git a/lib/tracker_app_web/live/register_live.html.leex b/lib/tracker_app_web/live/register_live.html.leex new file mode 100644 index 0000000..47bac89 --- /dev/null +++ b/lib/tracker_app_web/live/register_live.html.leex @@ -0,0 +1,2 @@ +<%= render(TrackerAppWeb.TermsView, "terms.html", []) %> + diff --git a/lib/tracker_app_web/router.ex b/lib/tracker_app_web/router.ex new file mode 100644 index 0000000..3d727d9 --- /dev/null +++ b/lib/tracker_app_web/router.ex @@ -0,0 +1,53 @@ +defmodule TrackerAppWeb.Router do + use TrackerAppWeb, :router + + pipeline :browser do + plug :accepts, ["html"] + plug :fetch_session + plug :fetch_live_flash + plug :put_root_layout, {TrackerAppWeb.LayoutView, :root} + plug :protect_from_forgery + plug :put_secure_browser_headers + end + + pipeline :api do + plug :accepts, ["json"] + end + + scope "/", TrackerAppWeb do + pipe_through :browser + + live "/", PageLive, :index + live "/register/free", RegisterLive, :free + live "/register/paid", RegisterLive, :paid + + live "/free", AccountLive, :free + live "/account/:accountId", AccountLive, :paid + + live "/free/item/new", ItemLive, :free + live "/free/item/:itemId", ItemLive, :free + live "/account/:accountId/item/new", ItemLive, :paid + live "/account/:accountId/item/:itemId", ItemLive, :paid + end + + # Other scopes may use custom stacks. + # scope "/api", TrackerAppWeb do + # pipe_through :api + # end + + # Enables LiveDashboard only for development + # + # If you want to use the LiveDashboard in production, you should put + # it behind authentication and allow only admins to access it. + # If your application does not have an admins-only section yet, + # you can use Plug.BasicAuth to set up some basic authentication + # as long as you are also using SSL (which you should anyway). + if Mix.env() in [:dev, :test] do + import Phoenix.LiveDashboard.Router + + scope "/" do + pipe_through :browser + live_dashboard "/dashboard", metrics: TrackerAppWeb.Telemetry + end + end +end diff --git a/lib/tracker_app_web/telemetry.ex b/lib/tracker_app_web/telemetry.ex new file mode 100644 index 0000000..a988bb0 --- /dev/null +++ b/lib/tracker_app_web/telemetry.ex @@ -0,0 +1,48 @@ +defmodule TrackerAppWeb.Telemetry do + use Supervisor + import Telemetry.Metrics + + def start_link(arg) do + Supervisor.start_link(__MODULE__, arg, name: __MODULE__) + end + + @impl true + def init(_arg) do + children = [ + # Telemetry poller will execute the given period measurements + # every 10_000ms. Learn more here: https://hexdocs.pm/telemetry_metrics + {:telemetry_poller, measurements: periodic_measurements(), period: 10_000} + # Add reporters as children of your supervision tree. + # {Telemetry.Metrics.ConsoleReporter, metrics: metrics()} + ] + + Supervisor.init(children, strategy: :one_for_one) + end + + def metrics do + [ + # Phoenix Metrics + summary("phoenix.endpoint.stop.duration", + unit: {:native, :millisecond} + ), + summary("phoenix.router_dispatch.stop.duration", + tags: [:route], + unit: {:native, :millisecond} + ), + + # VM Metrics + summary("vm.memory.total", unit: {:byte, :kilobyte}), + summary("vm.total_run_queue_lengths.total"), + summary("vm.total_run_queue_lengths.cpu"), + summary("vm.total_run_queue_lengths.io") + ] + end + + defp periodic_measurements do + [ + # A module, function and arguments to be invoked periodically. + # This function must call :telemetry.execute/3 and a metric must be added above. + # {TrackerAppWeb, :count_users, []} + ] + end +end diff --git a/lib/tracker_app_web/templates/layout/app.html.eex b/lib/tracker_app_web/templates/layout/app.html.eex new file mode 100644 index 0000000..09ffdad --- /dev/null +++ b/lib/tracker_app_web/templates/layout/app.html.eex @@ -0,0 +1,5 @@ +
+ + + <%= @inner_content %> +
diff --git a/lib/tracker_app_web/templates/layout/live.html.leex b/lib/tracker_app_web/templates/layout/live.html.leex new file mode 100644 index 0000000..3cfbbe1 --- /dev/null +++ b/lib/tracker_app_web/templates/layout/live.html.leex @@ -0,0 +1,20 @@ +
+ +
+
+ + + + + <%= @inner_content %> +
diff --git a/lib/tracker_app_web/templates/layout/root.html.leex b/lib/tracker_app_web/templates/layout/root.html.leex new file mode 100644 index 0000000..6f9ee0e --- /dev/null +++ b/lib/tracker_app_web/templates/layout/root.html.leex @@ -0,0 +1,15 @@ + + + + + + + <%= csrf_meta_tag() %> + <%= live_title_tag assigns[:page_title] || "TrackerApp", suffix: " ยท Phoenix Framework" %> + "/> + + + + <%= @inner_content %> + + diff --git a/lib/tracker_app_web/templates/terms/terms.html.leex b/lib/tracker_app_web/templates/terms/terms.html.leex new file mode 100644 index 0000000..789c5d9 --- /dev/null +++ b/lib/tracker_app_web/templates/terms/terms.html.leex @@ -0,0 +1,3 @@ +

Legal terms

+ +

Something something blah blah blah.

diff --git a/lib/tracker_app_web/templates/utils/account_not_found.html.leex b/lib/tracker_app_web/templates/utils/account_not_found.html.leex new file mode 100644 index 0000000..09c738c --- /dev/null +++ b/lib/tracker_app_web/templates/utils/account_not_found.html.leex @@ -0,0 +1,3 @@ + diff --git a/lib/tracker_app_web/views/error_helpers.ex b/lib/tracker_app_web/views/error_helpers.ex new file mode 100644 index 0000000..beafc2c --- /dev/null +++ b/lib/tracker_app_web/views/error_helpers.ex @@ -0,0 +1,47 @@ +defmodule TrackerAppWeb.ErrorHelpers do + @moduledoc """ + Conveniences for translating and building error messages. + """ + + use Phoenix.HTML + + @doc """ + Generates tag for inlined form input errors. + """ + def error_tag(form, field) do + Enum.map(Keyword.get_values(form.errors, field), fn error -> + content_tag(:span, translate_error(error), + class: "invalid-feedback", + phx_feedback_for: input_id(form, field) + ) + end) + end + + @doc """ + Translates an error message using gettext. + """ + def translate_error({msg, opts}) do + # When using gettext, we typically pass the strings we want + # to translate as a static argument: + # + # # Translate "is invalid" in the "errors" domain + # dgettext("errors", "is invalid") + # + # # Translate the number of files with plural rules + # dngettext("errors", "1 file", "%{count} files", count) + # + # Because the error messages we show in our forms and APIs + # are defined inside Ecto, we need to translate them dynamically. + # This requires us to call the Gettext module passing our gettext + # backend as first argument. + # + # Note we use the "errors" domain, which means translations + # should be written to the errors.po file. The :count option is + # set by Ecto and indicates we should also apply plural rules. + if count = opts[:count] do + Gettext.dngettext(TrackerAppWeb.Gettext, "errors", msg, msg, count, opts) + else + Gettext.dgettext(TrackerAppWeb.Gettext, "errors", msg, opts) + end + end +end diff --git a/lib/tracker_app_web/views/error_view.ex b/lib/tracker_app_web/views/error_view.ex new file mode 100644 index 0000000..dec3ce0 --- /dev/null +++ b/lib/tracker_app_web/views/error_view.ex @@ -0,0 +1,16 @@ +defmodule TrackerAppWeb.ErrorView do + use TrackerAppWeb, :view + + # If you want to customize a particular status code + # for a certain format, you may uncomment below. + # def render("500.html", _assigns) do + # "Internal Server Error" + # end + + # By default, Phoenix returns the status message from + # the template name. For example, "404.html" becomes + # "Not Found". + def template_not_found(template, _assigns) do + Phoenix.Controller.status_message_from_template(template) + end +end diff --git a/lib/tracker_app_web/views/layout_view.ex b/lib/tracker_app_web/views/layout_view.ex new file mode 100644 index 0000000..9165b3f --- /dev/null +++ b/lib/tracker_app_web/views/layout_view.ex @@ -0,0 +1,3 @@ +defmodule TrackerAppWeb.LayoutView do + use TrackerAppWeb, :view +end diff --git a/lib/tracker_app_web/views/terms_view.ex b/lib/tracker_app_web/views/terms_view.ex new file mode 100644 index 0000000..c371381 --- /dev/null +++ b/lib/tracker_app_web/views/terms_view.ex @@ -0,0 +1,3 @@ +defmodule TrackerAppWeb.TermsView do + use TrackerAppWeb, :view +end diff --git a/lib/tracker_app_web/views/utils_view.ex b/lib/tracker_app_web/views/utils_view.ex new file mode 100644 index 0000000..be28dae --- /dev/null +++ b/lib/tracker_app_web/views/utils_view.ex @@ -0,0 +1,3 @@ +defmodule TrackerAppWeb.UtilsView do + use TrackerAppWeb, :view +end diff --git a/mix.exs b/mix.exs new file mode 100644 index 0000000..6aa068c --- /dev/null +++ b/mix.exs @@ -0,0 +1,62 @@ +defmodule TrackerApp.MixProject do + use Mix.Project + + def project do + [ + app: :tracker_app, + version: "0.1.0", + elixir: "~> 1.11", + elixirc_paths: elixirc_paths(Mix.env()), + compilers: [:phoenix, :gettext] ++ Mix.compilers(), + start_permanent: Mix.env() == :prod, + aliases: aliases(), + deps: deps() + ] + end + + # Configuration for the OTP application. + # + # Type `mix help compile.app` for more information. + def application do + [ + mod: {TrackerApp.Application, []}, + extra_applications: [:logger, :runtime_tools, :os_mon] + ] + end + + # Specifies which paths to compile per environment. + defp elixirc_paths(:test), do: ["lib", "test/support"] + defp elixirc_paths(_), do: ["lib"] + + # Specifies your project dependencies. + # + # Type `mix help deps` for examples and options. + defp deps do + [ + {:phoenix, "~> 1.5.5"}, + {:phoenix_live_view, "~> 0.15.3"}, + {:floki, ">= 0.27.0", only: :test}, + {:phoenix_html, "~> 2.11"}, + {:phoenix_live_reload, "~> 1.2", only: :dev}, + {:phoenix_live_dashboard, "~> 0.2"}, + {:telemetry_metrics, "~> 0.6"}, + {:telemetry_poller, "~> 0.4"}, + {:gettext, "~> 0.11"}, + {:jason, "~> 1.0"}, + {:plug_cowboy, "~> 2.0"}, + {:elixir_uuid, "~> 1.2"} + ] + end + + # Aliases are shortcuts or tasks specific to the current project. + # For example, to install project dependencies and perform other setup tasks, run: + # + # $ mix setup + # + # See the documentation for `Mix` for more info on aliases. + defp aliases do + [ + setup: ["deps.get", "cmd npm install --prefix assets"] + ] + end +end diff --git a/mix.lock b/mix.lock new file mode 100644 index 0000000..07caf80 --- /dev/null +++ b/mix.lock @@ -0,0 +1,25 @@ +%{ + "cowboy": {:hex, :cowboy, "2.8.0", "f3dc62e35797ecd9ac1b50db74611193c29815401e53bac9a5c0577bd7bc667d", [:rebar3], [{:cowlib, "~> 2.9.1", [hex: :cowlib, repo: "hexpm", optional: false]}, {:ranch, "~> 1.7.1", [hex: :ranch, repo: "hexpm", optional: false]}], "hexpm", "4643e4fba74ac96d4d152c75803de6fad0b3fa5df354c71afdd6cbeeb15fac8a"}, + "cowboy_telemetry": {:hex, :cowboy_telemetry, "0.3.1", "ebd1a1d7aff97f27c66654e78ece187abdc646992714164380d8a041eda16754", [:rebar3], [{:cowboy, "~> 2.7", [hex: :cowboy, repo: "hexpm", optional: false]}, {:telemetry, "~> 0.4", [hex: :telemetry, repo: "hexpm", optional: false]}], "hexpm", "3a6efd3366130eab84ca372cbd4a7d3c3a97bdfcfb4911233b035d117063f0af"}, + "cowlib": {:hex, :cowlib, "2.9.1", "61a6c7c50cf07fdd24b2f45b89500bb93b6686579b069a89f88cb211e1125c78", [:rebar3], [], "hexpm", "e4175dc240a70d996156160891e1c62238ede1729e45740bdd38064dad476170"}, + "elixir_uuid": {:hex, :elixir_uuid, "1.2.1", "dce506597acb7e6b0daeaff52ff6a9043f5919a4c3315abb4143f0b00378c097", [:mix], [], "hexpm", "f7eba2ea6c3555cea09706492716b0d87397b88946e6380898c2889d68585752"}, + "file_system": {:hex, :file_system, "0.2.10", "fb082005a9cd1711c05b5248710f8826b02d7d1784e7c3451f9c1231d4fc162d", [:mix], [], "hexpm", "41195edbfb562a593726eda3b3e8b103a309b733ad25f3d642ba49696bf715dc"}, + "floki": {:hex, :floki, "0.29.0", "b1710d8c93a2f860dc2d7adc390dd808dc2fb8f78ee562304457b75f4c640881", [:mix], [{:html_entities, "~> 0.5.0", [hex: :html_entities, repo: "hexpm", optional: false]}], "hexpm", "008585ce64b9f74c07d32958ec9866f4b8a124bf4da1e2941b28e41384edaaad"}, + "gettext": {:hex, :gettext, "0.18.2", "7df3ea191bb56c0309c00a783334b288d08a879f53a7014341284635850a6e55", [:mix], [], "hexpm", "f9f537b13d4fdd30f3039d33cb80144c3aa1f8d9698e47d7bcbcc8df93b1f5c5"}, + "html_entities": {:hex, :html_entities, "0.5.1", "1c9715058b42c35a2ab65edc5b36d0ea66dd083767bef6e3edb57870ef556549", [:mix], [], "hexpm", "30efab070904eb897ff05cd52fa61c1025d7f8ef3a9ca250bc4e6513d16c32de"}, + "jason": {:hex, :jason, "1.2.2", "ba43e3f2709fd1aa1dce90aaabfd039d000469c05c56f0b8e31978e03fa39052", [:mix], [{:decimal, "~> 1.0 or ~> 2.0", [hex: :decimal, repo: "hexpm", optional: true]}], "hexpm", "18a228f5f0058ee183f29f9eae0805c6e59d61c3b006760668d8d18ff0d12179"}, + "mime": {:hex, :mime, "1.5.0", "203ef35ef3389aae6d361918bf3f952fa17a09e8e43b5aa592b93eba05d0fb8d", [:mix], [], "hexpm", "55a94c0f552249fc1a3dd9cd2d3ab9de9d3c89b559c2bd01121f824834f24746"}, + "phoenix": {:hex, :phoenix, "1.5.7", "2923bb3af924f184459fe4fa4b100bd25fa6468e69b2803dfae82698269aa5e0", [:mix], [{:jason, "~> 1.0", [hex: :jason, repo: "hexpm", optional: true]}, {:phoenix_html, "~> 2.13", [hex: :phoenix_html, repo: "hexpm", optional: true]}, {:phoenix_pubsub, "~> 2.0", [hex: :phoenix_pubsub, repo: "hexpm", optional: false]}, {:plug, "~> 1.10", [hex: :plug, repo: "hexpm", optional: false]}, {:plug_cowboy, "~> 1.0 or ~> 2.2", [hex: :plug_cowboy, repo: "hexpm", optional: true]}, {:plug_crypto, "~> 1.1.2 or ~> 1.2", [hex: :plug_crypto, repo: "hexpm", optional: false]}, {:telemetry, "~> 0.4", [hex: :telemetry, repo: "hexpm", optional: false]}], "hexpm", "774cd64417c5a3788414fdbb2be2eb9bcd0c048d9e6ad11a0c1fd67b7c0d0978"}, + "phoenix_html": {:hex, :phoenix_html, "2.14.3", "51f720d0d543e4e157ff06b65de38e13303d5778a7919bcc696599e5934271b8", [:mix], [{:plug, "~> 1.5", [hex: :plug, repo: "hexpm", optional: false]}], "hexpm", "efd697a7fff35a13eeeb6b43db884705cba353a1a41d127d118fda5f90c8e80f"}, + "phoenix_live_dashboard": {:hex, :phoenix_live_dashboard, "0.4.0", "87990e68b60213d7487e65814046f9a2bed4a67886c943270125913499b3e5c3", [:mix], [{:ecto_psql_extras, "~> 0.4.1 or ~> 0.5", [hex: :ecto_psql_extras, repo: "hexpm", optional: true]}, {:phoenix_html, "~> 2.14.1 or ~> 2.15", [hex: :phoenix_html, repo: "hexpm", optional: false]}, {:phoenix_live_view, "~> 0.15.0", [hex: :phoenix_live_view, repo: "hexpm", optional: false]}, {:telemetry_metrics, "~> 0.4.0 or ~> 0.5.0 or ~> 0.6.0", [hex: :telemetry_metrics, repo: "hexpm", optional: false]}], "hexpm", "8d52149e58188e9e4497cc0d8900ab94d9b66f96998ec38c47c7a4f8f4f50e57"}, + "phoenix_live_reload": {:hex, :phoenix_live_reload, "1.3.0", "f35f61c3f959c9a01b36defaa1f0624edd55b87e236b606664a556d6f72fd2e7", [:mix], [{:file_system, "~> 0.2.1 or ~> 0.3", [hex: :file_system, repo: "hexpm", optional: false]}, {:phoenix, "~> 1.4", [hex: :phoenix, repo: "hexpm", optional: false]}], "hexpm", "02c1007ae393f2b76ec61c1a869b1e617179877984678babde131d716f95b582"}, + "phoenix_live_view": {:hex, :phoenix_live_view, "0.15.3", "70c7917e5c421e32d1a1c8ddf8123378bb741748cd8091eb9d557fb4be92a94f", [:mix], [{:jason, "~> 1.0", [hex: :jason, repo: "hexpm", optional: true]}, {:phoenix, "~> 1.5.7", [hex: :phoenix, repo: "hexpm", optional: false]}, {:phoenix_html, "~> 2.14", [hex: :phoenix_html, repo: "hexpm", optional: false]}, {:telemetry, "~> 0.4.2 or ~> 0.5", [hex: :telemetry, repo: "hexpm", optional: false]}], "hexpm", "cabcfb6738419a08600009219a5f0d861de97507fc1232121e1d5221aba849bd"}, + "phoenix_pubsub": {:hex, :phoenix_pubsub, "2.0.0", "a1ae76717bb168cdeb10ec9d92d1480fec99e3080f011402c0a2d68d47395ffb", [:mix], [], "hexpm", "c52d948c4f261577b9c6fa804be91884b381a7f8f18450c5045975435350f771"}, + "plug": {:hex, :plug, "1.11.0", "f17217525597628298998bc3baed9f8ea1fa3f1160aa9871aee6df47a6e4d38e", [:mix], [{:mime, "~> 1.0", [hex: :mime, repo: "hexpm", optional: false]}, {:plug_crypto, "~> 1.1.1 or ~> 1.2", [hex: :plug_crypto, repo: "hexpm", optional: false]}, {:telemetry, "~> 0.4", [hex: :telemetry, repo: "hexpm", optional: false]}], "hexpm", "2d9c633f0499f9dc5c2fd069161af4e2e7756890b81adcbb2ceaa074e8308876"}, + "plug_cowboy": {:hex, :plug_cowboy, "2.4.1", "779ba386c0915027f22e14a48919a9545714f849505fa15af2631a0d298abf0f", [:mix], [{:cowboy, "~> 2.7", [hex: :cowboy, repo: "hexpm", optional: false]}, {:cowboy_telemetry, "~> 0.3", [hex: :cowboy_telemetry, repo: "hexpm", optional: false]}, {:plug, "~> 1.7", [hex: :plug, repo: "hexpm", optional: false]}, {:telemetry, "~> 0.4", [hex: :telemetry, repo: "hexpm", optional: false]}], "hexpm", "d72113b6dff7b37a7d9b2a5b68892808e3a9a752f2bf7e503240945385b70507"}, + "plug_crypto": {:hex, :plug_crypto, "1.2.0", "1cb20793aa63a6c619dd18bb33d7a3aa94818e5fd39ad357051a67f26dfa2df6", [:mix], [], "hexpm", "a48b538ae8bf381ffac344520755f3007cc10bd8e90b240af98ea29b69683fc2"}, + "ranch": {:hex, :ranch, "1.7.1", "6b1fab51b49196860b733a49c07604465a47bdb78aa10c1c16a3d199f7f8c881", [:rebar3], [], "hexpm", "451d8527787df716d99dc36162fca05934915db0b6141bbdac2ea8d3c7afc7d7"}, + "telemetry": {:hex, :telemetry, "0.4.2", "2808c992455e08d6177322f14d3bdb6b625fbcfd233a73505870d8738a2f4599", [:rebar3], [], "hexpm", "2d1419bd9dda6a206d7b5852179511722e2b18812310d304620c7bd92a13fcef"}, + "telemetry_metrics": {:hex, :telemetry_metrics, "0.6.0", "da9d49ee7e6bb1c259d36ce6539cd45ae14d81247a2b0c90edf55e2b50507f7b", [:mix], [{:telemetry, "~> 0.4", [hex: :telemetry, repo: "hexpm", optional: false]}], "hexpm", "5cfe67ad464b243835512aa44321cee91faed6ea868d7fb761d7016e02915c3d"}, + "telemetry_poller": {:hex, :telemetry_poller, "0.5.1", "21071cc2e536810bac5628b935521ff3e28f0303e770951158c73eaaa01e962a", [:rebar3], [{:telemetry, "~> 0.4", [hex: :telemetry, repo: "hexpm", optional: false]}], "hexpm", "4cab72069210bc6e7a080cec9afffad1b33370149ed5d379b81c7c5f0c663fd4"}, +} diff --git a/priv/gettext/en/LC_MESSAGES/errors.po b/priv/gettext/en/LC_MESSAGES/errors.po new file mode 100644 index 0000000..cdec3a1 --- /dev/null +++ b/priv/gettext/en/LC_MESSAGES/errors.po @@ -0,0 +1,11 @@ +## `msgid`s in this file come from POT (.pot) files. +## +## Do not add, change, or remove `msgid`s manually here as +## they're tied to the ones in the corresponding POT file +## (with the same domain). +## +## Use `mix gettext.extract --merge` or `mix gettext.merge` +## to merge POT files into PO files. +msgid "" +msgstr "" +"Language: en\n" diff --git a/priv/gettext/errors.pot b/priv/gettext/errors.pot new file mode 100644 index 0000000..d6f47fa --- /dev/null +++ b/priv/gettext/errors.pot @@ -0,0 +1,10 @@ +## This is a PO Template file. +## +## `msgid`s here are often extracted from source code. +## Add new translations manually only if they're dynamic +## translations that can't be statically extracted. +## +## Run `mix gettext.extract` to bring this file up to +## date. Leave `msgstr`s empty as changing them here has no +## effect: edit them in PO (`.po`) files instead. + diff --git a/priv/static/css/app.css b/priv/static/css/app.css new file mode 100644 index 0000000..b93e479 --- /dev/null +++ b/priv/static/css/app.css @@ -0,0 +1,257 @@ +/* Includes some default style for the starter application. + * This can be safely deleted to start fresh. + */ + +/* Milligram v1.3.0 https://milligram.github.io + * Copyright (c) 2017 CJ Patoilo Licensed under the MIT license + */ + +*,*:after,*:before{box-sizing:inherit}html{box-sizing:border-box;font-size:62.5%}body{color:#000000;font-family:'Helvetica', 'Arial', sans-serif;font-size:1.6em;font-weight:300;line-height:1.6}blockquote{border-left:0.3rem solid #d1d1d1;margin-left:0;margin-right:0;padding:1rem 1.5rem}blockquote *:last-child{margin-bottom:0}.button,button,input[type='button'],input[type='reset'],input[type='submit']{background-color:#0069d9;border:0.1rem solid #0069d9;border-radius:.4rem;color:#fff;cursor:pointer;display:inline-block;font-size:1.1rem;font-weight:700;height:3.8rem;letter-spacing:.1rem;line-height:3.8rem;padding:0 3.0rem;text-align:center;text-decoration:none;text-transform:uppercase;white-space:nowrap}.button:focus,.button:hover,button:focus,button:hover,input[type='button']:focus,input[type='button']:hover,input[type='reset']:focus,input[type='reset']:hover,input[type='submit']:focus,input[type='submit']:hover{background-color:#606c76;border-color:#606c76;color:#fff;outline:0}.button[disabled],button[disabled],input[type='button'][disabled],input[type='reset'][disabled],input[type='submit'][disabled]{cursor:default;opacity:.5}.button[disabled]:focus,.button[disabled]:hover,button[disabled]:focus,button[disabled]:hover,input[type='button'][disabled]:focus,input[type='button'][disabled]:hover,input[type='reset'][disabled]:focus,input[type='reset'][disabled]:hover,input[type='submit'][disabled]:focus,input[type='submit'][disabled]:hover{background-color:#0069d9;border-color:#0069d9}.button.button-outline,button.button-outline,input[type='button'].button-outline,input[type='reset'].button-outline,input[type='submit'].button-outline{background-color:transparent;color:#0069d9}.button.button-outline:focus,.button.button-outline:hover,button.button-outline:focus,button.button-outline:hover,input[type='button'].button-outline:focus,input[type='button'].button-outline:hover,input[type='reset'].button-outline:focus,input[type='reset'].button-outline:hover,input[type='submit'].button-outline:focus,input[type='submit'].button-outline:hover{background-color:transparent;border-color:#606c76;color:#606c76}.button.button-outline[disabled]:focus,.button.button-outline[disabled]:hover,button.button-outline[disabled]:focus,button.button-outline[disabled]:hover,input[type='button'].button-outline[disabled]:focus,input[type='button'].button-outline[disabled]:hover,input[type='reset'].button-outline[disabled]:focus,input[type='reset'].button-outline[disabled]:hover,input[type='submit'].button-outline[disabled]:focus,input[type='submit'].button-outline[disabled]:hover{border-color:inherit;color:#0069d9}.button.button-clear,button.button-clear,input[type='button'].button-clear,input[type='reset'].button-clear,input[type='submit'].button-clear{background-color:transparent;border-color:transparent;color:#0069d9}.button.button-clear:focus,.button.button-clear:hover,button.button-clear:focus,button.button-clear:hover,input[type='button'].button-clear:focus,input[type='button'].button-clear:hover,input[type='reset'].button-clear:focus,input[type='reset'].button-clear:hover,input[type='submit'].button-clear:focus,input[type='submit'].button-clear:hover{background-color:transparent;border-color:transparent;color:#606c76}.button.button-clear[disabled]:focus,.button.button-clear[disabled]:hover,button.button-clear[disabled]:focus,button.button-clear[disabled]:hover,input[type='button'].button-clear[disabled]:focus,input[type='button'].button-clear[disabled]:hover,input[type='reset'].button-clear[disabled]:focus,input[type='reset'].button-clear[disabled]:hover,input[type='submit'].button-clear[disabled]:focus,input[type='submit'].button-clear[disabled]:hover{color:#0069d9}code{background:#f4f5f6;border-radius:.4rem;font-size:86%;margin:0 .2rem;padding:.2rem .5rem;white-space:nowrap}pre{background:#f4f5f6;border-left:0.3rem solid #0069d9;overflow-y:hidden}pre>code{border-radius:0;display:block;padding:1rem 1.5rem;white-space:pre}hr{border:0;border-top:0.1rem solid #f4f5f6;margin:3.0rem 0}input[type='email'],input[type='number'],input[type='password'],input[type='search'],input[type='tel'],input[type='text'],input[type='url'],textarea,select{-webkit-appearance:none;-moz-appearance:none;appearance:none;background-color:transparent;border:0.1rem solid #d1d1d1;border-radius:.4rem;box-shadow:none;box-sizing:inherit;height:3.8rem;padding:.6rem 1.0rem;width:100%}input[type='email']:focus,input[type='number']:focus,input[type='password']:focus,input[type='search']:focus,input[type='tel']:focus,input[type='text']:focus,input[type='url']:focus,textarea:focus,select:focus{border-color:#0069d9;outline:0}select{background:url('data:image/svg+xml;utf8,') center right no-repeat;padding-right:3.0rem}select:focus{background-image:url('data:image/svg+xml;utf8,')}textarea{min-height:6.5rem}label,legend{display:block;font-size:1.6rem;font-weight:700;margin-bottom:.5rem}fieldset{border-width:0;padding:0}input[type='checkbox'],input[type='radio']{display:inline}.label-inline{display:inline-block;font-weight:normal;margin-left:.5rem}.row{display:flex;flex-direction:column;padding:0;width:100%}.row.row-no-padding{padding:0}.row.row-no-padding>.column{padding:0}.row.row-wrap{flex-wrap:wrap}.row.row-top{align-items:flex-start}.row.row-bottom{align-items:flex-end}.row.row-center{align-items:center}.row.row-stretch{align-items:stretch}.row.row-baseline{align-items:baseline}.row .column{display:block;flex:1 1 auto;margin-left:0;max-width:100%;width:100%}.row .column.column-offset-10{margin-left:10%}.row .column.column-offset-20{margin-left:20%}.row .column.column-offset-25{margin-left:25%}.row .column.column-offset-33,.row .column.column-offset-34{margin-left:33.3333%}.row .column.column-offset-50{margin-left:50%}.row .column.column-offset-66,.row .column.column-offset-67{margin-left:66.6666%}.row .column.column-offset-75{margin-left:75%}.row .column.column-offset-80{margin-left:80%}.row .column.column-offset-90{margin-left:90%}.row .column.column-10{flex:0 0 10%;max-width:10%}.row .column.column-20{flex:0 0 20%;max-width:20%}.row .column.column-25{flex:0 0 25%;max-width:25%}.row .column.column-33,.row .column.column-34{flex:0 0 33.3333%;max-width:33.3333%}.row .column.column-40{flex:0 0 40%;max-width:40%}.row .column.column-50{flex:0 0 50%;max-width:50%}.row .column.column-60{flex:0 0 60%;max-width:60%}.row .column.column-66,.row .column.column-67{flex:0 0 66.6666%;max-width:66.6666%}.row .column.column-75{flex:0 0 75%;max-width:75%}.row .column.column-80{flex:0 0 80%;max-width:80%}.row .column.column-90{flex:0 0 90%;max-width:90%}.row .column .column-top{align-self:flex-start}.row .column .column-bottom{align-self:flex-end}.row .column .column-center{-ms-grid-row-align:center;align-self:center}@media (min-width: 40rem){.row{flex-direction:row;margin-left:-1.0rem;width:calc(100% + 2.0rem)}.row .column{margin-bottom:inherit;padding:0 1.0rem}}a{color:#0069d9;text-decoration:none}a:focus,a:hover{color:#606c76}dl,ol,ul{list-style:none;margin-top:0;padding-left:0}dl dl,dl ol,dl ul,ol dl,ol ol,ol ul,ul dl,ul ol,ul ul{font-size:90%;margin:1.5rem 0 1.5rem 3.0rem}ol{list-style:decimal inside}ul{list-style:circle inside}.button,button,dd,dt,li{margin-bottom:1.0rem}fieldset,input,select,textarea{margin-bottom:1.5rem}blockquote,dl,figure,form,ol,p,pre,table,ul{margin-bottom:2.5rem}table{border-spacing:0;width:100%}td,th{border-bottom:0.1rem solid #e1e1e1;padding:1.2rem 1.5rem;text-align:left}td:first-child,th:first-child{padding-left:0}td:last-child,th:last-child{padding-right:0}b,strong{font-weight:bold}p{margin-top:0}h1,h2,h3,h4,h5,h6{font-weight:300;letter-spacing:-.1rem;margin-bottom:2.0rem;margin-top:0}h1{font-size:4.6rem;line-height:1.2}h2{font-size:3.6rem;line-height:1.25}h3{font-size:2.8rem;line-height:1.3}h4{font-size:2.2rem;letter-spacing:-.08rem;line-height:1.35}h5{font-size:1.8rem;letter-spacing:-.05rem;line-height:1.5}h6{font-size:1.6rem;letter-spacing:0;line-height:1.4}img{max-width:100%}.clearfix:after{clear:both;content:' ';display:table}.float-left{float:left}.float-right{float:right} + +/* General style */ +h1{font-size: 3.6rem; line-height: 1.25} +h2{font-size: 2.8rem; line-height: 1.3} +h3{font-size: 2.2rem; letter-spacing: -.08rem; line-height: 1.35} +h4{font-size: 1.8rem; letter-spacing: -.05rem; line-height: 1.5} +h5{font-size: 1.6rem; letter-spacing: 0; line-height: 1.4} +h6{font-size: 1.4rem; letter-spacing: 0; line-height: 1.2} +pre{padding: 1em;} + +.container{ + margin: 0 auto; + max-width: 80.0rem; + padding: 0 2.0rem; + position: relative; + width: 100% +} +select { + width: auto; +} + +/* Phoenix promo and logo */ +.phx-hero { + text-align: center; + border-bottom: 1px solid #e3e3e3; + background: #eee; + border-radius: 6px; + padding: 3em 3em 1em; + margin-bottom: 3rem; + font-weight: 200; + font-size: 120%; +} +.phx-hero input { + background: #ffffff; +} +.phx-logo { + min-width: 300px; + margin: 1rem; + display: block; +} +.phx-logo img { + width: auto; + display: block; +} + +/* Headers */ +header { + width: 100%; + background: #fdfdfd; + border-bottom: 1px solid #eaeaea; + margin-bottom: 2rem; +} +header section { + align-items: center; + display: flex; + flex-direction: column; + justify-content: space-between; +} +header section :first-child { + order: 2; +} +header section :last-child { + order: 1; +} +header nav ul, +header nav li { + margin: 0; + padding: 0; + display: block; + text-align: right; + white-space: nowrap; +} +header nav ul { + margin: 1rem; + margin-top: 0; +} +header nav a { + display: block; +} + +@media (min-width: 40.0rem) { /* Small devices (landscape phones, 576px and up) */ + header section { + flex-direction: row; + } + header nav ul { + margin: 1rem; + } + .phx-logo { + flex-basis: 527px; + margin: 2rem 1rem; + } +} + +/* Make clicks pass-through */ +#nprogress { + pointer-events: none; +} + +#nprogress .bar { + background: #29d; + + position: fixed; + z-index: 1031; + top: 0; + left: 0; + + width: 100%; + height: 2px; +} + +/* Fancy blur effect */ +#nprogress .peg { + display: block; + position: absolute; + right: 0px; + width: 100px; + height: 100%; + box-shadow: 0 0 10px #29d, 0 0 5px #29d; + opacity: 1.0; + + -webkit-transform: rotate(3deg) translate(0px, -4px); + -ms-transform: rotate(3deg) translate(0px, -4px); + transform: rotate(3deg) translate(0px, -4px); +} + +/* Remove these to get rid of the spinner */ +#nprogress .spinner { + display: block; + position: fixed; + z-index: 1031; + top: 15px; + right: 15px; +} + +#nprogress .spinner-icon { + width: 18px; + height: 18px; + box-sizing: border-box; + + border: solid 2px transparent; + border-top-color: #29d; + border-left-color: #29d; + border-radius: 50%; + + -webkit-animation: nprogress-spinner 400ms linear infinite; + animation: nprogress-spinner 400ms linear infinite; +} + +.nprogress-custom-parent { + overflow: hidden; + position: relative; +} + +.nprogress-custom-parent #nprogress .spinner, +.nprogress-custom-parent #nprogress .bar { + position: absolute; +} + +@-webkit-keyframes nprogress-spinner { + 0% { -webkit-transform: rotate(0deg); } + 100% { -webkit-transform: rotate(360deg); } +} +@keyframes nprogress-spinner { + 0% { transform: rotate(0deg); } + 100% { transform: rotate(360deg); } +} + + +/* This file is for your main application css. */ +/* LiveView specific classes for your customizations */ +.phx-no-feedback.invalid-feedback, +.phx-no-feedback .invalid-feedback { + display: none; } + +.phx-click-loading { + opacity: 0.5; + transition: opacity 1s ease-out; } + +.phx-disconnected { + cursor: wait; } + +.phx-disconnected * { + pointer-events: none; } + +.phx-modal { + opacity: 1 !important; + position: fixed; + z-index: 1; + left: 0; + top: 0; + width: 100%; + height: 100%; + overflow: auto; + background-color: black; + background-color: rgba(0, 0, 0, 0.4); } + +.phx-modal-content { + background-color: #fefefe; + margin: 15% auto; + padding: 20px; + border: 1px solid #888; + width: 80%; } + +.phx-modal-close { + color: #aaa; + float: right; + font-size: 28px; + font-weight: bold; } + +.phx-modal-close:hover, +.phx-modal-close:focus { + color: black; + text-decoration: none; + cursor: pointer; } + +/* Alerts and form errors */ +.alert { + padding: 15px; + margin-bottom: 20px; + border: 1px solid transparent; + border-radius: 4px; } + +.alert-info { + color: #31708f; + background-color: #d9edf7; + border-color: #bce8f1; } + +.alert-warning { + color: #8a6d3b; + background-color: #fcf8e3; + border-color: #faebcc; } + +.alert-danger { + color: #a94442; + background-color: #f2dede; + border-color: #ebccd1; } + +.alert p { + margin-bottom: 0; } + +.alert:empty { + display: none; } + +.invalid-feedback { + color: #a94442; + display: block; + margin: -1rem 0 2rem; } + diff --git a/priv/static/favicon.ico b/priv/static/favicon.ico new file mode 100644 index 0000000..73de524 Binary files /dev/null and b/priv/static/favicon.ico differ diff --git a/priv/static/images/phoenix.png b/priv/static/images/phoenix.png new file mode 100644 index 0000000..9c81075 Binary files /dev/null and b/priv/static/images/phoenix.png differ diff --git a/priv/static/js/app.js b/priv/static/js/app.js new file mode 100644 index 0000000..937f739 --- /dev/null +++ b/priv/static/js/app.js @@ -0,0 +1,169 @@ +/******/ (function(modules) { // webpackBootstrap +/******/ // The module cache +/******/ var installedModules = {}; +/******/ +/******/ // The require function +/******/ function __webpack_require__(moduleId) { +/******/ +/******/ // Check if module is in cache +/******/ if(installedModules[moduleId]) { +/******/ return installedModules[moduleId].exports; +/******/ } +/******/ // Create a new module (and put it into the cache) +/******/ var module = installedModules[moduleId] = { +/******/ i: moduleId, +/******/ l: false, +/******/ exports: {} +/******/ }; +/******/ +/******/ // Execute the module function +/******/ modules[moduleId].call(module.exports, module, module.exports, __webpack_require__); +/******/ +/******/ // Flag the module as loaded +/******/ module.l = true; +/******/ +/******/ // Return the exports of the module +/******/ return module.exports; +/******/ } +/******/ +/******/ +/******/ // expose the modules object (__webpack_modules__) +/******/ __webpack_require__.m = modules; +/******/ +/******/ // expose the module cache +/******/ __webpack_require__.c = installedModules; +/******/ +/******/ // define getter function for harmony exports +/******/ __webpack_require__.d = function(exports, name, getter) { +/******/ if(!__webpack_require__.o(exports, name)) { +/******/ Object.defineProperty(exports, name, { enumerable: true, get: getter }); +/******/ } +/******/ }; +/******/ +/******/ // define __esModule on exports +/******/ __webpack_require__.r = function(exports) { +/******/ if(typeof Symbol !== 'undefined' && Symbol.toStringTag) { +/******/ Object.defineProperty(exports, Symbol.toStringTag, { value: 'Module' }); +/******/ } +/******/ Object.defineProperty(exports, '__esModule', { value: true }); +/******/ }; +/******/ +/******/ // create a fake namespace object +/******/ // mode & 1: value is a module id, require it +/******/ // mode & 2: merge all properties of value into the ns +/******/ // mode & 4: return value when already ns object +/******/ // mode & 8|1: behave like require +/******/ __webpack_require__.t = function(value, mode) { +/******/ if(mode & 1) value = __webpack_require__(value); +/******/ if(mode & 8) return value; +/******/ if((mode & 4) && typeof value === 'object' && value && value.__esModule) return value; +/******/ var ns = Object.create(null); +/******/ __webpack_require__.r(ns); +/******/ Object.defineProperty(ns, 'default', { enumerable: true, value: value }); +/******/ if(mode & 2 && typeof value != 'string') for(var key in value) __webpack_require__.d(ns, key, function(key) { return value[key]; }.bind(null, key)); +/******/ return ns; +/******/ }; +/******/ +/******/ // getDefaultExport function for compatibility with non-harmony modules +/******/ __webpack_require__.n = function(module) { +/******/ var getter = module && module.__esModule ? +/******/ function getDefault() { return module['default']; } : +/******/ function getModuleExports() { return module; }; +/******/ __webpack_require__.d(getter, 'a', getter); +/******/ return getter; +/******/ }; +/******/ +/******/ // Object.prototype.hasOwnProperty.call +/******/ __webpack_require__.o = function(object, property) { return Object.prototype.hasOwnProperty.call(object, property); }; +/******/ +/******/ // __webpack_public_path__ +/******/ __webpack_require__.p = "/js/"; +/******/ +/******/ +/******/ // Load entry module and return exports +/******/ return __webpack_require__(__webpack_require__.s = 0); +/******/ }) +/************************************************************************/ +/******/ ({ + +/***/ "../deps/phoenix/priv/static/phoenix.js": +/*!**********************************************!*\ + !*** ../deps/phoenix/priv/static/phoenix.js ***! + \**********************************************/ +/*! no static exports found */ +/***/ (function(module, exports, __webpack_require__) { + + eval("!function (e, t) {\n true ? module.exports = t() : undefined;\n}(this, function () {\n return function (e) {\n var t = {};\n\n function n(i) {\n if (t[i]) return t[i].exports;\n var o = t[i] = {\n i: i,\n l: !1,\n exports: {}\n };\n return e[i].call(o.exports, o, o.exports, n), o.l = !0, o.exports;\n }\n\n return n.m = e, n.c = t, n.d = function (e, t, i) {\n n.o(e, t) || Object.defineProperty(e, t, {\n enumerable: !0,\n get: i\n });\n }, n.r = function (e) {\n \"undefined\" != typeof Symbol && Symbol.toStringTag && Object.defineProperty(e, Symbol.toStringTag, {\n value: \"Module\"\n }), Object.defineProperty(e, \"__esModule\", {\n value: !0\n });\n }, n.t = function (e, t) {\n if (1 & t && (e = n(e)), 8 & t) return e;\n if (4 & t && \"object\" == typeof e && e && e.__esModule) return e;\n var i = Object.create(null);\n if (n.r(i), Object.defineProperty(i, \"default\", {\n enumerable: !0,\n value: e\n }), 2 & t && \"string\" != typeof e) for (var o in e) n.d(i, o, function (t) {\n return e[t];\n }.bind(null, o));\n return i;\n }, n.n = function (e) {\n var t = e && e.__esModule ? function () {\n return e.default;\n } : function () {\n return e;\n };\n return n.d(t, \"a\", t), t;\n }, n.o = function (e, t) {\n return Object.prototype.hasOwnProperty.call(e, t);\n }, n.p = \"\", n(n.s = 0);\n }([function (e, t, n) {\n (function (t) {\n e.exports = t.Phoenix = n(2);\n }).call(this, n(1));\n }, function (e, t) {\n var n;\n\n n = function () {\n return this;\n }();\n\n try {\n n = n || new Function(\"return this\")();\n } catch (e) {\n \"object\" == typeof window && (n = window);\n }\n\n e.exports = n;\n }, function (e, t, n) {\n \"use strict\";\n\n function i(e) {\n return function (e) {\n if (Array.isArray(e)) return a(e);\n }(e) || function (e) {\n if (\"undefined\" != typeof Symbol && Symbol.iterator in Object(e)) return Array.from(e);\n }(e) || s(e) || function () {\n throw new TypeError(\"Invalid attempt to spread non-iterable instance.\\nIn order to be iterable, non-array objects must have a [Symbol.iterator]() method.\");\n }();\n }\n\n function o(e) {\n return (o = \"function\" == typeof Symbol && \"symbol\" == typeof Symbol.iterator ? function (e) {\n return typeof e;\n } : function (e) {\n return e && \"function\" == typeof Symbol && e.constructor === Symbol && e !== Symbol.prototype ? \"symbol\" : typeof e;\n })(e);\n }\n\n function r(e, t) {\n return function (e) {\n if (Array.isArray(e)) return e;\n }(e) || function (e, t) {\n if (\"undefined\" == typeof Symbol || !(Symbol.iterator in Object(e))) return;\n var n = [],\n i = !0,\n o = !1,\n r = void 0;\n\n try {\n for (var s, a = e[Symbol.iterator](); !(i = (s = a.next()).done) && (n.push(s.value), !t || n.length !== t); i = !0);\n } catch (e) {\n o = !0, r = e;\n } finally {\n try {\n i || null == a.return || a.return();\n } finally {\n if (o) throw r;\n }\n }\n\n return n;\n }(e, t) || s(e, t) || function () {\n throw new TypeError(\"Invalid attempt to destructure non-iterable instance.\\nIn order to be iterable, non-array objects must have a [Symbol.iterator]() method.\");\n }();\n }\n\n function s(e, t) {\n if (e) {\n if (\"string\" == typeof e) return a(e, t);\n var n = Object.prototype.toString.call(e).slice(8, -1);\n return \"Object\" === n && e.constructor && (n = e.constructor.name), \"Map\" === n || \"Set\" === n ? Array.from(n) : \"Arguments\" === n || /^(?:Ui|I)nt(?:8|16|32)(?:Clamped)?Array$/.test(n) ? a(e, t) : void 0;\n }\n }\n\n function a(e, t) {\n (null == t || t > e.length) && (t = e.length);\n\n for (var n = 0, i = new Array(t); n < t; n++) i[n] = e[n];\n\n return i;\n }\n\n function c(e, t) {\n if (!(e instanceof t)) throw new TypeError(\"Cannot call a class as a function\");\n }\n\n function u(e, t) {\n for (var n = 0; n < t.length; n++) {\n var i = t[n];\n i.enumerable = i.enumerable || !1, i.configurable = !0, \"value\" in i && (i.writable = !0), Object.defineProperty(e, i.key, i);\n }\n }\n\n function h(e, t, n) {\n return t && u(e.prototype, t), n && u(e, n), e;\n }\n\n n.r(t), n.d(t, \"Channel\", function () {\n return _;\n }), n.d(t, \"Serializer\", function () {\n return H;\n }), n.d(t, \"Socket\", function () {\n return U;\n }), n.d(t, \"LongPoll\", function () {\n return D;\n }), n.d(t, \"Ajax\", function () {\n return M;\n }), n.d(t, \"Presence\", function () {\n return N;\n });\n\n var l = \"undefined\" != typeof self ? self : null,\n f = \"undefined\" != typeof window ? window : null,\n d = l || f || void 0,\n p = 0,\n v = 1,\n y = 2,\n m = 3,\n g = \"closed\",\n k = \"errored\",\n b = \"joined\",\n j = \"joining\",\n C = \"leaving\",\n E = \"phx_close\",\n R = \"phx_error\",\n T = \"phx_join\",\n S = \"phx_reply\",\n w = \"phx_leave\",\n A = [E, R, T, S, w],\n L = \"longpoll\",\n x = \"websocket\",\n O = function (e) {\n if (\"function\" == typeof e) return e;\n return function () {\n return e;\n };\n },\n P = function () {\n function e(t, n, i, o) {\n c(this, e), this.channel = t, this.event = n, this.payload = i || function () {\n return {};\n }, this.receivedResp = null, this.timeout = o, this.timeoutTimer = null, this.recHooks = [], this.sent = !1;\n }\n\n return h(e, [{\n key: \"resend\",\n value: function (e) {\n this.timeout = e, this.reset(), this.send();\n }\n }, {\n key: \"send\",\n value: function () {\n this.hasReceived(\"timeout\") || (this.startTimeout(), this.sent = !0, this.channel.socket.push({\n topic: this.channel.topic,\n event: this.event,\n payload: this.payload(),\n ref: this.ref,\n join_ref: this.channel.joinRef()\n }));\n }\n }, {\n key: \"receive\",\n value: function (e, t) {\n return this.hasReceived(e) && t(this.receivedResp.response), this.recHooks.push({\n status: e,\n callback: t\n }), this;\n }\n }, {\n key: \"reset\",\n value: function () {\n this.cancelRefEvent(), this.ref = null, this.refEvent = null, this.receivedResp = null, this.sent = !1;\n }\n }, {\n key: \"matchReceive\",\n value: function (e) {\n var t = e.status,\n n = e.response;\n e.ref;\n this.recHooks.filter(function (e) {\n return e.status === t;\n }).forEach(function (e) {\n return e.callback(n);\n });\n }\n }, {\n key: \"cancelRefEvent\",\n value: function () {\n this.refEvent && this.channel.off(this.refEvent);\n }\n }, {\n key: \"cancelTimeout\",\n value: function () {\n clearTimeout(this.timeoutTimer), this.timeoutTimer = null;\n }\n }, {\n key: \"startTimeout\",\n value: function () {\n var e = this;\n this.timeoutTimer && this.cancelTimeout(), this.ref = this.channel.socket.makeRef(), this.refEvent = this.channel.replyEventName(this.ref), this.channel.on(this.refEvent, function (t) {\n e.cancelRefEvent(), e.cancelTimeout(), e.receivedResp = t, e.matchReceive(t);\n }), this.timeoutTimer = setTimeout(function () {\n e.trigger(\"timeout\", {});\n }, this.timeout);\n }\n }, {\n key: \"hasReceived\",\n value: function (e) {\n return this.receivedResp && this.receivedResp.status === e;\n }\n }, {\n key: \"trigger\",\n value: function (e, t) {\n this.channel.trigger(this.refEvent, {\n status: e,\n response: t\n });\n }\n }]), e;\n }(),\n _ = function () {\n function e(t, n, i) {\n var o = this;\n c(this, e), this.state = g, this.topic = t, this.params = O(n || {}), this.socket = i, this.bindings = [], this.bindingRef = 0, this.timeout = this.socket.timeout, this.joinedOnce = !1, this.joinPush = new P(this, T, this.params, this.timeout), this.pushBuffer = [], this.stateChangeRefs = [], this.rejoinTimer = new J(function () {\n o.socket.isConnected() && o.rejoin();\n }, this.socket.rejoinAfterMs), this.stateChangeRefs.push(this.socket.onError(function () {\n return o.rejoinTimer.reset();\n })), this.stateChangeRefs.push(this.socket.onOpen(function () {\n o.rejoinTimer.reset(), o.isErrored() && o.rejoin();\n })), this.joinPush.receive(\"ok\", function () {\n o.state = b, o.rejoinTimer.reset(), o.pushBuffer.forEach(function (e) {\n return e.send();\n }), o.pushBuffer = [];\n }), this.joinPush.receive(\"error\", function () {\n o.state = k, o.socket.isConnected() && o.rejoinTimer.scheduleTimeout();\n }), this.onClose(function () {\n o.rejoinTimer.reset(), o.socket.hasLogger() && o.socket.log(\"channel\", \"close \".concat(o.topic, \" \").concat(o.joinRef())), o.state = g, o.socket.remove(o);\n }), this.onError(function (e) {\n o.socket.hasLogger() && o.socket.log(\"channel\", \"error \".concat(o.topic), e), o.isJoining() && o.joinPush.reset(), o.state = k, o.socket.isConnected() && o.rejoinTimer.scheduleTimeout();\n }), this.joinPush.receive(\"timeout\", function () {\n o.socket.hasLogger() && o.socket.log(\"channel\", \"timeout \".concat(o.topic, \" (\").concat(o.joinRef(), \")\"), o.joinPush.timeout), new P(o, w, O({}), o.timeout).send(), o.state = k, o.joinPush.reset(), o.socket.isConnected() && o.rejoinTimer.scheduleTimeout();\n }), this.on(S, function (e, t) {\n o.trigger(o.replyEventName(t), e);\n });\n }\n\n return h(e, [{\n key: \"join\",\n value: function () {\n var e = arguments.length > 0 && void 0 !== arguments[0] ? arguments[0] : this.timeout;\n if (this.joinedOnce) throw new Error(\"tried to join multiple times. 'join' can only be called a single time per channel instance\");\n return this.timeout = e, this.joinedOnce = !0, this.rejoin(), this.joinPush;\n }\n }, {\n key: \"onClose\",\n value: function (e) {\n this.on(E, e);\n }\n }, {\n key: \"onError\",\n value: function (e) {\n return this.on(R, function (t) {\n return e(t);\n });\n }\n }, {\n key: \"on\",\n value: function (e, t) {\n var n = this.bindingRef++;\n return this.bindings.push({\n event: e,\n ref: n,\n callback: t\n }), n;\n }\n }, {\n key: \"off\",\n value: function (e, t) {\n this.bindings = this.bindings.filter(function (n) {\n return !(n.event === e && (void 0 === t || t === n.ref));\n });\n }\n }, {\n key: \"canPush\",\n value: function () {\n return this.socket.isConnected() && this.isJoined();\n }\n }, {\n key: \"push\",\n value: function (e, t) {\n var n = arguments.length > 2 && void 0 !== arguments[2] ? arguments[2] : this.timeout;\n if (!this.joinedOnce) throw new Error(\"tried to push '\".concat(e, \"' to '\").concat(this.topic, \"' before joining. Use channel.join() before pushing events\"));\n var i = new P(this, e, function () {\n return t;\n }, n);\n return this.canPush() ? i.send() : (i.startTimeout(), this.pushBuffer.push(i)), i;\n }\n }, {\n key: \"leave\",\n value: function () {\n var e = this,\n t = arguments.length > 0 && void 0 !== arguments[0] ? arguments[0] : this.timeout;\n this.rejoinTimer.reset(), this.joinPush.cancelTimeout(), this.state = C;\n\n var n = function () {\n e.socket.hasLogger() && e.socket.log(\"channel\", \"leave \".concat(e.topic)), e.trigger(E, \"leave\");\n },\n i = new P(this, w, O({}), t);\n\n return i.receive(\"ok\", function () {\n return n();\n }).receive(\"timeout\", function () {\n return n();\n }), i.send(), this.canPush() || i.trigger(\"ok\", {}), i;\n }\n }, {\n key: \"onMessage\",\n value: function (e, t, n) {\n return t;\n }\n }, {\n key: \"isLifecycleEvent\",\n value: function (e) {\n return A.indexOf(e) >= 0;\n }\n }, {\n key: \"isMember\",\n value: function (e, t, n, i) {\n return this.topic === e && (!i || i === this.joinRef() || !this.isLifecycleEvent(t) || (this.socket.hasLogger() && this.socket.log(\"channel\", \"dropping outdated message\", {\n topic: e,\n event: t,\n payload: n,\n joinRef: i\n }), !1));\n }\n }, {\n key: \"joinRef\",\n value: function () {\n return this.joinPush.ref;\n }\n }, {\n key: \"rejoin\",\n value: function () {\n var e = arguments.length > 0 && void 0 !== arguments[0] ? arguments[0] : this.timeout;\n this.isLeaving() || (this.socket.leaveOpenTopic(this.topic), this.state = j, this.joinPush.resend(e));\n }\n }, {\n key: \"trigger\",\n value: function (e, t, n, i) {\n var o = this.onMessage(e, t, n, i);\n if (t && !o) throw new Error(\"channel onMessage callbacks must return the payload, modified or unmodified\");\n\n for (var r = this.bindings.filter(function (t) {\n return t.event === e;\n }), s = 0; s < r.length; s++) {\n r[s].callback(o, n, i || this.joinRef());\n }\n }\n }, {\n key: \"replyEventName\",\n value: function (e) {\n return \"chan_reply_\".concat(e);\n }\n }, {\n key: \"isClosed\",\n value: function () {\n return this.state === g;\n }\n }, {\n key: \"isErrored\",\n value: function () {\n return this.state === k;\n }\n }, {\n key: \"isJoined\",\n value: function () {\n return this.state === b;\n }\n }, {\n key: \"isJoining\",\n value: function () {\n return this.state === j;\n }\n }, {\n key: \"isLeaving\",\n value: function () {\n return this.state === C;\n }\n }]), e;\n }(),\n H = {\n HEADER_LENGTH: 1,\n META_LENGTH: 4,\n KINDS: {\n push: 0,\n reply: 1,\n broadcast: 2\n },\n encode: function (e, t) {\n if (e.payload.constructor === ArrayBuffer) return t(this.binaryEncode(e));\n var n = [e.join_ref, e.ref, e.topic, e.event, e.payload];\n return t(JSON.stringify(n));\n },\n decode: function (e, t) {\n if (e.constructor === ArrayBuffer) return t(this.binaryDecode(e));\n var n = r(JSON.parse(e), 5);\n return t({\n join_ref: n[0],\n ref: n[1],\n topic: n[2],\n event: n[3],\n payload: n[4]\n });\n },\n binaryEncode: function (e) {\n var t = e.join_ref,\n n = e.ref,\n i = e.event,\n o = e.topic,\n r = e.payload,\n s = this.META_LENGTH + t.length + n.length + o.length + i.length,\n a = new ArrayBuffer(this.HEADER_LENGTH + s),\n c = new DataView(a),\n u = 0;\n c.setUint8(u++, this.KINDS.push), c.setUint8(u++, t.length), c.setUint8(u++, n.length), c.setUint8(u++, o.length), c.setUint8(u++, i.length), Array.from(t, function (e) {\n return c.setUint8(u++, e.charCodeAt(0));\n }), Array.from(n, function (e) {\n return c.setUint8(u++, e.charCodeAt(0));\n }), Array.from(o, function (e) {\n return c.setUint8(u++, e.charCodeAt(0));\n }), Array.from(i, function (e) {\n return c.setUint8(u++, e.charCodeAt(0));\n });\n var h = new Uint8Array(a.byteLength + r.byteLength);\n return h.set(new Uint8Array(a), 0), h.set(new Uint8Array(r), a.byteLength), h.buffer;\n },\n binaryDecode: function (e) {\n var t = new DataView(e),\n n = t.getUint8(0),\n i = new TextDecoder();\n\n switch (n) {\n case this.KINDS.push:\n return this.decodePush(e, t, i);\n\n case this.KINDS.reply:\n return this.decodeReply(e, t, i);\n\n case this.KINDS.broadcast:\n return this.decodeBroadcast(e, t, i);\n }\n },\n decodePush: function (e, t, n) {\n var i = t.getUint8(1),\n o = t.getUint8(2),\n r = t.getUint8(3),\n s = this.HEADER_LENGTH + this.META_LENGTH - 1,\n a = n.decode(e.slice(s, s + i));\n s += i;\n var c = n.decode(e.slice(s, s + o));\n s += o;\n var u = n.decode(e.slice(s, s + r));\n return s += r, {\n join_ref: a,\n ref: null,\n topic: c,\n event: u,\n payload: e.slice(s, e.byteLength)\n };\n },\n decodeReply: function (e, t, n) {\n var i = t.getUint8(1),\n o = t.getUint8(2),\n r = t.getUint8(3),\n s = t.getUint8(4),\n a = this.HEADER_LENGTH + this.META_LENGTH,\n c = n.decode(e.slice(a, a + i));\n a += i;\n var u = n.decode(e.slice(a, a + o));\n a += o;\n var h = n.decode(e.slice(a, a + r));\n a += r;\n var l = n.decode(e.slice(a, a + s));\n a += s;\n var f = e.slice(a, e.byteLength);\n return {\n join_ref: c,\n ref: u,\n topic: h,\n event: S,\n payload: {\n status: l,\n response: f\n }\n };\n },\n decodeBroadcast: function (e, t, n) {\n var i = t.getUint8(1),\n o = t.getUint8(2),\n r = this.HEADER_LENGTH + 2,\n s = n.decode(e.slice(r, r + i));\n r += i;\n var a = n.decode(e.slice(r, r + o));\n return r += o, {\n join_ref: null,\n ref: null,\n topic: s,\n event: a,\n payload: e.slice(r, e.byteLength)\n };\n }\n },\n U = function () {\n function e(t) {\n var n = this,\n i = arguments.length > 1 && void 0 !== arguments[1] ? arguments[1] : {};\n c(this, e), this.stateChangeCallbacks = {\n open: [],\n close: [],\n error: [],\n message: []\n }, this.channels = [], this.sendBuffer = [], this.ref = 0, this.timeout = i.timeout || 1e4, this.transport = i.transport || d.WebSocket || D, this.defaultEncoder = H.encode.bind(H), this.defaultDecoder = H.decode.bind(H), this.closeWasClean = !1, this.unloaded = !1, this.binaryType = i.binaryType || \"arraybuffer\", this.transport !== D ? (this.encode = i.encode || this.defaultEncoder, this.decode = i.decode || this.defaultDecoder) : (this.encode = this.defaultEncoder, this.decode = this.defaultDecoder), f && f.addEventListener && f.addEventListener(\"unload\", function (e) {\n n.conn && (n.unloaded = !0, n.abnormalClose(\"unloaded\"));\n }), this.heartbeatIntervalMs = i.heartbeatIntervalMs || 3e4, this.rejoinAfterMs = function (e) {\n return i.rejoinAfterMs ? i.rejoinAfterMs(e) : [1e3, 2e3, 5e3][e - 1] || 1e4;\n }, this.reconnectAfterMs = function (e) {\n return n.unloaded ? 100 : i.reconnectAfterMs ? i.reconnectAfterMs(e) : [10, 50, 100, 150, 200, 250, 500, 1e3, 2e3][e - 1] || 5e3;\n }, this.logger = i.logger || null, this.longpollerTimeout = i.longpollerTimeout || 2e4, this.params = O(i.params || {}), this.endPoint = \"\".concat(t, \"/\").concat(x), this.vsn = i.vsn || \"2.0.0\", this.heartbeatTimer = null, this.pendingHeartbeatRef = null, this.reconnectTimer = new J(function () {\n n.teardown(function () {\n return n.connect();\n });\n }, this.reconnectAfterMs);\n }\n\n return h(e, [{\n key: \"protocol\",\n value: function () {\n return location.protocol.match(/^https/) ? \"wss\" : \"ws\";\n }\n }, {\n key: \"endPointURL\",\n value: function () {\n var e = M.appendParams(M.appendParams(this.endPoint, this.params()), {\n vsn: this.vsn\n });\n return \"/\" !== e.charAt(0) ? e : \"/\" === e.charAt(1) ? \"\".concat(this.protocol(), \":\").concat(e) : \"\".concat(this.protocol(), \"://\").concat(location.host).concat(e);\n }\n }, {\n key: \"disconnect\",\n value: function (e, t, n) {\n this.closeWasClean = !0, this.reconnectTimer.reset(), this.teardown(e, t, n);\n }\n }, {\n key: \"connect\",\n value: function (e) {\n var t = this;\n e && (console && console.log(\"passing params to connect is deprecated. Instead pass :params to the Socket constructor\"), this.params = O(e)), this.conn || (this.closeWasClean = !1, this.conn = new this.transport(this.endPointURL()), this.conn.binaryType = this.binaryType, this.conn.timeout = this.longpollerTimeout, this.conn.onopen = function () {\n return t.onConnOpen();\n }, this.conn.onerror = function (e) {\n return t.onConnError(e);\n }, this.conn.onmessage = function (e) {\n return t.onConnMessage(e);\n }, this.conn.onclose = function (e) {\n return t.onConnClose(e);\n });\n }\n }, {\n key: \"log\",\n value: function (e, t, n) {\n this.logger(e, t, n);\n }\n }, {\n key: \"hasLogger\",\n value: function () {\n return null !== this.logger;\n }\n }, {\n key: \"onOpen\",\n value: function (e) {\n var t = this.makeRef();\n return this.stateChangeCallbacks.open.push([t, e]), t;\n }\n }, {\n key: \"onClose\",\n value: function (e) {\n var t = this.makeRef();\n return this.stateChangeCallbacks.close.push([t, e]), t;\n }\n }, {\n key: \"onError\",\n value: function (e) {\n var t = this.makeRef();\n return this.stateChangeCallbacks.error.push([t, e]), t;\n }\n }, {\n key: \"onMessage\",\n value: function (e) {\n var t = this.makeRef();\n return this.stateChangeCallbacks.message.push([t, e]), t;\n }\n }, {\n key: \"onConnOpen\",\n value: function () {\n this.hasLogger() && this.log(\"transport\", \"connected to \".concat(this.endPointURL())), this.unloaded = !1, this.closeWasClean = !1, this.flushSendBuffer(), this.reconnectTimer.reset(), this.resetHeartbeat(), this.stateChangeCallbacks.open.forEach(function (e) {\n return (0, r(e, 2)[1])();\n });\n }\n }, {\n key: \"resetHeartbeat\",\n value: function () {\n var e = this;\n this.conn && this.conn.skipHeartbeat || (this.pendingHeartbeatRef = null, clearInterval(this.heartbeatTimer), this.heartbeatTimer = setInterval(function () {\n return e.sendHeartbeat();\n }, this.heartbeatIntervalMs));\n }\n }, {\n key: \"teardown\",\n value: function (e, t, n) {\n var i = this;\n if (!this.conn) return e && e();\n this.waitForBufferDone(function () {\n i.conn && (t ? i.conn.close(t, n || \"\") : i.conn.close()), i.waitForSocketClosed(function () {\n i.conn && (i.conn.onclose = function () {}, i.conn = null), e && e();\n });\n });\n }\n }, {\n key: \"waitForBufferDone\",\n value: function (e) {\n var t = this,\n n = arguments.length > 1 && void 0 !== arguments[1] ? arguments[1] : 1;\n 5 !== n && this.conn && this.conn.bufferedAmount ? setTimeout(function () {\n t.waitForBufferDone(e, n + 1);\n }, 150 * n) : e();\n }\n }, {\n key: \"waitForSocketClosed\",\n value: function (e) {\n var t = this,\n n = arguments.length > 1 && void 0 !== arguments[1] ? arguments[1] : 1;\n 5 !== n && this.conn && this.conn.readyState !== m ? setTimeout(function () {\n t.waitForSocketClosed(e, n + 1);\n }, 150 * n) : e();\n }\n }, {\n key: \"onConnClose\",\n value: function (e) {\n this.hasLogger() && this.log(\"transport\", \"close\", e), this.triggerChanError(), clearInterval(this.heartbeatTimer), this.closeWasClean || this.reconnectTimer.scheduleTimeout(), this.stateChangeCallbacks.close.forEach(function (t) {\n return (0, r(t, 2)[1])(e);\n });\n }\n }, {\n key: \"onConnError\",\n value: function (e) {\n this.hasLogger() && this.log(\"transport\", e), this.triggerChanError(), this.stateChangeCallbacks.error.forEach(function (t) {\n return (0, r(t, 2)[1])(e);\n });\n }\n }, {\n key: \"triggerChanError\",\n value: function () {\n this.channels.forEach(function (e) {\n e.isErrored() || e.isLeaving() || e.isClosed() || e.trigger(R);\n });\n }\n }, {\n key: \"connectionState\",\n value: function () {\n switch (this.conn && this.conn.readyState) {\n case p:\n return \"connecting\";\n\n case v:\n return \"open\";\n\n case y:\n return \"closing\";\n\n default:\n return \"closed\";\n }\n }\n }, {\n key: \"isConnected\",\n value: function () {\n return \"open\" === this.connectionState();\n }\n }, {\n key: \"remove\",\n value: function (e) {\n this.off(e.stateChangeRefs), this.channels = this.channels.filter(function (t) {\n return t.joinRef() !== e.joinRef();\n });\n }\n }, {\n key: \"off\",\n value: function (e) {\n for (var t in this.stateChangeCallbacks) this.stateChangeCallbacks[t] = this.stateChangeCallbacks[t].filter(function (t) {\n var n = r(t, 1)[0];\n return -1 === e.indexOf(n);\n });\n }\n }, {\n key: \"channel\",\n value: function (e) {\n var t = arguments.length > 1 && void 0 !== arguments[1] ? arguments[1] : {},\n n = new _(e, t, this);\n return this.channels.push(n), n;\n }\n }, {\n key: \"push\",\n value: function (e) {\n var t = this;\n\n if (this.hasLogger()) {\n var n = e.topic,\n i = e.event,\n o = e.payload,\n r = e.ref,\n s = e.join_ref;\n this.log(\"push\", \"\".concat(n, \" \").concat(i, \" (\").concat(s, \", \").concat(r, \")\"), o);\n }\n\n this.isConnected() ? this.encode(e, function (e) {\n return t.conn.send(e);\n }) : this.sendBuffer.push(function () {\n return t.encode(e, function (e) {\n return t.conn.send(e);\n });\n });\n }\n }, {\n key: \"makeRef\",\n value: function () {\n var e = this.ref + 1;\n return e === this.ref ? this.ref = 0 : this.ref = e, this.ref.toString();\n }\n }, {\n key: \"sendHeartbeat\",\n value: function () {\n if (this.isConnected()) {\n if (this.pendingHeartbeatRef) return this.pendingHeartbeatRef = null, this.hasLogger() && this.log(\"transport\", \"heartbeat timeout. Attempting to re-establish connection\"), void this.abnormalClose(\"heartbeat timeout\");\n this.pendingHeartbeatRef = this.makeRef(), this.push({\n topic: \"phoenix\",\n event: \"heartbeat\",\n payload: {},\n ref: this.pendingHeartbeatRef\n });\n }\n }\n }, {\n key: \"abnormalClose\",\n value: function (e) {\n this.closeWasClean = !1, this.conn.readyState === v && this.conn.close(1e3, e);\n }\n }, {\n key: \"flushSendBuffer\",\n value: function () {\n this.isConnected() && this.sendBuffer.length > 0 && (this.sendBuffer.forEach(function (e) {\n return e();\n }), this.sendBuffer = []);\n }\n }, {\n key: \"onConnMessage\",\n value: function (e) {\n var t = this;\n this.decode(e.data, function (e) {\n var n = e.topic,\n i = e.event,\n o = e.payload,\n s = e.ref,\n a = e.join_ref;\n s && s === t.pendingHeartbeatRef && (t.pendingHeartbeatRef = null), t.hasLogger() && t.log(\"receive\", \"\".concat(o.status || \"\", \" \").concat(n, \" \").concat(i, \" \").concat(s && \"(\" + s + \")\" || \"\"), o);\n\n for (var c = 0; c < t.channels.length; c++) {\n var u = t.channels[c];\n u.isMember(n, i, o, a) && u.trigger(i, o, s, a);\n }\n\n for (var h = 0; h < t.stateChangeCallbacks.message.length; h++) {\n (0, r(t.stateChangeCallbacks.message[h], 2)[1])(e);\n }\n });\n }\n }, {\n key: \"leaveOpenTopic\",\n value: function (e) {\n var t = this.channels.find(function (t) {\n return t.topic === e && (t.isJoined() || t.isJoining());\n });\n t && (this.hasLogger() && this.log(\"transport\", 'leaving duplicate topic \"'.concat(e, '\"')), t.leave());\n }\n }]), e;\n }(),\n D = function () {\n function e(t) {\n c(this, e), this.endPoint = null, this.token = null, this.skipHeartbeat = !0, this.onopen = function () {}, this.onerror = function () {}, this.onmessage = function () {}, this.onclose = function () {}, this.pollEndpoint = this.normalizeEndpoint(t), this.readyState = p, this.poll();\n }\n\n return h(e, [{\n key: \"normalizeEndpoint\",\n value: function (e) {\n return e.replace(\"ws://\", \"http://\").replace(\"wss://\", \"https://\").replace(new RegExp(\"(.*)/\" + x), \"$1/\" + L);\n }\n }, {\n key: \"endpointURL\",\n value: function () {\n return M.appendParams(this.pollEndpoint, {\n token: this.token\n });\n }\n }, {\n key: \"closeAndRetry\",\n value: function () {\n this.close(), this.readyState = p;\n }\n }, {\n key: \"ontimeout\",\n value: function () {\n this.onerror(\"timeout\"), this.closeAndRetry();\n }\n }, {\n key: \"poll\",\n value: function () {\n var e = this;\n this.readyState !== v && this.readyState !== p || M.request(\"GET\", this.endpointURL(), \"application/json\", null, this.timeout, this.ontimeout.bind(this), function (t) {\n if (t) {\n var n = t.status,\n i = t.token,\n o = t.messages;\n e.token = i;\n } else n = 0;\n\n switch (n) {\n case 200:\n o.forEach(function (t) {\n return e.onmessage({\n data: t\n });\n }), e.poll();\n break;\n\n case 204:\n e.poll();\n break;\n\n case 410:\n e.readyState = v, e.onopen(), e.poll();\n break;\n\n case 403:\n e.onerror(), e.close();\n break;\n\n case 0:\n case 500:\n e.onerror(), e.closeAndRetry();\n break;\n\n default:\n throw new Error(\"unhandled poll status \".concat(n));\n }\n });\n }\n }, {\n key: \"send\",\n value: function (e) {\n var t = this;\n M.request(\"POST\", this.endpointURL(), \"application/json\", e, this.timeout, this.onerror.bind(this, \"timeout\"), function (e) {\n e && 200 === e.status || (t.onerror(e && e.status), t.closeAndRetry());\n });\n }\n }, {\n key: \"close\",\n value: function (e, t) {\n this.readyState = m, this.onclose();\n }\n }]), e;\n }(),\n M = function () {\n function e() {\n c(this, e);\n }\n\n return h(e, null, [{\n key: \"request\",\n value: function (e, t, n, i, o, r, s) {\n if (d.XDomainRequest) {\n var a = new XDomainRequest();\n this.xdomainRequest(a, e, t, i, o, r, s);\n } else {\n var c = new d.XMLHttpRequest();\n this.xhrRequest(c, e, t, n, i, o, r, s);\n }\n }\n }, {\n key: \"xdomainRequest\",\n value: function (e, t, n, i, o, r, s) {\n var a = this;\n e.timeout = o, e.open(t, n), e.onload = function () {\n var t = a.parseJSON(e.responseText);\n s && s(t);\n }, r && (e.ontimeout = r), e.onprogress = function () {}, e.send(i);\n }\n }, {\n key: \"xhrRequest\",\n value: function (e, t, n, i, o, r, s, a) {\n var c = this;\n e.open(t, n, !0), e.timeout = r, e.setRequestHeader(\"Content-Type\", i), e.onerror = function () {\n a && a(null);\n }, e.onreadystatechange = function () {\n if (e.readyState === c.states.complete && a) {\n var t = c.parseJSON(e.responseText);\n a(t);\n }\n }, s && (e.ontimeout = s), e.send(o);\n }\n }, {\n key: \"parseJSON\",\n value: function (e) {\n if (!e || \"\" === e) return null;\n\n try {\n return JSON.parse(e);\n } catch (t) {\n return console && console.log(\"failed to parse JSON response\", e), null;\n }\n }\n }, {\n key: \"serialize\",\n value: function (e, t) {\n var n = [];\n\n for (var i in e) if (e.hasOwnProperty(i)) {\n var r = t ? \"\".concat(t, \"[\").concat(i, \"]\") : i,\n s = e[i];\n \"object\" === o(s) ? n.push(this.serialize(s, r)) : n.push(encodeURIComponent(r) + \"=\" + encodeURIComponent(s));\n }\n\n return n.join(\"&\");\n }\n }, {\n key: \"appendParams\",\n value: function (e, t) {\n if (0 === Object.keys(t).length) return e;\n var n = e.match(/\\?/) ? \"&\" : \"?\";\n return \"\".concat(e).concat(n).concat(this.serialize(t));\n }\n }]), e;\n }();\n\n M.states = {\n complete: 4\n };\n\n var N = function () {\n function e(t) {\n var n = this,\n i = arguments.length > 1 && void 0 !== arguments[1] ? arguments[1] : {};\n c(this, e);\n var o = i.events || {\n state: \"presence_state\",\n diff: \"presence_diff\"\n };\n this.state = {}, this.pendingDiffs = [], this.channel = t, this.joinRef = null, this.caller = {\n onJoin: function () {},\n onLeave: function () {},\n onSync: function () {}\n }, this.channel.on(o.state, function (t) {\n var i = n.caller,\n o = i.onJoin,\n r = i.onLeave,\n s = i.onSync;\n n.joinRef = n.channel.joinRef(), n.state = e.syncState(n.state, t, o, r), n.pendingDiffs.forEach(function (t) {\n n.state = e.syncDiff(n.state, t, o, r);\n }), n.pendingDiffs = [], s();\n }), this.channel.on(o.diff, function (t) {\n var i = n.caller,\n o = i.onJoin,\n r = i.onLeave,\n s = i.onSync;\n n.inPendingSyncState() ? n.pendingDiffs.push(t) : (n.state = e.syncDiff(n.state, t, o, r), s());\n });\n }\n\n return h(e, [{\n key: \"onJoin\",\n value: function (e) {\n this.caller.onJoin = e;\n }\n }, {\n key: \"onLeave\",\n value: function (e) {\n this.caller.onLeave = e;\n }\n }, {\n key: \"onSync\",\n value: function (e) {\n this.caller.onSync = e;\n }\n }, {\n key: \"list\",\n value: function (t) {\n return e.list(this.state, t);\n }\n }, {\n key: \"inPendingSyncState\",\n value: function () {\n return !this.joinRef || this.joinRef !== this.channel.joinRef();\n }\n }], [{\n key: \"syncState\",\n value: function (e, t, n, i) {\n var o = this,\n r = this.clone(e),\n s = {},\n a = {};\n return this.map(r, function (e, n) {\n t[e] || (a[e] = n);\n }), this.map(t, function (e, t) {\n var n = r[e];\n\n if (n) {\n var i = t.metas.map(function (e) {\n return e.phx_ref;\n }),\n c = n.metas.map(function (e) {\n return e.phx_ref;\n }),\n u = t.metas.filter(function (e) {\n return c.indexOf(e.phx_ref) < 0;\n }),\n h = n.metas.filter(function (e) {\n return i.indexOf(e.phx_ref) < 0;\n });\n u.length > 0 && (s[e] = t, s[e].metas = u), h.length > 0 && (a[e] = o.clone(n), a[e].metas = h);\n } else s[e] = t;\n }), this.syncDiff(r, {\n joins: s,\n leaves: a\n }, n, i);\n }\n }, {\n key: \"syncDiff\",\n value: function (e, t, n, o) {\n var r = t.joins,\n s = t.leaves,\n a = this.clone(e);\n return n || (n = function () {}), o || (o = function () {}), this.map(r, function (e, t) {\n var o = a[e];\n\n if (a[e] = t, o) {\n var r,\n s = a[e].metas.map(function (e) {\n return e.phx_ref;\n }),\n c = o.metas.filter(function (e) {\n return s.indexOf(e.phx_ref) < 0;\n });\n (r = a[e].metas).unshift.apply(r, i(c));\n }\n\n n(e, o, t);\n }), this.map(s, function (e, t) {\n var n = a[e];\n\n if (n) {\n var i = t.metas.map(function (e) {\n return e.phx_ref;\n });\n n.metas = n.metas.filter(function (e) {\n return i.indexOf(e.phx_ref) < 0;\n }), o(e, n, t), 0 === n.metas.length && delete a[e];\n }\n }), a;\n }\n }, {\n key: \"list\",\n value: function (e, t) {\n return t || (t = function (e, t) {\n return t;\n }), this.map(e, function (e, n) {\n return t(e, n);\n });\n }\n }, {\n key: \"map\",\n value: function (e, t) {\n return Object.getOwnPropertyNames(e).map(function (n) {\n return t(n, e[n]);\n });\n }\n }, {\n key: \"clone\",\n value: function (e) {\n return JSON.parse(JSON.stringify(e));\n }\n }]), e;\n }(),\n J = function () {\n function e(t, n) {\n c(this, e), this.callback = t, this.timerCalc = n, this.timer = null, this.tries = 0;\n }\n\n return h(e, [{\n key: \"reset\",\n value: function () {\n this.tries = 0, clearTimeout(this.timer);\n }\n }, {\n key: \"scheduleTimeout\",\n value: function () {\n var e = this;\n clearTimeout(this.timer), this.timer = setTimeout(function () {\n e.tries = e.tries + 1, e.callback();\n }, this.timerCalc(this.tries + 1));\n }\n }]), e;\n }();\n }]);\n});//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiLi4vZGVwcy9waG9lbml4L3ByaXYvc3RhdGljL3Bob2VuaXguanMuanMiLCJzb3VyY2VzIjpbIndlYnBhY2s6Ly8vLi4vZGVwcy9waG9lbml4L3ByaXYvc3RhdGljL3Bob2VuaXguanM/MzFiYiJdLCJzb3VyY2VzQ29udGVudCI6WyIhZnVuY3Rpb24oZSx0KXtcIm9iamVjdFwiPT10eXBlb2YgZXhwb3J0cyYmXCJvYmplY3RcIj09dHlwZW9mIG1vZHVsZT9tb2R1bGUuZXhwb3J0cz10KCk6XCJmdW5jdGlvblwiPT10eXBlb2YgZGVmaW5lJiZkZWZpbmUuYW1kP2RlZmluZShbXSx0KTpcIm9iamVjdFwiPT10eXBlb2YgZXhwb3J0cz9leHBvcnRzLlBob2VuaXg9dCgpOmUuUGhvZW5peD10KCl9KHRoaXMsKGZ1bmN0aW9uKCl7cmV0dXJuIGZ1bmN0aW9uKGUpe3ZhciB0PXt9O2Z1bmN0aW9uIG4oaSl7aWYodFtpXSlyZXR1cm4gdFtpXS5leHBvcnRzO3ZhciBvPXRbaV09e2k6aSxsOiExLGV4cG9ydHM6e319O3JldHVybiBlW2ldLmNhbGwoby5leHBvcnRzLG8sby5leHBvcnRzLG4pLG8ubD0hMCxvLmV4cG9ydHN9cmV0dXJuIG4ubT1lLG4uYz10LG4uZD1mdW5jdGlvbihlLHQsaSl7bi5vKGUsdCl8fE9iamVjdC5kZWZpbmVQcm9wZXJ0eShlLHQse2VudW1lcmFibGU6ITAsZ2V0Oml9KX0sbi5yPWZ1bmN0aW9uKGUpe1widW5kZWZpbmVkXCIhPXR5cGVvZiBTeW1ib2wmJlN5bWJvbC50b1N0cmluZ1RhZyYmT2JqZWN0LmRlZmluZVByb3BlcnR5KGUsU3ltYm9sLnRvU3RyaW5nVGFnLHt2YWx1ZTpcIk1vZHVsZVwifSksT2JqZWN0LmRlZmluZVByb3BlcnR5KGUsXCJfX2VzTW9kdWxlXCIse3ZhbHVlOiEwfSl9LG4udD1mdW5jdGlvbihlLHQpe2lmKDEmdCYmKGU9bihlKSksOCZ0KXJldHVybiBlO2lmKDQmdCYmXCJvYmplY3RcIj09dHlwZW9mIGUmJmUmJmUuX19lc01vZHVsZSlyZXR1cm4gZTt2YXIgaT1PYmplY3QuY3JlYXRlKG51bGwpO2lmKG4ucihpKSxPYmplY3QuZGVmaW5lUHJvcGVydHkoaSxcImRlZmF1bHRcIix7ZW51bWVyYWJsZTohMCx2YWx1ZTplfSksMiZ0JiZcInN0cmluZ1wiIT10eXBlb2YgZSlmb3IodmFyIG8gaW4gZSluLmQoaSxvLGZ1bmN0aW9uKHQpe3JldHVybiBlW3RdfS5iaW5kKG51bGwsbykpO3JldHVybiBpfSxuLm49ZnVuY3Rpb24oZSl7dmFyIHQ9ZSYmZS5fX2VzTW9kdWxlP2Z1bmN0aW9uKCl7cmV0dXJuIGUuZGVmYXVsdH06ZnVuY3Rpb24oKXtyZXR1cm4gZX07cmV0dXJuIG4uZCh0LFwiYVwiLHQpLHR9LG4ubz1mdW5jdGlvbihlLHQpe3JldHVybiBPYmplY3QucHJvdG90eXBlLmhhc093blByb3BlcnR5LmNhbGwoZSx0KX0sbi5wPVwiXCIsbihuLnM9MCl9KFtmdW5jdGlvbihlLHQsbil7KGZ1bmN0aW9uKHQpe2UuZXhwb3J0cz10LlBob2VuaXg9bigyKX0pLmNhbGwodGhpcyxuKDEpKX0sZnVuY3Rpb24oZSx0KXt2YXIgbjtuPWZ1bmN0aW9uKCl7cmV0dXJuIHRoaXN9KCk7dHJ5e249bnx8bmV3IEZ1bmN0aW9uKFwicmV0dXJuIHRoaXNcIikoKX1jYXRjaChlKXtcIm9iamVjdFwiPT10eXBlb2Ygd2luZG93JiYobj13aW5kb3cpfWUuZXhwb3J0cz1ufSxmdW5jdGlvbihlLHQsbil7XCJ1c2Ugc3RyaWN0XCI7ZnVuY3Rpb24gaShlKXtyZXR1cm4gZnVuY3Rpb24oZSl7aWYoQXJyYXkuaXNBcnJheShlKSlyZXR1cm4gYShlKX0oZSl8fGZ1bmN0aW9uKGUpe2lmKFwidW5kZWZpbmVkXCIhPXR5cGVvZiBTeW1ib2wmJlN5bWJvbC5pdGVyYXRvciBpbiBPYmplY3QoZSkpcmV0dXJuIEFycmF5LmZyb20oZSl9KGUpfHxzKGUpfHxmdW5jdGlvbigpe3Rocm93IG5ldyBUeXBlRXJyb3IoXCJJbnZhbGlkIGF0dGVtcHQgdG8gc3ByZWFkIG5vbi1pdGVyYWJsZSBpbnN0YW5jZS5cXG5JbiBvcmRlciB0byBiZSBpdGVyYWJsZSwgbm9uLWFycmF5IG9iamVjdHMgbXVzdCBoYXZlIGEgW1N5bWJvbC5pdGVyYXRvcl0oKSBtZXRob2QuXCIpfSgpfWZ1bmN0aW9uIG8oZSl7cmV0dXJuKG89XCJmdW5jdGlvblwiPT10eXBlb2YgU3ltYm9sJiZcInN5bWJvbFwiPT10eXBlb2YgU3ltYm9sLml0ZXJhdG9yP2Z1bmN0aW9uKGUpe3JldHVybiB0eXBlb2YgZX06ZnVuY3Rpb24oZSl7cmV0dXJuIGUmJlwiZnVuY3Rpb25cIj09dHlwZW9mIFN5bWJvbCYmZS5jb25zdHJ1Y3Rvcj09PVN5bWJvbCYmZSE9PVN5bWJvbC5wcm90b3R5cGU/XCJzeW1ib2xcIjp0eXBlb2YgZX0pKGUpfWZ1bmN0aW9uIHIoZSx0KXtyZXR1cm4gZnVuY3Rpb24oZSl7aWYoQXJyYXkuaXNBcnJheShlKSlyZXR1cm4gZX0oZSl8fGZ1bmN0aW9uKGUsdCl7aWYoXCJ1bmRlZmluZWRcIj09dHlwZW9mIFN5bWJvbHx8IShTeW1ib2wuaXRlcmF0b3IgaW4gT2JqZWN0KGUpKSlyZXR1cm47dmFyIG49W10saT0hMCxvPSExLHI9dm9pZCAwO3RyeXtmb3IodmFyIHMsYT1lW1N5bWJvbC5pdGVyYXRvcl0oKTshKGk9KHM9YS5uZXh0KCkpLmRvbmUpJiYobi5wdXNoKHMudmFsdWUpLCF0fHxuLmxlbmd0aCE9PXQpO2k9ITApO31jYXRjaChlKXtvPSEwLHI9ZX1maW5hbGx5e3RyeXtpfHxudWxsPT1hLnJldHVybnx8YS5yZXR1cm4oKX1maW5hbGx5e2lmKG8pdGhyb3cgcn19cmV0dXJuIG59KGUsdCl8fHMoZSx0KXx8ZnVuY3Rpb24oKXt0aHJvdyBuZXcgVHlwZUVycm9yKFwiSW52YWxpZCBhdHRlbXB0IHRvIGRlc3RydWN0dXJlIG5vbi1pdGVyYWJsZSBpbnN0YW5jZS5cXG5JbiBvcmRlciB0byBiZSBpdGVyYWJsZSwgbm9uLWFycmF5IG9iamVjdHMgbXVzdCBoYXZlIGEgW1N5bWJvbC5pdGVyYXRvcl0oKSBtZXRob2QuXCIpfSgpfWZ1bmN0aW9uIHMoZSx0KXtpZihlKXtpZihcInN0cmluZ1wiPT10eXBlb2YgZSlyZXR1cm4gYShlLHQpO3ZhciBuPU9iamVjdC5wcm90b3R5cGUudG9TdHJpbmcuY2FsbChlKS5zbGljZSg4LC0xKTtyZXR1cm5cIk9iamVjdFwiPT09biYmZS5jb25zdHJ1Y3RvciYmKG49ZS5jb25zdHJ1Y3Rvci5uYW1lKSxcIk1hcFwiPT09bnx8XCJTZXRcIj09PW4/QXJyYXkuZnJvbShuKTpcIkFyZ3VtZW50c1wiPT09bnx8L14oPzpVaXxJKW50KD86OHwxNnwzMikoPzpDbGFtcGVkKT9BcnJheSQvLnRlc3Qobik/YShlLHQpOnZvaWQgMH19ZnVuY3Rpb24gYShlLHQpeyhudWxsPT10fHx0PmUubGVuZ3RoKSYmKHQ9ZS5sZW5ndGgpO2Zvcih2YXIgbj0wLGk9bmV3IEFycmF5KHQpO248dDtuKyspaVtuXT1lW25dO3JldHVybiBpfWZ1bmN0aW9uIGMoZSx0KXtpZighKGUgaW5zdGFuY2VvZiB0KSl0aHJvdyBuZXcgVHlwZUVycm9yKFwiQ2Fubm90IGNhbGwgYSBjbGFzcyBhcyBhIGZ1bmN0aW9uXCIpfWZ1bmN0aW9uIHUoZSx0KXtmb3IodmFyIG49MDtuPHQubGVuZ3RoO24rKyl7dmFyIGk9dFtuXTtpLmVudW1lcmFibGU9aS5lbnVtZXJhYmxlfHwhMSxpLmNvbmZpZ3VyYWJsZT0hMCxcInZhbHVlXCJpbiBpJiYoaS53cml0YWJsZT0hMCksT2JqZWN0LmRlZmluZVByb3BlcnR5KGUsaS5rZXksaSl9fWZ1bmN0aW9uIGgoZSx0LG4pe3JldHVybiB0JiZ1KGUucHJvdG90eXBlLHQpLG4mJnUoZSxuKSxlfW4ucih0KSxuLmQodCxcIkNoYW5uZWxcIiwoZnVuY3Rpb24oKXtyZXR1cm4gX30pKSxuLmQodCxcIlNlcmlhbGl6ZXJcIiwoZnVuY3Rpb24oKXtyZXR1cm4gSH0pKSxuLmQodCxcIlNvY2tldFwiLChmdW5jdGlvbigpe3JldHVybiBVfSkpLG4uZCh0LFwiTG9uZ1BvbGxcIiwoZnVuY3Rpb24oKXtyZXR1cm4gRH0pKSxuLmQodCxcIkFqYXhcIiwoZnVuY3Rpb24oKXtyZXR1cm4gTX0pKSxuLmQodCxcIlByZXNlbmNlXCIsKGZ1bmN0aW9uKCl7cmV0dXJuIE59KSk7dmFyIGw9XCJ1bmRlZmluZWRcIiE9dHlwZW9mIHNlbGY/c2VsZjpudWxsLGY9XCJ1bmRlZmluZWRcIiE9dHlwZW9mIHdpbmRvdz93aW5kb3c6bnVsbCxkPWx8fGZ8fHZvaWQgMCxwPTAsdj0xLHk9MixtPTMsZz1cImNsb3NlZFwiLGs9XCJlcnJvcmVkXCIsYj1cImpvaW5lZFwiLGo9XCJqb2luaW5nXCIsQz1cImxlYXZpbmdcIixFPVwicGh4X2Nsb3NlXCIsUj1cInBoeF9lcnJvclwiLFQ9XCJwaHhfam9pblwiLFM9XCJwaHhfcmVwbHlcIix3PVwicGh4X2xlYXZlXCIsQT1bRSxSLFQsUyx3XSxMPVwibG9uZ3BvbGxcIix4PVwid2Vic29ja2V0XCIsTz1mdW5jdGlvbihlKXtpZihcImZ1bmN0aW9uXCI9PXR5cGVvZiBlKXJldHVybiBlO3JldHVybiBmdW5jdGlvbigpe3JldHVybiBlfX0sUD1mdW5jdGlvbigpe2Z1bmN0aW9uIGUodCxuLGksbyl7Yyh0aGlzLGUpLHRoaXMuY2hhbm5lbD10LHRoaXMuZXZlbnQ9bix0aGlzLnBheWxvYWQ9aXx8ZnVuY3Rpb24oKXtyZXR1cm57fX0sdGhpcy5yZWNlaXZlZFJlc3A9bnVsbCx0aGlzLnRpbWVvdXQ9byx0aGlzLnRpbWVvdXRUaW1lcj1udWxsLHRoaXMucmVjSG9va3M9W10sdGhpcy5zZW50PSExfXJldHVybiBoKGUsW3trZXk6XCJyZXNlbmRcIix2YWx1ZTpmdW5jdGlvbihlKXt0aGlzLnRpbWVvdXQ9ZSx0aGlzLnJlc2V0KCksdGhpcy5zZW5kKCl9fSx7a2V5Olwic2VuZFwiLHZhbHVlOmZ1bmN0aW9uKCl7dGhpcy5oYXNSZWNlaXZlZChcInRpbWVvdXRcIil8fCh0aGlzLnN0YXJ0VGltZW91dCgpLHRoaXMuc2VudD0hMCx0aGlzLmNoYW5uZWwuc29ja2V0LnB1c2goe3RvcGljOnRoaXMuY2hhbm5lbC50b3BpYyxldmVudDp0aGlzLmV2ZW50LHBheWxvYWQ6dGhpcy5wYXlsb2FkKCkscmVmOnRoaXMucmVmLGpvaW5fcmVmOnRoaXMuY2hhbm5lbC5qb2luUmVmKCl9KSl9fSx7a2V5OlwicmVjZWl2ZVwiLHZhbHVlOmZ1bmN0aW9uKGUsdCl7cmV0dXJuIHRoaXMuaGFzUmVjZWl2ZWQoZSkmJnQodGhpcy5yZWNlaXZlZFJlc3AucmVzcG9uc2UpLHRoaXMucmVjSG9va3MucHVzaCh7c3RhdHVzOmUsY2FsbGJhY2s6dH0pLHRoaXN9fSx7a2V5OlwicmVzZXRcIix2YWx1ZTpmdW5jdGlvbigpe3RoaXMuY2FuY2VsUmVmRXZlbnQoKSx0aGlzLnJlZj1udWxsLHRoaXMucmVmRXZlbnQ9bnVsbCx0aGlzLnJlY2VpdmVkUmVzcD1udWxsLHRoaXMuc2VudD0hMX19LHtrZXk6XCJtYXRjaFJlY2VpdmVcIix2YWx1ZTpmdW5jdGlvbihlKXt2YXIgdD1lLnN0YXR1cyxuPWUucmVzcG9uc2U7ZS5yZWY7dGhpcy5yZWNIb29rcy5maWx0ZXIoKGZ1bmN0aW9uKGUpe3JldHVybiBlLnN0YXR1cz09PXR9KSkuZm9yRWFjaCgoZnVuY3Rpb24oZSl7cmV0dXJuIGUuY2FsbGJhY2sobil9KSl9fSx7a2V5OlwiY2FuY2VsUmVmRXZlbnRcIix2YWx1ZTpmdW5jdGlvbigpe3RoaXMucmVmRXZlbnQmJnRoaXMuY2hhbm5lbC5vZmYodGhpcy5yZWZFdmVudCl9fSx7a2V5OlwiY2FuY2VsVGltZW91dFwiLHZhbHVlOmZ1bmN0aW9uKCl7Y2xlYXJUaW1lb3V0KHRoaXMudGltZW91dFRpbWVyKSx0aGlzLnRpbWVvdXRUaW1lcj1udWxsfX0se2tleTpcInN0YXJ0VGltZW91dFwiLHZhbHVlOmZ1bmN0aW9uKCl7dmFyIGU9dGhpczt0aGlzLnRpbWVvdXRUaW1lciYmdGhpcy5jYW5jZWxUaW1lb3V0KCksdGhpcy5yZWY9dGhpcy5jaGFubmVsLnNvY2tldC5tYWtlUmVmKCksdGhpcy5yZWZFdmVudD10aGlzLmNoYW5uZWwucmVwbHlFdmVudE5hbWUodGhpcy5yZWYpLHRoaXMuY2hhbm5lbC5vbih0aGlzLnJlZkV2ZW50LChmdW5jdGlvbih0KXtlLmNhbmNlbFJlZkV2ZW50KCksZS5jYW5jZWxUaW1lb3V0KCksZS5yZWNlaXZlZFJlc3A9dCxlLm1hdGNoUmVjZWl2ZSh0KX0pKSx0aGlzLnRpbWVvdXRUaW1lcj1zZXRUaW1lb3V0KChmdW5jdGlvbigpe2UudHJpZ2dlcihcInRpbWVvdXRcIix7fSl9KSx0aGlzLnRpbWVvdXQpfX0se2tleTpcImhhc1JlY2VpdmVkXCIsdmFsdWU6ZnVuY3Rpb24oZSl7cmV0dXJuIHRoaXMucmVjZWl2ZWRSZXNwJiZ0aGlzLnJlY2VpdmVkUmVzcC5zdGF0dXM9PT1lfX0se2tleTpcInRyaWdnZXJcIix2YWx1ZTpmdW5jdGlvbihlLHQpe3RoaXMuY2hhbm5lbC50cmlnZ2VyKHRoaXMucmVmRXZlbnQse3N0YXR1czplLHJlc3BvbnNlOnR9KX19XSksZX0oKSxfPWZ1bmN0aW9uKCl7ZnVuY3Rpb24gZSh0LG4saSl7dmFyIG89dGhpcztjKHRoaXMsZSksdGhpcy5zdGF0ZT1nLHRoaXMudG9waWM9dCx0aGlzLnBhcmFtcz1PKG58fHt9KSx0aGlzLnNvY2tldD1pLHRoaXMuYmluZGluZ3M9W10sdGhpcy5iaW5kaW5nUmVmPTAsdGhpcy50aW1lb3V0PXRoaXMuc29ja2V0LnRpbWVvdXQsdGhpcy5qb2luZWRPbmNlPSExLHRoaXMuam9pblB1c2g9bmV3IFAodGhpcyxULHRoaXMucGFyYW1zLHRoaXMudGltZW91dCksdGhpcy5wdXNoQnVmZmVyPVtdLHRoaXMuc3RhdGVDaGFuZ2VSZWZzPVtdLHRoaXMucmVqb2luVGltZXI9bmV3IEooKGZ1bmN0aW9uKCl7by5zb2NrZXQuaXNDb25uZWN0ZWQoKSYmby5yZWpvaW4oKX0pLHRoaXMuc29ja2V0LnJlam9pbkFmdGVyTXMpLHRoaXMuc3RhdGVDaGFuZ2VSZWZzLnB1c2godGhpcy5zb2NrZXQub25FcnJvcigoZnVuY3Rpb24oKXtyZXR1cm4gby5yZWpvaW5UaW1lci5yZXNldCgpfSkpKSx0aGlzLnN0YXRlQ2hhbmdlUmVmcy5wdXNoKHRoaXMuc29ja2V0Lm9uT3BlbigoZnVuY3Rpb24oKXtvLnJlam9pblRpbWVyLnJlc2V0KCksby5pc0Vycm9yZWQoKSYmby5yZWpvaW4oKX0pKSksdGhpcy5qb2luUHVzaC5yZWNlaXZlKFwib2tcIiwoZnVuY3Rpb24oKXtvLnN0YXRlPWIsby5yZWpvaW5UaW1lci5yZXNldCgpLG8ucHVzaEJ1ZmZlci5mb3JFYWNoKChmdW5jdGlvbihlKXtyZXR1cm4gZS5zZW5kKCl9KSksby5wdXNoQnVmZmVyPVtdfSkpLHRoaXMuam9pblB1c2gucmVjZWl2ZShcImVycm9yXCIsKGZ1bmN0aW9uKCl7by5zdGF0ZT1rLG8uc29ja2V0LmlzQ29ubmVjdGVkKCkmJm8ucmVqb2luVGltZXIuc2NoZWR1bGVUaW1lb3V0KCl9KSksdGhpcy5vbkNsb3NlKChmdW5jdGlvbigpe28ucmVqb2luVGltZXIucmVzZXQoKSxvLnNvY2tldC5oYXNMb2dnZXIoKSYmby5zb2NrZXQubG9nKFwiY2hhbm5lbFwiLFwiY2xvc2UgXCIuY29uY2F0KG8udG9waWMsXCIgXCIpLmNvbmNhdChvLmpvaW5SZWYoKSkpLG8uc3RhdGU9ZyxvLnNvY2tldC5yZW1vdmUobyl9KSksdGhpcy5vbkVycm9yKChmdW5jdGlvbihlKXtvLnNvY2tldC5oYXNMb2dnZXIoKSYmby5zb2NrZXQubG9nKFwiY2hhbm5lbFwiLFwiZXJyb3IgXCIuY29uY2F0KG8udG9waWMpLGUpLG8uaXNKb2luaW5nKCkmJm8uam9pblB1c2gucmVzZXQoKSxvLnN0YXRlPWssby5zb2NrZXQuaXNDb25uZWN0ZWQoKSYmby5yZWpvaW5UaW1lci5zY2hlZHVsZVRpbWVvdXQoKX0pKSx0aGlzLmpvaW5QdXNoLnJlY2VpdmUoXCJ0aW1lb3V0XCIsKGZ1bmN0aW9uKCl7by5zb2NrZXQuaGFzTG9nZ2VyKCkmJm8uc29ja2V0LmxvZyhcImNoYW5uZWxcIixcInRpbWVvdXQgXCIuY29uY2F0KG8udG9waWMsXCIgKFwiKS5jb25jYXQoby5qb2luUmVmKCksXCIpXCIpLG8uam9pblB1c2gudGltZW91dCksbmV3IFAobyx3LE8oe30pLG8udGltZW91dCkuc2VuZCgpLG8uc3RhdGU9ayxvLmpvaW5QdXNoLnJlc2V0KCksby5zb2NrZXQuaXNDb25uZWN0ZWQoKSYmby5yZWpvaW5UaW1lci5zY2hlZHVsZVRpbWVvdXQoKX0pKSx0aGlzLm9uKFMsKGZ1bmN0aW9uKGUsdCl7by50cmlnZ2VyKG8ucmVwbHlFdmVudE5hbWUodCksZSl9KSl9cmV0dXJuIGgoZSxbe2tleTpcImpvaW5cIix2YWx1ZTpmdW5jdGlvbigpe3ZhciBlPWFyZ3VtZW50cy5sZW5ndGg+MCYmdm9pZCAwIT09YXJndW1lbnRzWzBdP2FyZ3VtZW50c1swXTp0aGlzLnRpbWVvdXQ7aWYodGhpcy5qb2luZWRPbmNlKXRocm93IG5ldyBFcnJvcihcInRyaWVkIHRvIGpvaW4gbXVsdGlwbGUgdGltZXMuICdqb2luJyBjYW4gb25seSBiZSBjYWxsZWQgYSBzaW5nbGUgdGltZSBwZXIgY2hhbm5lbCBpbnN0YW5jZVwiKTtyZXR1cm4gdGhpcy50aW1lb3V0PWUsdGhpcy5qb2luZWRPbmNlPSEwLHRoaXMucmVqb2luKCksdGhpcy5qb2luUHVzaH19LHtrZXk6XCJvbkNsb3NlXCIsdmFsdWU6ZnVuY3Rpb24oZSl7dGhpcy5vbihFLGUpfX0se2tleTpcIm9uRXJyb3JcIix2YWx1ZTpmdW5jdGlvbihlKXtyZXR1cm4gdGhpcy5vbihSLChmdW5jdGlvbih0KXtyZXR1cm4gZSh0KX0pKX19LHtrZXk6XCJvblwiLHZhbHVlOmZ1bmN0aW9uKGUsdCl7dmFyIG49dGhpcy5iaW5kaW5nUmVmKys7cmV0dXJuIHRoaXMuYmluZGluZ3MucHVzaCh7ZXZlbnQ6ZSxyZWY6bixjYWxsYmFjazp0fSksbn19LHtrZXk6XCJvZmZcIix2YWx1ZTpmdW5jdGlvbihlLHQpe3RoaXMuYmluZGluZ3M9dGhpcy5iaW5kaW5ncy5maWx0ZXIoKGZ1bmN0aW9uKG4pe3JldHVybiEobi5ldmVudD09PWUmJih2b2lkIDA9PT10fHx0PT09bi5yZWYpKX0pKX19LHtrZXk6XCJjYW5QdXNoXCIsdmFsdWU6ZnVuY3Rpb24oKXtyZXR1cm4gdGhpcy5zb2NrZXQuaXNDb25uZWN0ZWQoKSYmdGhpcy5pc0pvaW5lZCgpfX0se2tleTpcInB1c2hcIix2YWx1ZTpmdW5jdGlvbihlLHQpe3ZhciBuPWFyZ3VtZW50cy5sZW5ndGg+MiYmdm9pZCAwIT09YXJndW1lbnRzWzJdP2FyZ3VtZW50c1syXTp0aGlzLnRpbWVvdXQ7aWYoIXRoaXMuam9pbmVkT25jZSl0aHJvdyBuZXcgRXJyb3IoXCJ0cmllZCB0byBwdXNoICdcIi5jb25jYXQoZSxcIicgdG8gJ1wiKS5jb25jYXQodGhpcy50b3BpYyxcIicgYmVmb3JlIGpvaW5pbmcuIFVzZSBjaGFubmVsLmpvaW4oKSBiZWZvcmUgcHVzaGluZyBldmVudHNcIikpO3ZhciBpPW5ldyBQKHRoaXMsZSwoZnVuY3Rpb24oKXtyZXR1cm4gdH0pLG4pO3JldHVybiB0aGlzLmNhblB1c2goKT9pLnNlbmQoKTooaS5zdGFydFRpbWVvdXQoKSx0aGlzLnB1c2hCdWZmZXIucHVzaChpKSksaX19LHtrZXk6XCJsZWF2ZVwiLHZhbHVlOmZ1bmN0aW9uKCl7dmFyIGU9dGhpcyx0PWFyZ3VtZW50cy5sZW5ndGg+MCYmdm9pZCAwIT09YXJndW1lbnRzWzBdP2FyZ3VtZW50c1swXTp0aGlzLnRpbWVvdXQ7dGhpcy5yZWpvaW5UaW1lci5yZXNldCgpLHRoaXMuam9pblB1c2guY2FuY2VsVGltZW91dCgpLHRoaXMuc3RhdGU9Qzt2YXIgbj1mdW5jdGlvbigpe2Uuc29ja2V0Lmhhc0xvZ2dlcigpJiZlLnNvY2tldC5sb2coXCJjaGFubmVsXCIsXCJsZWF2ZSBcIi5jb25jYXQoZS50b3BpYykpLGUudHJpZ2dlcihFLFwibGVhdmVcIil9LGk9bmV3IFAodGhpcyx3LE8oe30pLHQpO3JldHVybiBpLnJlY2VpdmUoXCJva1wiLChmdW5jdGlvbigpe3JldHVybiBuKCl9KSkucmVjZWl2ZShcInRpbWVvdXRcIiwoZnVuY3Rpb24oKXtyZXR1cm4gbigpfSkpLGkuc2VuZCgpLHRoaXMuY2FuUHVzaCgpfHxpLnRyaWdnZXIoXCJva1wiLHt9KSxpfX0se2tleTpcIm9uTWVzc2FnZVwiLHZhbHVlOmZ1bmN0aW9uKGUsdCxuKXtyZXR1cm4gdH19LHtrZXk6XCJpc0xpZmVjeWNsZUV2ZW50XCIsdmFsdWU6ZnVuY3Rpb24oZSl7cmV0dXJuIEEuaW5kZXhPZihlKT49MH19LHtrZXk6XCJpc01lbWJlclwiLHZhbHVlOmZ1bmN0aW9uKGUsdCxuLGkpe3JldHVybiB0aGlzLnRvcGljPT09ZSYmKCFpfHxpPT09dGhpcy5qb2luUmVmKCl8fCF0aGlzLmlzTGlmZWN5Y2xlRXZlbnQodCl8fCh0aGlzLnNvY2tldC5oYXNMb2dnZXIoKSYmdGhpcy5zb2NrZXQubG9nKFwiY2hhbm5lbFwiLFwiZHJvcHBpbmcgb3V0ZGF0ZWQgbWVzc2FnZVwiLHt0b3BpYzplLGV2ZW50OnQscGF5bG9hZDpuLGpvaW5SZWY6aX0pLCExKSl9fSx7a2V5Olwiam9pblJlZlwiLHZhbHVlOmZ1bmN0aW9uKCl7cmV0dXJuIHRoaXMuam9pblB1c2gucmVmfX0se2tleTpcInJlam9pblwiLHZhbHVlOmZ1bmN0aW9uKCl7dmFyIGU9YXJndW1lbnRzLmxlbmd0aD4wJiZ2b2lkIDAhPT1hcmd1bWVudHNbMF0/YXJndW1lbnRzWzBdOnRoaXMudGltZW91dDt0aGlzLmlzTGVhdmluZygpfHwodGhpcy5zb2NrZXQubGVhdmVPcGVuVG9waWModGhpcy50b3BpYyksdGhpcy5zdGF0ZT1qLHRoaXMuam9pblB1c2gucmVzZW5kKGUpKX19LHtrZXk6XCJ0cmlnZ2VyXCIsdmFsdWU6ZnVuY3Rpb24oZSx0LG4saSl7dmFyIG89dGhpcy5vbk1lc3NhZ2UoZSx0LG4saSk7aWYodCYmIW8pdGhyb3cgbmV3IEVycm9yKFwiY2hhbm5lbCBvbk1lc3NhZ2UgY2FsbGJhY2tzIG11c3QgcmV0dXJuIHRoZSBwYXlsb2FkLCBtb2RpZmllZCBvciB1bm1vZGlmaWVkXCIpO2Zvcih2YXIgcj10aGlzLmJpbmRpbmdzLmZpbHRlcigoZnVuY3Rpb24odCl7cmV0dXJuIHQuZXZlbnQ9PT1lfSkpLHM9MDtzPHIubGVuZ3RoO3MrKyl7cltzXS5jYWxsYmFjayhvLG4saXx8dGhpcy5qb2luUmVmKCkpfX19LHtrZXk6XCJyZXBseUV2ZW50TmFtZVwiLHZhbHVlOmZ1bmN0aW9uKGUpe3JldHVyblwiY2hhbl9yZXBseV9cIi5jb25jYXQoZSl9fSx7a2V5OlwiaXNDbG9zZWRcIix2YWx1ZTpmdW5jdGlvbigpe3JldHVybiB0aGlzLnN0YXRlPT09Z319LHtrZXk6XCJpc0Vycm9yZWRcIix2YWx1ZTpmdW5jdGlvbigpe3JldHVybiB0aGlzLnN0YXRlPT09a319LHtrZXk6XCJpc0pvaW5lZFwiLHZhbHVlOmZ1bmN0aW9uKCl7cmV0dXJuIHRoaXMuc3RhdGU9PT1ifX0se2tleTpcImlzSm9pbmluZ1wiLHZhbHVlOmZ1bmN0aW9uKCl7cmV0dXJuIHRoaXMuc3RhdGU9PT1qfX0se2tleTpcImlzTGVhdmluZ1wiLHZhbHVlOmZ1bmN0aW9uKCl7cmV0dXJuIHRoaXMuc3RhdGU9PT1DfX1dKSxlfSgpLEg9e0hFQURFUl9MRU5HVEg6MSxNRVRBX0xFTkdUSDo0LEtJTkRTOntwdXNoOjAscmVwbHk6MSxicm9hZGNhc3Q6Mn0sZW5jb2RlOmZ1bmN0aW9uKGUsdCl7aWYoZS5wYXlsb2FkLmNvbnN0cnVjdG9yPT09QXJyYXlCdWZmZXIpcmV0dXJuIHQodGhpcy5iaW5hcnlFbmNvZGUoZSkpO3ZhciBuPVtlLmpvaW5fcmVmLGUucmVmLGUudG9waWMsZS5ldmVudCxlLnBheWxvYWRdO3JldHVybiB0KEpTT04uc3RyaW5naWZ5KG4pKX0sZGVjb2RlOmZ1bmN0aW9uKGUsdCl7aWYoZS5jb25zdHJ1Y3Rvcj09PUFycmF5QnVmZmVyKXJldHVybiB0KHRoaXMuYmluYXJ5RGVjb2RlKGUpKTt2YXIgbj1yKEpTT04ucGFyc2UoZSksNSk7cmV0dXJuIHQoe2pvaW5fcmVmOm5bMF0scmVmOm5bMV0sdG9waWM6blsyXSxldmVudDpuWzNdLHBheWxvYWQ6bls0XX0pfSxiaW5hcnlFbmNvZGU6ZnVuY3Rpb24oZSl7dmFyIHQ9ZS5qb2luX3JlZixuPWUucmVmLGk9ZS5ldmVudCxvPWUudG9waWMscj1lLnBheWxvYWQscz10aGlzLk1FVEFfTEVOR1RIK3QubGVuZ3RoK24ubGVuZ3RoK28ubGVuZ3RoK2kubGVuZ3RoLGE9bmV3IEFycmF5QnVmZmVyKHRoaXMuSEVBREVSX0xFTkdUSCtzKSxjPW5ldyBEYXRhVmlldyhhKSx1PTA7Yy5zZXRVaW50OCh1KyssdGhpcy5LSU5EUy5wdXNoKSxjLnNldFVpbnQ4KHUrKyx0Lmxlbmd0aCksYy5zZXRVaW50OCh1Kyssbi5sZW5ndGgpLGMuc2V0VWludDgodSsrLG8ubGVuZ3RoKSxjLnNldFVpbnQ4KHUrKyxpLmxlbmd0aCksQXJyYXkuZnJvbSh0LChmdW5jdGlvbihlKXtyZXR1cm4gYy5zZXRVaW50OCh1KyssZS5jaGFyQ29kZUF0KDApKX0pKSxBcnJheS5mcm9tKG4sKGZ1bmN0aW9uKGUpe3JldHVybiBjLnNldFVpbnQ4KHUrKyxlLmNoYXJDb2RlQXQoMCkpfSkpLEFycmF5LmZyb20obywoZnVuY3Rpb24oZSl7cmV0dXJuIGMuc2V0VWludDgodSsrLGUuY2hhckNvZGVBdCgwKSl9KSksQXJyYXkuZnJvbShpLChmdW5jdGlvbihlKXtyZXR1cm4gYy5zZXRVaW50OCh1KyssZS5jaGFyQ29kZUF0KDApKX0pKTt2YXIgaD1uZXcgVWludDhBcnJheShhLmJ5dGVMZW5ndGgrci5ieXRlTGVuZ3RoKTtyZXR1cm4gaC5zZXQobmV3IFVpbnQ4QXJyYXkoYSksMCksaC5zZXQobmV3IFVpbnQ4QXJyYXkociksYS5ieXRlTGVuZ3RoKSxoLmJ1ZmZlcn0sYmluYXJ5RGVjb2RlOmZ1bmN0aW9uKGUpe3ZhciB0PW5ldyBEYXRhVmlldyhlKSxuPXQuZ2V0VWludDgoMCksaT1uZXcgVGV4dERlY29kZXI7c3dpdGNoKG4pe2Nhc2UgdGhpcy5LSU5EUy5wdXNoOnJldHVybiB0aGlzLmRlY29kZVB1c2goZSx0LGkpO2Nhc2UgdGhpcy5LSU5EUy5yZXBseTpyZXR1cm4gdGhpcy5kZWNvZGVSZXBseShlLHQsaSk7Y2FzZSB0aGlzLktJTkRTLmJyb2FkY2FzdDpyZXR1cm4gdGhpcy5kZWNvZGVCcm9hZGNhc3QoZSx0LGkpfX0sZGVjb2RlUHVzaDpmdW5jdGlvbihlLHQsbil7dmFyIGk9dC5nZXRVaW50OCgxKSxvPXQuZ2V0VWludDgoMikscj10LmdldFVpbnQ4KDMpLHM9dGhpcy5IRUFERVJfTEVOR1RIK3RoaXMuTUVUQV9MRU5HVEgtMSxhPW4uZGVjb2RlKGUuc2xpY2UocyxzK2kpKTtzKz1pO3ZhciBjPW4uZGVjb2RlKGUuc2xpY2UocyxzK28pKTtzKz1vO3ZhciB1PW4uZGVjb2RlKGUuc2xpY2UocyxzK3IpKTtyZXR1cm4gcys9cix7am9pbl9yZWY6YSxyZWY6bnVsbCx0b3BpYzpjLGV2ZW50OnUscGF5bG9hZDplLnNsaWNlKHMsZS5ieXRlTGVuZ3RoKX19LGRlY29kZVJlcGx5OmZ1bmN0aW9uKGUsdCxuKXt2YXIgaT10LmdldFVpbnQ4KDEpLG89dC5nZXRVaW50OCgyKSxyPXQuZ2V0VWludDgoMykscz10LmdldFVpbnQ4KDQpLGE9dGhpcy5IRUFERVJfTEVOR1RIK3RoaXMuTUVUQV9MRU5HVEgsYz1uLmRlY29kZShlLnNsaWNlKGEsYStpKSk7YSs9aTt2YXIgdT1uLmRlY29kZShlLnNsaWNlKGEsYStvKSk7YSs9bzt2YXIgaD1uLmRlY29kZShlLnNsaWNlKGEsYStyKSk7YSs9cjt2YXIgbD1uLmRlY29kZShlLnNsaWNlKGEsYStzKSk7YSs9czt2YXIgZj1lLnNsaWNlKGEsZS5ieXRlTGVuZ3RoKTtyZXR1cm57am9pbl9yZWY6YyxyZWY6dSx0b3BpYzpoLGV2ZW50OlMscGF5bG9hZDp7c3RhdHVzOmwscmVzcG9uc2U6Zn19fSxkZWNvZGVCcm9hZGNhc3Q6ZnVuY3Rpb24oZSx0LG4pe3ZhciBpPXQuZ2V0VWludDgoMSksbz10LmdldFVpbnQ4KDIpLHI9dGhpcy5IRUFERVJfTEVOR1RIKzIscz1uLmRlY29kZShlLnNsaWNlKHIscitpKSk7cis9aTt2YXIgYT1uLmRlY29kZShlLnNsaWNlKHIscitvKSk7cmV0dXJuIHIrPW8se2pvaW5fcmVmOm51bGwscmVmOm51bGwsdG9waWM6cyxldmVudDphLHBheWxvYWQ6ZS5zbGljZShyLGUuYnl0ZUxlbmd0aCl9fX0sVT1mdW5jdGlvbigpe2Z1bmN0aW9uIGUodCl7dmFyIG49dGhpcyxpPWFyZ3VtZW50cy5sZW5ndGg+MSYmdm9pZCAwIT09YXJndW1lbnRzWzFdP2FyZ3VtZW50c1sxXTp7fTtjKHRoaXMsZSksdGhpcy5zdGF0ZUNoYW5nZUNhbGxiYWNrcz17b3BlbjpbXSxjbG9zZTpbXSxlcnJvcjpbXSxtZXNzYWdlOltdfSx0aGlzLmNoYW5uZWxzPVtdLHRoaXMuc2VuZEJ1ZmZlcj1bXSx0aGlzLnJlZj0wLHRoaXMudGltZW91dD1pLnRpbWVvdXR8fDFlNCx0aGlzLnRyYW5zcG9ydD1pLnRyYW5zcG9ydHx8ZC5XZWJTb2NrZXR8fEQsdGhpcy5kZWZhdWx0RW5jb2Rlcj1ILmVuY29kZS5iaW5kKEgpLHRoaXMuZGVmYXVsdERlY29kZXI9SC5kZWNvZGUuYmluZChIKSx0aGlzLmNsb3NlV2FzQ2xlYW49ITEsdGhpcy51bmxvYWRlZD0hMSx0aGlzLmJpbmFyeVR5cGU9aS5iaW5hcnlUeXBlfHxcImFycmF5YnVmZmVyXCIsdGhpcy50cmFuc3BvcnQhPT1EPyh0aGlzLmVuY29kZT1pLmVuY29kZXx8dGhpcy5kZWZhdWx0RW5jb2Rlcix0aGlzLmRlY29kZT1pLmRlY29kZXx8dGhpcy5kZWZhdWx0RGVjb2Rlcik6KHRoaXMuZW5jb2RlPXRoaXMuZGVmYXVsdEVuY29kZXIsdGhpcy5kZWNvZGU9dGhpcy5kZWZhdWx0RGVjb2RlciksZiYmZi5hZGRFdmVudExpc3RlbmVyJiZmLmFkZEV2ZW50TGlzdGVuZXIoXCJ1bmxvYWRcIiwoZnVuY3Rpb24oZSl7bi5jb25uJiYobi51bmxvYWRlZD0hMCxuLmFibm9ybWFsQ2xvc2UoXCJ1bmxvYWRlZFwiKSl9KSksdGhpcy5oZWFydGJlYXRJbnRlcnZhbE1zPWkuaGVhcnRiZWF0SW50ZXJ2YWxNc3x8M2U0LHRoaXMucmVqb2luQWZ0ZXJNcz1mdW5jdGlvbihlKXtyZXR1cm4gaS5yZWpvaW5BZnRlck1zP2kucmVqb2luQWZ0ZXJNcyhlKTpbMWUzLDJlMyw1ZTNdW2UtMV18fDFlNH0sdGhpcy5yZWNvbm5lY3RBZnRlck1zPWZ1bmN0aW9uKGUpe3JldHVybiBuLnVubG9hZGVkPzEwMDppLnJlY29ubmVjdEFmdGVyTXM/aS5yZWNvbm5lY3RBZnRlck1zKGUpOlsxMCw1MCwxMDAsMTUwLDIwMCwyNTAsNTAwLDFlMywyZTNdW2UtMV18fDVlM30sdGhpcy5sb2dnZXI9aS5sb2dnZXJ8fG51bGwsdGhpcy5sb25ncG9sbGVyVGltZW91dD1pLmxvbmdwb2xsZXJUaW1lb3V0fHwyZTQsdGhpcy5wYXJhbXM9TyhpLnBhcmFtc3x8e30pLHRoaXMuZW5kUG9pbnQ9XCJcIi5jb25jYXQodCxcIi9cIikuY29uY2F0KHgpLHRoaXMudnNuPWkudnNufHxcIjIuMC4wXCIsdGhpcy5oZWFydGJlYXRUaW1lcj1udWxsLHRoaXMucGVuZGluZ0hlYXJ0YmVhdFJlZj1udWxsLHRoaXMucmVjb25uZWN0VGltZXI9bmV3IEooKGZ1bmN0aW9uKCl7bi50ZWFyZG93bigoZnVuY3Rpb24oKXtyZXR1cm4gbi5jb25uZWN0KCl9KSl9KSx0aGlzLnJlY29ubmVjdEFmdGVyTXMpfXJldHVybiBoKGUsW3trZXk6XCJwcm90b2NvbFwiLHZhbHVlOmZ1bmN0aW9uKCl7cmV0dXJuIGxvY2F0aW9uLnByb3RvY29sLm1hdGNoKC9eaHR0cHMvKT9cIndzc1wiOlwid3NcIn19LHtrZXk6XCJlbmRQb2ludFVSTFwiLHZhbHVlOmZ1bmN0aW9uKCl7dmFyIGU9TS5hcHBlbmRQYXJhbXMoTS5hcHBlbmRQYXJhbXModGhpcy5lbmRQb2ludCx0aGlzLnBhcmFtcygpKSx7dnNuOnRoaXMudnNufSk7cmV0dXJuXCIvXCIhPT1lLmNoYXJBdCgwKT9lOlwiL1wiPT09ZS5jaGFyQXQoMSk/XCJcIi5jb25jYXQodGhpcy5wcm90b2NvbCgpLFwiOlwiKS5jb25jYXQoZSk6XCJcIi5jb25jYXQodGhpcy5wcm90b2NvbCgpLFwiOi8vXCIpLmNvbmNhdChsb2NhdGlvbi5ob3N0KS5jb25jYXQoZSl9fSx7a2V5OlwiZGlzY29ubmVjdFwiLHZhbHVlOmZ1bmN0aW9uKGUsdCxuKXt0aGlzLmNsb3NlV2FzQ2xlYW49ITAsdGhpcy5yZWNvbm5lY3RUaW1lci5yZXNldCgpLHRoaXMudGVhcmRvd24oZSx0LG4pfX0se2tleTpcImNvbm5lY3RcIix2YWx1ZTpmdW5jdGlvbihlKXt2YXIgdD10aGlzO2UmJihjb25zb2xlJiZjb25zb2xlLmxvZyhcInBhc3NpbmcgcGFyYW1zIHRvIGNvbm5lY3QgaXMgZGVwcmVjYXRlZC4gSW5zdGVhZCBwYXNzIDpwYXJhbXMgdG8gdGhlIFNvY2tldCBjb25zdHJ1Y3RvclwiKSx0aGlzLnBhcmFtcz1PKGUpKSx0aGlzLmNvbm58fCh0aGlzLmNsb3NlV2FzQ2xlYW49ITEsdGhpcy5jb25uPW5ldyB0aGlzLnRyYW5zcG9ydCh0aGlzLmVuZFBvaW50VVJMKCkpLHRoaXMuY29ubi5iaW5hcnlUeXBlPXRoaXMuYmluYXJ5VHlwZSx0aGlzLmNvbm4udGltZW91dD10aGlzLmxvbmdwb2xsZXJUaW1lb3V0LHRoaXMuY29ubi5vbm9wZW49ZnVuY3Rpb24oKXtyZXR1cm4gdC5vbkNvbm5PcGVuKCl9LHRoaXMuY29ubi5vbmVycm9yPWZ1bmN0aW9uKGUpe3JldHVybiB0Lm9uQ29ubkVycm9yKGUpfSx0aGlzLmNvbm4ub25tZXNzYWdlPWZ1bmN0aW9uKGUpe3JldHVybiB0Lm9uQ29ubk1lc3NhZ2UoZSl9LHRoaXMuY29ubi5vbmNsb3NlPWZ1bmN0aW9uKGUpe3JldHVybiB0Lm9uQ29ubkNsb3NlKGUpfSl9fSx7a2V5OlwibG9nXCIsdmFsdWU6ZnVuY3Rpb24oZSx0LG4pe3RoaXMubG9nZ2VyKGUsdCxuKX19LHtrZXk6XCJoYXNMb2dnZXJcIix2YWx1ZTpmdW5jdGlvbigpe3JldHVybiBudWxsIT09dGhpcy5sb2dnZXJ9fSx7a2V5Olwib25PcGVuXCIsdmFsdWU6ZnVuY3Rpb24oZSl7dmFyIHQ9dGhpcy5tYWtlUmVmKCk7cmV0dXJuIHRoaXMuc3RhdGVDaGFuZ2VDYWxsYmFja3Mub3Blbi5wdXNoKFt0LGVdKSx0fX0se2tleTpcIm9uQ2xvc2VcIix2YWx1ZTpmdW5jdGlvbihlKXt2YXIgdD10aGlzLm1ha2VSZWYoKTtyZXR1cm4gdGhpcy5zdGF0ZUNoYW5nZUNhbGxiYWNrcy5jbG9zZS5wdXNoKFt0LGVdKSx0fX0se2tleTpcIm9uRXJyb3JcIix2YWx1ZTpmdW5jdGlvbihlKXt2YXIgdD10aGlzLm1ha2VSZWYoKTtyZXR1cm4gdGhpcy5zdGF0ZUNoYW5nZUNhbGxiYWNrcy5lcnJvci5wdXNoKFt0LGVdKSx0fX0se2tleTpcIm9uTWVzc2FnZVwiLHZhbHVlOmZ1bmN0aW9uKGUpe3ZhciB0PXRoaXMubWFrZVJlZigpO3JldHVybiB0aGlzLnN0YXRlQ2hhbmdlQ2FsbGJhY2tzLm1lc3NhZ2UucHVzaChbdCxlXSksdH19LHtrZXk6XCJvbkNvbm5PcGVuXCIsdmFsdWU6ZnVuY3Rpb24oKXt0aGlzLmhhc0xvZ2dlcigpJiZ0aGlzLmxvZyhcInRyYW5zcG9ydFwiLFwiY29ubmVjdGVkIHRvIFwiLmNvbmNhdCh0aGlzLmVuZFBvaW50VVJMKCkpKSx0aGlzLnVubG9hZGVkPSExLHRoaXMuY2xvc2VXYXNDbGVhbj0hMSx0aGlzLmZsdXNoU2VuZEJ1ZmZlcigpLHRoaXMucmVjb25uZWN0VGltZXIucmVzZXQoKSx0aGlzLnJlc2V0SGVhcnRiZWF0KCksdGhpcy5zdGF0ZUNoYW5nZUNhbGxiYWNrcy5vcGVuLmZvckVhY2goKGZ1bmN0aW9uKGUpe3JldHVybigwLHIoZSwyKVsxXSkoKX0pKX19LHtrZXk6XCJyZXNldEhlYXJ0YmVhdFwiLHZhbHVlOmZ1bmN0aW9uKCl7dmFyIGU9dGhpczt0aGlzLmNvbm4mJnRoaXMuY29ubi5za2lwSGVhcnRiZWF0fHwodGhpcy5wZW5kaW5nSGVhcnRiZWF0UmVmPW51bGwsY2xlYXJJbnRlcnZhbCh0aGlzLmhlYXJ0YmVhdFRpbWVyKSx0aGlzLmhlYXJ0YmVhdFRpbWVyPXNldEludGVydmFsKChmdW5jdGlvbigpe3JldHVybiBlLnNlbmRIZWFydGJlYXQoKX0pLHRoaXMuaGVhcnRiZWF0SW50ZXJ2YWxNcykpfX0se2tleTpcInRlYXJkb3duXCIsdmFsdWU6ZnVuY3Rpb24oZSx0LG4pe3ZhciBpPXRoaXM7aWYoIXRoaXMuY29ubilyZXR1cm4gZSYmZSgpO3RoaXMud2FpdEZvckJ1ZmZlckRvbmUoKGZ1bmN0aW9uKCl7aS5jb25uJiYodD9pLmNvbm4uY2xvc2UodCxufHxcIlwiKTppLmNvbm4uY2xvc2UoKSksaS53YWl0Rm9yU29ja2V0Q2xvc2VkKChmdW5jdGlvbigpe2kuY29ubiYmKGkuY29ubi5vbmNsb3NlPWZ1bmN0aW9uKCl7fSxpLmNvbm49bnVsbCksZSYmZSgpfSkpfSkpfX0se2tleTpcIndhaXRGb3JCdWZmZXJEb25lXCIsdmFsdWU6ZnVuY3Rpb24oZSl7dmFyIHQ9dGhpcyxuPWFyZ3VtZW50cy5sZW5ndGg+MSYmdm9pZCAwIT09YXJndW1lbnRzWzFdP2FyZ3VtZW50c1sxXToxOzUhPT1uJiZ0aGlzLmNvbm4mJnRoaXMuY29ubi5idWZmZXJlZEFtb3VudD9zZXRUaW1lb3V0KChmdW5jdGlvbigpe3Qud2FpdEZvckJ1ZmZlckRvbmUoZSxuKzEpfSksMTUwKm4pOmUoKX19LHtrZXk6XCJ3YWl0Rm9yU29ja2V0Q2xvc2VkXCIsdmFsdWU6ZnVuY3Rpb24oZSl7dmFyIHQ9dGhpcyxuPWFyZ3VtZW50cy5sZW5ndGg+MSYmdm9pZCAwIT09YXJndW1lbnRzWzFdP2FyZ3VtZW50c1sxXToxOzUhPT1uJiZ0aGlzLmNvbm4mJnRoaXMuY29ubi5yZWFkeVN0YXRlIT09bT9zZXRUaW1lb3V0KChmdW5jdGlvbigpe3Qud2FpdEZvclNvY2tldENsb3NlZChlLG4rMSl9KSwxNTAqbik6ZSgpfX0se2tleTpcIm9uQ29ubkNsb3NlXCIsdmFsdWU6ZnVuY3Rpb24oZSl7dGhpcy5oYXNMb2dnZXIoKSYmdGhpcy5sb2coXCJ0cmFuc3BvcnRcIixcImNsb3NlXCIsZSksdGhpcy50cmlnZ2VyQ2hhbkVycm9yKCksY2xlYXJJbnRlcnZhbCh0aGlzLmhlYXJ0YmVhdFRpbWVyKSx0aGlzLmNsb3NlV2FzQ2xlYW58fHRoaXMucmVjb25uZWN0VGltZXIuc2NoZWR1bGVUaW1lb3V0KCksdGhpcy5zdGF0ZUNoYW5nZUNhbGxiYWNrcy5jbG9zZS5mb3JFYWNoKChmdW5jdGlvbih0KXtyZXR1cm4oMCxyKHQsMilbMV0pKGUpfSkpfX0se2tleTpcIm9uQ29ubkVycm9yXCIsdmFsdWU6ZnVuY3Rpb24oZSl7dGhpcy5oYXNMb2dnZXIoKSYmdGhpcy5sb2coXCJ0cmFuc3BvcnRcIixlKSx0aGlzLnRyaWdnZXJDaGFuRXJyb3IoKSx0aGlzLnN0YXRlQ2hhbmdlQ2FsbGJhY2tzLmVycm9yLmZvckVhY2goKGZ1bmN0aW9uKHQpe3JldHVybigwLHIodCwyKVsxXSkoZSl9KSl9fSx7a2V5OlwidHJpZ2dlckNoYW5FcnJvclwiLHZhbHVlOmZ1bmN0aW9uKCl7dGhpcy5jaGFubmVscy5mb3JFYWNoKChmdW5jdGlvbihlKXtlLmlzRXJyb3JlZCgpfHxlLmlzTGVhdmluZygpfHxlLmlzQ2xvc2VkKCl8fGUudHJpZ2dlcihSKX0pKX19LHtrZXk6XCJjb25uZWN0aW9uU3RhdGVcIix2YWx1ZTpmdW5jdGlvbigpe3N3aXRjaCh0aGlzLmNvbm4mJnRoaXMuY29ubi5yZWFkeVN0YXRlKXtjYXNlIHA6cmV0dXJuXCJjb25uZWN0aW5nXCI7Y2FzZSB2OnJldHVyblwib3BlblwiO2Nhc2UgeTpyZXR1cm5cImNsb3NpbmdcIjtkZWZhdWx0OnJldHVyblwiY2xvc2VkXCJ9fX0se2tleTpcImlzQ29ubmVjdGVkXCIsdmFsdWU6ZnVuY3Rpb24oKXtyZXR1cm5cIm9wZW5cIj09PXRoaXMuY29ubmVjdGlvblN0YXRlKCl9fSx7a2V5OlwicmVtb3ZlXCIsdmFsdWU6ZnVuY3Rpb24oZSl7dGhpcy5vZmYoZS5zdGF0ZUNoYW5nZVJlZnMpLHRoaXMuY2hhbm5lbHM9dGhpcy5jaGFubmVscy5maWx0ZXIoKGZ1bmN0aW9uKHQpe3JldHVybiB0LmpvaW5SZWYoKSE9PWUuam9pblJlZigpfSkpfX0se2tleTpcIm9mZlwiLHZhbHVlOmZ1bmN0aW9uKGUpe2Zvcih2YXIgdCBpbiB0aGlzLnN0YXRlQ2hhbmdlQ2FsbGJhY2tzKXRoaXMuc3RhdGVDaGFuZ2VDYWxsYmFja3NbdF09dGhpcy5zdGF0ZUNoYW5nZUNhbGxiYWNrc1t0XS5maWx0ZXIoKGZ1bmN0aW9uKHQpe3ZhciBuPXIodCwxKVswXTtyZXR1cm4tMT09PWUuaW5kZXhPZihuKX0pKX19LHtrZXk6XCJjaGFubmVsXCIsdmFsdWU6ZnVuY3Rpb24oZSl7dmFyIHQ9YXJndW1lbnRzLmxlbmd0aD4xJiZ2b2lkIDAhPT1hcmd1bWVudHNbMV0/YXJndW1lbnRzWzFdOnt9LG49bmV3IF8oZSx0LHRoaXMpO3JldHVybiB0aGlzLmNoYW5uZWxzLnB1c2gobiksbn19LHtrZXk6XCJwdXNoXCIsdmFsdWU6ZnVuY3Rpb24oZSl7dmFyIHQ9dGhpcztpZih0aGlzLmhhc0xvZ2dlcigpKXt2YXIgbj1lLnRvcGljLGk9ZS5ldmVudCxvPWUucGF5bG9hZCxyPWUucmVmLHM9ZS5qb2luX3JlZjt0aGlzLmxvZyhcInB1c2hcIixcIlwiLmNvbmNhdChuLFwiIFwiKS5jb25jYXQoaSxcIiAoXCIpLmNvbmNhdChzLFwiLCBcIikuY29uY2F0KHIsXCIpXCIpLG8pfXRoaXMuaXNDb25uZWN0ZWQoKT90aGlzLmVuY29kZShlLChmdW5jdGlvbihlKXtyZXR1cm4gdC5jb25uLnNlbmQoZSl9KSk6dGhpcy5zZW5kQnVmZmVyLnB1c2goKGZ1bmN0aW9uKCl7cmV0dXJuIHQuZW5jb2RlKGUsKGZ1bmN0aW9uKGUpe3JldHVybiB0LmNvbm4uc2VuZChlKX0pKX0pKX19LHtrZXk6XCJtYWtlUmVmXCIsdmFsdWU6ZnVuY3Rpb24oKXt2YXIgZT10aGlzLnJlZisxO3JldHVybiBlPT09dGhpcy5yZWY/dGhpcy5yZWY9MDp0aGlzLnJlZj1lLHRoaXMucmVmLnRvU3RyaW5nKCl9fSx7a2V5Olwic2VuZEhlYXJ0YmVhdFwiLHZhbHVlOmZ1bmN0aW9uKCl7aWYodGhpcy5pc0Nvbm5lY3RlZCgpKXtpZih0aGlzLnBlbmRpbmdIZWFydGJlYXRSZWYpcmV0dXJuIHRoaXMucGVuZGluZ0hlYXJ0YmVhdFJlZj1udWxsLHRoaXMuaGFzTG9nZ2VyKCkmJnRoaXMubG9nKFwidHJhbnNwb3J0XCIsXCJoZWFydGJlYXQgdGltZW91dC4gQXR0ZW1wdGluZyB0byByZS1lc3RhYmxpc2ggY29ubmVjdGlvblwiKSx2b2lkIHRoaXMuYWJub3JtYWxDbG9zZShcImhlYXJ0YmVhdCB0aW1lb3V0XCIpO3RoaXMucGVuZGluZ0hlYXJ0YmVhdFJlZj10aGlzLm1ha2VSZWYoKSx0aGlzLnB1c2goe3RvcGljOlwicGhvZW5peFwiLGV2ZW50OlwiaGVhcnRiZWF0XCIscGF5bG9hZDp7fSxyZWY6dGhpcy5wZW5kaW5nSGVhcnRiZWF0UmVmfSl9fX0se2tleTpcImFibm9ybWFsQ2xvc2VcIix2YWx1ZTpmdW5jdGlvbihlKXt0aGlzLmNsb3NlV2FzQ2xlYW49ITEsdGhpcy5jb25uLnJlYWR5U3RhdGU9PT12JiZ0aGlzLmNvbm4uY2xvc2UoMWUzLGUpfX0se2tleTpcImZsdXNoU2VuZEJ1ZmZlclwiLHZhbHVlOmZ1bmN0aW9uKCl7dGhpcy5pc0Nvbm5lY3RlZCgpJiZ0aGlzLnNlbmRCdWZmZXIubGVuZ3RoPjAmJih0aGlzLnNlbmRCdWZmZXIuZm9yRWFjaCgoZnVuY3Rpb24oZSl7cmV0dXJuIGUoKX0pKSx0aGlzLnNlbmRCdWZmZXI9W10pfX0se2tleTpcIm9uQ29ubk1lc3NhZ2VcIix2YWx1ZTpmdW5jdGlvbihlKXt2YXIgdD10aGlzO3RoaXMuZGVjb2RlKGUuZGF0YSwoZnVuY3Rpb24oZSl7dmFyIG49ZS50b3BpYyxpPWUuZXZlbnQsbz1lLnBheWxvYWQscz1lLnJlZixhPWUuam9pbl9yZWY7cyYmcz09PXQucGVuZGluZ0hlYXJ0YmVhdFJlZiYmKHQucGVuZGluZ0hlYXJ0YmVhdFJlZj1udWxsKSx0Lmhhc0xvZ2dlcigpJiZ0LmxvZyhcInJlY2VpdmVcIixcIlwiLmNvbmNhdChvLnN0YXR1c3x8XCJcIixcIiBcIikuY29uY2F0KG4sXCIgXCIpLmNvbmNhdChpLFwiIFwiKS5jb25jYXQocyYmXCIoXCIrcytcIilcInx8XCJcIiksbyk7Zm9yKHZhciBjPTA7Yzx0LmNoYW5uZWxzLmxlbmd0aDtjKyspe3ZhciB1PXQuY2hhbm5lbHNbY107dS5pc01lbWJlcihuLGksbyxhKSYmdS50cmlnZ2VyKGksbyxzLGEpfWZvcih2YXIgaD0wO2g8dC5zdGF0ZUNoYW5nZUNhbGxiYWNrcy5tZXNzYWdlLmxlbmd0aDtoKyspeygwLHIodC5zdGF0ZUNoYW5nZUNhbGxiYWNrcy5tZXNzYWdlW2hdLDIpWzFdKShlKX19KSl9fSx7a2V5OlwibGVhdmVPcGVuVG9waWNcIix2YWx1ZTpmdW5jdGlvbihlKXt2YXIgdD10aGlzLmNoYW5uZWxzLmZpbmQoKGZ1bmN0aW9uKHQpe3JldHVybiB0LnRvcGljPT09ZSYmKHQuaXNKb2luZWQoKXx8dC5pc0pvaW5pbmcoKSl9KSk7dCYmKHRoaXMuaGFzTG9nZ2VyKCkmJnRoaXMubG9nKFwidHJhbnNwb3J0XCIsJ2xlYXZpbmcgZHVwbGljYXRlIHRvcGljIFwiJy5jb25jYXQoZSwnXCInKSksdC5sZWF2ZSgpKX19XSksZX0oKSxEPWZ1bmN0aW9uKCl7ZnVuY3Rpb24gZSh0KXtjKHRoaXMsZSksdGhpcy5lbmRQb2ludD1udWxsLHRoaXMudG9rZW49bnVsbCx0aGlzLnNraXBIZWFydGJlYXQ9ITAsdGhpcy5vbm9wZW49ZnVuY3Rpb24oKXt9LHRoaXMub25lcnJvcj1mdW5jdGlvbigpe30sdGhpcy5vbm1lc3NhZ2U9ZnVuY3Rpb24oKXt9LHRoaXMub25jbG9zZT1mdW5jdGlvbigpe30sdGhpcy5wb2xsRW5kcG9pbnQ9dGhpcy5ub3JtYWxpemVFbmRwb2ludCh0KSx0aGlzLnJlYWR5U3RhdGU9cCx0aGlzLnBvbGwoKX1yZXR1cm4gaChlLFt7a2V5Olwibm9ybWFsaXplRW5kcG9pbnRcIix2YWx1ZTpmdW5jdGlvbihlKXtyZXR1cm4gZS5yZXBsYWNlKFwid3M6Ly9cIixcImh0dHA6Ly9cIikucmVwbGFjZShcIndzczovL1wiLFwiaHR0cHM6Ly9cIikucmVwbGFjZShuZXcgUmVnRXhwKFwiKC4qKS9cIit4KSxcIiQxL1wiK0wpfX0se2tleTpcImVuZHBvaW50VVJMXCIsdmFsdWU6ZnVuY3Rpb24oKXtyZXR1cm4gTS5hcHBlbmRQYXJhbXModGhpcy5wb2xsRW5kcG9pbnQse3Rva2VuOnRoaXMudG9rZW59KX19LHtrZXk6XCJjbG9zZUFuZFJldHJ5XCIsdmFsdWU6ZnVuY3Rpb24oKXt0aGlzLmNsb3NlKCksdGhpcy5yZWFkeVN0YXRlPXB9fSx7a2V5Olwib250aW1lb3V0XCIsdmFsdWU6ZnVuY3Rpb24oKXt0aGlzLm9uZXJyb3IoXCJ0aW1lb3V0XCIpLHRoaXMuY2xvc2VBbmRSZXRyeSgpfX0se2tleTpcInBvbGxcIix2YWx1ZTpmdW5jdGlvbigpe3ZhciBlPXRoaXM7dGhpcy5yZWFkeVN0YXRlIT09diYmdGhpcy5yZWFkeVN0YXRlIT09cHx8TS5yZXF1ZXN0KFwiR0VUXCIsdGhpcy5lbmRwb2ludFVSTCgpLFwiYXBwbGljYXRpb24vanNvblwiLG51bGwsdGhpcy50aW1lb3V0LHRoaXMub250aW1lb3V0LmJpbmQodGhpcyksKGZ1bmN0aW9uKHQpe2lmKHQpe3ZhciBuPXQuc3RhdHVzLGk9dC50b2tlbixvPXQubWVzc2FnZXM7ZS50b2tlbj1pfWVsc2Ugbj0wO3N3aXRjaChuKXtjYXNlIDIwMDpvLmZvckVhY2goKGZ1bmN0aW9uKHQpe3JldHVybiBlLm9ubWVzc2FnZSh7ZGF0YTp0fSl9KSksZS5wb2xsKCk7YnJlYWs7Y2FzZSAyMDQ6ZS5wb2xsKCk7YnJlYWs7Y2FzZSA0MTA6ZS5yZWFkeVN0YXRlPXYsZS5vbm9wZW4oKSxlLnBvbGwoKTticmVhaztjYXNlIDQwMzplLm9uZXJyb3IoKSxlLmNsb3NlKCk7YnJlYWs7Y2FzZSAwOmNhc2UgNTAwOmUub25lcnJvcigpLGUuY2xvc2VBbmRSZXRyeSgpO2JyZWFrO2RlZmF1bHQ6dGhyb3cgbmV3IEVycm9yKFwidW5oYW5kbGVkIHBvbGwgc3RhdHVzIFwiLmNvbmNhdChuKSl9fSkpfX0se2tleTpcInNlbmRcIix2YWx1ZTpmdW5jdGlvbihlKXt2YXIgdD10aGlzO00ucmVxdWVzdChcIlBPU1RcIix0aGlzLmVuZHBvaW50VVJMKCksXCJhcHBsaWNhdGlvbi9qc29uXCIsZSx0aGlzLnRpbWVvdXQsdGhpcy5vbmVycm9yLmJpbmQodGhpcyxcInRpbWVvdXRcIiksKGZ1bmN0aW9uKGUpe2UmJjIwMD09PWUuc3RhdHVzfHwodC5vbmVycm9yKGUmJmUuc3RhdHVzKSx0LmNsb3NlQW5kUmV0cnkoKSl9KSl9fSx7a2V5OlwiY2xvc2VcIix2YWx1ZTpmdW5jdGlvbihlLHQpe3RoaXMucmVhZHlTdGF0ZT1tLHRoaXMub25jbG9zZSgpfX1dKSxlfSgpLE09ZnVuY3Rpb24oKXtmdW5jdGlvbiBlKCl7Yyh0aGlzLGUpfXJldHVybiBoKGUsbnVsbCxbe2tleTpcInJlcXVlc3RcIix2YWx1ZTpmdW5jdGlvbihlLHQsbixpLG8scixzKXtpZihkLlhEb21haW5SZXF1ZXN0KXt2YXIgYT1uZXcgWERvbWFpblJlcXVlc3Q7dGhpcy54ZG9tYWluUmVxdWVzdChhLGUsdCxpLG8scixzKX1lbHNle3ZhciBjPW5ldyBkLlhNTEh0dHBSZXF1ZXN0O3RoaXMueGhyUmVxdWVzdChjLGUsdCxuLGksbyxyLHMpfX19LHtrZXk6XCJ4ZG9tYWluUmVxdWVzdFwiLHZhbHVlOmZ1bmN0aW9uKGUsdCxuLGksbyxyLHMpe3ZhciBhPXRoaXM7ZS50aW1lb3V0PW8sZS5vcGVuKHQsbiksZS5vbmxvYWQ9ZnVuY3Rpb24oKXt2YXIgdD1hLnBhcnNlSlNPTihlLnJlc3BvbnNlVGV4dCk7cyYmcyh0KX0sciYmKGUub250aW1lb3V0PXIpLGUub25wcm9ncmVzcz1mdW5jdGlvbigpe30sZS5zZW5kKGkpfX0se2tleTpcInhoclJlcXVlc3RcIix2YWx1ZTpmdW5jdGlvbihlLHQsbixpLG8scixzLGEpe3ZhciBjPXRoaXM7ZS5vcGVuKHQsbiwhMCksZS50aW1lb3V0PXIsZS5zZXRSZXF1ZXN0SGVhZGVyKFwiQ29udGVudC1UeXBlXCIsaSksZS5vbmVycm9yPWZ1bmN0aW9uKCl7YSYmYShudWxsKX0sZS5vbnJlYWR5c3RhdGVjaGFuZ2U9ZnVuY3Rpb24oKXtpZihlLnJlYWR5U3RhdGU9PT1jLnN0YXRlcy5jb21wbGV0ZSYmYSl7dmFyIHQ9Yy5wYXJzZUpTT04oZS5yZXNwb25zZVRleHQpO2EodCl9fSxzJiYoZS5vbnRpbWVvdXQ9cyksZS5zZW5kKG8pfX0se2tleTpcInBhcnNlSlNPTlwiLHZhbHVlOmZ1bmN0aW9uKGUpe2lmKCFlfHxcIlwiPT09ZSlyZXR1cm4gbnVsbDt0cnl7cmV0dXJuIEpTT04ucGFyc2UoZSl9Y2F0Y2godCl7cmV0dXJuIGNvbnNvbGUmJmNvbnNvbGUubG9nKFwiZmFpbGVkIHRvIHBhcnNlIEpTT04gcmVzcG9uc2VcIixlKSxudWxsfX19LHtrZXk6XCJzZXJpYWxpemVcIix2YWx1ZTpmdW5jdGlvbihlLHQpe3ZhciBuPVtdO2Zvcih2YXIgaSBpbiBlKWlmKGUuaGFzT3duUHJvcGVydHkoaSkpe3ZhciByPXQ/XCJcIi5jb25jYXQodCxcIltcIikuY29uY2F0KGksXCJdXCIpOmkscz1lW2ldO1wib2JqZWN0XCI9PT1vKHMpP24ucHVzaCh0aGlzLnNlcmlhbGl6ZShzLHIpKTpuLnB1c2goZW5jb2RlVVJJQ29tcG9uZW50KHIpK1wiPVwiK2VuY29kZVVSSUNvbXBvbmVudChzKSl9cmV0dXJuIG4uam9pbihcIiZcIil9fSx7a2V5OlwiYXBwZW5kUGFyYW1zXCIsdmFsdWU6ZnVuY3Rpb24oZSx0KXtpZigwPT09T2JqZWN0LmtleXModCkubGVuZ3RoKXJldHVybiBlO3ZhciBuPWUubWF0Y2goL1xcPy8pP1wiJlwiOlwiP1wiO3JldHVyblwiXCIuY29uY2F0KGUpLmNvbmNhdChuKS5jb25jYXQodGhpcy5zZXJpYWxpemUodCkpfX1dKSxlfSgpO00uc3RhdGVzPXtjb21wbGV0ZTo0fTt2YXIgTj1mdW5jdGlvbigpe2Z1bmN0aW9uIGUodCl7dmFyIG49dGhpcyxpPWFyZ3VtZW50cy5sZW5ndGg+MSYmdm9pZCAwIT09YXJndW1lbnRzWzFdP2FyZ3VtZW50c1sxXTp7fTtjKHRoaXMsZSk7dmFyIG89aS5ldmVudHN8fHtzdGF0ZTpcInByZXNlbmNlX3N0YXRlXCIsZGlmZjpcInByZXNlbmNlX2RpZmZcIn07dGhpcy5zdGF0ZT17fSx0aGlzLnBlbmRpbmdEaWZmcz1bXSx0aGlzLmNoYW5uZWw9dCx0aGlzLmpvaW5SZWY9bnVsbCx0aGlzLmNhbGxlcj17b25Kb2luOmZ1bmN0aW9uKCl7fSxvbkxlYXZlOmZ1bmN0aW9uKCl7fSxvblN5bmM6ZnVuY3Rpb24oKXt9fSx0aGlzLmNoYW5uZWwub24oby5zdGF0ZSwoZnVuY3Rpb24odCl7dmFyIGk9bi5jYWxsZXIsbz1pLm9uSm9pbixyPWkub25MZWF2ZSxzPWkub25TeW5jO24uam9pblJlZj1uLmNoYW5uZWwuam9pblJlZigpLG4uc3RhdGU9ZS5zeW5jU3RhdGUobi5zdGF0ZSx0LG8sciksbi5wZW5kaW5nRGlmZnMuZm9yRWFjaCgoZnVuY3Rpb24odCl7bi5zdGF0ZT1lLnN5bmNEaWZmKG4uc3RhdGUsdCxvLHIpfSkpLG4ucGVuZGluZ0RpZmZzPVtdLHMoKX0pKSx0aGlzLmNoYW5uZWwub24oby5kaWZmLChmdW5jdGlvbih0KXt2YXIgaT1uLmNhbGxlcixvPWkub25Kb2luLHI9aS5vbkxlYXZlLHM9aS5vblN5bmM7bi5pblBlbmRpbmdTeW5jU3RhdGUoKT9uLnBlbmRpbmdEaWZmcy5wdXNoKHQpOihuLnN0YXRlPWUuc3luY0RpZmYobi5zdGF0ZSx0LG8scikscygpKX0pKX1yZXR1cm4gaChlLFt7a2V5Olwib25Kb2luXCIsdmFsdWU6ZnVuY3Rpb24oZSl7dGhpcy5jYWxsZXIub25Kb2luPWV9fSx7a2V5Olwib25MZWF2ZVwiLHZhbHVlOmZ1bmN0aW9uKGUpe3RoaXMuY2FsbGVyLm9uTGVhdmU9ZX19LHtrZXk6XCJvblN5bmNcIix2YWx1ZTpmdW5jdGlvbihlKXt0aGlzLmNhbGxlci5vblN5bmM9ZX19LHtrZXk6XCJsaXN0XCIsdmFsdWU6ZnVuY3Rpb24odCl7cmV0dXJuIGUubGlzdCh0aGlzLnN0YXRlLHQpfX0se2tleTpcImluUGVuZGluZ1N5bmNTdGF0ZVwiLHZhbHVlOmZ1bmN0aW9uKCl7cmV0dXJuIXRoaXMuam9pblJlZnx8dGhpcy5qb2luUmVmIT09dGhpcy5jaGFubmVsLmpvaW5SZWYoKX19XSxbe2tleTpcInN5bmNTdGF0ZVwiLHZhbHVlOmZ1bmN0aW9uKGUsdCxuLGkpe3ZhciBvPXRoaXMscj10aGlzLmNsb25lKGUpLHM9e30sYT17fTtyZXR1cm4gdGhpcy5tYXAociwoZnVuY3Rpb24oZSxuKXt0W2VdfHwoYVtlXT1uKX0pKSx0aGlzLm1hcCh0LChmdW5jdGlvbihlLHQpe3ZhciBuPXJbZV07aWYobil7dmFyIGk9dC5tZXRhcy5tYXAoKGZ1bmN0aW9uKGUpe3JldHVybiBlLnBoeF9yZWZ9KSksYz1uLm1ldGFzLm1hcCgoZnVuY3Rpb24oZSl7cmV0dXJuIGUucGh4X3JlZn0pKSx1PXQubWV0YXMuZmlsdGVyKChmdW5jdGlvbihlKXtyZXR1cm4gYy5pbmRleE9mKGUucGh4X3JlZik8MH0pKSxoPW4ubWV0YXMuZmlsdGVyKChmdW5jdGlvbihlKXtyZXR1cm4gaS5pbmRleE9mKGUucGh4X3JlZik8MH0pKTt1Lmxlbmd0aD4wJiYoc1tlXT10LHNbZV0ubWV0YXM9dSksaC5sZW5ndGg+MCYmKGFbZV09by5jbG9uZShuKSxhW2VdLm1ldGFzPWgpfWVsc2Ugc1tlXT10fSkpLHRoaXMuc3luY0RpZmYocix7am9pbnM6cyxsZWF2ZXM6YX0sbixpKX19LHtrZXk6XCJzeW5jRGlmZlwiLHZhbHVlOmZ1bmN0aW9uKGUsdCxuLG8pe3ZhciByPXQuam9pbnMscz10LmxlYXZlcyxhPXRoaXMuY2xvbmUoZSk7cmV0dXJuIG58fChuPWZ1bmN0aW9uKCl7fSksb3x8KG89ZnVuY3Rpb24oKXt9KSx0aGlzLm1hcChyLChmdW5jdGlvbihlLHQpe3ZhciBvPWFbZV07aWYoYVtlXT10LG8pe3ZhciByLHM9YVtlXS5tZXRhcy5tYXAoKGZ1bmN0aW9uKGUpe3JldHVybiBlLnBoeF9yZWZ9KSksYz1vLm1ldGFzLmZpbHRlcigoZnVuY3Rpb24oZSl7cmV0dXJuIHMuaW5kZXhPZihlLnBoeF9yZWYpPDB9KSk7KHI9YVtlXS5tZXRhcykudW5zaGlmdC5hcHBseShyLGkoYykpfW4oZSxvLHQpfSkpLHRoaXMubWFwKHMsKGZ1bmN0aW9uKGUsdCl7dmFyIG49YVtlXTtpZihuKXt2YXIgaT10Lm1ldGFzLm1hcCgoZnVuY3Rpb24oZSl7cmV0dXJuIGUucGh4X3JlZn0pKTtuLm1ldGFzPW4ubWV0YXMuZmlsdGVyKChmdW5jdGlvbihlKXtyZXR1cm4gaS5pbmRleE9mKGUucGh4X3JlZik8MH0pKSxvKGUsbix0KSwwPT09bi5tZXRhcy5sZW5ndGgmJmRlbGV0ZSBhW2VdfX0pKSxhfX0se2tleTpcImxpc3RcIix2YWx1ZTpmdW5jdGlvbihlLHQpe3JldHVybiB0fHwodD1mdW5jdGlvbihlLHQpe3JldHVybiB0fSksdGhpcy5tYXAoZSwoZnVuY3Rpb24oZSxuKXtyZXR1cm4gdChlLG4pfSkpfX0se2tleTpcIm1hcFwiLHZhbHVlOmZ1bmN0aW9uKGUsdCl7cmV0dXJuIE9iamVjdC5nZXRPd25Qcm9wZXJ0eU5hbWVzKGUpLm1hcCgoZnVuY3Rpb24obil7cmV0dXJuIHQobixlW25dKX0pKX19LHtrZXk6XCJjbG9uZVwiLHZhbHVlOmZ1bmN0aW9uKGUpe3JldHVybiBKU09OLnBhcnNlKEpTT04uc3RyaW5naWZ5KGUpKX19XSksZX0oKSxKPWZ1bmN0aW9uKCl7ZnVuY3Rpb24gZSh0LG4pe2ModGhpcyxlKSx0aGlzLmNhbGxiYWNrPXQsdGhpcy50aW1lckNhbGM9bix0aGlzLnRpbWVyPW51bGwsdGhpcy50cmllcz0wfXJldHVybiBoKGUsW3trZXk6XCJyZXNldFwiLHZhbHVlOmZ1bmN0aW9uKCl7dGhpcy50cmllcz0wLGNsZWFyVGltZW91dCh0aGlzLnRpbWVyKX19LHtrZXk6XCJzY2hlZHVsZVRpbWVvdXRcIix2YWx1ZTpmdW5jdGlvbigpe3ZhciBlPXRoaXM7Y2xlYXJUaW1lb3V0KHRoaXMudGltZXIpLHRoaXMudGltZXI9c2V0VGltZW91dCgoZnVuY3Rpb24oKXtlLnRyaWVzPWUudHJpZXMrMSxlLmNhbGxiYWNrKCl9KSx0aGlzLnRpbWVyQ2FsYyh0aGlzLnRyaWVzKzEpKX19XSksZX0oKX1dKX0pKTsiXSwibWFwcGluZ3MiOiJBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFDQTtBQURBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUNBO0FBREE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUNBO0FBREE7QUFBQTtBQUFBO0FBQ0E7QUFEQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQ0E7QUFEQTtBQUFBO0FBQUE7QUFDQTtBQURBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUNBO0FBREE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFDQTtBQURBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUNBO0FBREE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUNBO0FBREE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUNBO0FBREE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFDQTtBQURBO0FBQUE7QUFDQTtBQURBO0FBQ0E7QUFEQTtBQUFBO0FBQ0E7QUFEQTtBQUFBO0FBQUE7QUFDQTtBQURBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUNBO0FBREE7QUFBQTtBQUFBO0FBQ0E7QUFEQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUNBO0FBREE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUNBO0FBREE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUNBO0FBREE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQ0E7QUFEQTtBQUFBO0FBQUE7QUFBQTtBQUNBO0FBREE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQ0E7QUFEQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUNBO0FBREE7QUFBQTtBQUFBO0FBQ0E7QUFEQTtBQUFBO0FBQ0E7QUFEQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUNBO0FBREE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUNBO0FBREE7QUFBQTtBQUNBO0FBREE7QUFBQTtBQUNBO0FBREE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUNBO0FBREE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUNBO0FBREE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUNBO0FBREE7QUFBQTtBQUFBO0FBQUE7QUFDQTtBQURBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQ0E7QUFEQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFDQTtBQURBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFDQTtBQURBO0FBQUE7QUFBQTtBQUNBO0FBREE7QUFBQTtBQUFBO0FBQ0E7QUFEQTtBQUFBO0FBQUE7QUFDQTtBQURBO0FBQUE7QUFBQTtBQUFBO0FBQ0E7QUFEQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFDQTtBQURBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQ0E7QUFEQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUNBO0FBREE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUNBO0FBREE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUNBO0FBREE7QUFBQTtBQUFBO0FBQ0E7QUFEQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQ0E7QUFEQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFDQTtBQURBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQ0E7QUFEQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUNBO0FBREE7QUFBQTtBQUFBO0FBQ0E7QUFEQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUNBO0FBREE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQSIsInNvdXJjZVJvb3QiOiIifQ==\n//# sourceURL=webpack-internal:///../deps/phoenix/priv/static/phoenix.js\n"); + + /***/ }), + + /***/ "../deps/phoenix_html/priv/static/phoenix_html.js": + /*!********************************************************!*\ + !*** ../deps/phoenix_html/priv/static/phoenix_html.js ***! + \********************************************************/ + /*! no static exports found */ + /***/ (function(module, exports, __webpack_require__) { + + "use strict"; + eval("\n\n(function () {\n var PolyfillEvent = eventConstructor();\n\n function eventConstructor() {\n if (typeof window.CustomEvent === \"function\") return window.CustomEvent; // IE<=9 Support\n\n function CustomEvent(event, params) {\n params = params || {\n bubbles: false,\n cancelable: false,\n detail: undefined\n };\n var evt = document.createEvent('CustomEvent');\n evt.initCustomEvent(event, params.bubbles, params.cancelable, params.detail);\n return evt;\n }\n\n CustomEvent.prototype = window.Event.prototype;\n return CustomEvent;\n }\n\n function buildHiddenInput(name, value) {\n var input = document.createElement(\"input\");\n input.type = \"hidden\";\n input.name = name;\n input.value = value;\n return input;\n }\n\n function handleClick(element) {\n var to = element.getAttribute(\"data-to\"),\n method = buildHiddenInput(\"_method\", element.getAttribute(\"data-method\")),\n csrf = buildHiddenInput(\"_csrf_token\", element.getAttribute(\"data-csrf\")),\n form = document.createElement(\"form\"),\n target = element.getAttribute(\"target\");\n form.method = element.getAttribute(\"data-method\") === \"get\" ? \"get\" : \"post\";\n form.action = to;\n form.style.display = \"hidden\";\n if (target) form.target = target;\n form.appendChild(csrf);\n form.appendChild(method);\n document.body.appendChild(form);\n form.submit();\n }\n\n window.addEventListener(\"click\", function (e) {\n var element = e.target;\n\n while (element && element.getAttribute) {\n var phoenixLinkEvent = new PolyfillEvent('phoenix.link.click', {\n \"bubbles\": true,\n \"cancelable\": true\n });\n\n if (!element.dispatchEvent(phoenixLinkEvent)) {\n e.preventDefault();\n e.stopImmediatePropagation();\n return false;\n }\n\n if (element.getAttribute(\"data-method\")) {\n handleClick(element);\n e.preventDefault();\n return false;\n } else {\n element = element.parentNode;\n }\n }\n }, false);\n window.addEventListener('phoenix.link.click', function (e) {\n var message = e.target.getAttribute(\"data-confirm\");\n\n if (message && !window.confirm(message)) {\n e.preventDefault();\n }\n }, false);\n})();//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiLi4vZGVwcy9waG9lbml4X2h0bWwvcHJpdi9zdGF0aWMvcGhvZW5peF9odG1sLmpzLmpzIiwic291cmNlcyI6WyJ3ZWJwYWNrOi8vLy4uL2RlcHMvcGhvZW5peF9odG1sL3ByaXYvc3RhdGljL3Bob2VuaXhfaHRtbC5qcz80N2Q4Il0sInNvdXJjZXNDb250ZW50IjpbIlwidXNlIHN0cmljdFwiO1xuXG4oZnVuY3Rpb24oKSB7XG4gIHZhciBQb2x5ZmlsbEV2ZW50ID0gZXZlbnRDb25zdHJ1Y3RvcigpO1xuXG4gIGZ1bmN0aW9uIGV2ZW50Q29uc3RydWN0b3IoKSB7XG4gICAgaWYgKHR5cGVvZiB3aW5kb3cuQ3VzdG9tRXZlbnQgPT09IFwiZnVuY3Rpb25cIikgcmV0dXJuIHdpbmRvdy5DdXN0b21FdmVudDtcbiAgICAvLyBJRTw9OSBTdXBwb3J0XG4gICAgZnVuY3Rpb24gQ3VzdG9tRXZlbnQoZXZlbnQsIHBhcmFtcykge1xuICAgICAgcGFyYW1zID0gcGFyYW1zIHx8IHtidWJibGVzOiBmYWxzZSwgY2FuY2VsYWJsZTogZmFsc2UsIGRldGFpbDogdW5kZWZpbmVkfTtcbiAgICAgIHZhciBldnQgPSBkb2N1bWVudC5jcmVhdGVFdmVudCgnQ3VzdG9tRXZlbnQnKTtcbiAgICAgIGV2dC5pbml0Q3VzdG9tRXZlbnQoZXZlbnQsIHBhcmFtcy5idWJibGVzLCBwYXJhbXMuY2FuY2VsYWJsZSwgcGFyYW1zLmRldGFpbCk7XG4gICAgICByZXR1cm4gZXZ0O1xuICAgIH1cbiAgICBDdXN0b21FdmVudC5wcm90b3R5cGUgPSB3aW5kb3cuRXZlbnQucHJvdG90eXBlO1xuICAgIHJldHVybiBDdXN0b21FdmVudDtcbiAgfVxuXG4gIGZ1bmN0aW9uIGJ1aWxkSGlkZGVuSW5wdXQobmFtZSwgdmFsdWUpIHtcbiAgICB2YXIgaW5wdXQgPSBkb2N1bWVudC5jcmVhdGVFbGVtZW50KFwiaW5wdXRcIik7XG4gICAgaW5wdXQudHlwZSA9IFwiaGlkZGVuXCI7XG4gICAgaW5wdXQubmFtZSA9IG5hbWU7XG4gICAgaW5wdXQudmFsdWUgPSB2YWx1ZTtcbiAgICByZXR1cm4gaW5wdXQ7XG4gIH1cblxuICBmdW5jdGlvbiBoYW5kbGVDbGljayhlbGVtZW50KSB7XG4gICAgdmFyIHRvID0gZWxlbWVudC5nZXRBdHRyaWJ1dGUoXCJkYXRhLXRvXCIpLFxuICAgICAgICBtZXRob2QgPSBidWlsZEhpZGRlbklucHV0KFwiX21ldGhvZFwiLCBlbGVtZW50LmdldEF0dHJpYnV0ZShcImRhdGEtbWV0aG9kXCIpKSxcbiAgICAgICAgY3NyZiA9IGJ1aWxkSGlkZGVuSW5wdXQoXCJfY3NyZl90b2tlblwiLCBlbGVtZW50LmdldEF0dHJpYnV0ZShcImRhdGEtY3NyZlwiKSksXG4gICAgICAgIGZvcm0gPSBkb2N1bWVudC5jcmVhdGVFbGVtZW50KFwiZm9ybVwiKSxcbiAgICAgICAgdGFyZ2V0ID0gZWxlbWVudC5nZXRBdHRyaWJ1dGUoXCJ0YXJnZXRcIik7XG5cbiAgICBmb3JtLm1ldGhvZCA9IChlbGVtZW50LmdldEF0dHJpYnV0ZShcImRhdGEtbWV0aG9kXCIpID09PSBcImdldFwiKSA/IFwiZ2V0XCIgOiBcInBvc3RcIjtcbiAgICBmb3JtLmFjdGlvbiA9IHRvO1xuICAgIGZvcm0uc3R5bGUuZGlzcGxheSA9IFwiaGlkZGVuXCI7XG5cbiAgICBpZiAodGFyZ2V0KSBmb3JtLnRhcmdldCA9IHRhcmdldDtcblxuICAgIGZvcm0uYXBwZW5kQ2hpbGQoY3NyZik7XG4gICAgZm9ybS5hcHBlbmRDaGlsZChtZXRob2QpO1xuICAgIGRvY3VtZW50LmJvZHkuYXBwZW5kQ2hpbGQoZm9ybSk7XG4gICAgZm9ybS5zdWJtaXQoKTtcbiAgfVxuXG4gIHdpbmRvdy5hZGRFdmVudExpc3RlbmVyKFwiY2xpY2tcIiwgZnVuY3Rpb24oZSkge1xuICAgIHZhciBlbGVtZW50ID0gZS50YXJnZXQ7XG5cbiAgICB3aGlsZSAoZWxlbWVudCAmJiBlbGVtZW50LmdldEF0dHJpYnV0ZSkge1xuICAgICAgdmFyIHBob2VuaXhMaW5rRXZlbnQgPSBuZXcgUG9seWZpbGxFdmVudCgncGhvZW5peC5saW5rLmNsaWNrJywge1xuICAgICAgICBcImJ1YmJsZXNcIjogdHJ1ZSwgXCJjYW5jZWxhYmxlXCI6IHRydWVcbiAgICAgIH0pO1xuXG4gICAgICBpZiAoIWVsZW1lbnQuZGlzcGF0Y2hFdmVudChwaG9lbml4TGlua0V2ZW50KSkge1xuICAgICAgICBlLnByZXZlbnREZWZhdWx0KCk7XG4gICAgICAgIGUuc3RvcEltbWVkaWF0ZVByb3BhZ2F0aW9uKCk7XG4gICAgICAgIHJldHVybiBmYWxzZTtcbiAgICAgIH1cblxuICAgICAgaWYgKGVsZW1lbnQuZ2V0QXR0cmlidXRlKFwiZGF0YS1tZXRob2RcIikpIHtcbiAgICAgICAgaGFuZGxlQ2xpY2soZWxlbWVudCk7XG4gICAgICAgIGUucHJldmVudERlZmF1bHQoKTtcbiAgICAgICAgcmV0dXJuIGZhbHNlO1xuICAgICAgfSBlbHNlIHtcbiAgICAgICAgZWxlbWVudCA9IGVsZW1lbnQucGFyZW50Tm9kZTtcbiAgICAgIH1cbiAgICB9XG4gIH0sIGZhbHNlKTtcblxuICB3aW5kb3cuYWRkRXZlbnRMaXN0ZW5lcigncGhvZW5peC5saW5rLmNsaWNrJywgZnVuY3Rpb24gKGUpIHtcbiAgICB2YXIgbWVzc2FnZSA9IGUudGFyZ2V0LmdldEF0dHJpYnV0ZShcImRhdGEtY29uZmlybVwiKTtcbiAgICBpZihtZXNzYWdlICYmICF3aW5kb3cuY29uZmlybShtZXNzYWdlKSkge1xuICAgICAgZS5wcmV2ZW50RGVmYXVsdCgpO1xuICAgIH1cbiAgfSwgZmFsc2UpO1xufSkoKTtcbiJdLCJtYXBwaW5ncyI6IkFBQUE7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFBQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFNQTtBQUNBO0FBQ0E7QUFFQTtBQUVBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUFBO0FBREE7QUFDQTtBQUdBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUVBO0FBQ0E7QUFDQTtBQUFBO0FBQ0E7QUFDQTtBQUNBO0FBQ0EiLCJzb3VyY2VSb290IjoiIn0=\n//# sourceURL=webpack-internal:///../deps/phoenix_html/priv/static/phoenix_html.js\n"); + + /***/ }), + + /***/ "../deps/phoenix_live_view/priv/static/phoenix_live_view.js": + /*!******************************************************************!*\ + !*** ../deps/phoenix_live_view/priv/static/phoenix_live_view.js ***! + \******************************************************************/ + /*! no static exports found */ + /***/ (function(module, exports, __webpack_require__) { + + eval("!function (e, t) {\n true ? module.exports = t() : undefined;\n}(this, function () {\n return function (e) {\n var t = {};\n\n function n(i) {\n if (t[i]) return t[i].exports;\n var r = t[i] = {\n i: i,\n l: !1,\n exports: {}\n };\n return e[i].call(r.exports, r, r.exports, n), r.l = !0, r.exports;\n }\n\n return n.m = e, n.c = t, n.d = function (e, t, i) {\n n.o(e, t) || Object.defineProperty(e, t, {\n configurable: !1,\n enumerable: !0,\n get: i\n });\n }, n.r = function (e) {\n Object.defineProperty(e, \"__esModule\", {\n value: !0\n });\n }, n.n = function (e) {\n var t = e && e.__esModule ? function () {\n return e.default;\n } : function () {\n return e;\n };\n return n.d(t, \"a\", t), t;\n }, n.o = function (e, t) {\n return Object.prototype.hasOwnProperty.call(e, t);\n }, n.p = \"\", n(n.s = 2);\n }([function (e, t, n) {\n \"use strict\";\n\n n.r(t);\n var i,\n r = 11;\n var o = \"http://www.w3.org/1999/xhtml\",\n a = \"undefined\" == typeof document ? void 0 : document,\n u = !!a && \"content\" in a.createElement(\"template\"),\n s = !!a && a.createRange && \"createContextualFragment\" in a.createRange();\n\n function c(e) {\n return e = e.trim(), u ? function (e) {\n var t = a.createElement(\"template\");\n return t.innerHTML = e, t.content.childNodes[0];\n }(e) : s ? function (e) {\n return i || (i = a.createRange()).selectNode(a.body), i.createContextualFragment(e).childNodes[0];\n }(e) : function (e) {\n var t = a.createElement(\"body\");\n return t.innerHTML = e, t.childNodes[0];\n }(e);\n }\n\n function l(e, t) {\n var n,\n i,\n r = e.nodeName,\n o = t.nodeName;\n return r === o || (n = r.charCodeAt(0), i = o.charCodeAt(0), n <= 90 && i >= 97 ? r === o.toUpperCase() : i <= 90 && n >= 97 && o === r.toUpperCase());\n }\n\n function d(e, t, n) {\n e[n] !== t[n] && (e[n] = t[n], e[n] ? e.setAttribute(n, \"\") : e.removeAttribute(n));\n }\n\n var h = {\n OPTION: function (e, t) {\n var n = e.parentNode;\n\n if (n) {\n var i = n.nodeName.toUpperCase();\n \"OPTGROUP\" === i && (i = (n = n.parentNode) && n.nodeName.toUpperCase()), \"SELECT\" !== i || n.hasAttribute(\"multiple\") || (e.hasAttribute(\"selected\") && !t.selected && (e.setAttribute(\"selected\", \"selected\"), e.removeAttribute(\"selected\")), n.selectedIndex = -1);\n }\n\n d(e, t, \"selected\");\n },\n INPUT: function (e, t) {\n d(e, t, \"checked\"), d(e, t, \"disabled\"), e.value !== t.value && (e.value = t.value), t.hasAttribute(\"value\") || e.removeAttribute(\"value\");\n },\n TEXTAREA: function (e, t) {\n var n = t.value;\n e.value !== n && (e.value = n);\n var i = e.firstChild;\n\n if (i) {\n var r = i.nodeValue;\n if (r == n || !n && r == e.placeholder) return;\n i.nodeValue = n;\n }\n },\n SELECT: function (e, t) {\n if (!t.hasAttribute(\"multiple\")) {\n for (var n, i, r = -1, o = 0, a = e.firstChild; a;) if (\"OPTGROUP\" === (i = a.nodeName && a.nodeName.toUpperCase())) a = (n = a).firstChild;else {\n if (\"OPTION\" === i) {\n if (a.hasAttribute(\"selected\")) {\n r = o;\n break;\n }\n\n o++;\n }\n\n !(a = a.nextSibling) && n && (a = n.nextSibling, n = null);\n }\n\n e.selectedIndex = r;\n }\n }\n },\n f = 1,\n v = 11,\n p = 3,\n g = 8;\n\n function m() {}\n\n function y(e) {\n if (e) return e.getAttribute && e.getAttribute(\"id\") || e.id;\n }\n\n var b = function (e) {\n return function (t, n, i) {\n if (i || (i = {}), \"string\" == typeof n) if (\"#document\" === t.nodeName || \"HTML\" === t.nodeName || \"BODY\" === t.nodeName) {\n var r = n;\n (n = a.createElement(\"html\")).innerHTML = r;\n } else n = c(n);\n var u = i.getNodeKey || y,\n s = i.onBeforeNodeAdded || m,\n d = i.onNodeAdded || m,\n b = i.onBeforeElUpdated || m,\n k = i.onElUpdated || m,\n w = i.onBeforeNodeDiscarded || m,\n E = i.onNodeDiscarded || m,\n A = i.onBeforeElChildrenUpdated || m,\n S = !0 === i.childrenOnly,\n x = Object.create(null),\n C = [];\n\n function P(e) {\n C.push(e);\n }\n\n function L(e, t, n) {\n !1 !== w(e) && (t && t.removeChild(e), E(e), function e(t, n) {\n if (t.nodeType === f) for (var i = t.firstChild; i;) {\n var r = void 0;\n n && (r = u(i)) ? P(r) : (E(i), i.firstChild && e(i, n)), i = i.nextSibling;\n }\n }(e, n));\n }\n\n function I(e) {\n d(e);\n\n for (var t = e.firstChild; t;) {\n var n = t.nextSibling,\n i = u(t);\n\n if (i) {\n var r = x[i];\n r && l(t, r) ? (t.parentNode.replaceChild(r, t), T(r, t)) : I(t);\n } else I(t);\n\n t = n;\n }\n }\n\n function T(t, n, i) {\n var r = u(n);\n\n if (r && delete x[r], !i) {\n if (!1 === b(t, n)) return;\n if (e(t, n), k(t), !1 === A(t, n)) return;\n }\n\n \"TEXTAREA\" !== t.nodeName ? function (e, t) {\n var n,\n i,\n r,\n o,\n c,\n d = t.firstChild,\n v = e.firstChild;\n\n e: for (; d;) {\n for (o = d.nextSibling, n = u(d); v;) {\n if (r = v.nextSibling, d.isSameNode && d.isSameNode(v)) {\n d = o, v = r;\n continue e;\n }\n\n i = u(v);\n var m = v.nodeType,\n y = void 0;\n\n if (m === d.nodeType && (m === f ? (n ? n !== i && ((c = x[n]) ? r === c ? y = !1 : (e.insertBefore(c, v), i ? P(i) : L(v, e, !0), v = c) : y = !1) : i && (y = !1), (y = !1 !== y && l(v, d)) && T(v, d)) : m !== p && m != g || (y = !0, v.nodeValue !== d.nodeValue && (v.nodeValue = d.nodeValue))), y) {\n d = o, v = r;\n continue e;\n }\n\n i ? P(i) : L(v, e, !0), v = r;\n }\n\n if (n && (c = x[n]) && l(c, d)) e.appendChild(c), T(c, d);else {\n var b = s(d);\n !1 !== b && (b && (d = b), d.actualize && (d = d.actualize(e.ownerDocument || a)), e.appendChild(d), I(d));\n }\n d = o, v = r;\n }\n\n !function (e, t, n) {\n for (; t;) {\n var i = t.nextSibling;\n (n = u(t)) ? P(n) : L(t, e, !0), t = i;\n }\n }(e, v, i);\n var k = h[e.nodeName];\n k && k(e, t);\n }(t, n) : h.TEXTAREA(t, n);\n }\n\n !function e(t) {\n if (t.nodeType === f || t.nodeType === v) for (var n = t.firstChild; n;) {\n var i = u(n);\n i && (x[i] = n), e(n), n = n.nextSibling;\n }\n }(t);\n var D = t,\n _ = D.nodeType,\n N = n.nodeType;\n if (!S) if (_ === f) N === f ? l(t, n) || (E(t), D = function (e, t) {\n for (var n = e.firstChild; n;) {\n var i = n.nextSibling;\n t.appendChild(n), n = i;\n }\n\n return t;\n }(t, function (e, t) {\n return t && t !== o ? a.createElementNS(t, e) : a.createElement(e);\n }(n.nodeName, n.namespaceURI))) : D = n;else if (_ === p || _ === g) {\n if (N === _) return D.nodeValue !== n.nodeValue && (D.nodeValue = n.nodeValue), D;\n D = n;\n }\n if (D === n) E(t);else {\n if (n.isSameNode && n.isSameNode(D)) return;\n if (T(D, n, S), C) for (var R = 0, O = C.length; R < O; R++) {\n var j = x[C[R]];\n j && L(j, j.parentNode, !1);\n }\n }\n return !S && D !== t && t.parentNode && (D.actualize && (D = D.actualize(t.ownerDocument || a)), t.parentNode.replaceChild(D, t)), D;\n };\n }(function (e, t) {\n var n,\n i,\n o,\n a,\n u = t.attributes;\n\n if (t.nodeType !== r && e.nodeType !== r) {\n for (var s = u.length - 1; s >= 0; s--) i = (n = u[s]).name, o = n.namespaceURI, a = n.value, o ? (i = n.localName || i, e.getAttributeNS(o, i) !== a && (\"xmlns\" === n.prefix && (i = n.name), e.setAttributeNS(o, i, a))) : e.getAttribute(i) !== a && e.setAttribute(i, a);\n\n for (var c = e.attributes, l = c.length - 1; l >= 0; l--) i = (n = c[l]).name, (o = n.namespaceURI) ? (i = n.localName || i, t.hasAttributeNS(o, i) || e.removeAttributeNS(o, i)) : t.hasAttribute(i) || e.removeAttribute(i);\n }\n });\n\n function k(e) {\n return P(e) || S(e) || L(e) || C();\n }\n\n function w(e, t) {\n var n = Object.keys(e);\n\n if (Object.getOwnPropertySymbols) {\n var i = Object.getOwnPropertySymbols(e);\n t && (i = i.filter(function (t) {\n return Object.getOwnPropertyDescriptor(e, t).enumerable;\n })), n.push.apply(n, i);\n }\n\n return n;\n }\n\n function E(e, t, n) {\n return t in e ? Object.defineProperty(e, t, {\n value: n,\n enumerable: !0,\n configurable: !0,\n writable: !0\n }) : e[t] = n, e;\n }\n\n function A(e) {\n return function (e) {\n if (Array.isArray(e)) return I(e);\n }(e) || S(e) || L(e) || function () {\n throw new TypeError(\"Invalid attempt to spread non-iterable instance.\\nIn order to be iterable, non-array objects must have a [Symbol.iterator]() method.\");\n }();\n }\n\n function S(e) {\n if (\"undefined\" != typeof Symbol && Symbol.iterator in Object(e)) return Array.from(e);\n }\n\n function x(e, t) {\n return P(e) || function (e, t) {\n if (\"undefined\" == typeof Symbol || !(Symbol.iterator in Object(e))) return;\n var n = [],\n i = !0,\n r = !1,\n o = void 0;\n\n try {\n for (var a, u = e[Symbol.iterator](); !(i = (a = u.next()).done) && (n.push(a.value), !t || n.length !== t); i = !0);\n } catch (e) {\n r = !0, o = e;\n } finally {\n try {\n i || null == u.return || u.return();\n } finally {\n if (r) throw o;\n }\n }\n\n return n;\n }(e, t) || L(e, t) || C();\n }\n\n function C() {\n throw new TypeError(\"Invalid attempt to destructure non-iterable instance.\\nIn order to be iterable, non-array objects must have a [Symbol.iterator]() method.\");\n }\n\n function P(e) {\n if (Array.isArray(e)) return e;\n }\n\n function L(e, t) {\n if (e) {\n if (\"string\" == typeof e) return I(e, t);\n var n = Object.prototype.toString.call(e).slice(8, -1);\n return \"Object\" === n && e.constructor && (n = e.constructor.name), \"Map\" === n || \"Set\" === n ? Array.from(e) : \"Arguments\" === n || /^(?:Ui|I)nt(?:8|16|32)(?:Clamped)?Array$/.test(n) ? I(e, t) : void 0;\n }\n }\n\n function I(e, t) {\n (null == t || t > e.length) && (t = e.length);\n\n for (var n = 0, i = new Array(t); n < t; n++) i[n] = e[n];\n\n return i;\n }\n\n function T(e, t) {\n if (!(e instanceof t)) throw new TypeError(\"Cannot call a class as a function\");\n }\n\n function D(e, t) {\n for (var n = 0; n < t.length; n++) {\n var i = t[n];\n i.enumerable = i.enumerable || !1, i.configurable = !0, \"value\" in i && (i.writable = !0), Object.defineProperty(e, i.key, i);\n }\n }\n\n function _(e, t, n) {\n return t && D(e.prototype, t), n && D(e, n), e;\n }\n\n function N(e) {\n \"@babel/helpers - typeof\";\n\n return (N = \"function\" == typeof Symbol && \"symbol\" == typeof Symbol.iterator ? function (e) {\n return typeof e;\n } : function (e) {\n return e && \"function\" == typeof Symbol && e.constructor === Symbol && e !== Symbol.prototype ? \"symbol\" : typeof e;\n })(e);\n }\n\n n.d(t, \"debug\", function () {\n return X;\n }), n.d(t, \"Rendered\", function () {\n return se;\n }), n.d(t, \"LiveSocket\", function () {\n return ce;\n }), n.d(t, \"Browser\", function () {\n return le;\n }), n.d(t, \"DOM\", function () {\n return de;\n }), n.d(t, \"View\", function () {\n return ve;\n });\n\n var R = [1e3, 3e3],\n O = \"data-phx-view\",\n j = [\"phx-click-loading\", \"phx-change-loading\", \"phx-submit-loading\", \"phx-keydown-loading\", \"phx-keyup-loading\", \"phx-blur-loading\", \"phx-focus-loading\"],\n H = \"data-phx-component\",\n F = \"data-phx-ref\",\n M = \"data-phx-upload-ref\",\n U = \"[\".concat(O, \"]\"),\n B = [\"text\", \"textarea\", \"number\", \"email\", \"password\", \"search\", \"tel\", \"url\", \"date\", \"time\"],\n J = [\"checkbox\", \"radio\"],\n V = \"data-phx-static\",\n W = 1,\n q = \"phx-\",\n z = {\n debounce: 300,\n throttle: 300\n },\n K = function (e, t) {\n return console.error && console.error(e, t);\n };\n\n var X = function (e, t, n, i) {\n e.liveSocket.isDebugEnabled() && console.log(\"\".concat(e.id, \" \").concat(t, \": \").concat(n, \" - \"), i);\n },\n $ = function (e) {\n return \"function\" == typeof e ? e : function () {\n return e;\n };\n },\n G = function (e) {\n return JSON.parse(JSON.stringify(e));\n },\n Y = function (e, t, n) {\n do {\n if (e.matches(\"[\".concat(t, \"]\"))) return e;\n e = e.parentElement || e.parentNode;\n } while (null !== e && 1 === e.nodeType && !(n && n.isSameNode(e) || e.matches(U)));\n\n return null;\n },\n Q = function (e) {\n return null !== e && \"object\" === N(e) && !(e instanceof Array);\n },\n Z = function (e) {\n for (var t in e) return !1;\n\n return !0;\n },\n ee = function (e, t) {\n return e && t(e);\n },\n te = function () {\n function e(t, n, i) {\n T(this, e), this.ref = re.genFileRef(n), this.fileEl = t, this.file = n, this.view = i, this.meta = null, this._isCancelled = !1, this._isDone = !1, this._progress = 0, this._onDone = function () {};\n }\n\n return _(e, null, [{\n key: \"isActive\",\n value: function (e, t) {\n var n = void 0 === t._phxRef,\n i = e.getAttribute(\"data-phx-active-refs\").split(\",\").indexOf(re.genFileRef(t)) >= 0;\n return t.size > 0 && (n || i);\n }\n }, {\n key: \"isPreflighted\",\n value: function (e, t) {\n var n = e.getAttribute(\"data-phx-preflighted-refs\").split(\",\").indexOf(re.genFileRef(t)) >= 0;\n return n && this.isActive(e, t);\n }\n }]), _(e, [{\n key: \"metadata\",\n value: function () {\n return this.meta;\n }\n }, {\n key: \"progress\",\n value: function (e) {\n var t = this;\n this._progress = Math.floor(e), this._progress >= 100 ? (this._progress = 100, this._isDone = !0, this.view.pushFileProgress(this.fileEl, this.ref, 100, function () {\n re.untrackFile(t.fileEl, t.file), t._onDone();\n })) : this.view.pushFileProgress(this.fileEl, this.ref, this._progress);\n }\n }, {\n key: \"cancel\",\n value: function () {\n this._isCancelled = !0, this._isDone = !0, this._onDone();\n }\n }, {\n key: \"isDone\",\n value: function () {\n return this._isDone;\n }\n }, {\n key: \"error\",\n value: function () {\n var e = arguments.length > 0 && void 0 !== arguments[0] ? arguments[0] : \"failed\";\n this.view.pushFileProgress(this.fileEl, this.ref, {\n error: e\n });\n }\n }, {\n key: \"onDone\",\n value: function (e) {\n this._onDone = e;\n }\n }, {\n key: \"toPreflightPayload\",\n value: function () {\n return {\n last_modified: this.file.lastModified,\n name: this.file.name,\n size: this.file.size,\n type: this.file.type,\n ref: this.ref\n };\n }\n }, {\n key: \"uploader\",\n value: function (e) {\n if (this.meta.uploader) {\n var t = e[this.meta.uploader] || K(\"no uploader configured for \".concat(this.meta.uploader));\n return {\n name: this.meta.uploader,\n callback: t\n };\n }\n\n return {\n name: \"channel\",\n callback: oe\n };\n }\n }, {\n key: \"zipPostFlight\",\n value: function (e) {\n this.meta = e.entries[this.ref], this.meta || K(\"no preflight upload response returned with ref \".concat(this.ref), {\n input: this.fileEl,\n response: e\n });\n }\n }]), e;\n }(),\n ne = {\n LiveFileUpload: {\n preflightedRefs: function () {\n return this.el.getAttribute(\"data-phx-preflighted-refs\");\n },\n mounted: function () {\n this.preflightedWas = this.preflightedRefs();\n },\n updated: function () {\n var e = this.preflightedRefs();\n this.preflightedWas !== e && (this.preflightedWas = e, \"\" === e && this.__view.cancelSubmit(this.el.form));\n }\n }\n };\n\n ne.LiveImgPreview = {\n mounted: function () {\n var e = this;\n this.ref = this.el.getAttribute(\"data-phx-entry-ref\"), this.inputEl = document.getElementById(this.el.getAttribute(M)), re.getEntryDataURL(this.inputEl, this.ref, function (t) {\n return e.el.src = t;\n });\n }\n };\n\n var ie = 0,\n re = function () {\n function e(t, n, i) {\n T(this, e), this.view = n, this.onComplete = i, this._entries = Array.from(e.filesAwaitingPreflight(t) || []).map(function (e) {\n return new te(t, e, n);\n }), this.numEntriesInProgress = this._entries.length;\n }\n\n return _(e, null, [{\n key: \"genFileRef\",\n value: function (e) {\n var t = e._phxRef;\n return void 0 !== t ? t : (e._phxRef = (ie++).toString(), e._phxRef);\n }\n }, {\n key: \"getEntryDataURL\",\n value: function (e, t, n) {\n var i = this,\n r = this.activeFiles(e).find(function (e) {\n return i.genFileRef(e) === t;\n }),\n o = new FileReader();\n o.onload = function (e) {\n return n(e.target.result);\n }, o.readAsDataURL(r);\n }\n }, {\n key: \"hasUploadsInProgress\",\n value: function (e) {\n var t = 0;\n return de.findUploadInputs(e).forEach(function (e) {\n e.getAttribute(\"data-phx-preflighted-refs\") !== e.getAttribute(\"data-phx-done-refs\") && t++;\n }), t > 0;\n }\n }, {\n key: \"serializeUploads\",\n value: function (e) {\n var t = this,\n n = {};\n return this.activeFiles(e, \"serialize\").forEach(function (i) {\n var r = {\n path: e.name\n },\n o = e.getAttribute(M);\n n[o] = n[o] || [], r.ref = t.genFileRef(i), r.name = i.name, r.type = i.type, r.size = i.size, n[o].push(r);\n }), n;\n }\n }, {\n key: \"clearFiles\",\n value: function (e) {\n e.value = null, de.putPrivate(e, \"files\", []);\n }\n }, {\n key: \"untrackFile\",\n value: function (e, t) {\n de.putPrivate(e, \"files\", de.private(e, \"files\").filter(function (e) {\n return !Object.is(e, t);\n }));\n }\n }, {\n key: \"trackFiles\",\n value: function (e, t) {\n var n = this;\n\n if (null !== e.getAttribute(\"multiple\")) {\n var i = t.filter(function (t) {\n return !n.activeFiles(e).find(function (e) {\n return Object.is(e, t);\n });\n });\n de.putPrivate(e, \"files\", this.activeFiles(e).concat(i)), e.value = null;\n } else de.putPrivate(e, \"files\", t);\n }\n }, {\n key: \"activeFileInputs\",\n value: function (e) {\n var t = this,\n n = de.findUploadInputs(e);\n return Array.from(n).filter(function (e) {\n return e.files && t.activeFiles(e).length > 0;\n });\n }\n }, {\n key: \"activeFiles\",\n value: function (e) {\n return (de.private(e, \"files\") || []).filter(function (t) {\n return te.isActive(e, t);\n });\n }\n }, {\n key: \"inputsAwaitingPreflight\",\n value: function (e) {\n var t = this,\n n = de.findUploadInputs(e);\n return Array.from(n).filter(function (e) {\n return t.filesAwaitingPreflight(e).length > 0;\n });\n }\n }, {\n key: \"filesAwaitingPreflight\",\n value: function (e) {\n return this.activeFiles(e).filter(function (t) {\n return !te.isPreflighted(e, t);\n });\n }\n }]), _(e, [{\n key: \"entries\",\n value: function () {\n return this._entries;\n }\n }, {\n key: \"initAdapterUpload\",\n value: function (e, t, n) {\n var i = this;\n this._entries = this._entries.map(function (t) {\n return t.zipPostFlight(e), t.onDone(function () {\n i.numEntriesInProgress--, 0 === i.numEntriesInProgress && i.onComplete();\n }), t;\n });\n\n var r = this._entries.reduce(function (e, t) {\n var i = t.uploader(n.uploaders),\n r = i.name,\n o = i.callback;\n return e[r] = e[r] || {\n callback: o,\n entries: []\n }, e[r].entries.push(t), e;\n }, {});\n\n for (var o in r) {\n var a = r[o];\n (0, a.callback)(a.entries, t, e, n);\n }\n }\n }]), e;\n }(),\n oe = function (e, t, n, i) {\n e.forEach(function (e) {\n new ae(e, n.config.chunk_size, i).upload();\n });\n },\n ae = function () {\n function e(t, n, i) {\n T(this, e), this.liveSocket = i, this.entry = t, this.offset = 0, this.chunkSize = n, this.uploadChannel = i.channel(\"lvu:\".concat(t.ref), {\n token: t.metadata()\n });\n }\n\n return _(e, [{\n key: \"upload\",\n value: function () {\n var e = this;\n this.uploadChannel.join().receive(\"ok\", function (t) {\n return e.readNextChunk();\n }).receive(\"error\", function (t) {\n e.uploadChannel.leave(), e.entry.error();\n });\n }\n }, {\n key: \"isDone\",\n value: function () {\n return this.offset >= this.entry.file.size;\n }\n }, {\n key: \"readNextChunk\",\n value: function () {\n var e = this,\n t = new window.FileReader(),\n n = this.entry.file.slice(this.offset, this.chunkSize + this.offset);\n t.onload = function (t) {\n if (null !== t.target.error) return K(\"Read error: \" + t.target.error);\n e.offset += t.target.result.byteLength, e.pushChunk(t.target.result);\n }, t.readAsArrayBuffer(n);\n }\n }, {\n key: \"pushChunk\",\n value: function (e) {\n var t = this;\n this.uploadChannel.isJoined() && this.uploadChannel.push(\"chunk\", e).receive(\"ok\", function () {\n t.entry.progress(t.offset / t.entry.file.size * 100), t.isDone() || setTimeout(function () {\n return t.readNextChunk();\n }, t.liveSocket.getLatencySim() || 0);\n });\n }\n }]), e;\n }(),\n ue = function (e) {\n var t = arguments.length > 1 && void 0 !== arguments[1] ? arguments[1] : {},\n n = new FormData(e),\n i = [];\n n.forEach(function (e, t, n) {\n e instanceof File && i.push(t);\n }), i.forEach(function (e) {\n return n.delete(e);\n });\n\n var r,\n o = new URLSearchParams(),\n a = function (e) {\n if (\"undefined\" == typeof Symbol || null == e[Symbol.iterator]) {\n if (Array.isArray(e) || (e = L(e))) {\n var t = 0,\n n = function () {};\n\n return {\n s: n,\n n: function () {\n return t >= e.length ? {\n done: !0\n } : {\n done: !1,\n value: e[t++]\n };\n },\n e: function (e) {\n throw e;\n },\n f: n\n };\n }\n\n throw new TypeError(\"Invalid attempt to iterate non-iterable instance.\\nIn order to be iterable, non-array objects must have a [Symbol.iterator]() method.\");\n }\n\n var i,\n r,\n o = !0,\n a = !1;\n return {\n s: function () {\n i = e[Symbol.iterator]();\n },\n n: function () {\n var e = i.next();\n return o = e.done, e;\n },\n e: function (e) {\n a = !0, r = e;\n },\n f: function () {\n try {\n o || null == i.return || i.return();\n } finally {\n if (a) throw r;\n }\n }\n };\n }(n.entries());\n\n try {\n for (a.s(); !(r = a.n()).done;) {\n var u = x(r.value, 2),\n s = u[0],\n c = u[1];\n o.append(s, c);\n }\n } catch (e) {\n a.e(e);\n } finally {\n a.f();\n }\n\n for (var l in t) o.append(l, t[l]);\n\n return o.toString();\n },\n se = function () {\n function e(t, n) {\n T(this, e), this.viewId = t, this.rendered = {}, this.mergeDiff(n);\n }\n\n return _(e, null, [{\n key: \"extract\",\n value: function (e) {\n var t = e.r,\n n = e.e,\n i = e.t;\n return delete e.r, delete e.e, delete e.t, {\n diff: e,\n title: i,\n reply: t || null,\n events: n || []\n };\n }\n }]), _(e, [{\n key: \"parentViewId\",\n value: function () {\n return this.viewId;\n }\n }, {\n key: \"toString\",\n value: function (e) {\n return this.recursiveToString(this.rendered, this.rendered.c, e);\n }\n }, {\n key: \"recursiveToString\",\n value: function (e) {\n var t = arguments.length > 1 && void 0 !== arguments[1] ? arguments[1] : e.c,\n n = arguments.length > 2 ? arguments[2] : void 0,\n i = {\n buffer: \"\",\n components: t,\n onlyCids: n = n ? new Set(n) : null\n };\n return this.toOutputBuffer(e, i), i.buffer;\n }\n }, {\n key: \"componentCIDs\",\n value: function (e) {\n return Object.keys(e.c || {}).map(function (e) {\n return parseInt(e);\n });\n }\n }, {\n key: \"isComponentOnlyDiff\",\n value: function (e) {\n return !!e.c && 1 === Object.keys(e).length;\n }\n }, {\n key: \"getComponent\",\n value: function (e, t) {\n return e.c[t];\n }\n }, {\n key: \"mergeDiff\",\n value: function (e) {\n var t = e.c;\n\n if (delete e.c, this.rendered = this.recursiveMerge(this.rendered, e), this.rendered.c = this.rendered.c || {}, t) {\n var n = this.rendered.c;\n\n for (var i in t) {\n var r = t[i],\n o = r,\n a = o.s;\n\n if (\"number\" == typeof a) {\n for (; \"number\" == typeof a;) a = (o = a > 0 ? t[a] : n[-a]).s;\n\n o = G(o), this.doRecursiveMerge(o, r), o.s = a;\n } else o = n[i] || {}, o = this.recursiveMerge(o, r);\n\n t[i] = o;\n }\n\n for (var u in t) n[u] = t[u];\n\n e.c = t;\n }\n }\n }, {\n key: \"recursiveMerge\",\n value: function (e, t) {\n return void 0 !== t.s ? t : (this.doRecursiveMerge(e, t), e);\n }\n }, {\n key: \"doRecursiveMerge\",\n value: function (e, t) {\n for (var n in t) {\n var i = t[n],\n r = e[n];\n Q(i) && void 0 === i.s && Q(r) ? this.doRecursiveMerge(r, i) : e[n] = i;\n }\n }\n }, {\n key: \"componentToString\",\n value: function (e) {\n return this.recursiveCIDToString(this.rendered.c, e);\n }\n }, {\n key: \"pruneCIDs\",\n value: function (e) {\n var t = this;\n e.forEach(function (e) {\n return delete t.rendered.c[e];\n });\n }\n }, {\n key: \"get\",\n value: function () {\n return this.rendered;\n }\n }, {\n key: \"isNewFingerprint\",\n value: function () {\n return !!(arguments.length > 0 && void 0 !== arguments[0] ? arguments[0] : {}).s;\n }\n }, {\n key: \"toOutputBuffer\",\n value: function (e, t) {\n if (e.d) return this.comprehensionToBuffer(e, t);\n var n = e.s;\n t.buffer += n[0];\n\n for (var i = 1; i < n.length; i++) this.dynamicToBuffer(e[i - 1], t), t.buffer += n[i];\n }\n }, {\n key: \"comprehensionToBuffer\",\n value: function (e, t) {\n for (var n = e.d, i = e.s, r = 0; r < n.length; r++) {\n var o = n[r];\n t.buffer += i[0];\n\n for (var a = 1; a < i.length; a++) this.dynamicToBuffer(o[a - 1], t), t.buffer += i[a];\n }\n }\n }, {\n key: \"dynamicToBuffer\",\n value: function (e, t) {\n \"number\" == typeof e ? t.buffer += this.recursiveCIDToString(t.components, e, t.onlyCids) : Q(e) ? this.toOutputBuffer(e, t) : t.buffer += e;\n }\n }, {\n key: \"recursiveCIDToString\",\n value: function (e, t, n) {\n var i = this,\n r = e[t] || K(\"no component for CID \".concat(t), e),\n o = document.createElement(\"template\");\n o.innerHTML = this.recursiveToString(r, e, n);\n var a = o.content,\n u = n && !n.has(t),\n s = x(Array.from(a.childNodes).reduce(function (e, n, r) {\n var a = x(e, 2),\n s = a[0],\n c = a[1];\n return n.nodeType === Node.ELEMENT_NODE ? n.getAttribute(H) ? [s, !0] : (n.setAttribute(H, t), n.id || (n.id = \"\".concat(i.parentViewId(), \"-\").concat(t, \"-\").concat(r)), u && (n.setAttribute(\"data-phx-skip\", \"\"), n.innerHTML = \"\"), [!0, c]) : \"\" !== n.nodeValue.trim() ? (K(\"only HTML element tags are allowed at the root of components.\\n\\n\" + 'got: \"'.concat(n.nodeValue.trim(), '\"\\n\\n') + \"within:\\n\", o.innerHTML.trim()), n.replaceWith(i.createSpan(n.nodeValue, t)), [!0, c]) : (n.remove(), [s, c]);\n }, [!1, !1]), 2),\n c = s[0],\n l = s[1];\n return c || l ? !c && l ? (K(\"expected at least one HTML element tag directly inside a component, but only subcomponents were found. A component must render at least one HTML tag directly inside itself.\", o.innerHTML.trim()), o.innerHTML) : o.innerHTML : (K(\"expected at least one HTML element tag inside a component, but the component is empty:\\n\", o.innerHTML.trim()), this.createSpan(\"\", t).outerHTML);\n }\n }, {\n key: \"createSpan\",\n value: function (e, t) {\n var n = document.createElement(\"span\");\n return n.innerText = e, n.setAttribute(H, t), n;\n }\n }]), e;\n }(),\n ce = function () {\n function e(t, n) {\n var i = this,\n r = arguments.length > 2 && void 0 !== arguments[2] ? arguments[2] : {};\n if (T(this, e), this.unloaded = !1, !n || \"Object\" === n.constructor.name) throw new Error('\\n a phoenix Socket must be provided as the second argument to the LiveSocket constructor. For example:\\n\\n import {Socket} from \"phoenix\"\\n import {LiveSocket} from \"phoenix_live_view\"\\n let liveSocket = new LiveSocket(\"/live\", Socket, {...})\\n ');\n this.socket = new n(t, r), this.bindingPrefix = r.bindingPrefix || q, this.opts = r, this.params = $(r.params || {}), this.viewLogger = r.viewLogger, this.metadataCallbacks = r.metadata || {}, this.defaults = Object.assign(G(z), r.defaults || {}), this.activeElement = null, this.prevActive = null, this.silenced = !1, this.main = null, this.linkRef = 0, this.roots = {}, this.href = window.location.href, this.pendingLink = null, this.currentLocation = G(window.location), this.hooks = r.hooks || {}, this.uploaders = r.uploaders || {}, this.loaderTimeout = r.loaderTimeout || W, this.boundTopLevelEvents = !1, this.domCallbacks = Object.assign({\n onNodeAdded: $(),\n onBeforeElUpdated: $()\n }, r.dom || {}), window.addEventListener(\"unload\", function (e) {\n i.unloaded = !0;\n }), this.socket.onOpen(function () {\n i.isUnloaded() && window.location.reload();\n });\n }\n\n return _(e, [{\n key: \"isProfileEnabled\",\n value: function () {\n return \"true\" === sessionStorage.getItem(\"phx:live-socket:profiling\");\n }\n }, {\n key: \"isDebugEnabled\",\n value: function () {\n return \"true\" === sessionStorage.getItem(\"phx:live-socket:debug\");\n }\n }, {\n key: \"enableDebug\",\n value: function () {\n sessionStorage.setItem(\"phx:live-socket:debug\", \"true\");\n }\n }, {\n key: \"enableProfiling\",\n value: function () {\n sessionStorage.setItem(\"phx:live-socket:profiling\", \"true\");\n }\n }, {\n key: \"disableDebug\",\n value: function () {\n sessionStorage.removeItem(\"phx:live-socket:debug\");\n }\n }, {\n key: \"disableProfiling\",\n value: function () {\n sessionStorage.removeItem(\"phx:live-socket:profiling\");\n }\n }, {\n key: \"enableLatencySim\",\n value: function (e) {\n this.enableDebug(), console.log(\"latency simulator enabled for the duration of this browser session. Call disableLatencySim() to disable\"), sessionStorage.setItem(\"phx:live-socket:latency-sim\", e);\n }\n }, {\n key: \"disableLatencySim\",\n value: function () {\n sessionStorage.removeItem(\"phx:live-socket:latency-sim\");\n }\n }, {\n key: \"getLatencySim\",\n value: function () {\n var e = sessionStorage.getItem(\"phx:live-socket:latency-sim\");\n return e ? parseInt(e) : null;\n }\n }, {\n key: \"getSocket\",\n value: function () {\n return this.socket;\n }\n }, {\n key: \"connect\",\n value: function () {\n var e = this,\n t = function () {\n e.joinRootViews() && (e.bindTopLevelEvents(), e.socket.connect());\n };\n\n [\"complete\", \"loaded\", \"interactive\"].indexOf(document.readyState) >= 0 ? t() : document.addEventListener(\"DOMContentLoaded\", function () {\n return t();\n });\n }\n }, {\n key: \"disconnect\",\n value: function (e) {\n this.socket.disconnect(e);\n }\n }, {\n key: \"triggerDOM\",\n value: function (e, t) {\n var n;\n (n = this.domCallbacks)[e].apply(n, A(t));\n }\n }, {\n key: \"time\",\n value: function (e, t) {\n if (!this.isProfileEnabled() || !console.time) return t();\n console.time(e);\n var n = t();\n return console.timeEnd(e), n;\n }\n }, {\n key: \"log\",\n value: function (e, t, n) {\n if (this.viewLogger) {\n var i = x(n(), 2),\n r = i[0],\n o = i[1];\n this.viewLogger(e, t, r, o);\n } else if (this.isDebugEnabled()) {\n var a = x(n(), 2),\n u = a[0],\n s = a[1];\n X(e, t, u, s);\n }\n }\n }, {\n key: \"onChannel\",\n value: function (e, t, n) {\n var i = this;\n e.on(t, function (e) {\n var t = i.getLatencySim();\n t ? (console.log(\"simulating \".concat(t, \"ms of latency from server to client\")), setTimeout(function () {\n return n(e);\n }, t)) : n(e);\n });\n }\n }, {\n key: \"wrapPush\",\n value: function (e, t, n) {\n var i = this,\n r = this.getLatencySim(),\n o = e.joinCount;\n if (!r) return t.timeout ? n().receive(\"timeout\", function () {\n e.joinCount === o && i.reloadWithJitter(e, function () {\n i.log(e, \"timeout\", function () {\n return [\"received timeout while communicating with server. Falling back to hard refresh for recovery\"];\n });\n });\n }) : n();\n console.log(\"simulating \".concat(r, \"ms of latency from client to server\"));\n var a = {\n receives: [],\n receive: function (e, t) {\n this.receives.push([e, t]);\n }\n };\n return setTimeout(function () {\n a.receives.reduce(function (e, t) {\n var n = x(t, 2),\n i = n[0],\n r = n[1];\n return e.receive(i, r);\n }, n());\n }, r), a;\n }\n }, {\n key: \"reloadWithJitter\",\n value: function (e, t) {\n var n = this;\n e.destroy(), this.disconnect();\n var i = R[0],\n r = R[1],\n o = Math.floor(Math.random() * (r - i + 1)) + i,\n a = le.updateLocal(e.name(), \"consecutive-reloads\", 0, function (e) {\n return e + 1;\n });\n t ? t() : this.log(e, \"join\", function () {\n return [\"encountered \".concat(a, \" consecutive reloads\")];\n }), a > 10 && (this.log(e, \"join\", function () {\n return [\"exceeded \".concat(10, \" consecutive reloads. Entering failsafe mode\")];\n }), o = 3e4), setTimeout(function () {\n n.hasPendingLink() ? window.location = n.pendingLink : window.location.reload();\n }, o);\n }\n }, {\n key: \"getHookCallbacks\",\n value: function (e) {\n return e && e.startsWith(\"Phoenix.\") ? ne[e.split(\".\")[1]] : this.hooks[e];\n }\n }, {\n key: \"isUnloaded\",\n value: function () {\n return this.unloaded;\n }\n }, {\n key: \"isConnected\",\n value: function () {\n return this.socket.isConnected();\n }\n }, {\n key: \"getBindingPrefix\",\n value: function () {\n return this.bindingPrefix;\n }\n }, {\n key: \"binding\",\n value: function (e) {\n return \"\".concat(this.getBindingPrefix()).concat(e);\n }\n }, {\n key: \"channel\",\n value: function (e, t) {\n return this.socket.channel(e, t);\n }\n }, {\n key: \"joinRootViews\",\n value: function () {\n var e = this,\n t = !1;\n return de.all(document, \"\".concat(U, \":not([\").concat(\"data-phx-parent-id\", \"])\"), function (n) {\n if (!e.getRootById(n.id)) {\n var i = e.joinRootView(n, e.getHref());\n e.root = e.root || i, n.getAttribute(\"data-phx-main\") && (e.main = i);\n }\n\n t = !0;\n }), t;\n }\n }, {\n key: \"redirect\",\n value: function (e, t) {\n this.disconnect(), le.redirect(e, t);\n }\n }, {\n key: \"replaceMain\",\n value: function (e, t) {\n var n = this,\n i = arguments.length > 2 && void 0 !== arguments[2] ? arguments[2] : null,\n r = arguments.length > 3 && void 0 !== arguments[3] ? arguments[3] : this.setPendingLink(e),\n o = this.main.el;\n this.main.showLoader(this.loaderTimeout), this.main.destroy(), le.fetchPage(e, function (a, u) {\n if (200 !== a) return n.redirect(e);\n var s = document.createElement(\"template\");\n s.innerHTML = u;\n var c = s.content.childNodes[0];\n if (!c || !n.isPhxView(c)) return n.redirect(e);\n n.joinRootView(c, e, t, function (e, t) {\n 1 === t && (n.commitPendingLink(r) ? (o.replaceWith(e.el), n.main = e, i && i()) : e.destroy());\n });\n });\n }\n }, {\n key: \"isPhxView\",\n value: function (e) {\n return e.getAttribute && null !== e.getAttribute(O);\n }\n }, {\n key: \"joinRootView\",\n value: function (e, t, n, i) {\n var r = new ve(e, this, null, t, n);\n return this.roots[r.id] = r, r.join(i), r;\n }\n }, {\n key: \"owner\",\n value: function (e, t) {\n var n = this,\n i = ee(e.closest(U), function (e) {\n return n.getViewByEl(e);\n });\n i && t(i);\n }\n }, {\n key: \"withinOwners\",\n value: function (e, t) {\n var n = this;\n this.owner(e, function (i) {\n var r = e.getAttribute(n.binding(\"target\"));\n null === r ? t(i, e) : i.withinTargets(r, t);\n });\n }\n }, {\n key: \"getViewByEl\",\n value: function (e) {\n var t = e.getAttribute(\"data-phx-root-id\");\n return ee(this.getRootById(t), function (t) {\n return t.getDescendentByEl(e);\n });\n }\n }, {\n key: \"getRootById\",\n value: function (e) {\n return this.roots[e];\n }\n }, {\n key: \"destroyAllViews\",\n value: function () {\n for (var e in this.roots) this.roots[e].destroy(), delete this.roots[e];\n }\n }, {\n key: \"destroyViewByEl\",\n value: function (e) {\n var t = this.getRootById(e.getAttribute(\"data-phx-root-id\"));\n t && t.destroyDescendent(e.id);\n }\n }, {\n key: \"setActiveElement\",\n value: function (e) {\n var t = this;\n\n if (this.activeElement !== e) {\n this.activeElement = e;\n\n var n = function () {\n e === t.activeElement && (t.activeElement = null), e.removeEventListener(\"mouseup\", t), e.removeEventListener(\"touchend\", t);\n };\n\n e.addEventListener(\"mouseup\", n), e.addEventListener(\"touchend\", n);\n }\n }\n }, {\n key: \"getActiveElement\",\n value: function () {\n return document.activeElement === document.body ? this.activeElement || document.activeElement : document.activeElement || document.body;\n }\n }, {\n key: \"dropActiveElement\",\n value: function (e) {\n this.prevActive && e.ownsElement(this.prevActive) && (this.prevActive = null);\n }\n }, {\n key: \"restorePreviouslyActiveFocus\",\n value: function () {\n this.prevActive && this.prevActive !== document.body && this.prevActive.focus();\n }\n }, {\n key: \"blurActiveElement\",\n value: function () {\n this.prevActive = this.getActiveElement(), this.prevActive !== document.body && this.prevActive.blur();\n }\n }, {\n key: \"bindTopLevelEvents\",\n value: function () {\n var e = this;\n this.boundTopLevelEvents || (this.boundTopLevelEvents = !0, document.body.addEventListener(\"click\", function () {}), window.addEventListener(\"pageshow\", function (t) {\n t.persisted && (e.withPageLoading({\n to: window.location.href,\n kind: \"redirect\"\n }), window.location.reload());\n }), this.bindClicks(), this.bindNav(), this.bindForms(), this.bind({\n keyup: \"keyup\",\n keydown: \"keydown\"\n }, function (t, n, i, r, o, a, u) {\n var s = r.getAttribute(e.binding(\"key\")),\n c = t.key && t.key.toLowerCase();\n s && s.toLowerCase() !== c || i.pushKey(r, o, n, a, function (e) {\n for (var t = 1; t < arguments.length; t++) {\n var n = null != arguments[t] ? arguments[t] : {};\n t % 2 ? w(Object(n), !0).forEach(function (t) {\n E(e, t, n[t]);\n }) : Object.getOwnPropertyDescriptors ? Object.defineProperties(e, Object.getOwnPropertyDescriptors(n)) : w(Object(n)).forEach(function (t) {\n Object.defineProperty(e, t, Object.getOwnPropertyDescriptor(n, t));\n });\n }\n\n return e;\n }({\n key: t.key\n }, e.eventMeta(n, t, r)));\n }), this.bind({\n blur: \"focusout\",\n focus: \"focusin\"\n }, function (t, n, i, r, o, a, u) {\n u || i.pushEvent(n, r, o, a, e.eventMeta(n, t, r));\n }), this.bind({\n blur: \"blur\",\n focus: \"focus\"\n }, function (t, n, i, r, o, a, u) {\n u && \"window\" !== !u && i.pushEvent(n, r, o, a, e.eventMeta(n, t, r));\n }), window.addEventListener(\"dragover\", function (e) {\n return e.preventDefault();\n }), window.addEventListener(\"drop\", function (t) {\n t.preventDefault();\n var n = ee(Y(t.target, e.binding(\"drop-target\")), function (t) {\n return t.getAttribute(e.binding(\"drop-target\"));\n }),\n i = n && document.getElementById(n),\n r = Array.from(t.dataTransfer.files || []);\n i && !i.disabled && 0 !== r.length && i.files instanceof FileList && (re.trackFiles(i, r), i.dispatchEvent(new Event(\"input\", {\n bubbles: !0\n })));\n }));\n }\n }, {\n key: \"eventMeta\",\n value: function (e, t, n) {\n var i = this.metadataCallbacks[e];\n return i ? i(t, n) : {};\n }\n }, {\n key: \"setPendingLink\",\n value: function (e) {\n return this.linkRef++, this.pendingLink = e, this.linkRef;\n }\n }, {\n key: \"commitPendingLink\",\n value: function (e) {\n return this.linkRef === e && (this.href = this.pendingLink, this.pendingLink = null, !0);\n }\n }, {\n key: \"getHref\",\n value: function () {\n return this.href;\n }\n }, {\n key: \"hasPendingLink\",\n value: function () {\n return !!this.pendingLink;\n }\n }, {\n key: \"bind\",\n value: function (e, t) {\n var n = this,\n i = function (i) {\n var r = e[i];\n n.on(r, function (e) {\n var r = n.binding(i),\n o = n.binding(\"window-\".concat(i)),\n a = e.target.getAttribute && e.target.getAttribute(r);\n a ? n.debounce(e.target, e, function () {\n n.withinOwners(e.target, function (n, r) {\n t(e, i, n, e.target, r, a, null);\n });\n }) : de.all(document, \"[\".concat(o, \"]\"), function (r) {\n var a = r.getAttribute(o);\n n.debounce(r, e, function () {\n n.withinOwners(r, function (n, o) {\n t(e, i, n, r, o, a, \"window\");\n });\n });\n });\n });\n };\n\n for (var r in e) i(r);\n }\n }, {\n key: \"bindClicks\",\n value: function () {\n this.bindClick(\"click\", \"click\", !1), this.bindClick(\"mousedown\", \"capture-click\", !0);\n }\n }, {\n key: \"bindClick\",\n value: function (e, t, n) {\n var i = this,\n r = this.binding(t);\n window.addEventListener(e, function (e) {\n if (i.isConnected()) {\n var t = null,\n o = (t = n ? e.target.matches(\"[\".concat(r, \"]\")) ? e.target : e.target.querySelector(\"[\".concat(r, \"]\")) : Y(e.target, r)) && t.getAttribute(r);\n o && (\"#\" === t.getAttribute(\"href\") && e.preventDefault(), i.debounce(t, e, function () {\n i.withinOwners(t, function (n, r) {\n n.pushEvent(\"click\", t, r, o, i.eventMeta(\"click\", e, t));\n });\n }));\n }\n }, n);\n }\n }, {\n key: \"bindNav\",\n value: function () {\n var e = this;\n\n if (le.canPushState()) {\n history.scrollRestoration && (history.scrollRestoration = \"manual\");\n var t = null;\n window.addEventListener(\"scroll\", function (e) {\n clearTimeout(t), t = setTimeout(function () {\n le.updateCurrentState(function (e) {\n return Object.assign(e, {\n scroll: window.scrollY\n });\n });\n }, 100);\n }), window.addEventListener(\"popstate\", function (t) {\n if (e.registerNewLocation(window.location)) {\n var n = t.state || {},\n i = n.type,\n r = n.id,\n o = n.root,\n a = n.scroll,\n u = window.location.href;\n e.main.isConnected() && \"patch\" === i && r === e.main.id ? e.main.pushLinkPatch(u, null) : e.replaceMain(u, null, function () {\n o && e.replaceRootHistory(), \"number\" == typeof a && setTimeout(function () {\n window.scrollTo(0, a);\n }, 0);\n });\n }\n }, !1), window.addEventListener(\"click\", function (t) {\n var n = Y(t.target, \"data-phx-link\"),\n i = n && n.getAttribute(\"data-phx-link\"),\n r = t.metaKey || t.ctrlKey || 1 === t.button;\n\n if (i && e.isConnected() && e.main && !r) {\n var o = n.href,\n a = n.getAttribute(\"data-phx-link-state\");\n if (t.preventDefault(), e.pendingLink !== o) if (\"patch\" === i) e.pushHistoryPatch(o, a, n);else {\n if (\"redirect\" !== i) throw new Error(\"expected \".concat(\"data-phx-link\", ' to be \"patch\" or \"redirect\", got: ').concat(i));\n e.historyRedirect(o, a);\n }\n }\n }, !1);\n }\n }\n }, {\n key: \"withPageLoading\",\n value: function (e, t) {\n de.dispatchEvent(window, \"phx:page-loading-start\", e);\n\n var n = function () {\n return de.dispatchEvent(window, \"phx:page-loading-stop\", e);\n };\n\n return t ? t(n) : n;\n }\n }, {\n key: \"pushHistoryPatch\",\n value: function (e, t, n) {\n var i = this;\n this.withPageLoading({\n to: e,\n kind: \"patch\"\n }, function (r) {\n i.main.pushLinkPatch(e, n, function () {\n i.historyPatch(e, t), r();\n });\n });\n }\n }, {\n key: \"historyPatch\",\n value: function (e, t) {\n le.pushState(t, {\n type: \"patch\",\n id: this.main.id\n }, e), this.registerNewLocation(window.location);\n }\n }, {\n key: \"historyRedirect\",\n value: function (e, t, n) {\n var i = this,\n r = window.scrollY;\n this.withPageLoading({\n to: e,\n kind: \"redirect\"\n }, function (o) {\n i.replaceMain(e, n, function () {\n le.pushState(t, {\n type: \"redirect\",\n id: i.main.id,\n scroll: r\n }, e), i.registerNewLocation(window.location), o();\n });\n });\n }\n }, {\n key: \"replaceRootHistory\",\n value: function () {\n le.pushState(\"replace\", {\n root: !0,\n type: \"patch\",\n id: this.main.id\n });\n }\n }, {\n key: \"registerNewLocation\",\n value: function (e) {\n var t = this.currentLocation;\n return t.pathname + t.search !== e.pathname + e.search && (this.currentLocation = G(e), !0);\n }\n }, {\n key: \"bindForms\",\n value: function () {\n var e = this,\n t = 0;\n this.on(\"submit\", function (t) {\n var n = t.target.getAttribute(e.binding(\"submit\"));\n n && (t.preventDefault(), t.target.disabled = !0, e.withinOwners(t.target, function (e, i) {\n return e.submitForm(t.target, i, n);\n }));\n }, !1);\n\n for (var n = function () {\n var n = r[i];\n e.on(n, function (i) {\n var r = i.target,\n o = r.form && r.form.getAttribute(e.binding(\"change\"));\n\n if (o && (\"number\" !== r.type || !r.validity || !r.validity.badInput)) {\n var a = t;\n t++;\n var u = de.private(r, \"prev-iteration\") || {},\n s = u.at,\n c = u.type;\n s === a - 1 && n !== c || (de.putPrivate(r, \"prev-iteration\", {\n at: a,\n type: n\n }), e.debounce(r, i, function () {\n e.withinOwners(r.form, function (t, n) {\n de.putPrivate(r, \"phx-has-focused\", !0), de.isTextualInput(r) || e.setActiveElement(r), t.pushInput(r, n, o, i.target);\n });\n }));\n }\n }, !1);\n }, i = 0, r = [\"change\", \"input\"]; i < r.length; i++) n();\n }\n }, {\n key: \"debounce\",\n value: function (e, t, n) {\n var i = this.binding(\"debounce\"),\n r = this.binding(\"throttle\"),\n o = this.defaults.debounce.toString(),\n a = this.defaults.throttle.toString();\n de.debounce(e, t, i, o, r, a, n);\n }\n }, {\n key: \"silenceEvents\",\n value: function (e) {\n this.silenced = !0, e(), this.silenced = !1;\n }\n }, {\n key: \"on\",\n value: function (e, t) {\n var n = this;\n window.addEventListener(e, function (e) {\n n.silenced || t(e);\n });\n }\n }]), e;\n }(),\n le = {\n canPushState: function () {\n return void 0 !== history.pushState;\n },\n dropLocal: function (e, t) {\n return window.localStorage.removeItem(this.localKey(e, t));\n },\n updateLocal: function (e, t, n, i) {\n var r = this.getLocal(e, t),\n o = this.localKey(e, t),\n a = null === r ? n : i(r);\n return window.localStorage.setItem(o, JSON.stringify(a)), a;\n },\n getLocal: function (e, t) {\n return JSON.parse(window.localStorage.getItem(this.localKey(e, t)));\n },\n fetchPage: function (e, t) {\n var n = new XMLHttpRequest();\n n.open(\"GET\", e, !0), n.timeout = 3e4, n.setRequestHeader(\"content-type\", \"text/html\"), n.setRequestHeader(\"cache-control\", \"max-age=0, no-cache, no-store, must-revalidate, post-check=0, pre-check=0\"), n.setRequestHeader(\"x-requested-with\", \"live-link\"), n.onerror = function () {\n return t(400);\n }, n.ontimeout = function () {\n return t(504);\n }, n.onreadystatechange = function () {\n if (4 === n.readyState) {\n var i = new URL(e),\n r = i.pathname + i.search,\n o = ee(n.getResponseHeader(\"x-response-url\") || n.responseURL, function (e) {\n return new URL(e);\n }),\n a = o ? o.pathname + o.search : null;\n return \"live-link\" !== n.getResponseHeader(\"x-requested-with\") ? t(400) : null === o || a != r ? t(302) : 200 !== n.status ? t(n.status) : void t(200, n.responseText);\n }\n }, n.send();\n },\n updateCurrentState: function (e) {\n this.canPushState() && history.replaceState(e(history.state || {}), \"\", window.location.href);\n },\n pushState: function (e, t, n) {\n if (this.canPushState()) {\n if (n !== window.location.href) {\n if (\"redirect\" == t.type && t.scroll) {\n var i = history.state || {};\n i.scroll = t.scroll, history.replaceState(i, \"\", window.location.href);\n }\n\n delete t.scroll, history[e + \"State\"](t, \"\", n || null);\n var r = this.getHashTargetEl(window.location.hash);\n r ? r.scrollIntoView() : \"redirect\" === t.type && window.scroll(0, 0);\n }\n } else this.redirect(n);\n },\n setCookie: function (e, t) {\n document.cookie = \"\".concat(e, \"=\").concat(t);\n },\n getCookie: function (e) {\n return document.cookie.replace(new RegExp(\"(?:(?:^|.*;s*)\".concat(e, \"s*=s*([^;]*).*$)|^.*$\")), \"$1\");\n },\n redirect: function (e, t) {\n t && le.setCookie(\"__phoenix_flash__\", t + \"; max-age=60000; path=/\"), window.location = e;\n },\n localKey: function (e, t) {\n return \"\".concat(e, \"-\").concat(t);\n },\n getHashTargetEl: function (e) {\n var t = e.toString().substring(1);\n if (\"\" !== t) return document.getElementById(t) || document.querySelector('a[name=\"'.concat(t, '\"]'));\n }\n },\n de = {\n byId: function (e) {\n return document.getElementById(e) || K(\"no id found for \".concat(e));\n },\n removeClass: function (e, t) {\n e.classList.remove(t), 0 === e.classList.length && e.removeAttribute(\"class\");\n },\n all: function (e, t, n) {\n var i = Array.from(e.querySelectorAll(t));\n return n ? i.forEach(n) : i;\n },\n childNodeLength: function (e) {\n var t = document.createElement(\"template\");\n return t.innerHTML = e, t.content.childElementCount;\n },\n isUploadInput: function (e) {\n return \"file\" === e.type && null !== e.getAttribute(M);\n },\n findUploadInputs: function (e) {\n return this.all(e, 'input[type=\"file\"]['.concat(M, \"]\"));\n },\n findComponentNodeList: function (e, t) {\n return this.filterWithinSameLiveView(this.all(e, \"[\".concat(H, '=\"').concat(t, '\"]')), e);\n },\n findPhxChildrenInFragment: function (e, t) {\n var n = document.createElement(\"template\");\n return n.innerHTML = e, this.findPhxChildren(n.content, t);\n },\n isIgnored: function (e, t) {\n return \"ignore\" === (e.getAttribute(t) || e.getAttribute(\"data-phx-update\"));\n },\n isPhxUpdate: function (e, t, n) {\n return e.getAttribute && n.indexOf(e.getAttribute(t)) >= 0;\n },\n findPhxChildren: function (e, t) {\n return this.all(e, \"\".concat(U, \"[\").concat(\"data-phx-parent-id\", '=\"').concat(t, '\"]'));\n },\n findParentCIDs: function (e, t) {\n var n = this,\n i = new Set(t);\n return t.reduce(function (t, i) {\n var r = \"[\".concat(H, '=\"').concat(i, '\"] [').concat(H, \"]\");\n return n.filterWithinSameLiveView(n.all(e, r), e).map(function (e) {\n return parseInt(e.getAttribute(H));\n }).forEach(function (e) {\n return t.delete(e);\n }), t;\n }, i);\n },\n filterWithinSameLiveView: function (e, t) {\n var n = this;\n return t.querySelector(U) ? e.filter(function (e) {\n return n.withinSameLiveView(e, t);\n }) : e;\n },\n withinSameLiveView: function (e, t) {\n for (; e = e.parentNode;) {\n if (e.isSameNode(t)) return !0;\n if (e.getAttribute(O)) return !1;\n }\n },\n private: function (e, t) {\n return e.phxPrivate && e.phxPrivate[t];\n },\n deletePrivate: function (e, t) {\n e.phxPrivate && delete e.phxPrivate[t];\n },\n putPrivate: function (e, t, n) {\n e.phxPrivate || (e.phxPrivate = {}), e.phxPrivate[t] = n;\n },\n copyPrivates: function (e, t) {\n t.phxPrivate && (e.phxPrivate = G(t.phxPrivate));\n },\n putTitle: function (e) {\n var t = document.querySelector(\"title\").dataset,\n n = t.prefix,\n i = t.suffix;\n document.title = \"\".concat(n || \"\").concat(e).concat(i || \"\");\n },\n debounce: function (e, t, n, i, r, o, a) {\n var u = this,\n s = e.getAttribute(n),\n c = e.getAttribute(r);\n \"\" === s && (s = i), \"\" === c && (c = o);\n var l = s || c;\n\n switch (l) {\n case null:\n return a();\n\n case \"blur\":\n return void (this.once(e, \"debounce-blur\") && e.addEventListener(\"blur\", function () {\n return a();\n }));\n\n default:\n var d = parseInt(l),\n h = this.incCycle(e, \"debounce-trigger\", function () {\n return c ? u.deletePrivate(e, \"throttled\") : a();\n });\n if (isNaN(d)) return K(\"invalid throttle/debounce value: \".concat(l));\n\n if (c) {\n var f = !1;\n\n if (\"keydown\" === t.type) {\n var v = this.private(e, \"debounce-prev-key\");\n this.putPrivate(e, \"debounce-prev-key\", t.key), f = v !== t.key;\n }\n\n if (!f && this.private(e, \"throttled\")) return !1;\n a(), this.putPrivate(e, \"throttled\", !0), setTimeout(function () {\n return u.triggerCycle(e, \"debounce-trigger\");\n }, d);\n } else setTimeout(function () {\n return u.triggerCycle(e, \"debounce-trigger\", h);\n }, d);\n\n e.form && this.once(e.form, \"bind-debounce\") && e.form.addEventListener(\"submit\", function (t) {\n Array.from(new FormData(e.form).entries(), function (t) {\n var n = x(t, 2),\n i = n[0],\n r = (n[1], e.form.querySelector('[name=\"'.concat(i, '\"]')));\n u.incCycle(r, \"debounce-trigger\"), u.deletePrivate(r, \"throttled\");\n });\n }), this.once(e, \"bind-debounce\") && e.addEventListener(\"blur\", function (t) {\n return u.triggerCycle(e, \"debounce-trigger\");\n });\n }\n },\n triggerCycle: function (e, t, n) {\n var i = x(this.private(e, t), 2),\n r = i[0],\n o = i[1];\n n || (n = r), n === r && (this.incCycle(e, t), o());\n },\n once: function (e, t) {\n return !0 !== this.private(e, t) && (this.putPrivate(e, t, !0), !0);\n },\n incCycle: function (e, t) {\n var n = arguments.length > 2 && void 0 !== arguments[2] ? arguments[2] : function () {},\n i = x(this.private(e, t) || [0, n], 2),\n r = i[0];\n i[1];\n return r++, this.putPrivate(e, t, [r, n]), r;\n },\n discardError: function (e, t, n) {\n var i = t.getAttribute && t.getAttribute(n),\n r = i && e.querySelector(\"#\".concat(i, ', [name=\"').concat(i, '\"]'));\n r && (this.private(r, \"phx-has-focused\") || this.private(r.form, \"phx-has-submitted\") || t.classList.add(\"phx-no-feedback\"));\n },\n showError: function (e, t) {\n var n = this;\n (e.id || e.name) && this.all(e.form, \"[\".concat(t, '=\"').concat(e.id, '\"], [').concat(t, '=\"').concat(e.name, '\"]'), function (e) {\n n.removeClass(e, \"phx-no-feedback\");\n });\n },\n isPhxChild: function (e) {\n return e.getAttribute && e.getAttribute(\"data-phx-parent-id\");\n },\n dispatchEvent: function (e, t) {\n var n = arguments.length > 2 && void 0 !== arguments[2] ? arguments[2] : {},\n i = new CustomEvent(t, {\n bubbles: !0,\n cancelable: !0,\n detail: n\n });\n e.dispatchEvent(i);\n },\n cloneNode: function (e, t) {\n if (void 0 === t) return e.cloneNode(!0);\n var n = e.cloneNode(!1);\n return n.innerHTML = t, n;\n },\n mergeAttrs: function (e, t) {\n for (var n = arguments.length > 2 && void 0 !== arguments[2] ? arguments[2] : {}, i = n.exclude || [], r = n.isIgnored, o = t.attributes, a = o.length - 1; a >= 0; a--) {\n var u = o[a].name;\n i.indexOf(u) < 0 && e.setAttribute(u, t.getAttribute(u));\n }\n\n for (var s = e.attributes, c = s.length - 1; c >= 0; c--) {\n var l = s[c].name;\n r ? l.startsWith(\"data-\") && !t.hasAttribute(l) && e.removeAttribute(l) : t.hasAttribute(l) || e.removeAttribute(l);\n }\n },\n mergeFocusedInput: function (e, t) {\n e instanceof HTMLSelectElement || de.mergeAttrs(e, t, {\n except: [\"value\"]\n }), t.readOnly ? e.setAttribute(\"readonly\", !0) : e.removeAttribute(\"readonly\");\n },\n hasSelectionRange: function (e) {\n return e.setSelectionRange && (\"text\" === e.type || \"textarea\" === e.type);\n },\n restoreFocus: function (e, t, n) {\n if (de.isTextualInput(e)) {\n var i = e.matches(\":focus\");\n e.readOnly && e.blur(), i || e.focus(), this.hasSelectionRange(e) && e.setSelectionRange(t, n);\n }\n },\n isFormInput: function (e) {\n return /^(?:input|select|textarea)$/i.test(e.tagName) && \"button\" !== e.type;\n },\n syncAttrsToProps: function (e) {\n e instanceof HTMLInputElement && J.indexOf(e.type.toLocaleLowerCase()) >= 0 && (e.checked = null !== e.getAttribute(\"checked\"));\n },\n isTextualInput: function (e) {\n return B.indexOf(e.type) >= 0;\n },\n isNowTriggerFormExternal: function (e, t) {\n return e.getAttribute && null !== e.getAttribute(t);\n },\n syncPendingRef: function (e, t, n) {\n var i = e.getAttribute(F);\n return null === i || (de.isFormInput(e) || null !== e.getAttribute(n) ? (de.isUploadInput(e) && de.mergeAttrs(e, t, {\n isIgnored: !0\n }), de.putPrivate(e, F, t), !1) : (j.forEach(function (n) {\n e.classList.contains(n) && t.classList.add(n);\n }), t.setAttribute(F, i), !0));\n },\n cleanChildNodes: function (e, t) {\n if (de.isPhxUpdate(e, t, [\"append\", \"prepend\"])) {\n var n = [];\n e.childNodes.forEach(function (e) {\n e.id || (e.nodeType === Node.TEXT_NODE && \"\" === e.nodeValue.trim() || K(\"only HTML element tags with an id are allowed inside containers with phx-update.\\n\\n\" + 'removing illegal node: \"'.concat((e.outerHTML || e.nodeValue).trim(), '\"\\n\\n')), n.push(e));\n }), n.forEach(function (e) {\n return e.remove();\n });\n }\n }\n },\n he = function () {\n function e(t, n, i) {\n T(this, e);\n var r = new Set(),\n o = new Set(A(n.children).map(function (e) {\n return e.id;\n })),\n a = [];\n Array.from(t.children).forEach(function (e) {\n if (e.id && (r.add(e.id), o.has(e.id))) {\n var t = e.previousElementSibling && e.previousElementSibling.id;\n a.push({\n elementId: e.id,\n previousElementId: t\n });\n }\n }), this.containerId = n.id, this.updateType = i, this.elementsToModify = a, this.elementIdsToAdd = A(o).filter(function (e) {\n return !r.has(e);\n });\n }\n\n return _(e, [{\n key: \"perform\",\n value: function () {\n var e = de.byId(this.containerId);\n this.elementsToModify.forEach(function (t) {\n t.previousElementId ? ee(document.getElementById(t.previousElementId), function (e) {\n ee(document.getElementById(t.elementId), function (t) {\n t.previousElementSibling && t.previousElementSibling.id == e.id || e.insertAdjacentElement(\"afterend\", t);\n });\n }) : ee(document.getElementById(t.elementId), function (t) {\n null == t.previousElementSibling || e.insertAdjacentElement(\"afterbegin\", t);\n });\n }), \"prepend\" == this.updateType && this.elementIdsToAdd.reverse().forEach(function (t) {\n ee(document.getElementById(t), function (t) {\n return e.insertAdjacentElement(\"afterbegin\", t);\n });\n });\n }\n }]), e;\n }(),\n fe = function () {\n function e(t, n, i, r, o) {\n T(this, e), this.view = t, this.liveSocket = t.liveSocket, this.container = n, this.id = i, this.rootID = t.root.id, this.html = r, this.targetCID = o, this.cidPatch = \"number\" == typeof this.targetCID, this.callbacks = {\n beforeadded: [],\n beforeupdated: [],\n beforephxChildAdded: [],\n afteradded: [],\n afterupdated: [],\n afterdiscarded: [],\n afterphxChildAdded: []\n };\n }\n\n return _(e, null, [{\n key: \"patchEl\",\n value: function (e, t, n) {\n b(e, t, {\n childrenOnly: !1,\n onBeforeElUpdated: function (e, t) {\n if (n && n.isSameNode(e) && de.isFormInput(e)) return de.mergeFocusedInput(e, t), !1;\n }\n });\n }\n }]), _(e, [{\n key: \"before\",\n value: function (e, t) {\n this.callbacks[\"before\".concat(e)].push(t);\n }\n }, {\n key: \"after\",\n value: function (e, t) {\n this.callbacks[\"after\".concat(e)].push(t);\n }\n }, {\n key: \"trackBefore\",\n value: function (e) {\n for (var t = arguments.length, n = new Array(t > 1 ? t - 1 : 0), i = 1; i < t; i++) n[i - 1] = arguments[i];\n\n this.callbacks[\"before\".concat(e)].forEach(function (e) {\n return e.apply(void 0, n);\n });\n }\n }, {\n key: \"trackAfter\",\n value: function (e) {\n for (var t = arguments.length, n = new Array(t > 1 ? t - 1 : 0), i = 1; i < t; i++) n[i - 1] = arguments[i];\n\n this.callbacks[\"after\".concat(e)].forEach(function (e) {\n return e.apply(void 0, n);\n });\n }\n }, {\n key: \"markPrunableContentForRemoval\",\n value: function () {\n de.all(this.container, \"[phx-update=append] > *, [phx-update=prepend] > *\", function (e) {\n e.setAttribute(\"data-phx-remove\", \"\");\n });\n }\n }, {\n key: \"perform\",\n value: function () {\n var e = this,\n t = this.view,\n n = this.liveSocket,\n i = this.container,\n r = this.html,\n o = this.isCIDPatch() ? this.targetCIDContainer(r) : i;\n\n if (!this.isCIDPatch() || o) {\n var a = n.getActiveElement(),\n u = a && de.hasSelectionRange(a) ? a : {},\n s = u.selectionStart,\n c = u.selectionEnd,\n l = n.binding(\"update\"),\n d = n.binding(\"feedback-for\"),\n h = n.binding(\"disable-with\"),\n f = n.binding(\"trigger-action\"),\n v = [],\n p = [],\n g = [],\n m = null,\n y = n.time(\"premorph container prep\", function () {\n return e.buildDiffHTML(i, r, l, o);\n });\n return this.trackBefore(\"added\", i), this.trackBefore(\"updated\", i, i), n.time(\"morphdom\", function () {\n b(o, y, {\n childrenOnly: null === o.getAttribute(H),\n getNodeKey: function (e) {\n return e.id && e.id + (e.getAttribute(\"data-phx-session\") || \"\");\n },\n onBeforeNodeAdded: function (t) {\n return de.discardError(o, t, d), e.trackBefore(\"added\", t), t;\n },\n onNodeAdded: function (n) {\n de.isNowTriggerFormExternal(n, f) && (m = n), de.isPhxChild(n) && t.ownsElement(n) && e.trackAfter(\"phxChildAdded\", n), v.push(n);\n },\n onNodeDiscarded: function (t) {\n de.isPhxChild(t) && n.destroyViewByEl(t), e.trackAfter(\"discarded\", t);\n },\n onBeforeNodeDiscarded: function (t) {\n return !(!t.getAttribute || null === t.getAttribute(\"data-phx-remove\")) || (null === t.parentNode || !de.isPhxUpdate(t.parentNode, l, [\"append\", \"prepend\"]) || !t.id) && !e.skipCIDSibling(t);\n },\n onElUpdated: function (e) {\n de.isNowTriggerFormExternal(e, f) && (m = e), p.push(e);\n },\n onBeforeElUpdated: function (t, n) {\n if (de.cleanChildNodes(n, l), e.skipCIDSibling(n)) return !1;\n if (de.isIgnored(t, l)) return e.trackBefore(\"updated\", t, n), de.mergeAttrs(t, n, {\n isIgnored: !0\n }), p.push(t), !1;\n if (\"number\" === t.type && t.validity && t.validity.badInput) return !1;\n if (!de.syncPendingRef(t, n, h)) return de.isUploadInput(t) && (e.trackBefore(\"updated\", t, n), p.push(t)), !1;\n\n if (de.isPhxChild(n)) {\n var i = t.getAttribute(V);\n return de.mergeAttrs(t, n), t.setAttribute(V, i), t.setAttribute(\"data-phx-root-id\", e.rootID), !1;\n }\n\n return de.copyPrivates(n, t), de.discardError(o, n, d), a && t.isSameNode(a) && de.isFormInput(t) && !e.forceFocusedSelectUpdate(t, n) ? (e.trackBefore(\"updated\", t, n), de.mergeFocusedInput(t, n), de.syncAttrsToProps(t), p.push(t), !1) : (de.isPhxUpdate(n, l, [\"append\", \"prepend\"]) && g.push(new he(t, n, n.getAttribute(l))), de.syncAttrsToProps(n), e.trackBefore(\"updated\", t, n), !0);\n }\n });\n }), n.isDebugEnabled() && function () {\n for (var e = new Set(), t = document.querySelectorAll(\"*[id]\"), n = 0, i = t.length; n < i; n++) e.has(t[n].id) ? console.error(\"Multiple IDs detected: \".concat(t[n].id, \". Ensure unique element ids.\")) : e.add(t[n].id);\n }(), g.length > 0 && n.time(\"post-morph append/prepend restoration\", function () {\n g.forEach(function (e) {\n return e.perform();\n });\n }), n.silenceEvents(function () {\n return de.restoreFocus(a, s, c);\n }), de.dispatchEvent(document, \"phx:update\"), v.forEach(function (t) {\n return e.trackAfter(\"added\", t);\n }), p.forEach(function (t) {\n return e.trackAfter(\"updated\", t);\n }), m && (n.disconnect(), m.submit()), !0;\n }\n }\n }, {\n key: \"forceFocusedSelectUpdate\",\n value: function (e, t) {\n return !0 === e.multiple || \"select\" === e.type && e.innerHTML != t.innerHTML;\n }\n }, {\n key: \"isCIDPatch\",\n value: function () {\n return this.cidPatch;\n }\n }, {\n key: \"skipCIDSibling\",\n value: function (e) {\n return e.nodeType === Node.ELEMENT_NODE && null !== e.getAttribute(\"data-phx-skip\");\n }\n }, {\n key: \"targetCIDContainer\",\n value: function (e) {\n if (this.isCIDPatch()) {\n var t = k(de.findComponentNodeList(this.container, this.targetCID)),\n n = t[0];\n return 0 === t.slice(1).length && 1 === de.childNodeLength(e) ? n : n && n.parentNode;\n }\n }\n }, {\n key: \"buildDiffHTML\",\n value: function (e, t, n, i) {\n var r = this,\n o = this.isCIDPatch(),\n a = o && i.getAttribute(H) === this.targetCID.toString();\n if (!o || a) return t;\n var u = null,\n s = document.createElement(\"template\");\n u = de.cloneNode(i);\n var c = k(de.findComponentNodeList(u, this.targetCID)),\n l = c[0],\n d = c.slice(1);\n return s.innerHTML = t, d.forEach(function (e) {\n return e.remove();\n }), Array.from(u.childNodes).forEach(function (e) {\n e.id && e.nodeType === Node.ELEMENT_NODE && e.getAttribute(H) !== r.targetCID.toString() && (e.setAttribute(\"data-phx-skip\", \"\"), e.innerHTML = \"\");\n }), Array.from(s.content.childNodes).forEach(function (e) {\n return u.insertBefore(e, l);\n }), l.remove(), u.outerHTML;\n }\n }]), e;\n }(),\n ve = function () {\n function e(t, n, i, r, o) {\n var a = this;\n T(this, e), this.liveSocket = n, this.flash = o, this.parent = i, this.root = i ? i.root : this, this.el = t, this.id = this.el.id, this.view = this.el.getAttribute(O), this.ref = 0, this.childJoins = 0, this.loaderTimer = null, this.pendingDiffs = [], this.pruningCIDs = [], this.href = r, this.joinCount = this.parent ? this.parent.joinCount - 1 : 0, this.joinPending = !0, this.destroyed = !1, this.joinCallback = function () {}, this.stopCallback = function () {}, this.pendingJoinOps = this.parent ? null : [], this.viewHooks = {}, this.uploaders = {}, this.formSubmits = [], this.children = this.parent ? null : {}, this.root.children[this.id] = {}, this.channel = this.liveSocket.channel(\"lv:\".concat(this.id), function () {\n return {\n url: a.href,\n params: a.connectParams(),\n session: a.getSession(),\n static: a.getStatic(),\n flash: a.flash\n };\n }), this.showLoader(this.liveSocket.loaderTimeout), this.bindChannel();\n }\n\n return _(e, [{\n key: \"isMain\",\n value: function () {\n return this.liveSocket.main === this;\n }\n }, {\n key: \"connectParams\",\n value: function () {\n var e = this.liveSocket.params(this.view),\n t = de.all(document, \"[\".concat(this.binding(\"track-static\"), \"]\")).map(function (e) {\n return e.src || e.href;\n }).filter(function (e) {\n return \"string\" == typeof e;\n });\n return t.length > 0 && (e._track_static = t), e._mounts = this.joinCount, e;\n }\n }, {\n key: \"name\",\n value: function () {\n return this.view;\n }\n }, {\n key: \"isConnected\",\n value: function () {\n return this.channel.canPush();\n }\n }, {\n key: \"getSession\",\n value: function () {\n return this.el.getAttribute(\"data-phx-session\");\n }\n }, {\n key: \"getStatic\",\n value: function () {\n var e = this.el.getAttribute(V);\n return \"\" === e ? null : e;\n }\n }, {\n key: \"destroy\",\n value: function () {\n var e = this,\n t = arguments.length > 0 && void 0 !== arguments[0] ? arguments[0] : function () {};\n this.destroyAllChildren(), this.destroyed = !0, delete this.root.children[this.id], this.parent && delete this.root.children[this.parent.id][this.id], clearTimeout(this.loaderTimer);\n\n var n = function () {\n for (var n in t(), e.viewHooks) e.destroyHook(e.viewHooks[n]);\n };\n\n this.log(\"destroyed\", function () {\n return [\"the child has been removed from the parent\"];\n }), this.channel.leave().receive(\"ok\", n).receive(\"error\", n).receive(\"timeout\", n);\n }\n }, {\n key: \"setContainerClasses\",\n value: function () {\n var e;\n this.el.classList.remove(\"phx-connected\", \"phx-disconnected\", \"phx-error\"), (e = this.el.classList).add.apply(e, arguments);\n }\n }, {\n key: \"isLoading\",\n value: function () {\n return this.el.classList.contains(\"phx-disconnected\");\n }\n }, {\n key: \"showLoader\",\n value: function (e) {\n var t = this;\n if (clearTimeout(this.loaderTimer), e) this.loaderTimer = setTimeout(function () {\n return t.showLoader();\n }, e);else {\n for (var n in this.viewHooks) this.viewHooks[n].__disconnected();\n\n this.setContainerClasses(\"phx-disconnected\");\n }\n }\n }, {\n key: \"hideLoader\",\n value: function () {\n clearTimeout(this.loaderTimer), this.setContainerClasses(\"phx-connected\");\n }\n }, {\n key: \"triggerReconnected\",\n value: function () {\n for (var e in this.viewHooks) this.viewHooks[e].__reconnected();\n }\n }, {\n key: \"log\",\n value: function (e, t) {\n this.liveSocket.log(this, e, t);\n }\n }, {\n key: \"withinTargets\",\n value: function (e, t) {\n var n = this;\n\n if (/^(0|[1-9]\\d*)$/.test(e)) {\n var i = de.findComponentNodeList(this.el, e);\n 0 === i.length ? K(\"no component found matching phx-target of \".concat(e)) : t(this, i[0]);\n } else {\n var r = Array.from(document.querySelectorAll(e));\n 0 === r.length && K('nothing found matching the phx-target selector \"'.concat(e, '\"')), r.forEach(function (e) {\n return n.liveSocket.owner(e, function (n) {\n return t(n, e);\n });\n });\n }\n }\n }, {\n key: \"applyDiff\",\n value: function (e, t, n) {\n this.log(e, function () {\n return [\"\", G(t)];\n });\n var i = se.extract(t),\n r = i.diff,\n o = i.reply,\n a = i.events,\n u = i.title;\n return u && de.putTitle(u), n({\n diff: r,\n reply: o,\n events: a\n }), o;\n }\n }, {\n key: \"onJoin\",\n value: function (e) {\n var t = this,\n n = e.rendered;\n this.childJoins = 0, this.joinPending = !0, this.flash = null, le.dropLocal(this.name(), \"consecutive-reloads\"), this.applyDiff(\"mount\", n, function (n) {\n var i = n.diff,\n r = n.events;\n t.rendered = new se(t.id, i);\n var o = t.renderContainer(null, \"join\");\n t.dropPendingRefs();\n var a = t.formsForRecovery(o);\n t.joinCount++, a.length > 0 ? a.forEach(function (e, n) {\n t.pushFormRecovery(e, function (e) {\n n === a.length - 1 && t.onJoinComplete(e, o, r);\n });\n }) : t.onJoinComplete(e, o, r);\n });\n }\n }, {\n key: \"dropPendingRefs\",\n value: function () {\n de.all(this.el, \"[\".concat(F, \"]\"), function (e) {\n return e.removeAttribute(F);\n });\n }\n }, {\n key: \"onJoinComplete\",\n value: function (e, t, n) {\n var i = this,\n r = e.live_patch;\n if (this.joinCount > 1 || this.parent && !this.parent.isJoinPending()) return this.applyJoinPatch(r, t, n);\n 0 === de.findPhxChildrenInFragment(t, this.id).filter(function (e) {\n var t = e.id && i.el.querySelector(\"#\".concat(e.id)),\n n = t && t.getAttribute(V);\n return n && e.setAttribute(V, n), i.joinChild(e);\n }).length ? this.parent ? (this.root.pendingJoinOps.push([this, function () {\n return i.applyJoinPatch(r, t, n);\n }]), this.parent.ackJoin(this)) : (this.onAllChildJoinsComplete(), this.applyJoinPatch(r, t, n)) : this.root.pendingJoinOps.push([this, function () {\n return i.applyJoinPatch(r, t, n);\n }]);\n }\n }, {\n key: \"attachTrueDocEl\",\n value: function () {\n this.el = de.byId(this.id), this.el.setAttribute(\"data-phx-root-id\", this.root.id);\n }\n }, {\n key: \"dispatchEvents\",\n value: function (e) {\n e.forEach(function (e) {\n var t = x(e, 2),\n n = t[0],\n i = t[1];\n window.dispatchEvent(new CustomEvent(\"phx:hook:\".concat(n), {\n detail: i\n }));\n });\n }\n }, {\n key: \"applyJoinPatch\",\n value: function (e, t, n) {\n var i = this;\n this.attachTrueDocEl();\n var r = new fe(this, this.el, this.id, t, null);\n\n if (r.markPrunableContentForRemoval(), this.performPatch(r, !1), this.joinNewChildren(), de.all(this.el, \"[\".concat(this.binding(\"hook\"), \"], [data-phx-\").concat(\"hook\", \"]\"), function (e) {\n var t = i.addHook(e);\n t && t.__mounted();\n }), this.joinPending = !1, this.dispatchEvents(n), this.applyPendingUpdates(), e) {\n var o = e.kind,\n a = e.to;\n this.liveSocket.historyPatch(a, o);\n }\n\n this.hideLoader(), this.joinCount > 1 && this.triggerReconnected(), this.stopCallback();\n }\n }, {\n key: \"triggerBeforeUpdateHook\",\n value: function (e, t) {\n this.liveSocket.triggerDOM(\"onBeforeElUpdated\", [e, t]);\n var n = this.getHook(e),\n i = n && de.isIgnored(e, this.binding(\"update\"));\n if (n && !e.isEqualNode(t) && (!i || !function (e, t) {\n return JSON.stringify(e) === JSON.stringify(t);\n }(e.dataset, t.dataset))) return n.__beforeUpdate(), n;\n }\n }, {\n key: \"performPatch\",\n value: function (e, t) {\n var n = this,\n i = [],\n r = !1,\n o = new Set();\n return e.after(\"added\", function (e) {\n n.liveSocket.triggerDOM(\"onNodeAdded\", [e]);\n var t = n.addHook(e);\n t && t.__mounted();\n }), e.after(\"phxChildAdded\", function (e) {\n return r = !0;\n }), e.before(\"updated\", function (e, t) {\n n.triggerBeforeUpdateHook(e, t) && o.add(e.id);\n }), e.after(\"updated\", function (e) {\n o.has(e.id) && n.getHook(e).__updated();\n }), e.after(\"discarded\", function (e) {\n var t = n.componentID(e);\n \"number\" == typeof t && -1 === i.indexOf(t) && i.push(t);\n var r = n.getHook(e);\n r && n.destroyHook(r);\n }), e.perform(), t && this.maybePushComponentsDestroyed(i), r;\n }\n }, {\n key: \"joinNewChildren\",\n value: function () {\n var e = this;\n de.findPhxChildren(this.el, this.id).forEach(function (t) {\n return e.joinChild(t);\n });\n }\n }, {\n key: \"getChildById\",\n value: function (e) {\n return this.root.children[this.id][e];\n }\n }, {\n key: \"getDescendentByEl\",\n value: function (e) {\n return e.id === this.id ? this : this.children[e.getAttribute(\"data-phx-parent-id\")][e.id];\n }\n }, {\n key: \"destroyDescendent\",\n value: function (e) {\n for (var t in this.root.children) for (var n in this.root.children[t]) if (n === e) return this.root.children[t][n].destroy();\n }\n }, {\n key: \"joinChild\",\n value: function (t) {\n if (!this.getChildById(t.id)) {\n var n = new e(t, this.liveSocket, this);\n return this.root.children[this.id][n.id] = n, n.join(), this.childJoins++, !0;\n }\n }\n }, {\n key: \"isJoinPending\",\n value: function () {\n return this.joinPending;\n }\n }, {\n key: \"ackJoin\",\n value: function (e) {\n this.childJoins--, 0 === this.childJoins && (this.parent ? this.parent.ackJoin(this) : this.onAllChildJoinsComplete());\n }\n }, {\n key: \"onAllChildJoinsComplete\",\n value: function () {\n this.joinCallback(), this.pendingJoinOps.forEach(function (e) {\n var t = x(e, 2),\n n = t[0],\n i = t[1];\n n.isDestroyed() || i();\n }), this.pendingJoinOps = [];\n }\n }, {\n key: \"update\",\n value: function (e, t) {\n var n = this;\n if (this.isJoinPending() || this.liveSocket.hasPendingLink()) return this.pendingDiffs.push({\n diff: e,\n events: t\n });\n this.rendered.mergeDiff(e);\n var i = !1;\n this.rendered.isComponentOnlyDiff(e) ? this.liveSocket.time(\"component patch complete\", function () {\n de.findParentCIDs(n.el, n.rendered.componentCIDs(e)).forEach(function (t) {\n n.componentPatch(n.rendered.getComponent(e, t), t) && (i = !0);\n });\n }) : Z(e) || this.liveSocket.time(\"full patch complete\", function () {\n var t = n.renderContainer(e, \"update\"),\n r = new fe(n, n.el, n.id, t, null);\n i = n.performPatch(r, !0);\n }), this.dispatchEvents(t), i && this.joinNewChildren();\n }\n }, {\n key: \"renderContainer\",\n value: function (e, t) {\n var n = this;\n return this.liveSocket.time(\"toString diff (\".concat(t, \")\"), function () {\n var t = n.el.tagName,\n i = e ? n.rendered.componentCIDs(e).concat(n.pruningCIDs) : null,\n r = n.rendered.toString(i);\n return \"<\".concat(t, \">\").concat(r, \"\");\n });\n }\n }, {\n key: \"componentPatch\",\n value: function (e, t) {\n if (Z(e)) return !1;\n var n = this.rendered.componentToString(t),\n i = new fe(this, this.el, this.id, n, t);\n return this.performPatch(i, !0);\n }\n }, {\n key: \"getHook\",\n value: function (e) {\n return this.viewHooks[ge.elementID(e)];\n }\n }, {\n key: \"addHook\",\n value: function (e) {\n if (!ge.elementID(e) && e.getAttribute) {\n var t = e.getAttribute(\"data-phx-\".concat(\"hook\")) || e.getAttribute(this.binding(\"hook\"));\n\n if (!t || this.ownsElement(e)) {\n var n = this.liveSocket.getHookCallbacks(t);\n\n if (n) {\n e.id || K('no DOM ID for hook \"'.concat(t, '\". Hooks require a unique ID on each element.'), e);\n var i = new ge(this, e, n);\n return this.viewHooks[ge.elementID(i.el)] = i, i;\n }\n\n null !== t && K('unknown hook found for \"'.concat(t, '\"'), e);\n }\n }\n }\n }, {\n key: \"destroyHook\",\n value: function (e) {\n e.__destroyed(), e.__cleanup__(), delete this.viewHooks[ge.elementID(e.el)];\n }\n }, {\n key: \"applyPendingUpdates\",\n value: function () {\n var e = this;\n this.pendingDiffs.forEach(function (t) {\n var n = t.diff,\n i = t.events;\n return e.update(n, i);\n }), this.pendingDiffs = [];\n }\n }, {\n key: \"onChannel\",\n value: function (e, t) {\n var n = this;\n this.liveSocket.onChannel(this.channel, e, function (e) {\n n.isJoinPending() ? n.root.pendingJoinOps.push([n, function () {\n return t(e);\n }]) : t(e);\n });\n }\n }, {\n key: \"bindChannel\",\n value: function () {\n var e = this;\n this.liveSocket.onChannel(this.channel, \"diff\", function (t) {\n e.applyDiff(\"update\", t, function (t) {\n var n = t.diff,\n i = t.events;\n return e.update(n, i);\n });\n }), this.onChannel(\"redirect\", function (t) {\n var n = t.to,\n i = t.flash;\n return e.onRedirect({\n to: n,\n flash: i\n });\n }), this.onChannel(\"live_patch\", function (t) {\n return e.onLivePatch(t);\n }), this.onChannel(\"live_redirect\", function (t) {\n return e.onLiveRedirect(t);\n }), this.channel.onError(function (t) {\n return e.onError(t);\n }), this.channel.onClose(function (t) {\n return e.onClose(t);\n });\n }\n }, {\n key: \"destroyAllChildren\",\n value: function () {\n for (var e in this.root.children[this.id]) this.getChildById(e).destroy();\n }\n }, {\n key: \"onLiveRedirect\",\n value: function (e) {\n var t = e.to,\n n = e.kind,\n i = e.flash,\n r = this.expandURL(t);\n this.liveSocket.historyRedirect(r, n, i);\n }\n }, {\n key: \"onLivePatch\",\n value: function (e) {\n var t = e.to,\n n = e.kind;\n this.href = this.expandURL(t), this.liveSocket.historyPatch(t, n);\n }\n }, {\n key: \"expandURL\",\n value: function (e) {\n return e.startsWith(\"/\") ? \"\".concat(window.location.protocol, \"//\").concat(window.location.host).concat(e) : e;\n }\n }, {\n key: \"onRedirect\",\n value: function (e) {\n var t = e.to,\n n = e.flash;\n this.liveSocket.redirect(t, n);\n }\n }, {\n key: \"isDestroyed\",\n value: function () {\n return this.destroyed;\n }\n }, {\n key: \"join\",\n value: function (e) {\n var t = this;\n this.parent || (this.stopCallback = this.liveSocket.withPageLoading({\n to: this.href,\n kind: \"initial\"\n })), this.joinCallback = function () {\n return e && e(t, t.joinCount);\n }, this.liveSocket.wrapPush(this, {\n timeout: !1\n }, function () {\n return t.channel.join().receive(\"ok\", function (e) {\n return t.onJoin(e);\n }).receive(\"error\", function (e) {\n return t.onJoinError(e);\n }).receive(\"timeout\", function () {\n return t.onJoinError({\n reason: \"timeout\"\n });\n });\n });\n }\n }, {\n key: \"onJoinError\",\n value: function (e) {\n return (e.redirect || e.live_redirect) && (this.joinPending = !1, this.channel.leave()), e.redirect ? this.onRedirect(e.redirect) : e.live_redirect ? this.onLiveRedirect(e.live_redirect) : (this.log(\"error\", function () {\n return [\"unable to join\", e];\n }), this.liveSocket.reloadWithJitter(this));\n }\n }, {\n key: \"onClose\",\n value: function (e) {\n if (!this.isDestroyed()) {\n if (this.isJoinPending() || this.liveSocket.hasPendingLink() && \"leave\" !== e) return this.liveSocket.reloadWithJitter(this);\n this.destroyAllChildren(), this.liveSocket.dropActiveElement(this), document.activeElement && document.activeElement.blur(), this.liveSocket.isUnloaded() && this.showLoader(200);\n }\n }\n }, {\n key: \"onError\",\n value: function (e) {\n this.onClose(e), this.log(\"error\", function () {\n return [\"view crashed\", e];\n }), this.liveSocket.isUnloaded() || this.displayError();\n }\n }, {\n key: \"displayError\",\n value: function () {\n this.isMain() && de.dispatchEvent(window, \"phx:page-loading-start\", {\n to: this.href,\n kind: \"error\"\n }), this.showLoader(), this.setContainerClasses(\"phx-disconnected\", \"phx-error\");\n }\n }, {\n key: \"pushWithReply\",\n value: function (e, t, n) {\n var i = this,\n r = arguments.length > 3 && void 0 !== arguments[3] ? arguments[3] : function () {},\n o = x(e ? e() : [null, []], 2),\n a = o[0],\n u = x(o[1], 1)[0],\n s = function () {};\n\n return u && null !== u.getAttribute(this.binding(\"page-loading\")) && (s = this.liveSocket.withPageLoading({\n kind: \"element\",\n target: u\n })), \"number\" != typeof n.cid && delete n.cid, this.liveSocket.wrapPush(this, {\n timeout: !0\n }, function () {\n return i.channel.push(t, n, 3e4).receive(\"ok\", function (e) {\n var t = null;\n null !== a && i.undoRefs(a), e.diff && (t = i.applyDiff(\"update\", e.diff, function (e) {\n var t = e.diff,\n n = e.events;\n i.update(t, n);\n })), e.redirect && i.onRedirect(e.redirect), e.live_patch && i.onLivePatch(e.live_patch), e.live_redirect && i.onLiveRedirect(e.live_redirect), s(), r(e, t);\n });\n });\n }\n }, {\n key: \"undoRefs\",\n value: function (e) {\n var t = this;\n de.all(this.el, \"[\".concat(F, '=\"').concat(e, '\"]'), function (e) {\n e.removeAttribute(F), null !== e.getAttribute(\"data-phx-readonly\") && (e.readOnly = !1, e.removeAttribute(\"data-phx-readonly\")), null !== e.getAttribute(\"data-phx-disabled\") && (e.disabled = !1, e.removeAttribute(\"data-phx-disabled\")), j.forEach(function (t) {\n return de.removeClass(e, t);\n });\n var n = e.getAttribute(\"data-phx-disable-with-restore\");\n null !== n && (e.innerText = n, e.removeAttribute(\"data-phx-disable-with-restore\"));\n var i = de.private(e, F);\n\n if (i) {\n var r = t.triggerBeforeUpdateHook(e, i);\n fe.patchEl(e, i, t.liveSocket.getActiveElement()), r && r.__updated(), de.deletePrivate(e, F);\n }\n });\n }\n }, {\n key: \"putRef\",\n value: function (e, t) {\n var n = this.ref++,\n i = this.binding(\"disable-with\");\n return e.forEach(function (e) {\n e.classList.add(\"phx-\".concat(t, \"-loading\")), e.setAttribute(F, n);\n var r = e.getAttribute(i);\n null !== r && (e.getAttribute(\"data-phx-disable-with-restore\") || e.setAttribute(\"data-phx-disable-with-restore\", e.innerText), e.innerText = r);\n }), [n, e];\n }\n }, {\n key: \"componentID\",\n value: function (e) {\n var t = e.getAttribute && e.getAttribute(H);\n return t ? parseInt(t) : null;\n }\n }, {\n key: \"targetComponentID\",\n value: function (e, t) {\n return e.getAttribute(this.binding(\"target\")) ? this.closestComponentID(t) : null;\n }\n }, {\n key: \"closestComponentID\",\n value: function (e) {\n var t = this;\n return e ? ee(e.closest(\"[\".concat(H, \"]\")), function (e) {\n return t.ownsElement(e) && t.componentID(e);\n }) : null;\n }\n }, {\n key: \"pushHookEvent\",\n value: function (e, t, n, i) {\n var r = x(this.putRef([], \"hook\"), 2),\n o = r[0],\n a = r[1];\n return this.pushWithReply(function () {\n return [o, a];\n }, \"event\", {\n type: \"hook\",\n event: t,\n value: n,\n cid: this.closestComponentID(e)\n }, function (e, t) {\n return i(t, o);\n }), o;\n }\n }, {\n key: \"extractMeta\",\n value: function (e, t) {\n for (var n = this.binding(\"value-\"), i = 0; i < e.attributes.length; i++) {\n var r = e.attributes[i].name;\n r.startsWith(n) && (t[r.replace(n, \"\")] = e.getAttribute(r));\n }\n\n return void 0 !== e.value && (t.value = e.value, \"INPUT\" === e.tagName && J.indexOf(e.type) >= 0 && !e.checked && delete t.value), t;\n }\n }, {\n key: \"pushEvent\",\n value: function (e, t, n, i, r) {\n var o = this;\n this.pushWithReply(function () {\n return o.putRef([t], e);\n }, \"event\", {\n type: e,\n event: i,\n value: this.extractMeta(t, r),\n cid: this.targetComponentID(t, n)\n });\n }\n }, {\n key: \"pushKey\",\n value: function (e, t, n, i, r) {\n var o = this;\n this.pushWithReply(function () {\n return o.putRef([e], n);\n }, \"event\", {\n type: n,\n event: i,\n value: this.extractMeta(e, r),\n cid: this.targetComponentID(e, t)\n });\n }\n }, {\n key: \"pushFileProgress\",\n value: function (e, t, n) {\n var i = arguments.length > 3 && void 0 !== arguments[3] ? arguments[3] : function () {};\n this.liveSocket.withinOwners(e.form, function (r, o) {\n r.pushWithReply(null, \"progress\", {\n event: e.getAttribute(r.binding(\"progress\")),\n ref: e.getAttribute(M),\n entry_ref: t,\n progress: n,\n cid: r.targetComponentID(e.form, o)\n }, i);\n });\n }\n }, {\n key: \"pushInput\",\n value: function (e, t, n, i, r) {\n var o = this,\n a = this.targetComponentID(e.form, t),\n u = function () {\n return o.putRef([e, e.form], \"change\");\n },\n s = ue(e.form, {\n _target: i.name\n });\n\n e.files && e.files.length > 0 && re.trackFiles(e, Array.from(e.files));\n var c = {\n type: \"form\",\n event: n,\n value: s,\n uploads: re.serializeUploads(e),\n cid: a\n };\n this.pushWithReply(u, \"event\", c, function (n) {\n if (de.showError(e, o.liveSocket.binding(\"feedback-for\")), de.isUploadInput(e) && null !== e.getAttribute(\"data-phx-auto-upload\")) {\n if (re.filesAwaitingPreflight(e).length > 0) {\n var i = x(u(), 2),\n s = i[0];\n i[1];\n o.uploadFiles(e.form, t, s, a, function (t) {\n r && r(n), o.triggerAwaitingSubmit(e.form);\n });\n }\n } else r && r(n);\n });\n }\n }, {\n key: \"triggerAwaitingSubmit\",\n value: function (e) {\n var t = this.getScheduledSubmit(e);\n\n if (t) {\n var n = x(t, 3),\n i = (n[0], n[1], n[2]);\n this.cancelSubmit(e), i();\n }\n }\n }, {\n key: \"getScheduledSubmit\",\n value: function (e) {\n return this.formSubmits.find(function (t) {\n var n = x(t, 2),\n i = n[0];\n n[1];\n return i.isSameNode(e);\n });\n }\n }, {\n key: \"scheduleSubmit\",\n value: function (e, t, n) {\n if (this.getScheduledSubmit(e)) return !0;\n this.formSubmits.push([e, t, n]);\n }\n }, {\n key: \"cancelSubmit\",\n value: function (e) {\n var t = this;\n this.formSubmits = this.formSubmits.filter(function (n) {\n var i = x(n, 3),\n r = i[0],\n o = i[1];\n i[2];\n return !r.isSameNode(e) || (t.undoRefs(o), !1);\n });\n }\n }, {\n key: \"pushFormSubmit\",\n value: function (e, t, n, i) {\n var r = this,\n o = function (e) {\n return !(Y(e, \"\".concat(r.binding(\"update\"), \"=ignore\"), e.form) || Y(e, \"data-phx-update=ignore\", e.form));\n },\n a = function (e) {\n return e.hasAttribute(r.binding(\"disable-with\"));\n },\n u = function (e) {\n return \"BUTTON\" == e.tagName;\n },\n s = function (e) {\n return [\"INPUT\", \"TEXTAREA\", \"SELECT\"].includes(e.tagName);\n },\n c = function () {\n var t = Array.from(e.elements),\n n = t.filter(a),\n i = t.filter(u).filter(o),\n c = t.filter(s).filter(o);\n return i.forEach(function (e) {\n e.setAttribute(\"data-phx-disabled\", e.disabled), e.disabled = !0;\n }), c.forEach(function (e) {\n e.setAttribute(\"data-phx-readonly\", e.readOnly), e.readOnly = !0, e.files && (e.setAttribute(\"data-phx-disabled\", e.disabled), e.disabled = !0);\n }), e.setAttribute(r.binding(\"page-loading\"), \"\"), r.putRef([e].concat(n).concat(i).concat(c), \"submit\");\n },\n l = this.targetComponentID(e, t);\n\n if (re.hasUploadsInProgress(e)) {\n var d = x(c(), 2),\n h = d[0];\n d[1];\n return this.scheduleSubmit(e, h, function () {\n return r.pushFormSubmit(e, t, n, i);\n });\n }\n\n if (re.inputsAwaitingPreflight(e).length > 0) {\n var f = x(c(), 2),\n v = f[0],\n p = f[1],\n g = function () {\n return [v, p];\n };\n\n this.uploadFiles(e, t, v, l, function (t) {\n var o = ue(e, {});\n r.pushWithReply(g, \"event\", {\n type: \"form\",\n event: n,\n value: o,\n cid: l\n }, i);\n });\n } else {\n var m = ue(e);\n this.pushWithReply(c, \"event\", {\n type: \"form\",\n event: n,\n value: m,\n cid: l\n }, i);\n }\n }\n }, {\n key: \"uploadFiles\",\n value: function (e, t, n, i, r) {\n var o = this,\n a = this.joinCount;\n re.activeFileInputs(e).forEach(function (e) {\n var i = new re(e, o, r);\n o.uploaders[e] = i;\n var u = i.entries().map(function (e) {\n return e.toPreflightPayload();\n }),\n s = {\n ref: e.getAttribute(M),\n entries: u,\n cid: o.targetComponentID(e.form, t)\n };\n o.log(\"upload\", function () {\n return [\"sending preflight request\", s];\n }), o.pushWithReply(null, \"allow_upload\", s, function (e) {\n if (o.log(\"upload\", function () {\n return [\"got preflight response\", e];\n }), e.error) {\n o.undoRefs(n);\n var t = x(e.error, 2),\n r = t[0],\n u = t[1];\n o.log(\"upload\", function () {\n return [\"error for entry \".concat(r), u];\n });\n } else {\n i.initAdapterUpload(e, function (e) {\n o.channel.onError(function () {\n o.joinCount === a && e();\n });\n }, o.liveSocket);\n }\n });\n });\n }\n }, {\n key: \"pushFormRecovery\",\n value: function (e, t) {\n var n = this;\n this.liveSocket.withinOwners(e, function (i, r) {\n var o = e.elements[0],\n a = e.getAttribute(n.binding(\"auto-recover\")) || e.getAttribute(n.binding(\"change\"));\n i.pushInput(o, r, a, o, t);\n });\n }\n }, {\n key: \"pushLinkPatch\",\n value: function (e, t, n) {\n var i = this,\n r = this.liveSocket.setPendingLink(e),\n o = t ? function () {\n return i.putRef([t], \"click\");\n } : null;\n this.pushWithReply(o, \"link\", {\n url: e\n }, function (t) {\n t.link_redirect ? i.liveSocket.replaceMain(e, null, n, r) : i.liveSocket.commitPendingLink(r) && (i.href = e, i.applyPendingUpdates(), n && n());\n }).receive(\"timeout\", function () {\n return i.liveSocket.redirect(window.location.href);\n });\n }\n }, {\n key: \"formsForRecovery\",\n value: function (e) {\n var t = this;\n if (0 === this.joinCount) return [];\n var n = this.binding(\"change\"),\n i = document.createElement(\"template\");\n return i.innerHTML = e, de.all(this.el, \"form[\".concat(n, \"]\")).filter(function (e) {\n return t.ownsElement(e);\n }).filter(function (e) {\n return e.elements.length > 0;\n }).filter(function (e) {\n return \"ignore\" !== e.getAttribute(t.binding(\"auto-recover\"));\n }).filter(function (e) {\n return i.content.querySelector(\"form[\".concat(n, '=\"').concat(e.getAttribute(n), '\"]'));\n });\n }\n }, {\n key: \"maybePushComponentsDestroyed\",\n value: function (e) {\n var t,\n n = this,\n i = e.filter(function (e) {\n return 0 === de.findComponentNodeList(n.el, e).length;\n });\n i.length > 0 && ((t = this.pruningCIDs).push.apply(t, A(i)), this.pushWithReply(null, \"cids_will_destroy\", {\n cids: i\n }, function () {\n n.pruningCIDs = n.pruningCIDs.filter(function (e) {\n return -1 !== i.indexOf(e);\n });\n var e = i.filter(function (e) {\n return 0 === de.findComponentNodeList(n.el, e).length;\n });\n e.length > 0 && n.pushWithReply(null, \"cids_destroyed\", {\n cids: e\n }, function (e) {\n n.rendered.pruneCIDs(e.cids);\n });\n }));\n }\n }, {\n key: \"ownsElement\",\n value: function (e) {\n return e.getAttribute(\"data-phx-parent-id\") === this.id || ee(e.closest(U), function (e) {\n return e.id;\n }) === this.id;\n }\n }, {\n key: \"submitForm\",\n value: function (e, t, n) {\n var i = this;\n de.putPrivate(e, \"phx-has-submitted\", !0), this.liveSocket.blurActiveElement(this), this.pushFormSubmit(e, t, n, function () {\n i.liveSocket.restorePreviouslyActiveFocus();\n });\n }\n }, {\n key: \"binding\",\n value: function (e) {\n return this.liveSocket.binding(e);\n }\n }]), e;\n }(),\n pe = 1,\n ge = function () {\n function e(t, n, i) {\n for (var r in T(this, e), this.__view = t, this.__liveSocket = t.liveSocket, this.__callbacks = i, this.__listeners = new Set(), this.__isDisconnected = !1, this.el = n, this.viewName = t.name(), this.el.phxHookId = this.constructor.makeID(), this.__callbacks) this[r] = this.__callbacks[r];\n }\n\n return _(e, null, [{\n key: \"makeID\",\n value: function () {\n return pe++;\n }\n }, {\n key: \"elementID\",\n value: function (e) {\n return e.phxHookId;\n }\n }]), _(e, [{\n key: \"__mounted\",\n value: function () {\n this.mounted && this.mounted();\n }\n }, {\n key: \"__updated\",\n value: function () {\n this.updated && this.updated();\n }\n }, {\n key: \"__beforeUpdate\",\n value: function () {\n this.beforeUpdate && this.beforeUpdate();\n }\n }, {\n key: \"__destroyed\",\n value: function () {\n this.destroyed && this.destroyed();\n }\n }, {\n key: \"__reconnected\",\n value: function () {\n this.__isDisconnected && (this.__isDisconnected = !1, this.reconnected && this.reconnected());\n }\n }, {\n key: \"__disconnected\",\n value: function () {\n this.__isDisconnected = !0, this.disconnected && this.disconnected();\n }\n }, {\n key: \"pushEvent\",\n value: function (e) {\n var t = arguments.length > 1 && void 0 !== arguments[1] ? arguments[1] : {},\n n = arguments.length > 2 && void 0 !== arguments[2] ? arguments[2] : function () {};\n return this.__view.pushHookEvent(null, e, t, n);\n }\n }, {\n key: \"pushEventTo\",\n value: function (e, t) {\n var n = arguments.length > 2 && void 0 !== arguments[2] ? arguments[2] : {},\n i = arguments.length > 3 && void 0 !== arguments[3] ? arguments[3] : function () {};\n return this.__view.withinTargets(e, function (e, r) {\n return e.pushHookEvent(r, t, n, i);\n });\n }\n }, {\n key: \"handleEvent\",\n value: function (e, t) {\n var n = function (n, i) {\n return i ? e : t(n.detail);\n };\n\n return window.addEventListener(\"phx:hook:\".concat(e), n), this.__listeners.add(n), n;\n }\n }, {\n key: \"removeHandleEvent\",\n value: function (e) {\n var t = e(null, !0);\n window.removeEventListener(\"phx:hook:\".concat(t), e), this.__listeners.delete(e);\n }\n }, {\n key: \"__cleanup__\",\n value: function () {\n var e = this;\n\n this.__listeners.forEach(function (t) {\n return e.removeHandleEvent(t);\n });\n }\n }]), e;\n }();\n\n t.default = ce;\n }, function (e, t) {\n var n;\n\n n = function () {\n return this;\n }();\n\n try {\n n = n || Function(\"return this\")() || (0, eval)(\"this\");\n } catch (e) {\n \"object\" == typeof window && (n = window);\n }\n\n e.exports = n;\n }, function (e, t, n) {\n (function (t) {\n t.Phoenix || (t.Phoenix = {}), e.exports = t.Phoenix.LiveView = n(0);\n }).call(this, n(1));\n }]);\n});//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiLi4vZGVwcy9waG9lbml4X2xpdmVfdmlldy9wcml2L3N0YXRpYy9waG9lbml4X2xpdmVfdmlldy5qcy5qcyIsInNvdXJjZXMiOlsid2VicGFjazovLy8uLi9kZXBzL3Bob2VuaXhfbGl2ZV92aWV3L3ByaXYvc3RhdGljL3Bob2VuaXhfbGl2ZV92aWV3LmpzPzJjOTAiXSwic291cmNlc0NvbnRlbnQiOlsiIWZ1bmN0aW9uKGUsdCl7XCJvYmplY3RcIj09dHlwZW9mIGV4cG9ydHMmJlwib2JqZWN0XCI9PXR5cGVvZiBtb2R1bGU/bW9kdWxlLmV4cG9ydHM9dCgpOlwiZnVuY3Rpb25cIj09dHlwZW9mIGRlZmluZSYmZGVmaW5lLmFtZD9kZWZpbmUoW10sdCk6XCJvYmplY3RcIj09dHlwZW9mIGV4cG9ydHM/ZXhwb3J0cy5waG9lbml4X2xpdmVfdmlldz10KCk6ZS5waG9lbml4X2xpdmVfdmlldz10KCl9KHRoaXMsZnVuY3Rpb24oKXtyZXR1cm4gZnVuY3Rpb24oZSl7dmFyIHQ9e307ZnVuY3Rpb24gbihpKXtpZih0W2ldKXJldHVybiB0W2ldLmV4cG9ydHM7dmFyIHI9dFtpXT17aTppLGw6ITEsZXhwb3J0czp7fX07cmV0dXJuIGVbaV0uY2FsbChyLmV4cG9ydHMscixyLmV4cG9ydHMsbiksci5sPSEwLHIuZXhwb3J0c31yZXR1cm4gbi5tPWUsbi5jPXQsbi5kPWZ1bmN0aW9uKGUsdCxpKXtuLm8oZSx0KXx8T2JqZWN0LmRlZmluZVByb3BlcnR5KGUsdCx7Y29uZmlndXJhYmxlOiExLGVudW1lcmFibGU6ITAsZ2V0Oml9KX0sbi5yPWZ1bmN0aW9uKGUpe09iamVjdC5kZWZpbmVQcm9wZXJ0eShlLFwiX19lc01vZHVsZVwiLHt2YWx1ZTohMH0pfSxuLm49ZnVuY3Rpb24oZSl7dmFyIHQ9ZSYmZS5fX2VzTW9kdWxlP2Z1bmN0aW9uKCl7cmV0dXJuIGUuZGVmYXVsdH06ZnVuY3Rpb24oKXtyZXR1cm4gZX07cmV0dXJuIG4uZCh0LFwiYVwiLHQpLHR9LG4ubz1mdW5jdGlvbihlLHQpe3JldHVybiBPYmplY3QucHJvdG90eXBlLmhhc093blByb3BlcnR5LmNhbGwoZSx0KX0sbi5wPVwiXCIsbihuLnM9Mil9KFtmdW5jdGlvbihlLHQsbil7XCJ1c2Ugc3RyaWN0XCI7bi5yKHQpO3ZhciBpLHI9MTE7dmFyIG89XCJodHRwOi8vd3d3LnczLm9yZy8xOTk5L3hodG1sXCIsYT1cInVuZGVmaW5lZFwiPT10eXBlb2YgZG9jdW1lbnQ/dm9pZCAwOmRvY3VtZW50LHU9ISFhJiZcImNvbnRlbnRcImluIGEuY3JlYXRlRWxlbWVudChcInRlbXBsYXRlXCIpLHM9ISFhJiZhLmNyZWF0ZVJhbmdlJiZcImNyZWF0ZUNvbnRleHR1YWxGcmFnbWVudFwiaW4gYS5jcmVhdGVSYW5nZSgpO2Z1bmN0aW9uIGMoZSl7cmV0dXJuIGU9ZS50cmltKCksdT9mdW5jdGlvbihlKXt2YXIgdD1hLmNyZWF0ZUVsZW1lbnQoXCJ0ZW1wbGF0ZVwiKTtyZXR1cm4gdC5pbm5lckhUTUw9ZSx0LmNvbnRlbnQuY2hpbGROb2Rlc1swXX0oZSk6cz9mdW5jdGlvbihlKXtyZXR1cm4gaXx8KGk9YS5jcmVhdGVSYW5nZSgpKS5zZWxlY3ROb2RlKGEuYm9keSksaS5jcmVhdGVDb250ZXh0dWFsRnJhZ21lbnQoZSkuY2hpbGROb2Rlc1swXX0oZSk6ZnVuY3Rpb24oZSl7dmFyIHQ9YS5jcmVhdGVFbGVtZW50KFwiYm9keVwiKTtyZXR1cm4gdC5pbm5lckhUTUw9ZSx0LmNoaWxkTm9kZXNbMF19KGUpfWZ1bmN0aW9uIGwoZSx0KXt2YXIgbixpLHI9ZS5ub2RlTmFtZSxvPXQubm9kZU5hbWU7cmV0dXJuIHI9PT1vfHwobj1yLmNoYXJDb2RlQXQoMCksaT1vLmNoYXJDb2RlQXQoMCksbjw9OTAmJmk+PTk3P3I9PT1vLnRvVXBwZXJDYXNlKCk6aTw9OTAmJm4+PTk3JiZvPT09ci50b1VwcGVyQ2FzZSgpKX1mdW5jdGlvbiBkKGUsdCxuKXtlW25dIT09dFtuXSYmKGVbbl09dFtuXSxlW25dP2Uuc2V0QXR0cmlidXRlKG4sXCJcIik6ZS5yZW1vdmVBdHRyaWJ1dGUobikpfXZhciBoPXtPUFRJT046ZnVuY3Rpb24oZSx0KXt2YXIgbj1lLnBhcmVudE5vZGU7aWYobil7dmFyIGk9bi5ub2RlTmFtZS50b1VwcGVyQ2FzZSgpO1wiT1BUR1JPVVBcIj09PWkmJihpPShuPW4ucGFyZW50Tm9kZSkmJm4ubm9kZU5hbWUudG9VcHBlckNhc2UoKSksXCJTRUxFQ1RcIiE9PWl8fG4uaGFzQXR0cmlidXRlKFwibXVsdGlwbGVcIil8fChlLmhhc0F0dHJpYnV0ZShcInNlbGVjdGVkXCIpJiYhdC5zZWxlY3RlZCYmKGUuc2V0QXR0cmlidXRlKFwic2VsZWN0ZWRcIixcInNlbGVjdGVkXCIpLGUucmVtb3ZlQXR0cmlidXRlKFwic2VsZWN0ZWRcIikpLG4uc2VsZWN0ZWRJbmRleD0tMSl9ZChlLHQsXCJzZWxlY3RlZFwiKX0sSU5QVVQ6ZnVuY3Rpb24oZSx0KXtkKGUsdCxcImNoZWNrZWRcIiksZChlLHQsXCJkaXNhYmxlZFwiKSxlLnZhbHVlIT09dC52YWx1ZSYmKGUudmFsdWU9dC52YWx1ZSksdC5oYXNBdHRyaWJ1dGUoXCJ2YWx1ZVwiKXx8ZS5yZW1vdmVBdHRyaWJ1dGUoXCJ2YWx1ZVwiKX0sVEVYVEFSRUE6ZnVuY3Rpb24oZSx0KXt2YXIgbj10LnZhbHVlO2UudmFsdWUhPT1uJiYoZS52YWx1ZT1uKTt2YXIgaT1lLmZpcnN0Q2hpbGQ7aWYoaSl7dmFyIHI9aS5ub2RlVmFsdWU7aWYocj09bnx8IW4mJnI9PWUucGxhY2Vob2xkZXIpcmV0dXJuO2kubm9kZVZhbHVlPW59fSxTRUxFQ1Q6ZnVuY3Rpb24oZSx0KXtpZighdC5oYXNBdHRyaWJ1dGUoXCJtdWx0aXBsZVwiKSl7Zm9yKHZhciBuLGkscj0tMSxvPTAsYT1lLmZpcnN0Q2hpbGQ7YTspaWYoXCJPUFRHUk9VUFwiPT09KGk9YS5ub2RlTmFtZSYmYS5ub2RlTmFtZS50b1VwcGVyQ2FzZSgpKSlhPShuPWEpLmZpcnN0Q2hpbGQ7ZWxzZXtpZihcIk9QVElPTlwiPT09aSl7aWYoYS5oYXNBdHRyaWJ1dGUoXCJzZWxlY3RlZFwiKSl7cj1vO2JyZWFrfW8rK30hKGE9YS5uZXh0U2libGluZykmJm4mJihhPW4ubmV4dFNpYmxpbmcsbj1udWxsKX1lLnNlbGVjdGVkSW5kZXg9cn19fSxmPTEsdj0xMSxwPTMsZz04O2Z1bmN0aW9uIG0oKXt9ZnVuY3Rpb24geShlKXtpZihlKXJldHVybiBlLmdldEF0dHJpYnV0ZSYmZS5nZXRBdHRyaWJ1dGUoXCJpZFwiKXx8ZS5pZH12YXIgYj1mdW5jdGlvbihlKXtyZXR1cm4gZnVuY3Rpb24odCxuLGkpe2lmKGl8fChpPXt9KSxcInN0cmluZ1wiPT10eXBlb2YgbilpZihcIiNkb2N1bWVudFwiPT09dC5ub2RlTmFtZXx8XCJIVE1MXCI9PT10Lm5vZGVOYW1lfHxcIkJPRFlcIj09PXQubm9kZU5hbWUpe3ZhciByPW47KG49YS5jcmVhdGVFbGVtZW50KFwiaHRtbFwiKSkuaW5uZXJIVE1MPXJ9ZWxzZSBuPWMobik7dmFyIHU9aS5nZXROb2RlS2V5fHx5LHM9aS5vbkJlZm9yZU5vZGVBZGRlZHx8bSxkPWkub25Ob2RlQWRkZWR8fG0sYj1pLm9uQmVmb3JlRWxVcGRhdGVkfHxtLGs9aS5vbkVsVXBkYXRlZHx8bSx3PWkub25CZWZvcmVOb2RlRGlzY2FyZGVkfHxtLEU9aS5vbk5vZGVEaXNjYXJkZWR8fG0sQT1pLm9uQmVmb3JlRWxDaGlsZHJlblVwZGF0ZWR8fG0sUz0hMD09PWkuY2hpbGRyZW5Pbmx5LHg9T2JqZWN0LmNyZWF0ZShudWxsKSxDPVtdO2Z1bmN0aW9uIFAoZSl7Qy5wdXNoKGUpfWZ1bmN0aW9uIEwoZSx0LG4peyExIT09dyhlKSYmKHQmJnQucmVtb3ZlQ2hpbGQoZSksRShlKSxmdW5jdGlvbiBlKHQsbil7aWYodC5ub2RlVHlwZT09PWYpZm9yKHZhciBpPXQuZmlyc3RDaGlsZDtpOyl7dmFyIHI9dm9pZCAwO24mJihyPXUoaSkpP1Aocik6KEUoaSksaS5maXJzdENoaWxkJiZlKGksbikpLGk9aS5uZXh0U2libGluZ319KGUsbikpfWZ1bmN0aW9uIEkoZSl7ZChlKTtmb3IodmFyIHQ9ZS5maXJzdENoaWxkO3Q7KXt2YXIgbj10Lm5leHRTaWJsaW5nLGk9dSh0KTtpZihpKXt2YXIgcj14W2ldO3ImJmwodCxyKT8odC5wYXJlbnROb2RlLnJlcGxhY2VDaGlsZChyLHQpLFQocix0KSk6SSh0KX1lbHNlIEkodCk7dD1ufX1mdW5jdGlvbiBUKHQsbixpKXt2YXIgcj11KG4pO2lmKHImJmRlbGV0ZSB4W3JdLCFpKXtpZighMT09PWIodCxuKSlyZXR1cm47aWYoZSh0LG4pLGsodCksITE9PT1BKHQsbikpcmV0dXJufVwiVEVYVEFSRUFcIiE9PXQubm9kZU5hbWU/ZnVuY3Rpb24oZSx0KXt2YXIgbixpLHIsbyxjLGQ9dC5maXJzdENoaWxkLHY9ZS5maXJzdENoaWxkO2U6Zm9yKDtkOyl7Zm9yKG89ZC5uZXh0U2libGluZyxuPXUoZCk7djspe2lmKHI9di5uZXh0U2libGluZyxkLmlzU2FtZU5vZGUmJmQuaXNTYW1lTm9kZSh2KSl7ZD1vLHY9cjtjb250aW51ZSBlfWk9dSh2KTt2YXIgbT12Lm5vZGVUeXBlLHk9dm9pZCAwO2lmKG09PT1kLm5vZGVUeXBlJiYobT09PWY/KG4/biE9PWkmJigoYz14W25dKT9yPT09Yz95PSExOihlLmluc2VydEJlZm9yZShjLHYpLGk/UChpKTpMKHYsZSwhMCksdj1jKTp5PSExKTppJiYoeT0hMSksKHk9ITEhPT15JiZsKHYsZCkpJiZUKHYsZCkpOm0hPT1wJiZtIT1nfHwoeT0hMCx2Lm5vZGVWYWx1ZSE9PWQubm9kZVZhbHVlJiYodi5ub2RlVmFsdWU9ZC5ub2RlVmFsdWUpKSkseSl7ZD1vLHY9cjtjb250aW51ZSBlfWk/UChpKTpMKHYsZSwhMCksdj1yfWlmKG4mJihjPXhbbl0pJiZsKGMsZCkpZS5hcHBlbmRDaGlsZChjKSxUKGMsZCk7ZWxzZXt2YXIgYj1zKGQpOyExIT09YiYmKGImJihkPWIpLGQuYWN0dWFsaXplJiYoZD1kLmFjdHVhbGl6ZShlLm93bmVyRG9jdW1lbnR8fGEpKSxlLmFwcGVuZENoaWxkKGQpLEkoZCkpfWQ9byx2PXJ9IWZ1bmN0aW9uKGUsdCxuKXtmb3IoO3Q7KXt2YXIgaT10Lm5leHRTaWJsaW5nOyhuPXUodCkpP1Aobik6TCh0LGUsITApLHQ9aX19KGUsdixpKTt2YXIgaz1oW2Uubm9kZU5hbWVdO2smJmsoZSx0KX0odCxuKTpoLlRFWFRBUkVBKHQsbil9IWZ1bmN0aW9uIGUodCl7aWYodC5ub2RlVHlwZT09PWZ8fHQubm9kZVR5cGU9PT12KWZvcih2YXIgbj10LmZpcnN0Q2hpbGQ7bjspe3ZhciBpPXUobik7aSYmKHhbaV09biksZShuKSxuPW4ubmV4dFNpYmxpbmd9fSh0KTt2YXIgRD10LF89RC5ub2RlVHlwZSxOPW4ubm9kZVR5cGU7aWYoIVMpaWYoXz09PWYpTj09PWY/bCh0LG4pfHwoRSh0KSxEPWZ1bmN0aW9uKGUsdCl7Zm9yKHZhciBuPWUuZmlyc3RDaGlsZDtuOyl7dmFyIGk9bi5uZXh0U2libGluZzt0LmFwcGVuZENoaWxkKG4pLG49aX1yZXR1cm4gdH0odCxmdW5jdGlvbihlLHQpe3JldHVybiB0JiZ0IT09bz9hLmNyZWF0ZUVsZW1lbnROUyh0LGUpOmEuY3JlYXRlRWxlbWVudChlKX0obi5ub2RlTmFtZSxuLm5hbWVzcGFjZVVSSSkpKTpEPW47ZWxzZSBpZihfPT09cHx8Xz09PWcpe2lmKE49PT1fKXJldHVybiBELm5vZGVWYWx1ZSE9PW4ubm9kZVZhbHVlJiYoRC5ub2RlVmFsdWU9bi5ub2RlVmFsdWUpLEQ7RD1ufWlmKEQ9PT1uKUUodCk7ZWxzZXtpZihuLmlzU2FtZU5vZGUmJm4uaXNTYW1lTm9kZShEKSlyZXR1cm47aWYoVChELG4sUyksQylmb3IodmFyIFI9MCxPPUMubGVuZ3RoO1I8TztSKyspe3ZhciBqPXhbQ1tSXV07aiYmTChqLGoucGFyZW50Tm9kZSwhMSl9fXJldHVybiFTJiZEIT09dCYmdC5wYXJlbnROb2RlJiYoRC5hY3R1YWxpemUmJihEPUQuYWN0dWFsaXplKHQub3duZXJEb2N1bWVudHx8YSkpLHQucGFyZW50Tm9kZS5yZXBsYWNlQ2hpbGQoRCx0KSksRH19KGZ1bmN0aW9uKGUsdCl7dmFyIG4saSxvLGEsdT10LmF0dHJpYnV0ZXM7aWYodC5ub2RlVHlwZSE9PXImJmUubm9kZVR5cGUhPT1yKXtmb3IodmFyIHM9dS5sZW5ndGgtMTtzPj0wO3MtLSlpPShuPXVbc10pLm5hbWUsbz1uLm5hbWVzcGFjZVVSSSxhPW4udmFsdWUsbz8oaT1uLmxvY2FsTmFtZXx8aSxlLmdldEF0dHJpYnV0ZU5TKG8saSkhPT1hJiYoXCJ4bWxuc1wiPT09bi5wcmVmaXgmJihpPW4ubmFtZSksZS5zZXRBdHRyaWJ1dGVOUyhvLGksYSkpKTplLmdldEF0dHJpYnV0ZShpKSE9PWEmJmUuc2V0QXR0cmlidXRlKGksYSk7Zm9yKHZhciBjPWUuYXR0cmlidXRlcyxsPWMubGVuZ3RoLTE7bD49MDtsLS0paT0obj1jW2xdKS5uYW1lLChvPW4ubmFtZXNwYWNlVVJJKT8oaT1uLmxvY2FsTmFtZXx8aSx0Lmhhc0F0dHJpYnV0ZU5TKG8saSl8fGUucmVtb3ZlQXR0cmlidXRlTlMobyxpKSk6dC5oYXNBdHRyaWJ1dGUoaSl8fGUucmVtb3ZlQXR0cmlidXRlKGkpfX0pO2Z1bmN0aW9uIGsoZSl7cmV0dXJuIFAoZSl8fFMoZSl8fEwoZSl8fEMoKX1mdW5jdGlvbiB3KGUsdCl7dmFyIG49T2JqZWN0LmtleXMoZSk7aWYoT2JqZWN0LmdldE93blByb3BlcnR5U3ltYm9scyl7dmFyIGk9T2JqZWN0LmdldE93blByb3BlcnR5U3ltYm9scyhlKTt0JiYoaT1pLmZpbHRlcihmdW5jdGlvbih0KXtyZXR1cm4gT2JqZWN0LmdldE93blByb3BlcnR5RGVzY3JpcHRvcihlLHQpLmVudW1lcmFibGV9KSksbi5wdXNoLmFwcGx5KG4saSl9cmV0dXJuIG59ZnVuY3Rpb24gRShlLHQsbil7cmV0dXJuIHQgaW4gZT9PYmplY3QuZGVmaW5lUHJvcGVydHkoZSx0LHt2YWx1ZTpuLGVudW1lcmFibGU6ITAsY29uZmlndXJhYmxlOiEwLHdyaXRhYmxlOiEwfSk6ZVt0XT1uLGV9ZnVuY3Rpb24gQShlKXtyZXR1cm4gZnVuY3Rpb24oZSl7aWYoQXJyYXkuaXNBcnJheShlKSlyZXR1cm4gSShlKX0oZSl8fFMoZSl8fEwoZSl8fGZ1bmN0aW9uKCl7dGhyb3cgbmV3IFR5cGVFcnJvcihcIkludmFsaWQgYXR0ZW1wdCB0byBzcHJlYWQgbm9uLWl0ZXJhYmxlIGluc3RhbmNlLlxcbkluIG9yZGVyIHRvIGJlIGl0ZXJhYmxlLCBub24tYXJyYXkgb2JqZWN0cyBtdXN0IGhhdmUgYSBbU3ltYm9sLml0ZXJhdG9yXSgpIG1ldGhvZC5cIil9KCl9ZnVuY3Rpb24gUyhlKXtpZihcInVuZGVmaW5lZFwiIT10eXBlb2YgU3ltYm9sJiZTeW1ib2wuaXRlcmF0b3IgaW4gT2JqZWN0KGUpKXJldHVybiBBcnJheS5mcm9tKGUpfWZ1bmN0aW9uIHgoZSx0KXtyZXR1cm4gUChlKXx8ZnVuY3Rpb24oZSx0KXtpZihcInVuZGVmaW5lZFwiPT10eXBlb2YgU3ltYm9sfHwhKFN5bWJvbC5pdGVyYXRvciBpbiBPYmplY3QoZSkpKXJldHVybjt2YXIgbj1bXSxpPSEwLHI9ITEsbz12b2lkIDA7dHJ5e2Zvcih2YXIgYSx1PWVbU3ltYm9sLml0ZXJhdG9yXSgpOyEoaT0oYT11Lm5leHQoKSkuZG9uZSkmJihuLnB1c2goYS52YWx1ZSksIXR8fG4ubGVuZ3RoIT09dCk7aT0hMCk7fWNhdGNoKGUpe3I9ITAsbz1lfWZpbmFsbHl7dHJ5e2l8fG51bGw9PXUucmV0dXJufHx1LnJldHVybigpfWZpbmFsbHl7aWYocil0aHJvdyBvfX1yZXR1cm4gbn0oZSx0KXx8TChlLHQpfHxDKCl9ZnVuY3Rpb24gQygpe3Rocm93IG5ldyBUeXBlRXJyb3IoXCJJbnZhbGlkIGF0dGVtcHQgdG8gZGVzdHJ1Y3R1cmUgbm9uLWl0ZXJhYmxlIGluc3RhbmNlLlxcbkluIG9yZGVyIHRvIGJlIGl0ZXJhYmxlLCBub24tYXJyYXkgb2JqZWN0cyBtdXN0IGhhdmUgYSBbU3ltYm9sLml0ZXJhdG9yXSgpIG1ldGhvZC5cIil9ZnVuY3Rpb24gUChlKXtpZihBcnJheS5pc0FycmF5KGUpKXJldHVybiBlfWZ1bmN0aW9uIEwoZSx0KXtpZihlKXtpZihcInN0cmluZ1wiPT10eXBlb2YgZSlyZXR1cm4gSShlLHQpO3ZhciBuPU9iamVjdC5wcm90b3R5cGUudG9TdHJpbmcuY2FsbChlKS5zbGljZSg4LC0xKTtyZXR1cm5cIk9iamVjdFwiPT09biYmZS5jb25zdHJ1Y3RvciYmKG49ZS5jb25zdHJ1Y3Rvci5uYW1lKSxcIk1hcFwiPT09bnx8XCJTZXRcIj09PW4/QXJyYXkuZnJvbShlKTpcIkFyZ3VtZW50c1wiPT09bnx8L14oPzpVaXxJKW50KD86OHwxNnwzMikoPzpDbGFtcGVkKT9BcnJheSQvLnRlc3Qobik/SShlLHQpOnZvaWQgMH19ZnVuY3Rpb24gSShlLHQpeyhudWxsPT10fHx0PmUubGVuZ3RoKSYmKHQ9ZS5sZW5ndGgpO2Zvcih2YXIgbj0wLGk9bmV3IEFycmF5KHQpO248dDtuKyspaVtuXT1lW25dO3JldHVybiBpfWZ1bmN0aW9uIFQoZSx0KXtpZighKGUgaW5zdGFuY2VvZiB0KSl0aHJvdyBuZXcgVHlwZUVycm9yKFwiQ2Fubm90IGNhbGwgYSBjbGFzcyBhcyBhIGZ1bmN0aW9uXCIpfWZ1bmN0aW9uIEQoZSx0KXtmb3IodmFyIG49MDtuPHQubGVuZ3RoO24rKyl7dmFyIGk9dFtuXTtpLmVudW1lcmFibGU9aS5lbnVtZXJhYmxlfHwhMSxpLmNvbmZpZ3VyYWJsZT0hMCxcInZhbHVlXCJpbiBpJiYoaS53cml0YWJsZT0hMCksT2JqZWN0LmRlZmluZVByb3BlcnR5KGUsaS5rZXksaSl9fWZ1bmN0aW9uIF8oZSx0LG4pe3JldHVybiB0JiZEKGUucHJvdG90eXBlLHQpLG4mJkQoZSxuKSxlfWZ1bmN0aW9uIE4oZSl7XCJAYmFiZWwvaGVscGVycyAtIHR5cGVvZlwiO3JldHVybihOPVwiZnVuY3Rpb25cIj09dHlwZW9mIFN5bWJvbCYmXCJzeW1ib2xcIj09dHlwZW9mIFN5bWJvbC5pdGVyYXRvcj9mdW5jdGlvbihlKXtyZXR1cm4gdHlwZW9mIGV9OmZ1bmN0aW9uKGUpe3JldHVybiBlJiZcImZ1bmN0aW9uXCI9PXR5cGVvZiBTeW1ib2wmJmUuY29uc3RydWN0b3I9PT1TeW1ib2wmJmUhPT1TeW1ib2wucHJvdG90eXBlP1wic3ltYm9sXCI6dHlwZW9mIGV9KShlKX1uLmQodCxcImRlYnVnXCIsZnVuY3Rpb24oKXtyZXR1cm4gWH0pLG4uZCh0LFwiUmVuZGVyZWRcIixmdW5jdGlvbigpe3JldHVybiBzZX0pLG4uZCh0LFwiTGl2ZVNvY2tldFwiLGZ1bmN0aW9uKCl7cmV0dXJuIGNlfSksbi5kKHQsXCJCcm93c2VyXCIsZnVuY3Rpb24oKXtyZXR1cm4gbGV9KSxuLmQodCxcIkRPTVwiLGZ1bmN0aW9uKCl7cmV0dXJuIGRlfSksbi5kKHQsXCJWaWV3XCIsZnVuY3Rpb24oKXtyZXR1cm4gdmV9KTt2YXIgUj1bMWUzLDNlM10sTz1cImRhdGEtcGh4LXZpZXdcIixqPVtcInBoeC1jbGljay1sb2FkaW5nXCIsXCJwaHgtY2hhbmdlLWxvYWRpbmdcIixcInBoeC1zdWJtaXQtbG9hZGluZ1wiLFwicGh4LWtleWRvd24tbG9hZGluZ1wiLFwicGh4LWtleXVwLWxvYWRpbmdcIixcInBoeC1ibHVyLWxvYWRpbmdcIixcInBoeC1mb2N1cy1sb2FkaW5nXCJdLEg9XCJkYXRhLXBoeC1jb21wb25lbnRcIixGPVwiZGF0YS1waHgtcmVmXCIsTT1cImRhdGEtcGh4LXVwbG9hZC1yZWZcIixVPVwiW1wiLmNvbmNhdChPLFwiXVwiKSxCPVtcInRleHRcIixcInRleHRhcmVhXCIsXCJudW1iZXJcIixcImVtYWlsXCIsXCJwYXNzd29yZFwiLFwic2VhcmNoXCIsXCJ0ZWxcIixcInVybFwiLFwiZGF0ZVwiLFwidGltZVwiXSxKPVtcImNoZWNrYm94XCIsXCJyYWRpb1wiXSxWPVwiZGF0YS1waHgtc3RhdGljXCIsVz0xLHE9XCJwaHgtXCIsej17ZGVib3VuY2U6MzAwLHRocm90dGxlOjMwMH0sSz1mdW5jdGlvbihlLHQpe3JldHVybiBjb25zb2xlLmVycm9yJiZjb25zb2xlLmVycm9yKGUsdCl9O3ZhciBYPWZ1bmN0aW9uKGUsdCxuLGkpe2UubGl2ZVNvY2tldC5pc0RlYnVnRW5hYmxlZCgpJiZjb25zb2xlLmxvZyhcIlwiLmNvbmNhdChlLmlkLFwiIFwiKS5jb25jYXQodCxcIjogXCIpLmNvbmNhdChuLFwiIC0gXCIpLGkpfSwkPWZ1bmN0aW9uKGUpe3JldHVyblwiZnVuY3Rpb25cIj09dHlwZW9mIGU/ZTpmdW5jdGlvbigpe3JldHVybiBlfX0sRz1mdW5jdGlvbihlKXtyZXR1cm4gSlNPTi5wYXJzZShKU09OLnN0cmluZ2lmeShlKSl9LFk9ZnVuY3Rpb24oZSx0LG4pe2Rve2lmKGUubWF0Y2hlcyhcIltcIi5jb25jYXQodCxcIl1cIikpKXJldHVybiBlO2U9ZS5wYXJlbnRFbGVtZW50fHxlLnBhcmVudE5vZGV9d2hpbGUobnVsbCE9PWUmJjE9PT1lLm5vZGVUeXBlJiYhKG4mJm4uaXNTYW1lTm9kZShlKXx8ZS5tYXRjaGVzKFUpKSk7cmV0dXJuIG51bGx9LFE9ZnVuY3Rpb24oZSl7cmV0dXJuIG51bGwhPT1lJiZcIm9iamVjdFwiPT09TihlKSYmIShlIGluc3RhbmNlb2YgQXJyYXkpfSxaPWZ1bmN0aW9uKGUpe2Zvcih2YXIgdCBpbiBlKXJldHVybiExO3JldHVybiEwfSxlZT1mdW5jdGlvbihlLHQpe3JldHVybiBlJiZ0KGUpfSx0ZT1mdW5jdGlvbigpe2Z1bmN0aW9uIGUodCxuLGkpe1QodGhpcyxlKSx0aGlzLnJlZj1yZS5nZW5GaWxlUmVmKG4pLHRoaXMuZmlsZUVsPXQsdGhpcy5maWxlPW4sdGhpcy52aWV3PWksdGhpcy5tZXRhPW51bGwsdGhpcy5faXNDYW5jZWxsZWQ9ITEsdGhpcy5faXNEb25lPSExLHRoaXMuX3Byb2dyZXNzPTAsdGhpcy5fb25Eb25lPWZ1bmN0aW9uKCl7fX1yZXR1cm4gXyhlLG51bGwsW3trZXk6XCJpc0FjdGl2ZVwiLHZhbHVlOmZ1bmN0aW9uKGUsdCl7dmFyIG49dm9pZCAwPT09dC5fcGh4UmVmLGk9ZS5nZXRBdHRyaWJ1dGUoXCJkYXRhLXBoeC1hY3RpdmUtcmVmc1wiKS5zcGxpdChcIixcIikuaW5kZXhPZihyZS5nZW5GaWxlUmVmKHQpKT49MDtyZXR1cm4gdC5zaXplPjAmJihufHxpKX19LHtrZXk6XCJpc1ByZWZsaWdodGVkXCIsdmFsdWU6ZnVuY3Rpb24oZSx0KXt2YXIgbj1lLmdldEF0dHJpYnV0ZShcImRhdGEtcGh4LXByZWZsaWdodGVkLXJlZnNcIikuc3BsaXQoXCIsXCIpLmluZGV4T2YocmUuZ2VuRmlsZVJlZih0KSk+PTA7cmV0dXJuIG4mJnRoaXMuaXNBY3RpdmUoZSx0KX19XSksXyhlLFt7a2V5OlwibWV0YWRhdGFcIix2YWx1ZTpmdW5jdGlvbigpe3JldHVybiB0aGlzLm1ldGF9fSx7a2V5OlwicHJvZ3Jlc3NcIix2YWx1ZTpmdW5jdGlvbihlKXt2YXIgdD10aGlzO3RoaXMuX3Byb2dyZXNzPU1hdGguZmxvb3IoZSksdGhpcy5fcHJvZ3Jlc3M+PTEwMD8odGhpcy5fcHJvZ3Jlc3M9MTAwLHRoaXMuX2lzRG9uZT0hMCx0aGlzLnZpZXcucHVzaEZpbGVQcm9ncmVzcyh0aGlzLmZpbGVFbCx0aGlzLnJlZiwxMDAsZnVuY3Rpb24oKXtyZS51bnRyYWNrRmlsZSh0LmZpbGVFbCx0LmZpbGUpLHQuX29uRG9uZSgpfSkpOnRoaXMudmlldy5wdXNoRmlsZVByb2dyZXNzKHRoaXMuZmlsZUVsLHRoaXMucmVmLHRoaXMuX3Byb2dyZXNzKX19LHtrZXk6XCJjYW5jZWxcIix2YWx1ZTpmdW5jdGlvbigpe3RoaXMuX2lzQ2FuY2VsbGVkPSEwLHRoaXMuX2lzRG9uZT0hMCx0aGlzLl9vbkRvbmUoKX19LHtrZXk6XCJpc0RvbmVcIix2YWx1ZTpmdW5jdGlvbigpe3JldHVybiB0aGlzLl9pc0RvbmV9fSx7a2V5OlwiZXJyb3JcIix2YWx1ZTpmdW5jdGlvbigpe3ZhciBlPWFyZ3VtZW50cy5sZW5ndGg+MCYmdm9pZCAwIT09YXJndW1lbnRzWzBdP2FyZ3VtZW50c1swXTpcImZhaWxlZFwiO3RoaXMudmlldy5wdXNoRmlsZVByb2dyZXNzKHRoaXMuZmlsZUVsLHRoaXMucmVmLHtlcnJvcjplfSl9fSx7a2V5Olwib25Eb25lXCIsdmFsdWU6ZnVuY3Rpb24oZSl7dGhpcy5fb25Eb25lPWV9fSx7a2V5OlwidG9QcmVmbGlnaHRQYXlsb2FkXCIsdmFsdWU6ZnVuY3Rpb24oKXtyZXR1cm57bGFzdF9tb2RpZmllZDp0aGlzLmZpbGUubGFzdE1vZGlmaWVkLG5hbWU6dGhpcy5maWxlLm5hbWUsc2l6ZTp0aGlzLmZpbGUuc2l6ZSx0eXBlOnRoaXMuZmlsZS50eXBlLHJlZjp0aGlzLnJlZn19fSx7a2V5OlwidXBsb2FkZXJcIix2YWx1ZTpmdW5jdGlvbihlKXtpZih0aGlzLm1ldGEudXBsb2FkZXIpe3ZhciB0PWVbdGhpcy5tZXRhLnVwbG9hZGVyXXx8SyhcIm5vIHVwbG9hZGVyIGNvbmZpZ3VyZWQgZm9yIFwiLmNvbmNhdCh0aGlzLm1ldGEudXBsb2FkZXIpKTtyZXR1cm57bmFtZTp0aGlzLm1ldGEudXBsb2FkZXIsY2FsbGJhY2s6dH19cmV0dXJue25hbWU6XCJjaGFubmVsXCIsY2FsbGJhY2s6b2V9fX0se2tleTpcInppcFBvc3RGbGlnaHRcIix2YWx1ZTpmdW5jdGlvbihlKXt0aGlzLm1ldGE9ZS5lbnRyaWVzW3RoaXMucmVmXSx0aGlzLm1ldGF8fEsoXCJubyBwcmVmbGlnaHQgdXBsb2FkIHJlc3BvbnNlIHJldHVybmVkIHdpdGggcmVmIFwiLmNvbmNhdCh0aGlzLnJlZikse2lucHV0OnRoaXMuZmlsZUVsLHJlc3BvbnNlOmV9KX19XSksZX0oKSxuZT17TGl2ZUZpbGVVcGxvYWQ6e3ByZWZsaWdodGVkUmVmczpmdW5jdGlvbigpe3JldHVybiB0aGlzLmVsLmdldEF0dHJpYnV0ZShcImRhdGEtcGh4LXByZWZsaWdodGVkLXJlZnNcIil9LG1vdW50ZWQ6ZnVuY3Rpb24oKXt0aGlzLnByZWZsaWdodGVkV2FzPXRoaXMucHJlZmxpZ2h0ZWRSZWZzKCl9LHVwZGF0ZWQ6ZnVuY3Rpb24oKXt2YXIgZT10aGlzLnByZWZsaWdodGVkUmVmcygpO3RoaXMucHJlZmxpZ2h0ZWRXYXMhPT1lJiYodGhpcy5wcmVmbGlnaHRlZFdhcz1lLFwiXCI9PT1lJiZ0aGlzLl9fdmlldy5jYW5jZWxTdWJtaXQodGhpcy5lbC5mb3JtKSl9fX07bmUuTGl2ZUltZ1ByZXZpZXc9e21vdW50ZWQ6ZnVuY3Rpb24oKXt2YXIgZT10aGlzO3RoaXMucmVmPXRoaXMuZWwuZ2V0QXR0cmlidXRlKFwiZGF0YS1waHgtZW50cnktcmVmXCIpLHRoaXMuaW5wdXRFbD1kb2N1bWVudC5nZXRFbGVtZW50QnlJZCh0aGlzLmVsLmdldEF0dHJpYnV0ZShNKSkscmUuZ2V0RW50cnlEYXRhVVJMKHRoaXMuaW5wdXRFbCx0aGlzLnJlZixmdW5jdGlvbih0KXtyZXR1cm4gZS5lbC5zcmM9dH0pfX07dmFyIGllPTAscmU9ZnVuY3Rpb24oKXtmdW5jdGlvbiBlKHQsbixpKXtUKHRoaXMsZSksdGhpcy52aWV3PW4sdGhpcy5vbkNvbXBsZXRlPWksdGhpcy5fZW50cmllcz1BcnJheS5mcm9tKGUuZmlsZXNBd2FpdGluZ1ByZWZsaWdodCh0KXx8W10pLm1hcChmdW5jdGlvbihlKXtyZXR1cm4gbmV3IHRlKHQsZSxuKX0pLHRoaXMubnVtRW50cmllc0luUHJvZ3Jlc3M9dGhpcy5fZW50cmllcy5sZW5ndGh9cmV0dXJuIF8oZSxudWxsLFt7a2V5OlwiZ2VuRmlsZVJlZlwiLHZhbHVlOmZ1bmN0aW9uKGUpe3ZhciB0PWUuX3BoeFJlZjtyZXR1cm4gdm9pZCAwIT09dD90OihlLl9waHhSZWY9KGllKyspLnRvU3RyaW5nKCksZS5fcGh4UmVmKX19LHtrZXk6XCJnZXRFbnRyeURhdGFVUkxcIix2YWx1ZTpmdW5jdGlvbihlLHQsbil7dmFyIGk9dGhpcyxyPXRoaXMuYWN0aXZlRmlsZXMoZSkuZmluZChmdW5jdGlvbihlKXtyZXR1cm4gaS5nZW5GaWxlUmVmKGUpPT09dH0pLG89bmV3IEZpbGVSZWFkZXI7by5vbmxvYWQ9ZnVuY3Rpb24oZSl7cmV0dXJuIG4oZS50YXJnZXQucmVzdWx0KX0sby5yZWFkQXNEYXRhVVJMKHIpfX0se2tleTpcImhhc1VwbG9hZHNJblByb2dyZXNzXCIsdmFsdWU6ZnVuY3Rpb24oZSl7dmFyIHQ9MDtyZXR1cm4gZGUuZmluZFVwbG9hZElucHV0cyhlKS5mb3JFYWNoKGZ1bmN0aW9uKGUpe2UuZ2V0QXR0cmlidXRlKFwiZGF0YS1waHgtcHJlZmxpZ2h0ZWQtcmVmc1wiKSE9PWUuZ2V0QXR0cmlidXRlKFwiZGF0YS1waHgtZG9uZS1yZWZzXCIpJiZ0Kyt9KSx0PjB9fSx7a2V5Olwic2VyaWFsaXplVXBsb2Fkc1wiLHZhbHVlOmZ1bmN0aW9uKGUpe3ZhciB0PXRoaXMsbj17fTtyZXR1cm4gdGhpcy5hY3RpdmVGaWxlcyhlLFwic2VyaWFsaXplXCIpLmZvckVhY2goZnVuY3Rpb24oaSl7dmFyIHI9e3BhdGg6ZS5uYW1lfSxvPWUuZ2V0QXR0cmlidXRlKE0pO25bb109bltvXXx8W10sci5yZWY9dC5nZW5GaWxlUmVmKGkpLHIubmFtZT1pLm5hbWUsci50eXBlPWkudHlwZSxyLnNpemU9aS5zaXplLG5bb10ucHVzaChyKX0pLG59fSx7a2V5OlwiY2xlYXJGaWxlc1wiLHZhbHVlOmZ1bmN0aW9uKGUpe2UudmFsdWU9bnVsbCxkZS5wdXRQcml2YXRlKGUsXCJmaWxlc1wiLFtdKX19LHtrZXk6XCJ1bnRyYWNrRmlsZVwiLHZhbHVlOmZ1bmN0aW9uKGUsdCl7ZGUucHV0UHJpdmF0ZShlLFwiZmlsZXNcIixkZS5wcml2YXRlKGUsXCJmaWxlc1wiKS5maWx0ZXIoZnVuY3Rpb24oZSl7cmV0dXJuIU9iamVjdC5pcyhlLHQpfSkpfX0se2tleTpcInRyYWNrRmlsZXNcIix2YWx1ZTpmdW5jdGlvbihlLHQpe3ZhciBuPXRoaXM7aWYobnVsbCE9PWUuZ2V0QXR0cmlidXRlKFwibXVsdGlwbGVcIikpe3ZhciBpPXQuZmlsdGVyKGZ1bmN0aW9uKHQpe3JldHVybiFuLmFjdGl2ZUZpbGVzKGUpLmZpbmQoZnVuY3Rpb24oZSl7cmV0dXJuIE9iamVjdC5pcyhlLHQpfSl9KTtkZS5wdXRQcml2YXRlKGUsXCJmaWxlc1wiLHRoaXMuYWN0aXZlRmlsZXMoZSkuY29uY2F0KGkpKSxlLnZhbHVlPW51bGx9ZWxzZSBkZS5wdXRQcml2YXRlKGUsXCJmaWxlc1wiLHQpfX0se2tleTpcImFjdGl2ZUZpbGVJbnB1dHNcIix2YWx1ZTpmdW5jdGlvbihlKXt2YXIgdD10aGlzLG49ZGUuZmluZFVwbG9hZElucHV0cyhlKTtyZXR1cm4gQXJyYXkuZnJvbShuKS5maWx0ZXIoZnVuY3Rpb24oZSl7cmV0dXJuIGUuZmlsZXMmJnQuYWN0aXZlRmlsZXMoZSkubGVuZ3RoPjB9KX19LHtrZXk6XCJhY3RpdmVGaWxlc1wiLHZhbHVlOmZ1bmN0aW9uKGUpe3JldHVybihkZS5wcml2YXRlKGUsXCJmaWxlc1wiKXx8W10pLmZpbHRlcihmdW5jdGlvbih0KXtyZXR1cm4gdGUuaXNBY3RpdmUoZSx0KX0pfX0se2tleTpcImlucHV0c0F3YWl0aW5nUHJlZmxpZ2h0XCIsdmFsdWU6ZnVuY3Rpb24oZSl7dmFyIHQ9dGhpcyxuPWRlLmZpbmRVcGxvYWRJbnB1dHMoZSk7cmV0dXJuIEFycmF5LmZyb20obikuZmlsdGVyKGZ1bmN0aW9uKGUpe3JldHVybiB0LmZpbGVzQXdhaXRpbmdQcmVmbGlnaHQoZSkubGVuZ3RoPjB9KX19LHtrZXk6XCJmaWxlc0F3YWl0aW5nUHJlZmxpZ2h0XCIsdmFsdWU6ZnVuY3Rpb24oZSl7cmV0dXJuIHRoaXMuYWN0aXZlRmlsZXMoZSkuZmlsdGVyKGZ1bmN0aW9uKHQpe3JldHVybiF0ZS5pc1ByZWZsaWdodGVkKGUsdCl9KX19XSksXyhlLFt7a2V5OlwiZW50cmllc1wiLHZhbHVlOmZ1bmN0aW9uKCl7cmV0dXJuIHRoaXMuX2VudHJpZXN9fSx7a2V5OlwiaW5pdEFkYXB0ZXJVcGxvYWRcIix2YWx1ZTpmdW5jdGlvbihlLHQsbil7dmFyIGk9dGhpczt0aGlzLl9lbnRyaWVzPXRoaXMuX2VudHJpZXMubWFwKGZ1bmN0aW9uKHQpe3JldHVybiB0LnppcFBvc3RGbGlnaHQoZSksdC5vbkRvbmUoZnVuY3Rpb24oKXtpLm51bUVudHJpZXNJblByb2dyZXNzLS0sMD09PWkubnVtRW50cmllc0luUHJvZ3Jlc3MmJmkub25Db21wbGV0ZSgpfSksdH0pO3ZhciByPXRoaXMuX2VudHJpZXMucmVkdWNlKGZ1bmN0aW9uKGUsdCl7dmFyIGk9dC51cGxvYWRlcihuLnVwbG9hZGVycykscj1pLm5hbWUsbz1pLmNhbGxiYWNrO3JldHVybiBlW3JdPWVbcl18fHtjYWxsYmFjazpvLGVudHJpZXM6W119LGVbcl0uZW50cmllcy5wdXNoKHQpLGV9LHt9KTtmb3IodmFyIG8gaW4gcil7dmFyIGE9cltvXTsoMCxhLmNhbGxiYWNrKShhLmVudHJpZXMsdCxlLG4pfX19XSksZX0oKSxvZT1mdW5jdGlvbihlLHQsbixpKXtlLmZvckVhY2goZnVuY3Rpb24oZSl7bmV3IGFlKGUsbi5jb25maWcuY2h1bmtfc2l6ZSxpKS51cGxvYWQoKX0pfSxhZT1mdW5jdGlvbigpe2Z1bmN0aW9uIGUodCxuLGkpe1QodGhpcyxlKSx0aGlzLmxpdmVTb2NrZXQ9aSx0aGlzLmVudHJ5PXQsdGhpcy5vZmZzZXQ9MCx0aGlzLmNodW5rU2l6ZT1uLHRoaXMudXBsb2FkQ2hhbm5lbD1pLmNoYW5uZWwoXCJsdnU6XCIuY29uY2F0KHQucmVmKSx7dG9rZW46dC5tZXRhZGF0YSgpfSl9cmV0dXJuIF8oZSxbe2tleTpcInVwbG9hZFwiLHZhbHVlOmZ1bmN0aW9uKCl7dmFyIGU9dGhpczt0aGlzLnVwbG9hZENoYW5uZWwuam9pbigpLnJlY2VpdmUoXCJva1wiLGZ1bmN0aW9uKHQpe3JldHVybiBlLnJlYWROZXh0Q2h1bmsoKX0pLnJlY2VpdmUoXCJlcnJvclwiLGZ1bmN0aW9uKHQpe2UudXBsb2FkQ2hhbm5lbC5sZWF2ZSgpLGUuZW50cnkuZXJyb3IoKX0pfX0se2tleTpcImlzRG9uZVwiLHZhbHVlOmZ1bmN0aW9uKCl7cmV0dXJuIHRoaXMub2Zmc2V0Pj10aGlzLmVudHJ5LmZpbGUuc2l6ZX19LHtrZXk6XCJyZWFkTmV4dENodW5rXCIsdmFsdWU6ZnVuY3Rpb24oKXt2YXIgZT10aGlzLHQ9bmV3IHdpbmRvdy5GaWxlUmVhZGVyLG49dGhpcy5lbnRyeS5maWxlLnNsaWNlKHRoaXMub2Zmc2V0LHRoaXMuY2h1bmtTaXplK3RoaXMub2Zmc2V0KTt0Lm9ubG9hZD1mdW5jdGlvbih0KXtpZihudWxsIT09dC50YXJnZXQuZXJyb3IpcmV0dXJuIEsoXCJSZWFkIGVycm9yOiBcIit0LnRhcmdldC5lcnJvcik7ZS5vZmZzZXQrPXQudGFyZ2V0LnJlc3VsdC5ieXRlTGVuZ3RoLGUucHVzaENodW5rKHQudGFyZ2V0LnJlc3VsdCl9LHQucmVhZEFzQXJyYXlCdWZmZXIobil9fSx7a2V5OlwicHVzaENodW5rXCIsdmFsdWU6ZnVuY3Rpb24oZSl7dmFyIHQ9dGhpczt0aGlzLnVwbG9hZENoYW5uZWwuaXNKb2luZWQoKSYmdGhpcy51cGxvYWRDaGFubmVsLnB1c2goXCJjaHVua1wiLGUpLnJlY2VpdmUoXCJva1wiLGZ1bmN0aW9uKCl7dC5lbnRyeS5wcm9ncmVzcyh0Lm9mZnNldC90LmVudHJ5LmZpbGUuc2l6ZSoxMDApLHQuaXNEb25lKCl8fHNldFRpbWVvdXQoZnVuY3Rpb24oKXtyZXR1cm4gdC5yZWFkTmV4dENodW5rKCl9LHQubGl2ZVNvY2tldC5nZXRMYXRlbmN5U2ltKCl8fDApfSl9fV0pLGV9KCksdWU9ZnVuY3Rpb24oZSl7dmFyIHQ9YXJndW1lbnRzLmxlbmd0aD4xJiZ2b2lkIDAhPT1hcmd1bWVudHNbMV0/YXJndW1lbnRzWzFdOnt9LG49bmV3IEZvcm1EYXRhKGUpLGk9W107bi5mb3JFYWNoKGZ1bmN0aW9uKGUsdCxuKXtlIGluc3RhbmNlb2YgRmlsZSYmaS5wdXNoKHQpfSksaS5mb3JFYWNoKGZ1bmN0aW9uKGUpe3JldHVybiBuLmRlbGV0ZShlKX0pO3ZhciByLG89bmV3IFVSTFNlYXJjaFBhcmFtcyxhPWZ1bmN0aW9uKGUpe2lmKFwidW5kZWZpbmVkXCI9PXR5cGVvZiBTeW1ib2x8fG51bGw9PWVbU3ltYm9sLml0ZXJhdG9yXSl7aWYoQXJyYXkuaXNBcnJheShlKXx8KGU9TChlKSkpe3ZhciB0PTAsbj1mdW5jdGlvbigpe307cmV0dXJue3M6bixuOmZ1bmN0aW9uKCl7cmV0dXJuIHQ+PWUubGVuZ3RoP3tkb25lOiEwfTp7ZG9uZTohMSx2YWx1ZTplW3QrK119fSxlOmZ1bmN0aW9uKGUpe3Rocm93IGV9LGY6bn19dGhyb3cgbmV3IFR5cGVFcnJvcihcIkludmFsaWQgYXR0ZW1wdCB0byBpdGVyYXRlIG5vbi1pdGVyYWJsZSBpbnN0YW5jZS5cXG5JbiBvcmRlciB0byBiZSBpdGVyYWJsZSwgbm9uLWFycmF5IG9iamVjdHMgbXVzdCBoYXZlIGEgW1N5bWJvbC5pdGVyYXRvcl0oKSBtZXRob2QuXCIpfXZhciBpLHIsbz0hMCxhPSExO3JldHVybntzOmZ1bmN0aW9uKCl7aT1lW1N5bWJvbC5pdGVyYXRvcl0oKX0sbjpmdW5jdGlvbigpe3ZhciBlPWkubmV4dCgpO3JldHVybiBvPWUuZG9uZSxlfSxlOmZ1bmN0aW9uKGUpe2E9ITAscj1lfSxmOmZ1bmN0aW9uKCl7dHJ5e298fG51bGw9PWkucmV0dXJufHxpLnJldHVybigpfWZpbmFsbHl7aWYoYSl0aHJvdyByfX19fShuLmVudHJpZXMoKSk7dHJ5e2ZvcihhLnMoKTshKHI9YS5uKCkpLmRvbmU7KXt2YXIgdT14KHIudmFsdWUsMikscz11WzBdLGM9dVsxXTtvLmFwcGVuZChzLGMpfX1jYXRjaChlKXthLmUoZSl9ZmluYWxseXthLmYoKX1mb3IodmFyIGwgaW4gdClvLmFwcGVuZChsLHRbbF0pO3JldHVybiBvLnRvU3RyaW5nKCl9LHNlPWZ1bmN0aW9uKCl7ZnVuY3Rpb24gZSh0LG4pe1QodGhpcyxlKSx0aGlzLnZpZXdJZD10LHRoaXMucmVuZGVyZWQ9e30sdGhpcy5tZXJnZURpZmYobil9cmV0dXJuIF8oZSxudWxsLFt7a2V5OlwiZXh0cmFjdFwiLHZhbHVlOmZ1bmN0aW9uKGUpe3ZhciB0PWUucixuPWUuZSxpPWUudDtyZXR1cm4gZGVsZXRlIGUucixkZWxldGUgZS5lLGRlbGV0ZSBlLnQse2RpZmY6ZSx0aXRsZTppLHJlcGx5OnR8fG51bGwsZXZlbnRzOm58fFtdfX19XSksXyhlLFt7a2V5OlwicGFyZW50Vmlld0lkXCIsdmFsdWU6ZnVuY3Rpb24oKXtyZXR1cm4gdGhpcy52aWV3SWR9fSx7a2V5OlwidG9TdHJpbmdcIix2YWx1ZTpmdW5jdGlvbihlKXtyZXR1cm4gdGhpcy5yZWN1cnNpdmVUb1N0cmluZyh0aGlzLnJlbmRlcmVkLHRoaXMucmVuZGVyZWQuYyxlKX19LHtrZXk6XCJyZWN1cnNpdmVUb1N0cmluZ1wiLHZhbHVlOmZ1bmN0aW9uKGUpe3ZhciB0PWFyZ3VtZW50cy5sZW5ndGg+MSYmdm9pZCAwIT09YXJndW1lbnRzWzFdP2FyZ3VtZW50c1sxXTplLmMsbj1hcmd1bWVudHMubGVuZ3RoPjI/YXJndW1lbnRzWzJdOnZvaWQgMCxpPXtidWZmZXI6XCJcIixjb21wb25lbnRzOnQsb25seUNpZHM6bj1uP25ldyBTZXQobik6bnVsbH07cmV0dXJuIHRoaXMudG9PdXRwdXRCdWZmZXIoZSxpKSxpLmJ1ZmZlcn19LHtrZXk6XCJjb21wb25lbnRDSURzXCIsdmFsdWU6ZnVuY3Rpb24oZSl7cmV0dXJuIE9iamVjdC5rZXlzKGUuY3x8e30pLm1hcChmdW5jdGlvbihlKXtyZXR1cm4gcGFyc2VJbnQoZSl9KX19LHtrZXk6XCJpc0NvbXBvbmVudE9ubHlEaWZmXCIsdmFsdWU6ZnVuY3Rpb24oZSl7cmV0dXJuISFlLmMmJjE9PT1PYmplY3Qua2V5cyhlKS5sZW5ndGh9fSx7a2V5OlwiZ2V0Q29tcG9uZW50XCIsdmFsdWU6ZnVuY3Rpb24oZSx0KXtyZXR1cm4gZS5jW3RdfX0se2tleTpcIm1lcmdlRGlmZlwiLHZhbHVlOmZ1bmN0aW9uKGUpe3ZhciB0PWUuYztpZihkZWxldGUgZS5jLHRoaXMucmVuZGVyZWQ9dGhpcy5yZWN1cnNpdmVNZXJnZSh0aGlzLnJlbmRlcmVkLGUpLHRoaXMucmVuZGVyZWQuYz10aGlzLnJlbmRlcmVkLmN8fHt9LHQpe3ZhciBuPXRoaXMucmVuZGVyZWQuYztmb3IodmFyIGkgaW4gdCl7dmFyIHI9dFtpXSxvPXIsYT1vLnM7aWYoXCJudW1iZXJcIj09dHlwZW9mIGEpe2Zvcig7XCJudW1iZXJcIj09dHlwZW9mIGE7KWE9KG89YT4wP3RbYV06blstYV0pLnM7bz1HKG8pLHRoaXMuZG9SZWN1cnNpdmVNZXJnZShvLHIpLG8ucz1hfWVsc2Ugbz1uW2ldfHx7fSxvPXRoaXMucmVjdXJzaXZlTWVyZ2UobyxyKTt0W2ldPW99Zm9yKHZhciB1IGluIHQpblt1XT10W3VdO2UuYz10fX19LHtrZXk6XCJyZWN1cnNpdmVNZXJnZVwiLHZhbHVlOmZ1bmN0aW9uKGUsdCl7cmV0dXJuIHZvaWQgMCE9PXQucz90Oih0aGlzLmRvUmVjdXJzaXZlTWVyZ2UoZSx0KSxlKX19LHtrZXk6XCJkb1JlY3Vyc2l2ZU1lcmdlXCIsdmFsdWU6ZnVuY3Rpb24oZSx0KXtmb3IodmFyIG4gaW4gdCl7dmFyIGk9dFtuXSxyPWVbbl07UShpKSYmdm9pZCAwPT09aS5zJiZRKHIpP3RoaXMuZG9SZWN1cnNpdmVNZXJnZShyLGkpOmVbbl09aX19fSx7a2V5OlwiY29tcG9uZW50VG9TdHJpbmdcIix2YWx1ZTpmdW5jdGlvbihlKXtyZXR1cm4gdGhpcy5yZWN1cnNpdmVDSURUb1N0cmluZyh0aGlzLnJlbmRlcmVkLmMsZSl9fSx7a2V5OlwicHJ1bmVDSURzXCIsdmFsdWU6ZnVuY3Rpb24oZSl7dmFyIHQ9dGhpcztlLmZvckVhY2goZnVuY3Rpb24oZSl7cmV0dXJuIGRlbGV0ZSB0LnJlbmRlcmVkLmNbZV19KX19LHtrZXk6XCJnZXRcIix2YWx1ZTpmdW5jdGlvbigpe3JldHVybiB0aGlzLnJlbmRlcmVkfX0se2tleTpcImlzTmV3RmluZ2VycHJpbnRcIix2YWx1ZTpmdW5jdGlvbigpe3JldHVybiEhKGFyZ3VtZW50cy5sZW5ndGg+MCYmdm9pZCAwIT09YXJndW1lbnRzWzBdP2FyZ3VtZW50c1swXTp7fSkuc319LHtrZXk6XCJ0b091dHB1dEJ1ZmZlclwiLHZhbHVlOmZ1bmN0aW9uKGUsdCl7aWYoZS5kKXJldHVybiB0aGlzLmNvbXByZWhlbnNpb25Ub0J1ZmZlcihlLHQpO3ZhciBuPWUuczt0LmJ1ZmZlcis9blswXTtmb3IodmFyIGk9MTtpPG4ubGVuZ3RoO2krKyl0aGlzLmR5bmFtaWNUb0J1ZmZlcihlW2ktMV0sdCksdC5idWZmZXIrPW5baV19fSx7a2V5OlwiY29tcHJlaGVuc2lvblRvQnVmZmVyXCIsdmFsdWU6ZnVuY3Rpb24oZSx0KXtmb3IodmFyIG49ZS5kLGk9ZS5zLHI9MDtyPG4ubGVuZ3RoO3IrKyl7dmFyIG89bltyXTt0LmJ1ZmZlcis9aVswXTtmb3IodmFyIGE9MTthPGkubGVuZ3RoO2ErKyl0aGlzLmR5bmFtaWNUb0J1ZmZlcihvW2EtMV0sdCksdC5idWZmZXIrPWlbYV19fX0se2tleTpcImR5bmFtaWNUb0J1ZmZlclwiLHZhbHVlOmZ1bmN0aW9uKGUsdCl7XCJudW1iZXJcIj09dHlwZW9mIGU/dC5idWZmZXIrPXRoaXMucmVjdXJzaXZlQ0lEVG9TdHJpbmcodC5jb21wb25lbnRzLGUsdC5vbmx5Q2lkcyk6UShlKT90aGlzLnRvT3V0cHV0QnVmZmVyKGUsdCk6dC5idWZmZXIrPWV9fSx7a2V5OlwicmVjdXJzaXZlQ0lEVG9TdHJpbmdcIix2YWx1ZTpmdW5jdGlvbihlLHQsbil7dmFyIGk9dGhpcyxyPWVbdF18fEsoXCJubyBjb21wb25lbnQgZm9yIENJRCBcIi5jb25jYXQodCksZSksbz1kb2N1bWVudC5jcmVhdGVFbGVtZW50KFwidGVtcGxhdGVcIik7by5pbm5lckhUTUw9dGhpcy5yZWN1cnNpdmVUb1N0cmluZyhyLGUsbik7dmFyIGE9by5jb250ZW50LHU9biYmIW4uaGFzKHQpLHM9eChBcnJheS5mcm9tKGEuY2hpbGROb2RlcykucmVkdWNlKGZ1bmN0aW9uKGUsbixyKXt2YXIgYT14KGUsMikscz1hWzBdLGM9YVsxXTtyZXR1cm4gbi5ub2RlVHlwZT09PU5vZGUuRUxFTUVOVF9OT0RFP24uZ2V0QXR0cmlidXRlKEgpP1tzLCEwXToobi5zZXRBdHRyaWJ1dGUoSCx0KSxuLmlkfHwobi5pZD1cIlwiLmNvbmNhdChpLnBhcmVudFZpZXdJZCgpLFwiLVwiKS5jb25jYXQodCxcIi1cIikuY29uY2F0KHIpKSx1JiYobi5zZXRBdHRyaWJ1dGUoXCJkYXRhLXBoeC1za2lwXCIsXCJcIiksbi5pbm5lckhUTUw9XCJcIiksWyEwLGNdKTpcIlwiIT09bi5ub2RlVmFsdWUudHJpbSgpPyhLKFwib25seSBIVE1MIGVsZW1lbnQgdGFncyBhcmUgYWxsb3dlZCBhdCB0aGUgcm9vdCBvZiBjb21wb25lbnRzLlxcblxcblwiKydnb3Q6IFwiJy5jb25jYXQobi5ub2RlVmFsdWUudHJpbSgpLCdcIlxcblxcbicpK1wid2l0aGluOlxcblwiLG8uaW5uZXJIVE1MLnRyaW0oKSksbi5yZXBsYWNlV2l0aChpLmNyZWF0ZVNwYW4obi5ub2RlVmFsdWUsdCkpLFshMCxjXSk6KG4ucmVtb3ZlKCksW3MsY10pfSxbITEsITFdKSwyKSxjPXNbMF0sbD1zWzFdO3JldHVybiBjfHxsPyFjJiZsPyhLKFwiZXhwZWN0ZWQgYXQgbGVhc3Qgb25lIEhUTUwgZWxlbWVudCB0YWcgZGlyZWN0bHkgaW5zaWRlIGEgY29tcG9uZW50LCBidXQgb25seSBzdWJjb21wb25lbnRzIHdlcmUgZm91bmQuIEEgY29tcG9uZW50IG11c3QgcmVuZGVyIGF0IGxlYXN0IG9uZSBIVE1MIHRhZyBkaXJlY3RseSBpbnNpZGUgaXRzZWxmLlwiLG8uaW5uZXJIVE1MLnRyaW0oKSksby5pbm5lckhUTUwpOm8uaW5uZXJIVE1MOihLKFwiZXhwZWN0ZWQgYXQgbGVhc3Qgb25lIEhUTUwgZWxlbWVudCB0YWcgaW5zaWRlIGEgY29tcG9uZW50LCBidXQgdGhlIGNvbXBvbmVudCBpcyBlbXB0eTpcXG5cIixvLmlubmVySFRNTC50cmltKCkpLHRoaXMuY3JlYXRlU3BhbihcIlwiLHQpLm91dGVySFRNTCl9fSx7a2V5OlwiY3JlYXRlU3BhblwiLHZhbHVlOmZ1bmN0aW9uKGUsdCl7dmFyIG49ZG9jdW1lbnQuY3JlYXRlRWxlbWVudChcInNwYW5cIik7cmV0dXJuIG4uaW5uZXJUZXh0PWUsbi5zZXRBdHRyaWJ1dGUoSCx0KSxufX1dKSxlfSgpLGNlPWZ1bmN0aW9uKCl7ZnVuY3Rpb24gZSh0LG4pe3ZhciBpPXRoaXMscj1hcmd1bWVudHMubGVuZ3RoPjImJnZvaWQgMCE9PWFyZ3VtZW50c1syXT9hcmd1bWVudHNbMl06e307aWYoVCh0aGlzLGUpLHRoaXMudW5sb2FkZWQ9ITEsIW58fFwiT2JqZWN0XCI9PT1uLmNvbnN0cnVjdG9yLm5hbWUpdGhyb3cgbmV3IEVycm9yKCdcXG4gICAgICBhIHBob2VuaXggU29ja2V0IG11c3QgYmUgcHJvdmlkZWQgYXMgdGhlIHNlY29uZCBhcmd1bWVudCB0byB0aGUgTGl2ZVNvY2tldCBjb25zdHJ1Y3Rvci4gRm9yIGV4YW1wbGU6XFxuXFxuICAgICAgICAgIGltcG9ydCB7U29ja2V0fSBmcm9tIFwicGhvZW5peFwiXFxuICAgICAgICAgIGltcG9ydCB7TGl2ZVNvY2tldH0gZnJvbSBcInBob2VuaXhfbGl2ZV92aWV3XCJcXG4gICAgICAgICAgbGV0IGxpdmVTb2NrZXQgPSBuZXcgTGl2ZVNvY2tldChcIi9saXZlXCIsIFNvY2tldCwgey4uLn0pXFxuICAgICAgJyk7dGhpcy5zb2NrZXQ9bmV3IG4odCxyKSx0aGlzLmJpbmRpbmdQcmVmaXg9ci5iaW5kaW5nUHJlZml4fHxxLHRoaXMub3B0cz1yLHRoaXMucGFyYW1zPSQoci5wYXJhbXN8fHt9KSx0aGlzLnZpZXdMb2dnZXI9ci52aWV3TG9nZ2VyLHRoaXMubWV0YWRhdGFDYWxsYmFja3M9ci5tZXRhZGF0YXx8e30sdGhpcy5kZWZhdWx0cz1PYmplY3QuYXNzaWduKEcoeiksci5kZWZhdWx0c3x8e30pLHRoaXMuYWN0aXZlRWxlbWVudD1udWxsLHRoaXMucHJldkFjdGl2ZT1udWxsLHRoaXMuc2lsZW5jZWQ9ITEsdGhpcy5tYWluPW51bGwsdGhpcy5saW5rUmVmPTAsdGhpcy5yb290cz17fSx0aGlzLmhyZWY9d2luZG93LmxvY2F0aW9uLmhyZWYsdGhpcy5wZW5kaW5nTGluaz1udWxsLHRoaXMuY3VycmVudExvY2F0aW9uPUcod2luZG93LmxvY2F0aW9uKSx0aGlzLmhvb2tzPXIuaG9va3N8fHt9LHRoaXMudXBsb2FkZXJzPXIudXBsb2FkZXJzfHx7fSx0aGlzLmxvYWRlclRpbWVvdXQ9ci5sb2FkZXJUaW1lb3V0fHxXLHRoaXMuYm91bmRUb3BMZXZlbEV2ZW50cz0hMSx0aGlzLmRvbUNhbGxiYWNrcz1PYmplY3QuYXNzaWduKHtvbk5vZGVBZGRlZDokKCksb25CZWZvcmVFbFVwZGF0ZWQ6JCgpfSxyLmRvbXx8e30pLHdpbmRvdy5hZGRFdmVudExpc3RlbmVyKFwidW5sb2FkXCIsZnVuY3Rpb24oZSl7aS51bmxvYWRlZD0hMH0pLHRoaXMuc29ja2V0Lm9uT3BlbihmdW5jdGlvbigpe2kuaXNVbmxvYWRlZCgpJiZ3aW5kb3cubG9jYXRpb24ucmVsb2FkKCl9KX1yZXR1cm4gXyhlLFt7a2V5OlwiaXNQcm9maWxlRW5hYmxlZFwiLHZhbHVlOmZ1bmN0aW9uKCl7cmV0dXJuXCJ0cnVlXCI9PT1zZXNzaW9uU3RvcmFnZS5nZXRJdGVtKFwicGh4OmxpdmUtc29ja2V0OnByb2ZpbGluZ1wiKX19LHtrZXk6XCJpc0RlYnVnRW5hYmxlZFwiLHZhbHVlOmZ1bmN0aW9uKCl7cmV0dXJuXCJ0cnVlXCI9PT1zZXNzaW9uU3RvcmFnZS5nZXRJdGVtKFwicGh4OmxpdmUtc29ja2V0OmRlYnVnXCIpfX0se2tleTpcImVuYWJsZURlYnVnXCIsdmFsdWU6ZnVuY3Rpb24oKXtzZXNzaW9uU3RvcmFnZS5zZXRJdGVtKFwicGh4OmxpdmUtc29ja2V0OmRlYnVnXCIsXCJ0cnVlXCIpfX0se2tleTpcImVuYWJsZVByb2ZpbGluZ1wiLHZhbHVlOmZ1bmN0aW9uKCl7c2Vzc2lvblN0b3JhZ2Uuc2V0SXRlbShcInBoeDpsaXZlLXNvY2tldDpwcm9maWxpbmdcIixcInRydWVcIil9fSx7a2V5OlwiZGlzYWJsZURlYnVnXCIsdmFsdWU6ZnVuY3Rpb24oKXtzZXNzaW9uU3RvcmFnZS5yZW1vdmVJdGVtKFwicGh4OmxpdmUtc29ja2V0OmRlYnVnXCIpfX0se2tleTpcImRpc2FibGVQcm9maWxpbmdcIix2YWx1ZTpmdW5jdGlvbigpe3Nlc3Npb25TdG9yYWdlLnJlbW92ZUl0ZW0oXCJwaHg6bGl2ZS1zb2NrZXQ6cHJvZmlsaW5nXCIpfX0se2tleTpcImVuYWJsZUxhdGVuY3lTaW1cIix2YWx1ZTpmdW5jdGlvbihlKXt0aGlzLmVuYWJsZURlYnVnKCksY29uc29sZS5sb2coXCJsYXRlbmN5IHNpbXVsYXRvciBlbmFibGVkIGZvciB0aGUgZHVyYXRpb24gb2YgdGhpcyBicm93c2VyIHNlc3Npb24uIENhbGwgZGlzYWJsZUxhdGVuY3lTaW0oKSB0byBkaXNhYmxlXCIpLHNlc3Npb25TdG9yYWdlLnNldEl0ZW0oXCJwaHg6bGl2ZS1zb2NrZXQ6bGF0ZW5jeS1zaW1cIixlKX19LHtrZXk6XCJkaXNhYmxlTGF0ZW5jeVNpbVwiLHZhbHVlOmZ1bmN0aW9uKCl7c2Vzc2lvblN0b3JhZ2UucmVtb3ZlSXRlbShcInBoeDpsaXZlLXNvY2tldDpsYXRlbmN5LXNpbVwiKX19LHtrZXk6XCJnZXRMYXRlbmN5U2ltXCIsdmFsdWU6ZnVuY3Rpb24oKXt2YXIgZT1zZXNzaW9uU3RvcmFnZS5nZXRJdGVtKFwicGh4OmxpdmUtc29ja2V0OmxhdGVuY3ktc2ltXCIpO3JldHVybiBlP3BhcnNlSW50KGUpOm51bGx9fSx7a2V5OlwiZ2V0U29ja2V0XCIsdmFsdWU6ZnVuY3Rpb24oKXtyZXR1cm4gdGhpcy5zb2NrZXR9fSx7a2V5OlwiY29ubmVjdFwiLHZhbHVlOmZ1bmN0aW9uKCl7dmFyIGU9dGhpcyx0PWZ1bmN0aW9uKCl7ZS5qb2luUm9vdFZpZXdzKCkmJihlLmJpbmRUb3BMZXZlbEV2ZW50cygpLGUuc29ja2V0LmNvbm5lY3QoKSl9O1tcImNvbXBsZXRlXCIsXCJsb2FkZWRcIixcImludGVyYWN0aXZlXCJdLmluZGV4T2YoZG9jdW1lbnQucmVhZHlTdGF0ZSk+PTA/dCgpOmRvY3VtZW50LmFkZEV2ZW50TGlzdGVuZXIoXCJET01Db250ZW50TG9hZGVkXCIsZnVuY3Rpb24oKXtyZXR1cm4gdCgpfSl9fSx7a2V5OlwiZGlzY29ubmVjdFwiLHZhbHVlOmZ1bmN0aW9uKGUpe3RoaXMuc29ja2V0LmRpc2Nvbm5lY3QoZSl9fSx7a2V5OlwidHJpZ2dlckRPTVwiLHZhbHVlOmZ1bmN0aW9uKGUsdCl7dmFyIG47KG49dGhpcy5kb21DYWxsYmFja3MpW2VdLmFwcGx5KG4sQSh0KSl9fSx7a2V5OlwidGltZVwiLHZhbHVlOmZ1bmN0aW9uKGUsdCl7aWYoIXRoaXMuaXNQcm9maWxlRW5hYmxlZCgpfHwhY29uc29sZS50aW1lKXJldHVybiB0KCk7Y29uc29sZS50aW1lKGUpO3ZhciBuPXQoKTtyZXR1cm4gY29uc29sZS50aW1lRW5kKGUpLG59fSx7a2V5OlwibG9nXCIsdmFsdWU6ZnVuY3Rpb24oZSx0LG4pe2lmKHRoaXMudmlld0xvZ2dlcil7dmFyIGk9eChuKCksMikscj1pWzBdLG89aVsxXTt0aGlzLnZpZXdMb2dnZXIoZSx0LHIsbyl9ZWxzZSBpZih0aGlzLmlzRGVidWdFbmFibGVkKCkpe3ZhciBhPXgobigpLDIpLHU9YVswXSxzPWFbMV07WChlLHQsdSxzKX19fSx7a2V5Olwib25DaGFubmVsXCIsdmFsdWU6ZnVuY3Rpb24oZSx0LG4pe3ZhciBpPXRoaXM7ZS5vbih0LGZ1bmN0aW9uKGUpe3ZhciB0PWkuZ2V0TGF0ZW5jeVNpbSgpO3Q/KGNvbnNvbGUubG9nKFwic2ltdWxhdGluZyBcIi5jb25jYXQodCxcIm1zIG9mIGxhdGVuY3kgZnJvbSBzZXJ2ZXIgdG8gY2xpZW50XCIpKSxzZXRUaW1lb3V0KGZ1bmN0aW9uKCl7cmV0dXJuIG4oZSl9LHQpKTpuKGUpfSl9fSx7a2V5Olwid3JhcFB1c2hcIix2YWx1ZTpmdW5jdGlvbihlLHQsbil7dmFyIGk9dGhpcyxyPXRoaXMuZ2V0TGF0ZW5jeVNpbSgpLG89ZS5qb2luQ291bnQ7aWYoIXIpcmV0dXJuIHQudGltZW91dD9uKCkucmVjZWl2ZShcInRpbWVvdXRcIixmdW5jdGlvbigpe2Uuam9pbkNvdW50PT09byYmaS5yZWxvYWRXaXRoSml0dGVyKGUsZnVuY3Rpb24oKXtpLmxvZyhlLFwidGltZW91dFwiLGZ1bmN0aW9uKCl7cmV0dXJuW1wicmVjZWl2ZWQgdGltZW91dCB3aGlsZSBjb21tdW5pY2F0aW5nIHdpdGggc2VydmVyLiBGYWxsaW5nIGJhY2sgdG8gaGFyZCByZWZyZXNoIGZvciByZWNvdmVyeVwiXX0pfSl9KTpuKCk7Y29uc29sZS5sb2coXCJzaW11bGF0aW5nIFwiLmNvbmNhdChyLFwibXMgb2YgbGF0ZW5jeSBmcm9tIGNsaWVudCB0byBzZXJ2ZXJcIikpO3ZhciBhPXtyZWNlaXZlczpbXSxyZWNlaXZlOmZ1bmN0aW9uKGUsdCl7dGhpcy5yZWNlaXZlcy5wdXNoKFtlLHRdKX19O3JldHVybiBzZXRUaW1lb3V0KGZ1bmN0aW9uKCl7YS5yZWNlaXZlcy5yZWR1Y2UoZnVuY3Rpb24oZSx0KXt2YXIgbj14KHQsMiksaT1uWzBdLHI9blsxXTtyZXR1cm4gZS5yZWNlaXZlKGkscil9LG4oKSl9LHIpLGF9fSx7a2V5OlwicmVsb2FkV2l0aEppdHRlclwiLHZhbHVlOmZ1bmN0aW9uKGUsdCl7dmFyIG49dGhpcztlLmRlc3Ryb3koKSx0aGlzLmRpc2Nvbm5lY3QoKTt2YXIgaT1SWzBdLHI9UlsxXSxvPU1hdGguZmxvb3IoTWF0aC5yYW5kb20oKSooci1pKzEpKStpLGE9bGUudXBkYXRlTG9jYWwoZS5uYW1lKCksXCJjb25zZWN1dGl2ZS1yZWxvYWRzXCIsMCxmdW5jdGlvbihlKXtyZXR1cm4gZSsxfSk7dD90KCk6dGhpcy5sb2coZSxcImpvaW5cIixmdW5jdGlvbigpe3JldHVybltcImVuY291bnRlcmVkIFwiLmNvbmNhdChhLFwiIGNvbnNlY3V0aXZlIHJlbG9hZHNcIildfSksYT4xMCYmKHRoaXMubG9nKGUsXCJqb2luXCIsZnVuY3Rpb24oKXtyZXR1cm5bXCJleGNlZWRlZCBcIi5jb25jYXQoMTAsXCIgY29uc2VjdXRpdmUgcmVsb2Fkcy4gRW50ZXJpbmcgZmFpbHNhZmUgbW9kZVwiKV19KSxvPTNlNCksc2V0VGltZW91dChmdW5jdGlvbigpe24uaGFzUGVuZGluZ0xpbmsoKT93aW5kb3cubG9jYXRpb249bi5wZW5kaW5nTGluazp3aW5kb3cubG9jYXRpb24ucmVsb2FkKCl9LG8pfX0se2tleTpcImdldEhvb2tDYWxsYmFja3NcIix2YWx1ZTpmdW5jdGlvbihlKXtyZXR1cm4gZSYmZS5zdGFydHNXaXRoKFwiUGhvZW5peC5cIik/bmVbZS5zcGxpdChcIi5cIilbMV1dOnRoaXMuaG9va3NbZV19fSx7a2V5OlwiaXNVbmxvYWRlZFwiLHZhbHVlOmZ1bmN0aW9uKCl7cmV0dXJuIHRoaXMudW5sb2FkZWR9fSx7a2V5OlwiaXNDb25uZWN0ZWRcIix2YWx1ZTpmdW5jdGlvbigpe3JldHVybiB0aGlzLnNvY2tldC5pc0Nvbm5lY3RlZCgpfX0se2tleTpcImdldEJpbmRpbmdQcmVmaXhcIix2YWx1ZTpmdW5jdGlvbigpe3JldHVybiB0aGlzLmJpbmRpbmdQcmVmaXh9fSx7a2V5OlwiYmluZGluZ1wiLHZhbHVlOmZ1bmN0aW9uKGUpe3JldHVyblwiXCIuY29uY2F0KHRoaXMuZ2V0QmluZGluZ1ByZWZpeCgpKS5jb25jYXQoZSl9fSx7a2V5OlwiY2hhbm5lbFwiLHZhbHVlOmZ1bmN0aW9uKGUsdCl7cmV0dXJuIHRoaXMuc29ja2V0LmNoYW5uZWwoZSx0KX19LHtrZXk6XCJqb2luUm9vdFZpZXdzXCIsdmFsdWU6ZnVuY3Rpb24oKXt2YXIgZT10aGlzLHQ9ITE7cmV0dXJuIGRlLmFsbChkb2N1bWVudCxcIlwiLmNvbmNhdChVLFwiOm5vdChbXCIpLmNvbmNhdChcImRhdGEtcGh4LXBhcmVudC1pZFwiLFwiXSlcIiksZnVuY3Rpb24obil7aWYoIWUuZ2V0Um9vdEJ5SWQobi5pZCkpe3ZhciBpPWUuam9pblJvb3RWaWV3KG4sZS5nZXRIcmVmKCkpO2Uucm9vdD1lLnJvb3R8fGksbi5nZXRBdHRyaWJ1dGUoXCJkYXRhLXBoeC1tYWluXCIpJiYoZS5tYWluPWkpfXQ9ITB9KSx0fX0se2tleTpcInJlZGlyZWN0XCIsdmFsdWU6ZnVuY3Rpb24oZSx0KXt0aGlzLmRpc2Nvbm5lY3QoKSxsZS5yZWRpcmVjdChlLHQpfX0se2tleTpcInJlcGxhY2VNYWluXCIsdmFsdWU6ZnVuY3Rpb24oZSx0KXt2YXIgbj10aGlzLGk9YXJndW1lbnRzLmxlbmd0aD4yJiZ2b2lkIDAhPT1hcmd1bWVudHNbMl0/YXJndW1lbnRzWzJdOm51bGwscj1hcmd1bWVudHMubGVuZ3RoPjMmJnZvaWQgMCE9PWFyZ3VtZW50c1szXT9hcmd1bWVudHNbM106dGhpcy5zZXRQZW5kaW5nTGluayhlKSxvPXRoaXMubWFpbi5lbDt0aGlzLm1haW4uc2hvd0xvYWRlcih0aGlzLmxvYWRlclRpbWVvdXQpLHRoaXMubWFpbi5kZXN0cm95KCksbGUuZmV0Y2hQYWdlKGUsZnVuY3Rpb24oYSx1KXtpZigyMDAhPT1hKXJldHVybiBuLnJlZGlyZWN0KGUpO3ZhciBzPWRvY3VtZW50LmNyZWF0ZUVsZW1lbnQoXCJ0ZW1wbGF0ZVwiKTtzLmlubmVySFRNTD11O3ZhciBjPXMuY29udGVudC5jaGlsZE5vZGVzWzBdO2lmKCFjfHwhbi5pc1BoeFZpZXcoYykpcmV0dXJuIG4ucmVkaXJlY3QoZSk7bi5qb2luUm9vdFZpZXcoYyxlLHQsZnVuY3Rpb24oZSx0KXsxPT09dCYmKG4uY29tbWl0UGVuZGluZ0xpbmsocik/KG8ucmVwbGFjZVdpdGgoZS5lbCksbi5tYWluPWUsaSYmaSgpKTplLmRlc3Ryb3koKSl9KX0pfX0se2tleTpcImlzUGh4Vmlld1wiLHZhbHVlOmZ1bmN0aW9uKGUpe3JldHVybiBlLmdldEF0dHJpYnV0ZSYmbnVsbCE9PWUuZ2V0QXR0cmlidXRlKE8pfX0se2tleTpcImpvaW5Sb290Vmlld1wiLHZhbHVlOmZ1bmN0aW9uKGUsdCxuLGkpe3ZhciByPW5ldyB2ZShlLHRoaXMsbnVsbCx0LG4pO3JldHVybiB0aGlzLnJvb3RzW3IuaWRdPXIsci5qb2luKGkpLHJ9fSx7a2V5Olwib3duZXJcIix2YWx1ZTpmdW5jdGlvbihlLHQpe3ZhciBuPXRoaXMsaT1lZShlLmNsb3Nlc3QoVSksZnVuY3Rpb24oZSl7cmV0dXJuIG4uZ2V0Vmlld0J5RWwoZSl9KTtpJiZ0KGkpfX0se2tleTpcIndpdGhpbk93bmVyc1wiLHZhbHVlOmZ1bmN0aW9uKGUsdCl7dmFyIG49dGhpczt0aGlzLm93bmVyKGUsZnVuY3Rpb24oaSl7dmFyIHI9ZS5nZXRBdHRyaWJ1dGUobi5iaW5kaW5nKFwidGFyZ2V0XCIpKTtudWxsPT09cj90KGksZSk6aS53aXRoaW5UYXJnZXRzKHIsdCl9KX19LHtrZXk6XCJnZXRWaWV3QnlFbFwiLHZhbHVlOmZ1bmN0aW9uKGUpe3ZhciB0PWUuZ2V0QXR0cmlidXRlKFwiZGF0YS1waHgtcm9vdC1pZFwiKTtyZXR1cm4gZWUodGhpcy5nZXRSb290QnlJZCh0KSxmdW5jdGlvbih0KXtyZXR1cm4gdC5nZXREZXNjZW5kZW50QnlFbChlKX0pfX0se2tleTpcImdldFJvb3RCeUlkXCIsdmFsdWU6ZnVuY3Rpb24oZSl7cmV0dXJuIHRoaXMucm9vdHNbZV19fSx7a2V5OlwiZGVzdHJveUFsbFZpZXdzXCIsdmFsdWU6ZnVuY3Rpb24oKXtmb3IodmFyIGUgaW4gdGhpcy5yb290cyl0aGlzLnJvb3RzW2VdLmRlc3Ryb3koKSxkZWxldGUgdGhpcy5yb290c1tlXX19LHtrZXk6XCJkZXN0cm95Vmlld0J5RWxcIix2YWx1ZTpmdW5jdGlvbihlKXt2YXIgdD10aGlzLmdldFJvb3RCeUlkKGUuZ2V0QXR0cmlidXRlKFwiZGF0YS1waHgtcm9vdC1pZFwiKSk7dCYmdC5kZXN0cm95RGVzY2VuZGVudChlLmlkKX19LHtrZXk6XCJzZXRBY3RpdmVFbGVtZW50XCIsdmFsdWU6ZnVuY3Rpb24oZSl7dmFyIHQ9dGhpcztpZih0aGlzLmFjdGl2ZUVsZW1lbnQhPT1lKXt0aGlzLmFjdGl2ZUVsZW1lbnQ9ZTt2YXIgbj1mdW5jdGlvbigpe2U9PT10LmFjdGl2ZUVsZW1lbnQmJih0LmFjdGl2ZUVsZW1lbnQ9bnVsbCksZS5yZW1vdmVFdmVudExpc3RlbmVyKFwibW91c2V1cFwiLHQpLGUucmVtb3ZlRXZlbnRMaXN0ZW5lcihcInRvdWNoZW5kXCIsdCl9O2UuYWRkRXZlbnRMaXN0ZW5lcihcIm1vdXNldXBcIixuKSxlLmFkZEV2ZW50TGlzdGVuZXIoXCJ0b3VjaGVuZFwiLG4pfX19LHtrZXk6XCJnZXRBY3RpdmVFbGVtZW50XCIsdmFsdWU6ZnVuY3Rpb24oKXtyZXR1cm4gZG9jdW1lbnQuYWN0aXZlRWxlbWVudD09PWRvY3VtZW50LmJvZHk/dGhpcy5hY3RpdmVFbGVtZW50fHxkb2N1bWVudC5hY3RpdmVFbGVtZW50OmRvY3VtZW50LmFjdGl2ZUVsZW1lbnR8fGRvY3VtZW50LmJvZHl9fSx7a2V5OlwiZHJvcEFjdGl2ZUVsZW1lbnRcIix2YWx1ZTpmdW5jdGlvbihlKXt0aGlzLnByZXZBY3RpdmUmJmUub3duc0VsZW1lbnQodGhpcy5wcmV2QWN0aXZlKSYmKHRoaXMucHJldkFjdGl2ZT1udWxsKX19LHtrZXk6XCJyZXN0b3JlUHJldmlvdXNseUFjdGl2ZUZvY3VzXCIsdmFsdWU6ZnVuY3Rpb24oKXt0aGlzLnByZXZBY3RpdmUmJnRoaXMucHJldkFjdGl2ZSE9PWRvY3VtZW50LmJvZHkmJnRoaXMucHJldkFjdGl2ZS5mb2N1cygpfX0se2tleTpcImJsdXJBY3RpdmVFbGVtZW50XCIsdmFsdWU6ZnVuY3Rpb24oKXt0aGlzLnByZXZBY3RpdmU9dGhpcy5nZXRBY3RpdmVFbGVtZW50KCksdGhpcy5wcmV2QWN0aXZlIT09ZG9jdW1lbnQuYm9keSYmdGhpcy5wcmV2QWN0aXZlLmJsdXIoKX19LHtrZXk6XCJiaW5kVG9wTGV2ZWxFdmVudHNcIix2YWx1ZTpmdW5jdGlvbigpe3ZhciBlPXRoaXM7dGhpcy5ib3VuZFRvcExldmVsRXZlbnRzfHwodGhpcy5ib3VuZFRvcExldmVsRXZlbnRzPSEwLGRvY3VtZW50LmJvZHkuYWRkRXZlbnRMaXN0ZW5lcihcImNsaWNrXCIsZnVuY3Rpb24oKXt9KSx3aW5kb3cuYWRkRXZlbnRMaXN0ZW5lcihcInBhZ2VzaG93XCIsZnVuY3Rpb24odCl7dC5wZXJzaXN0ZWQmJihlLndpdGhQYWdlTG9hZGluZyh7dG86d2luZG93LmxvY2F0aW9uLmhyZWYsa2luZDpcInJlZGlyZWN0XCJ9KSx3aW5kb3cubG9jYXRpb24ucmVsb2FkKCkpfSksdGhpcy5iaW5kQ2xpY2tzKCksdGhpcy5iaW5kTmF2KCksdGhpcy5iaW5kRm9ybXMoKSx0aGlzLmJpbmQoe2tleXVwOlwia2V5dXBcIixrZXlkb3duOlwia2V5ZG93blwifSxmdW5jdGlvbih0LG4saSxyLG8sYSx1KXt2YXIgcz1yLmdldEF0dHJpYnV0ZShlLmJpbmRpbmcoXCJrZXlcIikpLGM9dC5rZXkmJnQua2V5LnRvTG93ZXJDYXNlKCk7cyYmcy50b0xvd2VyQ2FzZSgpIT09Y3x8aS5wdXNoS2V5KHIsbyxuLGEsZnVuY3Rpb24oZSl7Zm9yKHZhciB0PTE7dDxhcmd1bWVudHMubGVuZ3RoO3QrKyl7dmFyIG49bnVsbCE9YXJndW1lbnRzW3RdP2FyZ3VtZW50c1t0XTp7fTt0JTI/dyhPYmplY3QobiksITApLmZvckVhY2goZnVuY3Rpb24odCl7RShlLHQsblt0XSl9KTpPYmplY3QuZ2V0T3duUHJvcGVydHlEZXNjcmlwdG9ycz9PYmplY3QuZGVmaW5lUHJvcGVydGllcyhlLE9iamVjdC5nZXRPd25Qcm9wZXJ0eURlc2NyaXB0b3JzKG4pKTp3KE9iamVjdChuKSkuZm9yRWFjaChmdW5jdGlvbih0KXtPYmplY3QuZGVmaW5lUHJvcGVydHkoZSx0LE9iamVjdC5nZXRPd25Qcm9wZXJ0eURlc2NyaXB0b3Iobix0KSl9KX1yZXR1cm4gZX0oe2tleTp0LmtleX0sZS5ldmVudE1ldGEobix0LHIpKSl9KSx0aGlzLmJpbmQoe2JsdXI6XCJmb2N1c291dFwiLGZvY3VzOlwiZm9jdXNpblwifSxmdW5jdGlvbih0LG4saSxyLG8sYSx1KXt1fHxpLnB1c2hFdmVudChuLHIsbyxhLGUuZXZlbnRNZXRhKG4sdCxyKSl9KSx0aGlzLmJpbmQoe2JsdXI6XCJibHVyXCIsZm9jdXM6XCJmb2N1c1wifSxmdW5jdGlvbih0LG4saSxyLG8sYSx1KXt1JiZcIndpbmRvd1wiIT09IXUmJmkucHVzaEV2ZW50KG4scixvLGEsZS5ldmVudE1ldGEobix0LHIpKX0pLHdpbmRvdy5hZGRFdmVudExpc3RlbmVyKFwiZHJhZ292ZXJcIixmdW5jdGlvbihlKXtyZXR1cm4gZS5wcmV2ZW50RGVmYXVsdCgpfSksd2luZG93LmFkZEV2ZW50TGlzdGVuZXIoXCJkcm9wXCIsZnVuY3Rpb24odCl7dC5wcmV2ZW50RGVmYXVsdCgpO3ZhciBuPWVlKFkodC50YXJnZXQsZS5iaW5kaW5nKFwiZHJvcC10YXJnZXRcIikpLGZ1bmN0aW9uKHQpe3JldHVybiB0LmdldEF0dHJpYnV0ZShlLmJpbmRpbmcoXCJkcm9wLXRhcmdldFwiKSl9KSxpPW4mJmRvY3VtZW50LmdldEVsZW1lbnRCeUlkKG4pLHI9QXJyYXkuZnJvbSh0LmRhdGFUcmFuc2Zlci5maWxlc3x8W10pO2kmJiFpLmRpc2FibGVkJiYwIT09ci5sZW5ndGgmJmkuZmlsZXMgaW5zdGFuY2VvZiBGaWxlTGlzdCYmKHJlLnRyYWNrRmlsZXMoaSxyKSxpLmRpc3BhdGNoRXZlbnQobmV3IEV2ZW50KFwiaW5wdXRcIix7YnViYmxlczohMH0pKSl9KSl9fSx7a2V5OlwiZXZlbnRNZXRhXCIsdmFsdWU6ZnVuY3Rpb24oZSx0LG4pe3ZhciBpPXRoaXMubWV0YWRhdGFDYWxsYmFja3NbZV07cmV0dXJuIGk/aSh0LG4pOnt9fX0se2tleTpcInNldFBlbmRpbmdMaW5rXCIsdmFsdWU6ZnVuY3Rpb24oZSl7cmV0dXJuIHRoaXMubGlua1JlZisrLHRoaXMucGVuZGluZ0xpbms9ZSx0aGlzLmxpbmtSZWZ9fSx7a2V5OlwiY29tbWl0UGVuZGluZ0xpbmtcIix2YWx1ZTpmdW5jdGlvbihlKXtyZXR1cm4gdGhpcy5saW5rUmVmPT09ZSYmKHRoaXMuaHJlZj10aGlzLnBlbmRpbmdMaW5rLHRoaXMucGVuZGluZ0xpbms9bnVsbCwhMCl9fSx7a2V5OlwiZ2V0SHJlZlwiLHZhbHVlOmZ1bmN0aW9uKCl7cmV0dXJuIHRoaXMuaHJlZn19LHtrZXk6XCJoYXNQZW5kaW5nTGlua1wiLHZhbHVlOmZ1bmN0aW9uKCl7cmV0dXJuISF0aGlzLnBlbmRpbmdMaW5rfX0se2tleTpcImJpbmRcIix2YWx1ZTpmdW5jdGlvbihlLHQpe3ZhciBuPXRoaXMsaT1mdW5jdGlvbihpKXt2YXIgcj1lW2ldO24ub24ocixmdW5jdGlvbihlKXt2YXIgcj1uLmJpbmRpbmcoaSksbz1uLmJpbmRpbmcoXCJ3aW5kb3ctXCIuY29uY2F0KGkpKSxhPWUudGFyZ2V0LmdldEF0dHJpYnV0ZSYmZS50YXJnZXQuZ2V0QXR0cmlidXRlKHIpO2E/bi5kZWJvdW5jZShlLnRhcmdldCxlLGZ1bmN0aW9uKCl7bi53aXRoaW5Pd25lcnMoZS50YXJnZXQsZnVuY3Rpb24obixyKXt0KGUsaSxuLGUudGFyZ2V0LHIsYSxudWxsKX0pfSk6ZGUuYWxsKGRvY3VtZW50LFwiW1wiLmNvbmNhdChvLFwiXVwiKSxmdW5jdGlvbihyKXt2YXIgYT1yLmdldEF0dHJpYnV0ZShvKTtuLmRlYm91bmNlKHIsZSxmdW5jdGlvbigpe24ud2l0aGluT3duZXJzKHIsZnVuY3Rpb24obixvKXt0KGUsaSxuLHIsbyxhLFwid2luZG93XCIpfSl9KX0pfSl9O2Zvcih2YXIgciBpbiBlKWkocil9fSx7a2V5OlwiYmluZENsaWNrc1wiLHZhbHVlOmZ1bmN0aW9uKCl7dGhpcy5iaW5kQ2xpY2soXCJjbGlja1wiLFwiY2xpY2tcIiwhMSksdGhpcy5iaW5kQ2xpY2soXCJtb3VzZWRvd25cIixcImNhcHR1cmUtY2xpY2tcIiwhMCl9fSx7a2V5OlwiYmluZENsaWNrXCIsdmFsdWU6ZnVuY3Rpb24oZSx0LG4pe3ZhciBpPXRoaXMscj10aGlzLmJpbmRpbmcodCk7d2luZG93LmFkZEV2ZW50TGlzdGVuZXIoZSxmdW5jdGlvbihlKXtpZihpLmlzQ29ubmVjdGVkKCkpe3ZhciB0PW51bGwsbz0odD1uP2UudGFyZ2V0Lm1hdGNoZXMoXCJbXCIuY29uY2F0KHIsXCJdXCIpKT9lLnRhcmdldDplLnRhcmdldC5xdWVyeVNlbGVjdG9yKFwiW1wiLmNvbmNhdChyLFwiXVwiKSk6WShlLnRhcmdldCxyKSkmJnQuZ2V0QXR0cmlidXRlKHIpO28mJihcIiNcIj09PXQuZ2V0QXR0cmlidXRlKFwiaHJlZlwiKSYmZS5wcmV2ZW50RGVmYXVsdCgpLGkuZGVib3VuY2UodCxlLGZ1bmN0aW9uKCl7aS53aXRoaW5Pd25lcnModCxmdW5jdGlvbihuLHIpe24ucHVzaEV2ZW50KFwiY2xpY2tcIix0LHIsbyxpLmV2ZW50TWV0YShcImNsaWNrXCIsZSx0KSl9KX0pKX19LG4pfX0se2tleTpcImJpbmROYXZcIix2YWx1ZTpmdW5jdGlvbigpe3ZhciBlPXRoaXM7aWYobGUuY2FuUHVzaFN0YXRlKCkpe2hpc3Rvcnkuc2Nyb2xsUmVzdG9yYXRpb24mJihoaXN0b3J5LnNjcm9sbFJlc3RvcmF0aW9uPVwibWFudWFsXCIpO3ZhciB0PW51bGw7d2luZG93LmFkZEV2ZW50TGlzdGVuZXIoXCJzY3JvbGxcIixmdW5jdGlvbihlKXtjbGVhclRpbWVvdXQodCksdD1zZXRUaW1lb3V0KGZ1bmN0aW9uKCl7bGUudXBkYXRlQ3VycmVudFN0YXRlKGZ1bmN0aW9uKGUpe3JldHVybiBPYmplY3QuYXNzaWduKGUse3Njcm9sbDp3aW5kb3cuc2Nyb2xsWX0pfSl9LDEwMCl9KSx3aW5kb3cuYWRkRXZlbnRMaXN0ZW5lcihcInBvcHN0YXRlXCIsZnVuY3Rpb24odCl7aWYoZS5yZWdpc3Rlck5ld0xvY2F0aW9uKHdpbmRvdy5sb2NhdGlvbikpe3ZhciBuPXQuc3RhdGV8fHt9LGk9bi50eXBlLHI9bi5pZCxvPW4ucm9vdCxhPW4uc2Nyb2xsLHU9d2luZG93LmxvY2F0aW9uLmhyZWY7ZS5tYWluLmlzQ29ubmVjdGVkKCkmJlwicGF0Y2hcIj09PWkmJnI9PT1lLm1haW4uaWQ/ZS5tYWluLnB1c2hMaW5rUGF0Y2godSxudWxsKTplLnJlcGxhY2VNYWluKHUsbnVsbCxmdW5jdGlvbigpe28mJmUucmVwbGFjZVJvb3RIaXN0b3J5KCksXCJudW1iZXJcIj09dHlwZW9mIGEmJnNldFRpbWVvdXQoZnVuY3Rpb24oKXt3aW5kb3cuc2Nyb2xsVG8oMCxhKX0sMCl9KX19LCExKSx3aW5kb3cuYWRkRXZlbnRMaXN0ZW5lcihcImNsaWNrXCIsZnVuY3Rpb24odCl7dmFyIG49WSh0LnRhcmdldCxcImRhdGEtcGh4LWxpbmtcIiksaT1uJiZuLmdldEF0dHJpYnV0ZShcImRhdGEtcGh4LWxpbmtcIikscj10Lm1ldGFLZXl8fHQuY3RybEtleXx8MT09PXQuYnV0dG9uO2lmKGkmJmUuaXNDb25uZWN0ZWQoKSYmZS5tYWluJiYhcil7dmFyIG89bi5ocmVmLGE9bi5nZXRBdHRyaWJ1dGUoXCJkYXRhLXBoeC1saW5rLXN0YXRlXCIpO2lmKHQucHJldmVudERlZmF1bHQoKSxlLnBlbmRpbmdMaW5rIT09bylpZihcInBhdGNoXCI9PT1pKWUucHVzaEhpc3RvcnlQYXRjaChvLGEsbik7ZWxzZXtpZihcInJlZGlyZWN0XCIhPT1pKXRocm93IG5ldyBFcnJvcihcImV4cGVjdGVkIFwiLmNvbmNhdChcImRhdGEtcGh4LWxpbmtcIiwnIHRvIGJlIFwicGF0Y2hcIiBvciBcInJlZGlyZWN0XCIsIGdvdDogJykuY29uY2F0KGkpKTtlLmhpc3RvcnlSZWRpcmVjdChvLGEpfX19LCExKX19fSx7a2V5Olwid2l0aFBhZ2VMb2FkaW5nXCIsdmFsdWU6ZnVuY3Rpb24oZSx0KXtkZS5kaXNwYXRjaEV2ZW50KHdpbmRvdyxcInBoeDpwYWdlLWxvYWRpbmctc3RhcnRcIixlKTt2YXIgbj1mdW5jdGlvbigpe3JldHVybiBkZS5kaXNwYXRjaEV2ZW50KHdpbmRvdyxcInBoeDpwYWdlLWxvYWRpbmctc3RvcFwiLGUpfTtyZXR1cm4gdD90KG4pOm59fSx7a2V5OlwicHVzaEhpc3RvcnlQYXRjaFwiLHZhbHVlOmZ1bmN0aW9uKGUsdCxuKXt2YXIgaT10aGlzO3RoaXMud2l0aFBhZ2VMb2FkaW5nKHt0bzplLGtpbmQ6XCJwYXRjaFwifSxmdW5jdGlvbihyKXtpLm1haW4ucHVzaExpbmtQYXRjaChlLG4sZnVuY3Rpb24oKXtpLmhpc3RvcnlQYXRjaChlLHQpLHIoKX0pfSl9fSx7a2V5OlwiaGlzdG9yeVBhdGNoXCIsdmFsdWU6ZnVuY3Rpb24oZSx0KXtsZS5wdXNoU3RhdGUodCx7dHlwZTpcInBhdGNoXCIsaWQ6dGhpcy5tYWluLmlkfSxlKSx0aGlzLnJlZ2lzdGVyTmV3TG9jYXRpb24od2luZG93LmxvY2F0aW9uKX19LHtrZXk6XCJoaXN0b3J5UmVkaXJlY3RcIix2YWx1ZTpmdW5jdGlvbihlLHQsbil7dmFyIGk9dGhpcyxyPXdpbmRvdy5zY3JvbGxZO3RoaXMud2l0aFBhZ2VMb2FkaW5nKHt0bzplLGtpbmQ6XCJyZWRpcmVjdFwifSxmdW5jdGlvbihvKXtpLnJlcGxhY2VNYWluKGUsbixmdW5jdGlvbigpe2xlLnB1c2hTdGF0ZSh0LHt0eXBlOlwicmVkaXJlY3RcIixpZDppLm1haW4uaWQsc2Nyb2xsOnJ9LGUpLGkucmVnaXN0ZXJOZXdMb2NhdGlvbih3aW5kb3cubG9jYXRpb24pLG8oKX0pfSl9fSx7a2V5OlwicmVwbGFjZVJvb3RIaXN0b3J5XCIsdmFsdWU6ZnVuY3Rpb24oKXtsZS5wdXNoU3RhdGUoXCJyZXBsYWNlXCIse3Jvb3Q6ITAsdHlwZTpcInBhdGNoXCIsaWQ6dGhpcy5tYWluLmlkfSl9fSx7a2V5OlwicmVnaXN0ZXJOZXdMb2NhdGlvblwiLHZhbHVlOmZ1bmN0aW9uKGUpe3ZhciB0PXRoaXMuY3VycmVudExvY2F0aW9uO3JldHVybiB0LnBhdGhuYW1lK3Quc2VhcmNoIT09ZS5wYXRobmFtZStlLnNlYXJjaCYmKHRoaXMuY3VycmVudExvY2F0aW9uPUcoZSksITApfX0se2tleTpcImJpbmRGb3Jtc1wiLHZhbHVlOmZ1bmN0aW9uKCl7dmFyIGU9dGhpcyx0PTA7dGhpcy5vbihcInN1Ym1pdFwiLGZ1bmN0aW9uKHQpe3ZhciBuPXQudGFyZ2V0LmdldEF0dHJpYnV0ZShlLmJpbmRpbmcoXCJzdWJtaXRcIikpO24mJih0LnByZXZlbnREZWZhdWx0KCksdC50YXJnZXQuZGlzYWJsZWQ9ITAsZS53aXRoaW5Pd25lcnModC50YXJnZXQsZnVuY3Rpb24oZSxpKXtyZXR1cm4gZS5zdWJtaXRGb3JtKHQudGFyZ2V0LGksbil9KSl9LCExKTtmb3IodmFyIG49ZnVuY3Rpb24oKXt2YXIgbj1yW2ldO2Uub24obixmdW5jdGlvbihpKXt2YXIgcj1pLnRhcmdldCxvPXIuZm9ybSYmci5mb3JtLmdldEF0dHJpYnV0ZShlLmJpbmRpbmcoXCJjaGFuZ2VcIikpO2lmKG8mJihcIm51bWJlclwiIT09ci50eXBlfHwhci52YWxpZGl0eXx8IXIudmFsaWRpdHkuYmFkSW5wdXQpKXt2YXIgYT10O3QrKzt2YXIgdT1kZS5wcml2YXRlKHIsXCJwcmV2LWl0ZXJhdGlvblwiKXx8e30scz11LmF0LGM9dS50eXBlO3M9PT1hLTEmJm4hPT1jfHwoZGUucHV0UHJpdmF0ZShyLFwicHJldi1pdGVyYXRpb25cIix7YXQ6YSx0eXBlOm59KSxlLmRlYm91bmNlKHIsaSxmdW5jdGlvbigpe2Uud2l0aGluT3duZXJzKHIuZm9ybSxmdW5jdGlvbih0LG4pe2RlLnB1dFByaXZhdGUocixcInBoeC1oYXMtZm9jdXNlZFwiLCEwKSxkZS5pc1RleHR1YWxJbnB1dChyKXx8ZS5zZXRBY3RpdmVFbGVtZW50KHIpLHQucHVzaElucHV0KHIsbixvLGkudGFyZ2V0KX0pfSkpfX0sITEpfSxpPTAscj1bXCJjaGFuZ2VcIixcImlucHV0XCJdO2k8ci5sZW5ndGg7aSsrKW4oKX19LHtrZXk6XCJkZWJvdW5jZVwiLHZhbHVlOmZ1bmN0aW9uKGUsdCxuKXt2YXIgaT10aGlzLmJpbmRpbmcoXCJkZWJvdW5jZVwiKSxyPXRoaXMuYmluZGluZyhcInRocm90dGxlXCIpLG89dGhpcy5kZWZhdWx0cy5kZWJvdW5jZS50b1N0cmluZygpLGE9dGhpcy5kZWZhdWx0cy50aHJvdHRsZS50b1N0cmluZygpO2RlLmRlYm91bmNlKGUsdCxpLG8scixhLG4pfX0se2tleTpcInNpbGVuY2VFdmVudHNcIix2YWx1ZTpmdW5jdGlvbihlKXt0aGlzLnNpbGVuY2VkPSEwLGUoKSx0aGlzLnNpbGVuY2VkPSExfX0se2tleTpcIm9uXCIsdmFsdWU6ZnVuY3Rpb24oZSx0KXt2YXIgbj10aGlzO3dpbmRvdy5hZGRFdmVudExpc3RlbmVyKGUsZnVuY3Rpb24oZSl7bi5zaWxlbmNlZHx8dChlKX0pfX1dKSxlfSgpLGxlPXtjYW5QdXNoU3RhdGU6ZnVuY3Rpb24oKXtyZXR1cm4gdm9pZCAwIT09aGlzdG9yeS5wdXNoU3RhdGV9LGRyb3BMb2NhbDpmdW5jdGlvbihlLHQpe3JldHVybiB3aW5kb3cubG9jYWxTdG9yYWdlLnJlbW92ZUl0ZW0odGhpcy5sb2NhbEtleShlLHQpKX0sdXBkYXRlTG9jYWw6ZnVuY3Rpb24oZSx0LG4saSl7dmFyIHI9dGhpcy5nZXRMb2NhbChlLHQpLG89dGhpcy5sb2NhbEtleShlLHQpLGE9bnVsbD09PXI/bjppKHIpO3JldHVybiB3aW5kb3cubG9jYWxTdG9yYWdlLnNldEl0ZW0obyxKU09OLnN0cmluZ2lmeShhKSksYX0sZ2V0TG9jYWw6ZnVuY3Rpb24oZSx0KXtyZXR1cm4gSlNPTi5wYXJzZSh3aW5kb3cubG9jYWxTdG9yYWdlLmdldEl0ZW0odGhpcy5sb2NhbEtleShlLHQpKSl9LGZldGNoUGFnZTpmdW5jdGlvbihlLHQpe3ZhciBuPW5ldyBYTUxIdHRwUmVxdWVzdDtuLm9wZW4oXCJHRVRcIixlLCEwKSxuLnRpbWVvdXQ9M2U0LG4uc2V0UmVxdWVzdEhlYWRlcihcImNvbnRlbnQtdHlwZVwiLFwidGV4dC9odG1sXCIpLG4uc2V0UmVxdWVzdEhlYWRlcihcImNhY2hlLWNvbnRyb2xcIixcIm1heC1hZ2U9MCwgbm8tY2FjaGUsIG5vLXN0b3JlLCBtdXN0LXJldmFsaWRhdGUsIHBvc3QtY2hlY2s9MCwgcHJlLWNoZWNrPTBcIiksbi5zZXRSZXF1ZXN0SGVhZGVyKFwieC1yZXF1ZXN0ZWQtd2l0aFwiLFwibGl2ZS1saW5rXCIpLG4ub25lcnJvcj1mdW5jdGlvbigpe3JldHVybiB0KDQwMCl9LG4ub250aW1lb3V0PWZ1bmN0aW9uKCl7cmV0dXJuIHQoNTA0KX0sbi5vbnJlYWR5c3RhdGVjaGFuZ2U9ZnVuY3Rpb24oKXtpZig0PT09bi5yZWFkeVN0YXRlKXt2YXIgaT1uZXcgVVJMKGUpLHI9aS5wYXRobmFtZStpLnNlYXJjaCxvPWVlKG4uZ2V0UmVzcG9uc2VIZWFkZXIoXCJ4LXJlc3BvbnNlLXVybFwiKXx8bi5yZXNwb25zZVVSTCxmdW5jdGlvbihlKXtyZXR1cm4gbmV3IFVSTChlKX0pLGE9bz9vLnBhdGhuYW1lK28uc2VhcmNoOm51bGw7cmV0dXJuXCJsaXZlLWxpbmtcIiE9PW4uZ2V0UmVzcG9uc2VIZWFkZXIoXCJ4LXJlcXVlc3RlZC13aXRoXCIpP3QoNDAwKTpudWxsPT09b3x8YSE9cj90KDMwMik6MjAwIT09bi5zdGF0dXM/dChuLnN0YXR1cyk6dm9pZCB0KDIwMCxuLnJlc3BvbnNlVGV4dCl9fSxuLnNlbmQoKX0sdXBkYXRlQ3VycmVudFN0YXRlOmZ1bmN0aW9uKGUpe3RoaXMuY2FuUHVzaFN0YXRlKCkmJmhpc3RvcnkucmVwbGFjZVN0YXRlKGUoaGlzdG9yeS5zdGF0ZXx8e30pLFwiXCIsd2luZG93LmxvY2F0aW9uLmhyZWYpfSxwdXNoU3RhdGU6ZnVuY3Rpb24oZSx0LG4pe2lmKHRoaXMuY2FuUHVzaFN0YXRlKCkpe2lmKG4hPT13aW5kb3cubG9jYXRpb24uaHJlZil7aWYoXCJyZWRpcmVjdFwiPT10LnR5cGUmJnQuc2Nyb2xsKXt2YXIgaT1oaXN0b3J5LnN0YXRlfHx7fTtpLnNjcm9sbD10LnNjcm9sbCxoaXN0b3J5LnJlcGxhY2VTdGF0ZShpLFwiXCIsd2luZG93LmxvY2F0aW9uLmhyZWYpfWRlbGV0ZSB0LnNjcm9sbCxoaXN0b3J5W2UrXCJTdGF0ZVwiXSh0LFwiXCIsbnx8bnVsbCk7dmFyIHI9dGhpcy5nZXRIYXNoVGFyZ2V0RWwod2luZG93LmxvY2F0aW9uLmhhc2gpO3I/ci5zY3JvbGxJbnRvVmlldygpOlwicmVkaXJlY3RcIj09PXQudHlwZSYmd2luZG93LnNjcm9sbCgwLDApfX1lbHNlIHRoaXMucmVkaXJlY3Qobil9LHNldENvb2tpZTpmdW5jdGlvbihlLHQpe2RvY3VtZW50LmNvb2tpZT1cIlwiLmNvbmNhdChlLFwiPVwiKS5jb25jYXQodCl9LGdldENvb2tpZTpmdW5jdGlvbihlKXtyZXR1cm4gZG9jdW1lbnQuY29va2llLnJlcGxhY2UobmV3IFJlZ0V4cChcIig/Oig/Ol58Lio7cyopXCIuY29uY2F0KGUsXCJzKj1zKihbXjtdKikuKiQpfF4uKiRcIikpLFwiJDFcIil9LHJlZGlyZWN0OmZ1bmN0aW9uKGUsdCl7dCYmbGUuc2V0Q29va2llKFwiX19waG9lbml4X2ZsYXNoX19cIix0K1wiOyBtYXgtYWdlPTYwMDAwOyBwYXRoPS9cIiksd2luZG93LmxvY2F0aW9uPWV9LGxvY2FsS2V5OmZ1bmN0aW9uKGUsdCl7cmV0dXJuXCJcIi5jb25jYXQoZSxcIi1cIikuY29uY2F0KHQpfSxnZXRIYXNoVGFyZ2V0RWw6ZnVuY3Rpb24oZSl7dmFyIHQ9ZS50b1N0cmluZygpLnN1YnN0cmluZygxKTtpZihcIlwiIT09dClyZXR1cm4gZG9jdW1lbnQuZ2V0RWxlbWVudEJ5SWQodCl8fGRvY3VtZW50LnF1ZXJ5U2VsZWN0b3IoJ2FbbmFtZT1cIicuY29uY2F0KHQsJ1wiXScpKX19LGRlPXtieUlkOmZ1bmN0aW9uKGUpe3JldHVybiBkb2N1bWVudC5nZXRFbGVtZW50QnlJZChlKXx8SyhcIm5vIGlkIGZvdW5kIGZvciBcIi5jb25jYXQoZSkpfSxyZW1vdmVDbGFzczpmdW5jdGlvbihlLHQpe2UuY2xhc3NMaXN0LnJlbW92ZSh0KSwwPT09ZS5jbGFzc0xpc3QubGVuZ3RoJiZlLnJlbW92ZUF0dHJpYnV0ZShcImNsYXNzXCIpfSxhbGw6ZnVuY3Rpb24oZSx0LG4pe3ZhciBpPUFycmF5LmZyb20oZS5xdWVyeVNlbGVjdG9yQWxsKHQpKTtyZXR1cm4gbj9pLmZvckVhY2gobik6aX0sY2hpbGROb2RlTGVuZ3RoOmZ1bmN0aW9uKGUpe3ZhciB0PWRvY3VtZW50LmNyZWF0ZUVsZW1lbnQoXCJ0ZW1wbGF0ZVwiKTtyZXR1cm4gdC5pbm5lckhUTUw9ZSx0LmNvbnRlbnQuY2hpbGRFbGVtZW50Q291bnR9LGlzVXBsb2FkSW5wdXQ6ZnVuY3Rpb24oZSl7cmV0dXJuXCJmaWxlXCI9PT1lLnR5cGUmJm51bGwhPT1lLmdldEF0dHJpYnV0ZShNKX0sZmluZFVwbG9hZElucHV0czpmdW5jdGlvbihlKXtyZXR1cm4gdGhpcy5hbGwoZSwnaW5wdXRbdHlwZT1cImZpbGVcIl1bJy5jb25jYXQoTSxcIl1cIikpfSxmaW5kQ29tcG9uZW50Tm9kZUxpc3Q6ZnVuY3Rpb24oZSx0KXtyZXR1cm4gdGhpcy5maWx0ZXJXaXRoaW5TYW1lTGl2ZVZpZXcodGhpcy5hbGwoZSxcIltcIi5jb25jYXQoSCwnPVwiJykuY29uY2F0KHQsJ1wiXScpKSxlKX0sZmluZFBoeENoaWxkcmVuSW5GcmFnbWVudDpmdW5jdGlvbihlLHQpe3ZhciBuPWRvY3VtZW50LmNyZWF0ZUVsZW1lbnQoXCJ0ZW1wbGF0ZVwiKTtyZXR1cm4gbi5pbm5lckhUTUw9ZSx0aGlzLmZpbmRQaHhDaGlsZHJlbihuLmNvbnRlbnQsdCl9LGlzSWdub3JlZDpmdW5jdGlvbihlLHQpe3JldHVyblwiaWdub3JlXCI9PT0oZS5nZXRBdHRyaWJ1dGUodCl8fGUuZ2V0QXR0cmlidXRlKFwiZGF0YS1waHgtdXBkYXRlXCIpKX0saXNQaHhVcGRhdGU6ZnVuY3Rpb24oZSx0LG4pe3JldHVybiBlLmdldEF0dHJpYnV0ZSYmbi5pbmRleE9mKGUuZ2V0QXR0cmlidXRlKHQpKT49MH0sZmluZFBoeENoaWxkcmVuOmZ1bmN0aW9uKGUsdCl7cmV0dXJuIHRoaXMuYWxsKGUsXCJcIi5jb25jYXQoVSxcIltcIikuY29uY2F0KFwiZGF0YS1waHgtcGFyZW50LWlkXCIsJz1cIicpLmNvbmNhdCh0LCdcIl0nKSl9LGZpbmRQYXJlbnRDSURzOmZ1bmN0aW9uKGUsdCl7dmFyIG49dGhpcyxpPW5ldyBTZXQodCk7cmV0dXJuIHQucmVkdWNlKGZ1bmN0aW9uKHQsaSl7dmFyIHI9XCJbXCIuY29uY2F0KEgsJz1cIicpLmNvbmNhdChpLCdcIl0gWycpLmNvbmNhdChILFwiXVwiKTtyZXR1cm4gbi5maWx0ZXJXaXRoaW5TYW1lTGl2ZVZpZXcobi5hbGwoZSxyKSxlKS5tYXAoZnVuY3Rpb24oZSl7cmV0dXJuIHBhcnNlSW50KGUuZ2V0QXR0cmlidXRlKEgpKX0pLmZvckVhY2goZnVuY3Rpb24oZSl7cmV0dXJuIHQuZGVsZXRlKGUpfSksdH0saSl9LGZpbHRlcldpdGhpblNhbWVMaXZlVmlldzpmdW5jdGlvbihlLHQpe3ZhciBuPXRoaXM7cmV0dXJuIHQucXVlcnlTZWxlY3RvcihVKT9lLmZpbHRlcihmdW5jdGlvbihlKXtyZXR1cm4gbi53aXRoaW5TYW1lTGl2ZVZpZXcoZSx0KX0pOmV9LHdpdGhpblNhbWVMaXZlVmlldzpmdW5jdGlvbihlLHQpe2Zvcig7ZT1lLnBhcmVudE5vZGU7KXtpZihlLmlzU2FtZU5vZGUodCkpcmV0dXJuITA7aWYoZS5nZXRBdHRyaWJ1dGUoTykpcmV0dXJuITF9fSxwcml2YXRlOmZ1bmN0aW9uKGUsdCl7cmV0dXJuIGUucGh4UHJpdmF0ZSYmZS5waHhQcml2YXRlW3RdfSxkZWxldGVQcml2YXRlOmZ1bmN0aW9uKGUsdCl7ZS5waHhQcml2YXRlJiZkZWxldGUgZS5waHhQcml2YXRlW3RdfSxwdXRQcml2YXRlOmZ1bmN0aW9uKGUsdCxuKXtlLnBoeFByaXZhdGV8fChlLnBoeFByaXZhdGU9e30pLGUucGh4UHJpdmF0ZVt0XT1ufSxjb3B5UHJpdmF0ZXM6ZnVuY3Rpb24oZSx0KXt0LnBoeFByaXZhdGUmJihlLnBoeFByaXZhdGU9Ryh0LnBoeFByaXZhdGUpKX0scHV0VGl0bGU6ZnVuY3Rpb24oZSl7dmFyIHQ9ZG9jdW1lbnQucXVlcnlTZWxlY3RvcihcInRpdGxlXCIpLmRhdGFzZXQsbj10LnByZWZpeCxpPXQuc3VmZml4O2RvY3VtZW50LnRpdGxlPVwiXCIuY29uY2F0KG58fFwiXCIpLmNvbmNhdChlKS5jb25jYXQoaXx8XCJcIil9LGRlYm91bmNlOmZ1bmN0aW9uKGUsdCxuLGkscixvLGEpe3ZhciB1PXRoaXMscz1lLmdldEF0dHJpYnV0ZShuKSxjPWUuZ2V0QXR0cmlidXRlKHIpO1wiXCI9PT1zJiYocz1pKSxcIlwiPT09YyYmKGM9byk7dmFyIGw9c3x8Yztzd2l0Y2gobCl7Y2FzZSBudWxsOnJldHVybiBhKCk7Y2FzZVwiYmx1clwiOnJldHVybiB2b2lkKHRoaXMub25jZShlLFwiZGVib3VuY2UtYmx1clwiKSYmZS5hZGRFdmVudExpc3RlbmVyKFwiYmx1clwiLGZ1bmN0aW9uKCl7cmV0dXJuIGEoKX0pKTtkZWZhdWx0OnZhciBkPXBhcnNlSW50KGwpLGg9dGhpcy5pbmNDeWNsZShlLFwiZGVib3VuY2UtdHJpZ2dlclwiLGZ1bmN0aW9uKCl7cmV0dXJuIGM/dS5kZWxldGVQcml2YXRlKGUsXCJ0aHJvdHRsZWRcIik6YSgpfSk7aWYoaXNOYU4oZCkpcmV0dXJuIEsoXCJpbnZhbGlkIHRocm90dGxlL2RlYm91bmNlIHZhbHVlOiBcIi5jb25jYXQobCkpO2lmKGMpe3ZhciBmPSExO2lmKFwia2V5ZG93blwiPT09dC50eXBlKXt2YXIgdj10aGlzLnByaXZhdGUoZSxcImRlYm91bmNlLXByZXYta2V5XCIpO3RoaXMucHV0UHJpdmF0ZShlLFwiZGVib3VuY2UtcHJldi1rZXlcIix0LmtleSksZj12IT09dC5rZXl9aWYoIWYmJnRoaXMucHJpdmF0ZShlLFwidGhyb3R0bGVkXCIpKXJldHVybiExO2EoKSx0aGlzLnB1dFByaXZhdGUoZSxcInRocm90dGxlZFwiLCEwKSxzZXRUaW1lb3V0KGZ1bmN0aW9uKCl7cmV0dXJuIHUudHJpZ2dlckN5Y2xlKGUsXCJkZWJvdW5jZS10cmlnZ2VyXCIpfSxkKX1lbHNlIHNldFRpbWVvdXQoZnVuY3Rpb24oKXtyZXR1cm4gdS50cmlnZ2VyQ3ljbGUoZSxcImRlYm91bmNlLXRyaWdnZXJcIixoKX0sZCk7ZS5mb3JtJiZ0aGlzLm9uY2UoZS5mb3JtLFwiYmluZC1kZWJvdW5jZVwiKSYmZS5mb3JtLmFkZEV2ZW50TGlzdGVuZXIoXCJzdWJtaXRcIixmdW5jdGlvbih0KXtBcnJheS5mcm9tKG5ldyBGb3JtRGF0YShlLmZvcm0pLmVudHJpZXMoKSxmdW5jdGlvbih0KXt2YXIgbj14KHQsMiksaT1uWzBdLHI9KG5bMV0sZS5mb3JtLnF1ZXJ5U2VsZWN0b3IoJ1tuYW1lPVwiJy5jb25jYXQoaSwnXCJdJykpKTt1LmluY0N5Y2xlKHIsXCJkZWJvdW5jZS10cmlnZ2VyXCIpLHUuZGVsZXRlUHJpdmF0ZShyLFwidGhyb3R0bGVkXCIpfSl9KSx0aGlzLm9uY2UoZSxcImJpbmQtZGVib3VuY2VcIikmJmUuYWRkRXZlbnRMaXN0ZW5lcihcImJsdXJcIixmdW5jdGlvbih0KXtyZXR1cm4gdS50cmlnZ2VyQ3ljbGUoZSxcImRlYm91bmNlLXRyaWdnZXJcIil9KX19LHRyaWdnZXJDeWNsZTpmdW5jdGlvbihlLHQsbil7dmFyIGk9eCh0aGlzLnByaXZhdGUoZSx0KSwyKSxyPWlbMF0sbz1pWzFdO258fChuPXIpLG49PT1yJiYodGhpcy5pbmNDeWNsZShlLHQpLG8oKSl9LG9uY2U6ZnVuY3Rpb24oZSx0KXtyZXR1cm4hMCE9PXRoaXMucHJpdmF0ZShlLHQpJiYodGhpcy5wdXRQcml2YXRlKGUsdCwhMCksITApfSxpbmNDeWNsZTpmdW5jdGlvbihlLHQpe3ZhciBuPWFyZ3VtZW50cy5sZW5ndGg+MiYmdm9pZCAwIT09YXJndW1lbnRzWzJdP2FyZ3VtZW50c1syXTpmdW5jdGlvbigpe30saT14KHRoaXMucHJpdmF0ZShlLHQpfHxbMCxuXSwyKSxyPWlbMF07aVsxXTtyZXR1cm4gcisrLHRoaXMucHV0UHJpdmF0ZShlLHQsW3Isbl0pLHJ9LGRpc2NhcmRFcnJvcjpmdW5jdGlvbihlLHQsbil7dmFyIGk9dC5nZXRBdHRyaWJ1dGUmJnQuZ2V0QXR0cmlidXRlKG4pLHI9aSYmZS5xdWVyeVNlbGVjdG9yKFwiI1wiLmNvbmNhdChpLCcsIFtuYW1lPVwiJykuY29uY2F0KGksJ1wiXScpKTtyJiYodGhpcy5wcml2YXRlKHIsXCJwaHgtaGFzLWZvY3VzZWRcIil8fHRoaXMucHJpdmF0ZShyLmZvcm0sXCJwaHgtaGFzLXN1Ym1pdHRlZFwiKXx8dC5jbGFzc0xpc3QuYWRkKFwicGh4LW5vLWZlZWRiYWNrXCIpKX0sc2hvd0Vycm9yOmZ1bmN0aW9uKGUsdCl7dmFyIG49dGhpczsoZS5pZHx8ZS5uYW1lKSYmdGhpcy5hbGwoZS5mb3JtLFwiW1wiLmNvbmNhdCh0LCc9XCInKS5jb25jYXQoZS5pZCwnXCJdLCBbJykuY29uY2F0KHQsJz1cIicpLmNvbmNhdChlLm5hbWUsJ1wiXScpLGZ1bmN0aW9uKGUpe24ucmVtb3ZlQ2xhc3MoZSxcInBoeC1uby1mZWVkYmFja1wiKX0pfSxpc1BoeENoaWxkOmZ1bmN0aW9uKGUpe3JldHVybiBlLmdldEF0dHJpYnV0ZSYmZS5nZXRBdHRyaWJ1dGUoXCJkYXRhLXBoeC1wYXJlbnQtaWRcIil9LGRpc3BhdGNoRXZlbnQ6ZnVuY3Rpb24oZSx0KXt2YXIgbj1hcmd1bWVudHMubGVuZ3RoPjImJnZvaWQgMCE9PWFyZ3VtZW50c1syXT9hcmd1bWVudHNbMl06e30saT1uZXcgQ3VzdG9tRXZlbnQodCx7YnViYmxlczohMCxjYW5jZWxhYmxlOiEwLGRldGFpbDpufSk7ZS5kaXNwYXRjaEV2ZW50KGkpfSxjbG9uZU5vZGU6ZnVuY3Rpb24oZSx0KXtpZih2b2lkIDA9PT10KXJldHVybiBlLmNsb25lTm9kZSghMCk7dmFyIG49ZS5jbG9uZU5vZGUoITEpO3JldHVybiBuLmlubmVySFRNTD10LG59LG1lcmdlQXR0cnM6ZnVuY3Rpb24oZSx0KXtmb3IodmFyIG49YXJndW1lbnRzLmxlbmd0aD4yJiZ2b2lkIDAhPT1hcmd1bWVudHNbMl0/YXJndW1lbnRzWzJdOnt9LGk9bi5leGNsdWRlfHxbXSxyPW4uaXNJZ25vcmVkLG89dC5hdHRyaWJ1dGVzLGE9by5sZW5ndGgtMTthPj0wO2EtLSl7dmFyIHU9b1thXS5uYW1lO2kuaW5kZXhPZih1KTwwJiZlLnNldEF0dHJpYnV0ZSh1LHQuZ2V0QXR0cmlidXRlKHUpKX1mb3IodmFyIHM9ZS5hdHRyaWJ1dGVzLGM9cy5sZW5ndGgtMTtjPj0wO2MtLSl7dmFyIGw9c1tjXS5uYW1lO3I/bC5zdGFydHNXaXRoKFwiZGF0YS1cIikmJiF0Lmhhc0F0dHJpYnV0ZShsKSYmZS5yZW1vdmVBdHRyaWJ1dGUobCk6dC5oYXNBdHRyaWJ1dGUobCl8fGUucmVtb3ZlQXR0cmlidXRlKGwpfX0sbWVyZ2VGb2N1c2VkSW5wdXQ6ZnVuY3Rpb24oZSx0KXtlIGluc3RhbmNlb2YgSFRNTFNlbGVjdEVsZW1lbnR8fGRlLm1lcmdlQXR0cnMoZSx0LHtleGNlcHQ6W1widmFsdWVcIl19KSx0LnJlYWRPbmx5P2Uuc2V0QXR0cmlidXRlKFwicmVhZG9ubHlcIiwhMCk6ZS5yZW1vdmVBdHRyaWJ1dGUoXCJyZWFkb25seVwiKX0saGFzU2VsZWN0aW9uUmFuZ2U6ZnVuY3Rpb24oZSl7cmV0dXJuIGUuc2V0U2VsZWN0aW9uUmFuZ2UmJihcInRleHRcIj09PWUudHlwZXx8XCJ0ZXh0YXJlYVwiPT09ZS50eXBlKX0scmVzdG9yZUZvY3VzOmZ1bmN0aW9uKGUsdCxuKXtpZihkZS5pc1RleHR1YWxJbnB1dChlKSl7dmFyIGk9ZS5tYXRjaGVzKFwiOmZvY3VzXCIpO2UucmVhZE9ubHkmJmUuYmx1cigpLGl8fGUuZm9jdXMoKSx0aGlzLmhhc1NlbGVjdGlvblJhbmdlKGUpJiZlLnNldFNlbGVjdGlvblJhbmdlKHQsbil9fSxpc0Zvcm1JbnB1dDpmdW5jdGlvbihlKXtyZXR1cm4vXig/OmlucHV0fHNlbGVjdHx0ZXh0YXJlYSkkL2kudGVzdChlLnRhZ05hbWUpJiZcImJ1dHRvblwiIT09ZS50eXBlfSxzeW5jQXR0cnNUb1Byb3BzOmZ1bmN0aW9uKGUpe2UgaW5zdGFuY2VvZiBIVE1MSW5wdXRFbGVtZW50JiZKLmluZGV4T2YoZS50eXBlLnRvTG9jYWxlTG93ZXJDYXNlKCkpPj0wJiYoZS5jaGVja2VkPW51bGwhPT1lLmdldEF0dHJpYnV0ZShcImNoZWNrZWRcIikpfSxpc1RleHR1YWxJbnB1dDpmdW5jdGlvbihlKXtyZXR1cm4gQi5pbmRleE9mKGUudHlwZSk+PTB9LGlzTm93VHJpZ2dlckZvcm1FeHRlcm5hbDpmdW5jdGlvbihlLHQpe3JldHVybiBlLmdldEF0dHJpYnV0ZSYmbnVsbCE9PWUuZ2V0QXR0cmlidXRlKHQpfSxzeW5jUGVuZGluZ1JlZjpmdW5jdGlvbihlLHQsbil7dmFyIGk9ZS5nZXRBdHRyaWJ1dGUoRik7cmV0dXJuIG51bGw9PT1pfHwoZGUuaXNGb3JtSW5wdXQoZSl8fG51bGwhPT1lLmdldEF0dHJpYnV0ZShuKT8oZGUuaXNVcGxvYWRJbnB1dChlKSYmZGUubWVyZ2VBdHRycyhlLHQse2lzSWdub3JlZDohMH0pLGRlLnB1dFByaXZhdGUoZSxGLHQpLCExKTooai5mb3JFYWNoKGZ1bmN0aW9uKG4pe2UuY2xhc3NMaXN0LmNvbnRhaW5zKG4pJiZ0LmNsYXNzTGlzdC5hZGQobil9KSx0LnNldEF0dHJpYnV0ZShGLGkpLCEwKSl9LGNsZWFuQ2hpbGROb2RlczpmdW5jdGlvbihlLHQpe2lmKGRlLmlzUGh4VXBkYXRlKGUsdCxbXCJhcHBlbmRcIixcInByZXBlbmRcIl0pKXt2YXIgbj1bXTtlLmNoaWxkTm9kZXMuZm9yRWFjaChmdW5jdGlvbihlKXtlLmlkfHwoZS5ub2RlVHlwZT09PU5vZGUuVEVYVF9OT0RFJiZcIlwiPT09ZS5ub2RlVmFsdWUudHJpbSgpfHxLKFwib25seSBIVE1MIGVsZW1lbnQgdGFncyB3aXRoIGFuIGlkIGFyZSBhbGxvd2VkIGluc2lkZSBjb250YWluZXJzIHdpdGggcGh4LXVwZGF0ZS5cXG5cXG5cIisncmVtb3ZpbmcgaWxsZWdhbCBub2RlOiBcIicuY29uY2F0KChlLm91dGVySFRNTHx8ZS5ub2RlVmFsdWUpLnRyaW0oKSwnXCJcXG5cXG4nKSksbi5wdXNoKGUpKX0pLG4uZm9yRWFjaChmdW5jdGlvbihlKXtyZXR1cm4gZS5yZW1vdmUoKX0pfX19LGhlPWZ1bmN0aW9uKCl7ZnVuY3Rpb24gZSh0LG4saSl7VCh0aGlzLGUpO3ZhciByPW5ldyBTZXQsbz1uZXcgU2V0KEEobi5jaGlsZHJlbikubWFwKGZ1bmN0aW9uKGUpe3JldHVybiBlLmlkfSkpLGE9W107QXJyYXkuZnJvbSh0LmNoaWxkcmVuKS5mb3JFYWNoKGZ1bmN0aW9uKGUpe2lmKGUuaWQmJihyLmFkZChlLmlkKSxvLmhhcyhlLmlkKSkpe3ZhciB0PWUucHJldmlvdXNFbGVtZW50U2libGluZyYmZS5wcmV2aW91c0VsZW1lbnRTaWJsaW5nLmlkO2EucHVzaCh7ZWxlbWVudElkOmUuaWQscHJldmlvdXNFbGVtZW50SWQ6dH0pfX0pLHRoaXMuY29udGFpbmVySWQ9bi5pZCx0aGlzLnVwZGF0ZVR5cGU9aSx0aGlzLmVsZW1lbnRzVG9Nb2RpZnk9YSx0aGlzLmVsZW1lbnRJZHNUb0FkZD1BKG8pLmZpbHRlcihmdW5jdGlvbihlKXtyZXR1cm4hci5oYXMoZSl9KX1yZXR1cm4gXyhlLFt7a2V5OlwicGVyZm9ybVwiLHZhbHVlOmZ1bmN0aW9uKCl7dmFyIGU9ZGUuYnlJZCh0aGlzLmNvbnRhaW5lcklkKTt0aGlzLmVsZW1lbnRzVG9Nb2RpZnkuZm9yRWFjaChmdW5jdGlvbih0KXt0LnByZXZpb3VzRWxlbWVudElkP2VlKGRvY3VtZW50LmdldEVsZW1lbnRCeUlkKHQucHJldmlvdXNFbGVtZW50SWQpLGZ1bmN0aW9uKGUpe2VlKGRvY3VtZW50LmdldEVsZW1lbnRCeUlkKHQuZWxlbWVudElkKSxmdW5jdGlvbih0KXt0LnByZXZpb3VzRWxlbWVudFNpYmxpbmcmJnQucHJldmlvdXNFbGVtZW50U2libGluZy5pZD09ZS5pZHx8ZS5pbnNlcnRBZGphY2VudEVsZW1lbnQoXCJhZnRlcmVuZFwiLHQpfSl9KTplZShkb2N1bWVudC5nZXRFbGVtZW50QnlJZCh0LmVsZW1lbnRJZCksZnVuY3Rpb24odCl7bnVsbD09dC5wcmV2aW91c0VsZW1lbnRTaWJsaW5nfHxlLmluc2VydEFkamFjZW50RWxlbWVudChcImFmdGVyYmVnaW5cIix0KX0pfSksXCJwcmVwZW5kXCI9PXRoaXMudXBkYXRlVHlwZSYmdGhpcy5lbGVtZW50SWRzVG9BZGQucmV2ZXJzZSgpLmZvckVhY2goZnVuY3Rpb24odCl7ZWUoZG9jdW1lbnQuZ2V0RWxlbWVudEJ5SWQodCksZnVuY3Rpb24odCl7cmV0dXJuIGUuaW5zZXJ0QWRqYWNlbnRFbGVtZW50KFwiYWZ0ZXJiZWdpblwiLHQpfSl9KX19XSksZX0oKSxmZT1mdW5jdGlvbigpe2Z1bmN0aW9uIGUodCxuLGkscixvKXtUKHRoaXMsZSksdGhpcy52aWV3PXQsdGhpcy5saXZlU29ja2V0PXQubGl2ZVNvY2tldCx0aGlzLmNvbnRhaW5lcj1uLHRoaXMuaWQ9aSx0aGlzLnJvb3RJRD10LnJvb3QuaWQsdGhpcy5odG1sPXIsdGhpcy50YXJnZXRDSUQ9byx0aGlzLmNpZFBhdGNoPVwibnVtYmVyXCI9PXR5cGVvZiB0aGlzLnRhcmdldENJRCx0aGlzLmNhbGxiYWNrcz17YmVmb3JlYWRkZWQ6W10sYmVmb3JldXBkYXRlZDpbXSxiZWZvcmVwaHhDaGlsZEFkZGVkOltdLGFmdGVyYWRkZWQ6W10sYWZ0ZXJ1cGRhdGVkOltdLGFmdGVyZGlzY2FyZGVkOltdLGFmdGVycGh4Q2hpbGRBZGRlZDpbXX19cmV0dXJuIF8oZSxudWxsLFt7a2V5OlwicGF0Y2hFbFwiLHZhbHVlOmZ1bmN0aW9uKGUsdCxuKXtiKGUsdCx7Y2hpbGRyZW5Pbmx5OiExLG9uQmVmb3JlRWxVcGRhdGVkOmZ1bmN0aW9uKGUsdCl7aWYobiYmbi5pc1NhbWVOb2RlKGUpJiZkZS5pc0Zvcm1JbnB1dChlKSlyZXR1cm4gZGUubWVyZ2VGb2N1c2VkSW5wdXQoZSx0KSwhMX19KX19XSksXyhlLFt7a2V5OlwiYmVmb3JlXCIsdmFsdWU6ZnVuY3Rpb24oZSx0KXt0aGlzLmNhbGxiYWNrc1tcImJlZm9yZVwiLmNvbmNhdChlKV0ucHVzaCh0KX19LHtrZXk6XCJhZnRlclwiLHZhbHVlOmZ1bmN0aW9uKGUsdCl7dGhpcy5jYWxsYmFja3NbXCJhZnRlclwiLmNvbmNhdChlKV0ucHVzaCh0KX19LHtrZXk6XCJ0cmFja0JlZm9yZVwiLHZhbHVlOmZ1bmN0aW9uKGUpe2Zvcih2YXIgdD1hcmd1bWVudHMubGVuZ3RoLG49bmV3IEFycmF5KHQ+MT90LTE6MCksaT0xO2k8dDtpKyspbltpLTFdPWFyZ3VtZW50c1tpXTt0aGlzLmNhbGxiYWNrc1tcImJlZm9yZVwiLmNvbmNhdChlKV0uZm9yRWFjaChmdW5jdGlvbihlKXtyZXR1cm4gZS5hcHBseSh2b2lkIDAsbil9KX19LHtrZXk6XCJ0cmFja0FmdGVyXCIsdmFsdWU6ZnVuY3Rpb24oZSl7Zm9yKHZhciB0PWFyZ3VtZW50cy5sZW5ndGgsbj1uZXcgQXJyYXkodD4xP3QtMTowKSxpPTE7aTx0O2krKyluW2ktMV09YXJndW1lbnRzW2ldO3RoaXMuY2FsbGJhY2tzW1wiYWZ0ZXJcIi5jb25jYXQoZSldLmZvckVhY2goZnVuY3Rpb24oZSl7cmV0dXJuIGUuYXBwbHkodm9pZCAwLG4pfSl9fSx7a2V5OlwibWFya1BydW5hYmxlQ29udGVudEZvclJlbW92YWxcIix2YWx1ZTpmdW5jdGlvbigpe2RlLmFsbCh0aGlzLmNvbnRhaW5lcixcIltwaHgtdXBkYXRlPWFwcGVuZF0gPiAqLCBbcGh4LXVwZGF0ZT1wcmVwZW5kXSA+ICpcIixmdW5jdGlvbihlKXtlLnNldEF0dHJpYnV0ZShcImRhdGEtcGh4LXJlbW92ZVwiLFwiXCIpfSl9fSx7a2V5OlwicGVyZm9ybVwiLHZhbHVlOmZ1bmN0aW9uKCl7dmFyIGU9dGhpcyx0PXRoaXMudmlldyxuPXRoaXMubGl2ZVNvY2tldCxpPXRoaXMuY29udGFpbmVyLHI9dGhpcy5odG1sLG89dGhpcy5pc0NJRFBhdGNoKCk/dGhpcy50YXJnZXRDSURDb250YWluZXIocik6aTtpZighdGhpcy5pc0NJRFBhdGNoKCl8fG8pe3ZhciBhPW4uZ2V0QWN0aXZlRWxlbWVudCgpLHU9YSYmZGUuaGFzU2VsZWN0aW9uUmFuZ2UoYSk/YTp7fSxzPXUuc2VsZWN0aW9uU3RhcnQsYz11LnNlbGVjdGlvbkVuZCxsPW4uYmluZGluZyhcInVwZGF0ZVwiKSxkPW4uYmluZGluZyhcImZlZWRiYWNrLWZvclwiKSxoPW4uYmluZGluZyhcImRpc2FibGUtd2l0aFwiKSxmPW4uYmluZGluZyhcInRyaWdnZXItYWN0aW9uXCIpLHY9W10scD1bXSxnPVtdLG09bnVsbCx5PW4udGltZShcInByZW1vcnBoIGNvbnRhaW5lciBwcmVwXCIsZnVuY3Rpb24oKXtyZXR1cm4gZS5idWlsZERpZmZIVE1MKGkscixsLG8pfSk7cmV0dXJuIHRoaXMudHJhY2tCZWZvcmUoXCJhZGRlZFwiLGkpLHRoaXMudHJhY2tCZWZvcmUoXCJ1cGRhdGVkXCIsaSxpKSxuLnRpbWUoXCJtb3JwaGRvbVwiLGZ1bmN0aW9uKCl7YihvLHkse2NoaWxkcmVuT25seTpudWxsPT09by5nZXRBdHRyaWJ1dGUoSCksZ2V0Tm9kZUtleTpmdW5jdGlvbihlKXtyZXR1cm4gZS5pZCYmZS5pZCsoZS5nZXRBdHRyaWJ1dGUoXCJkYXRhLXBoeC1zZXNzaW9uXCIpfHxcIlwiKX0sb25CZWZvcmVOb2RlQWRkZWQ6ZnVuY3Rpb24odCl7cmV0dXJuIGRlLmRpc2NhcmRFcnJvcihvLHQsZCksZS50cmFja0JlZm9yZShcImFkZGVkXCIsdCksdH0sb25Ob2RlQWRkZWQ6ZnVuY3Rpb24obil7ZGUuaXNOb3dUcmlnZ2VyRm9ybUV4dGVybmFsKG4sZikmJihtPW4pLGRlLmlzUGh4Q2hpbGQobikmJnQub3duc0VsZW1lbnQobikmJmUudHJhY2tBZnRlcihcInBoeENoaWxkQWRkZWRcIixuKSx2LnB1c2gobil9LG9uTm9kZURpc2NhcmRlZDpmdW5jdGlvbih0KXtkZS5pc1BoeENoaWxkKHQpJiZuLmRlc3Ryb3lWaWV3QnlFbCh0KSxlLnRyYWNrQWZ0ZXIoXCJkaXNjYXJkZWRcIix0KX0sb25CZWZvcmVOb2RlRGlzY2FyZGVkOmZ1bmN0aW9uKHQpe3JldHVybiEoIXQuZ2V0QXR0cmlidXRlfHxudWxsPT09dC5nZXRBdHRyaWJ1dGUoXCJkYXRhLXBoeC1yZW1vdmVcIikpfHwobnVsbD09PXQucGFyZW50Tm9kZXx8IWRlLmlzUGh4VXBkYXRlKHQucGFyZW50Tm9kZSxsLFtcImFwcGVuZFwiLFwicHJlcGVuZFwiXSl8fCF0LmlkKSYmIWUuc2tpcENJRFNpYmxpbmcodCl9LG9uRWxVcGRhdGVkOmZ1bmN0aW9uKGUpe2RlLmlzTm93VHJpZ2dlckZvcm1FeHRlcm5hbChlLGYpJiYobT1lKSxwLnB1c2goZSl9LG9uQmVmb3JlRWxVcGRhdGVkOmZ1bmN0aW9uKHQsbil7aWYoZGUuY2xlYW5DaGlsZE5vZGVzKG4sbCksZS5za2lwQ0lEU2libGluZyhuKSlyZXR1cm4hMTtpZihkZS5pc0lnbm9yZWQodCxsKSlyZXR1cm4gZS50cmFja0JlZm9yZShcInVwZGF0ZWRcIix0LG4pLGRlLm1lcmdlQXR0cnModCxuLHtpc0lnbm9yZWQ6ITB9KSxwLnB1c2godCksITE7aWYoXCJudW1iZXJcIj09PXQudHlwZSYmdC52YWxpZGl0eSYmdC52YWxpZGl0eS5iYWRJbnB1dClyZXR1cm4hMTtpZighZGUuc3luY1BlbmRpbmdSZWYodCxuLGgpKXJldHVybiBkZS5pc1VwbG9hZElucHV0KHQpJiYoZS50cmFja0JlZm9yZShcInVwZGF0ZWRcIix0LG4pLHAucHVzaCh0KSksITE7aWYoZGUuaXNQaHhDaGlsZChuKSl7dmFyIGk9dC5nZXRBdHRyaWJ1dGUoVik7cmV0dXJuIGRlLm1lcmdlQXR0cnModCxuKSx0LnNldEF0dHJpYnV0ZShWLGkpLHQuc2V0QXR0cmlidXRlKFwiZGF0YS1waHgtcm9vdC1pZFwiLGUucm9vdElEKSwhMX1yZXR1cm4gZGUuY29weVByaXZhdGVzKG4sdCksZGUuZGlzY2FyZEVycm9yKG8sbixkKSxhJiZ0LmlzU2FtZU5vZGUoYSkmJmRlLmlzRm9ybUlucHV0KHQpJiYhZS5mb3JjZUZvY3VzZWRTZWxlY3RVcGRhdGUodCxuKT8oZS50cmFja0JlZm9yZShcInVwZGF0ZWRcIix0LG4pLGRlLm1lcmdlRm9jdXNlZElucHV0KHQsbiksZGUuc3luY0F0dHJzVG9Qcm9wcyh0KSxwLnB1c2godCksITEpOihkZS5pc1BoeFVwZGF0ZShuLGwsW1wiYXBwZW5kXCIsXCJwcmVwZW5kXCJdKSYmZy5wdXNoKG5ldyBoZSh0LG4sbi5nZXRBdHRyaWJ1dGUobCkpKSxkZS5zeW5jQXR0cnNUb1Byb3BzKG4pLGUudHJhY2tCZWZvcmUoXCJ1cGRhdGVkXCIsdCxuKSwhMCl9fSl9KSxuLmlzRGVidWdFbmFibGVkKCkmJmZ1bmN0aW9uKCl7Zm9yKHZhciBlPW5ldyBTZXQsdD1kb2N1bWVudC5xdWVyeVNlbGVjdG9yQWxsKFwiKltpZF1cIiksbj0wLGk9dC5sZW5ndGg7bjxpO24rKyllLmhhcyh0W25dLmlkKT9jb25zb2xlLmVycm9yKFwiTXVsdGlwbGUgSURzIGRldGVjdGVkOiBcIi5jb25jYXQodFtuXS5pZCxcIi4gRW5zdXJlIHVuaXF1ZSBlbGVtZW50IGlkcy5cIikpOmUuYWRkKHRbbl0uaWQpfSgpLGcubGVuZ3RoPjAmJm4udGltZShcInBvc3QtbW9ycGggYXBwZW5kL3ByZXBlbmQgcmVzdG9yYXRpb25cIixmdW5jdGlvbigpe2cuZm9yRWFjaChmdW5jdGlvbihlKXtyZXR1cm4gZS5wZXJmb3JtKCl9KX0pLG4uc2lsZW5jZUV2ZW50cyhmdW5jdGlvbigpe3JldHVybiBkZS5yZXN0b3JlRm9jdXMoYSxzLGMpfSksZGUuZGlzcGF0Y2hFdmVudChkb2N1bWVudCxcInBoeDp1cGRhdGVcIiksdi5mb3JFYWNoKGZ1bmN0aW9uKHQpe3JldHVybiBlLnRyYWNrQWZ0ZXIoXCJhZGRlZFwiLHQpfSkscC5mb3JFYWNoKGZ1bmN0aW9uKHQpe3JldHVybiBlLnRyYWNrQWZ0ZXIoXCJ1cGRhdGVkXCIsdCl9KSxtJiYobi5kaXNjb25uZWN0KCksbS5zdWJtaXQoKSksITB9fX0se2tleTpcImZvcmNlRm9jdXNlZFNlbGVjdFVwZGF0ZVwiLHZhbHVlOmZ1bmN0aW9uKGUsdCl7cmV0dXJuITA9PT1lLm11bHRpcGxlfHxcInNlbGVjdFwiPT09ZS50eXBlJiZlLmlubmVySFRNTCE9dC5pbm5lckhUTUx9fSx7a2V5OlwiaXNDSURQYXRjaFwiLHZhbHVlOmZ1bmN0aW9uKCl7cmV0dXJuIHRoaXMuY2lkUGF0Y2h9fSx7a2V5Olwic2tpcENJRFNpYmxpbmdcIix2YWx1ZTpmdW5jdGlvbihlKXtyZXR1cm4gZS5ub2RlVHlwZT09PU5vZGUuRUxFTUVOVF9OT0RFJiZudWxsIT09ZS5nZXRBdHRyaWJ1dGUoXCJkYXRhLXBoeC1za2lwXCIpfX0se2tleTpcInRhcmdldENJRENvbnRhaW5lclwiLHZhbHVlOmZ1bmN0aW9uKGUpe2lmKHRoaXMuaXNDSURQYXRjaCgpKXt2YXIgdD1rKGRlLmZpbmRDb21wb25lbnROb2RlTGlzdCh0aGlzLmNvbnRhaW5lcix0aGlzLnRhcmdldENJRCkpLG49dFswXTtyZXR1cm4gMD09PXQuc2xpY2UoMSkubGVuZ3RoJiYxPT09ZGUuY2hpbGROb2RlTGVuZ3RoKGUpP246biYmbi5wYXJlbnROb2RlfX19LHtrZXk6XCJidWlsZERpZmZIVE1MXCIsdmFsdWU6ZnVuY3Rpb24oZSx0LG4saSl7dmFyIHI9dGhpcyxvPXRoaXMuaXNDSURQYXRjaCgpLGE9byYmaS5nZXRBdHRyaWJ1dGUoSCk9PT10aGlzLnRhcmdldENJRC50b1N0cmluZygpO2lmKCFvfHxhKXJldHVybiB0O3ZhciB1PW51bGwscz1kb2N1bWVudC5jcmVhdGVFbGVtZW50KFwidGVtcGxhdGVcIik7dT1kZS5jbG9uZU5vZGUoaSk7dmFyIGM9ayhkZS5maW5kQ29tcG9uZW50Tm9kZUxpc3QodSx0aGlzLnRhcmdldENJRCkpLGw9Y1swXSxkPWMuc2xpY2UoMSk7cmV0dXJuIHMuaW5uZXJIVE1MPXQsZC5mb3JFYWNoKGZ1bmN0aW9uKGUpe3JldHVybiBlLnJlbW92ZSgpfSksQXJyYXkuZnJvbSh1LmNoaWxkTm9kZXMpLmZvckVhY2goZnVuY3Rpb24oZSl7ZS5pZCYmZS5ub2RlVHlwZT09PU5vZGUuRUxFTUVOVF9OT0RFJiZlLmdldEF0dHJpYnV0ZShIKSE9PXIudGFyZ2V0Q0lELnRvU3RyaW5nKCkmJihlLnNldEF0dHJpYnV0ZShcImRhdGEtcGh4LXNraXBcIixcIlwiKSxlLmlubmVySFRNTD1cIlwiKX0pLEFycmF5LmZyb20ocy5jb250ZW50LmNoaWxkTm9kZXMpLmZvckVhY2goZnVuY3Rpb24oZSl7cmV0dXJuIHUuaW5zZXJ0QmVmb3JlKGUsbCl9KSxsLnJlbW92ZSgpLHUub3V0ZXJIVE1MfX1dKSxlfSgpLHZlPWZ1bmN0aW9uKCl7ZnVuY3Rpb24gZSh0LG4saSxyLG8pe3ZhciBhPXRoaXM7VCh0aGlzLGUpLHRoaXMubGl2ZVNvY2tldD1uLHRoaXMuZmxhc2g9byx0aGlzLnBhcmVudD1pLHRoaXMucm9vdD1pP2kucm9vdDp0aGlzLHRoaXMuZWw9dCx0aGlzLmlkPXRoaXMuZWwuaWQsdGhpcy52aWV3PXRoaXMuZWwuZ2V0QXR0cmlidXRlKE8pLHRoaXMucmVmPTAsdGhpcy5jaGlsZEpvaW5zPTAsdGhpcy5sb2FkZXJUaW1lcj1udWxsLHRoaXMucGVuZGluZ0RpZmZzPVtdLHRoaXMucHJ1bmluZ0NJRHM9W10sdGhpcy5ocmVmPXIsdGhpcy5qb2luQ291bnQ9dGhpcy5wYXJlbnQ/dGhpcy5wYXJlbnQuam9pbkNvdW50LTE6MCx0aGlzLmpvaW5QZW5kaW5nPSEwLHRoaXMuZGVzdHJveWVkPSExLHRoaXMuam9pbkNhbGxiYWNrPWZ1bmN0aW9uKCl7fSx0aGlzLnN0b3BDYWxsYmFjaz1mdW5jdGlvbigpe30sdGhpcy5wZW5kaW5nSm9pbk9wcz10aGlzLnBhcmVudD9udWxsOltdLHRoaXMudmlld0hvb2tzPXt9LHRoaXMudXBsb2FkZXJzPXt9LHRoaXMuZm9ybVN1Ym1pdHM9W10sdGhpcy5jaGlsZHJlbj10aGlzLnBhcmVudD9udWxsOnt9LHRoaXMucm9vdC5jaGlsZHJlblt0aGlzLmlkXT17fSx0aGlzLmNoYW5uZWw9dGhpcy5saXZlU29ja2V0LmNoYW5uZWwoXCJsdjpcIi5jb25jYXQodGhpcy5pZCksZnVuY3Rpb24oKXtyZXR1cm57dXJsOmEuaHJlZixwYXJhbXM6YS5jb25uZWN0UGFyYW1zKCksc2Vzc2lvbjphLmdldFNlc3Npb24oKSxzdGF0aWM6YS5nZXRTdGF0aWMoKSxmbGFzaDphLmZsYXNofX0pLHRoaXMuc2hvd0xvYWRlcih0aGlzLmxpdmVTb2NrZXQubG9hZGVyVGltZW91dCksdGhpcy5iaW5kQ2hhbm5lbCgpfXJldHVybiBfKGUsW3trZXk6XCJpc01haW5cIix2YWx1ZTpmdW5jdGlvbigpe3JldHVybiB0aGlzLmxpdmVTb2NrZXQubWFpbj09PXRoaXN9fSx7a2V5OlwiY29ubmVjdFBhcmFtc1wiLHZhbHVlOmZ1bmN0aW9uKCl7dmFyIGU9dGhpcy5saXZlU29ja2V0LnBhcmFtcyh0aGlzLnZpZXcpLHQ9ZGUuYWxsKGRvY3VtZW50LFwiW1wiLmNvbmNhdCh0aGlzLmJpbmRpbmcoXCJ0cmFjay1zdGF0aWNcIiksXCJdXCIpKS5tYXAoZnVuY3Rpb24oZSl7cmV0dXJuIGUuc3JjfHxlLmhyZWZ9KS5maWx0ZXIoZnVuY3Rpb24oZSl7cmV0dXJuXCJzdHJpbmdcIj09dHlwZW9mIGV9KTtyZXR1cm4gdC5sZW5ndGg+MCYmKGUuX3RyYWNrX3N0YXRpYz10KSxlLl9tb3VudHM9dGhpcy5qb2luQ291bnQsZX19LHtrZXk6XCJuYW1lXCIsdmFsdWU6ZnVuY3Rpb24oKXtyZXR1cm4gdGhpcy52aWV3fX0se2tleTpcImlzQ29ubmVjdGVkXCIsdmFsdWU6ZnVuY3Rpb24oKXtyZXR1cm4gdGhpcy5jaGFubmVsLmNhblB1c2goKX19LHtrZXk6XCJnZXRTZXNzaW9uXCIsdmFsdWU6ZnVuY3Rpb24oKXtyZXR1cm4gdGhpcy5lbC5nZXRBdHRyaWJ1dGUoXCJkYXRhLXBoeC1zZXNzaW9uXCIpfX0se2tleTpcImdldFN0YXRpY1wiLHZhbHVlOmZ1bmN0aW9uKCl7dmFyIGU9dGhpcy5lbC5nZXRBdHRyaWJ1dGUoVik7cmV0dXJuXCJcIj09PWU/bnVsbDplfX0se2tleTpcImRlc3Ryb3lcIix2YWx1ZTpmdW5jdGlvbigpe3ZhciBlPXRoaXMsdD1hcmd1bWVudHMubGVuZ3RoPjAmJnZvaWQgMCE9PWFyZ3VtZW50c1swXT9hcmd1bWVudHNbMF06ZnVuY3Rpb24oKXt9O3RoaXMuZGVzdHJveUFsbENoaWxkcmVuKCksdGhpcy5kZXN0cm95ZWQ9ITAsZGVsZXRlIHRoaXMucm9vdC5jaGlsZHJlblt0aGlzLmlkXSx0aGlzLnBhcmVudCYmZGVsZXRlIHRoaXMucm9vdC5jaGlsZHJlblt0aGlzLnBhcmVudC5pZF1bdGhpcy5pZF0sY2xlYXJUaW1lb3V0KHRoaXMubG9hZGVyVGltZXIpO3ZhciBuPWZ1bmN0aW9uKCl7Zm9yKHZhciBuIGluIHQoKSxlLnZpZXdIb29rcyllLmRlc3Ryb3lIb29rKGUudmlld0hvb2tzW25dKX07dGhpcy5sb2coXCJkZXN0cm95ZWRcIixmdW5jdGlvbigpe3JldHVybltcInRoZSBjaGlsZCBoYXMgYmVlbiByZW1vdmVkIGZyb20gdGhlIHBhcmVudFwiXX0pLHRoaXMuY2hhbm5lbC5sZWF2ZSgpLnJlY2VpdmUoXCJva1wiLG4pLnJlY2VpdmUoXCJlcnJvclwiLG4pLnJlY2VpdmUoXCJ0aW1lb3V0XCIsbil9fSx7a2V5Olwic2V0Q29udGFpbmVyQ2xhc3Nlc1wiLHZhbHVlOmZ1bmN0aW9uKCl7dmFyIGU7dGhpcy5lbC5jbGFzc0xpc3QucmVtb3ZlKFwicGh4LWNvbm5lY3RlZFwiLFwicGh4LWRpc2Nvbm5lY3RlZFwiLFwicGh4LWVycm9yXCIpLChlPXRoaXMuZWwuY2xhc3NMaXN0KS5hZGQuYXBwbHkoZSxhcmd1bWVudHMpfX0se2tleTpcImlzTG9hZGluZ1wiLHZhbHVlOmZ1bmN0aW9uKCl7cmV0dXJuIHRoaXMuZWwuY2xhc3NMaXN0LmNvbnRhaW5zKFwicGh4LWRpc2Nvbm5lY3RlZFwiKX19LHtrZXk6XCJzaG93TG9hZGVyXCIsdmFsdWU6ZnVuY3Rpb24oZSl7dmFyIHQ9dGhpcztpZihjbGVhclRpbWVvdXQodGhpcy5sb2FkZXJUaW1lciksZSl0aGlzLmxvYWRlclRpbWVyPXNldFRpbWVvdXQoZnVuY3Rpb24oKXtyZXR1cm4gdC5zaG93TG9hZGVyKCl9LGUpO2Vsc2V7Zm9yKHZhciBuIGluIHRoaXMudmlld0hvb2tzKXRoaXMudmlld0hvb2tzW25dLl9fZGlzY29ubmVjdGVkKCk7dGhpcy5zZXRDb250YWluZXJDbGFzc2VzKFwicGh4LWRpc2Nvbm5lY3RlZFwiKX19fSx7a2V5OlwiaGlkZUxvYWRlclwiLHZhbHVlOmZ1bmN0aW9uKCl7Y2xlYXJUaW1lb3V0KHRoaXMubG9hZGVyVGltZXIpLHRoaXMuc2V0Q29udGFpbmVyQ2xhc3NlcyhcInBoeC1jb25uZWN0ZWRcIil9fSx7a2V5OlwidHJpZ2dlclJlY29ubmVjdGVkXCIsdmFsdWU6ZnVuY3Rpb24oKXtmb3IodmFyIGUgaW4gdGhpcy52aWV3SG9va3MpdGhpcy52aWV3SG9va3NbZV0uX19yZWNvbm5lY3RlZCgpfX0se2tleTpcImxvZ1wiLHZhbHVlOmZ1bmN0aW9uKGUsdCl7dGhpcy5saXZlU29ja2V0LmxvZyh0aGlzLGUsdCl9fSx7a2V5Olwid2l0aGluVGFyZ2V0c1wiLHZhbHVlOmZ1bmN0aW9uKGUsdCl7dmFyIG49dGhpcztpZigvXigwfFsxLTldXFxkKikkLy50ZXN0KGUpKXt2YXIgaT1kZS5maW5kQ29tcG9uZW50Tm9kZUxpc3QodGhpcy5lbCxlKTswPT09aS5sZW5ndGg/SyhcIm5vIGNvbXBvbmVudCBmb3VuZCBtYXRjaGluZyBwaHgtdGFyZ2V0IG9mIFwiLmNvbmNhdChlKSk6dCh0aGlzLGlbMF0pfWVsc2V7dmFyIHI9QXJyYXkuZnJvbShkb2N1bWVudC5xdWVyeVNlbGVjdG9yQWxsKGUpKTswPT09ci5sZW5ndGgmJksoJ25vdGhpbmcgZm91bmQgbWF0Y2hpbmcgdGhlIHBoeC10YXJnZXQgc2VsZWN0b3IgXCInLmNvbmNhdChlLCdcIicpKSxyLmZvckVhY2goZnVuY3Rpb24oZSl7cmV0dXJuIG4ubGl2ZVNvY2tldC5vd25lcihlLGZ1bmN0aW9uKG4pe3JldHVybiB0KG4sZSl9KX0pfX19LHtrZXk6XCJhcHBseURpZmZcIix2YWx1ZTpmdW5jdGlvbihlLHQsbil7dGhpcy5sb2coZSxmdW5jdGlvbigpe3JldHVybltcIlwiLEcodCldfSk7dmFyIGk9c2UuZXh0cmFjdCh0KSxyPWkuZGlmZixvPWkucmVwbHksYT1pLmV2ZW50cyx1PWkudGl0bGU7cmV0dXJuIHUmJmRlLnB1dFRpdGxlKHUpLG4oe2RpZmY6cixyZXBseTpvLGV2ZW50czphfSksb319LHtrZXk6XCJvbkpvaW5cIix2YWx1ZTpmdW5jdGlvbihlKXt2YXIgdD10aGlzLG49ZS5yZW5kZXJlZDt0aGlzLmNoaWxkSm9pbnM9MCx0aGlzLmpvaW5QZW5kaW5nPSEwLHRoaXMuZmxhc2g9bnVsbCxsZS5kcm9wTG9jYWwodGhpcy5uYW1lKCksXCJjb25zZWN1dGl2ZS1yZWxvYWRzXCIpLHRoaXMuYXBwbHlEaWZmKFwibW91bnRcIixuLGZ1bmN0aW9uKG4pe3ZhciBpPW4uZGlmZixyPW4uZXZlbnRzO3QucmVuZGVyZWQ9bmV3IHNlKHQuaWQsaSk7dmFyIG89dC5yZW5kZXJDb250YWluZXIobnVsbCxcImpvaW5cIik7dC5kcm9wUGVuZGluZ1JlZnMoKTt2YXIgYT10LmZvcm1zRm9yUmVjb3Zlcnkobyk7dC5qb2luQ291bnQrKyxhLmxlbmd0aD4wP2EuZm9yRWFjaChmdW5jdGlvbihlLG4pe3QucHVzaEZvcm1SZWNvdmVyeShlLGZ1bmN0aW9uKGUpe249PT1hLmxlbmd0aC0xJiZ0Lm9uSm9pbkNvbXBsZXRlKGUsbyxyKX0pfSk6dC5vbkpvaW5Db21wbGV0ZShlLG8scil9KX19LHtrZXk6XCJkcm9wUGVuZGluZ1JlZnNcIix2YWx1ZTpmdW5jdGlvbigpe2RlLmFsbCh0aGlzLmVsLFwiW1wiLmNvbmNhdChGLFwiXVwiKSxmdW5jdGlvbihlKXtyZXR1cm4gZS5yZW1vdmVBdHRyaWJ1dGUoRil9KX19LHtrZXk6XCJvbkpvaW5Db21wbGV0ZVwiLHZhbHVlOmZ1bmN0aW9uKGUsdCxuKXt2YXIgaT10aGlzLHI9ZS5saXZlX3BhdGNoO2lmKHRoaXMuam9pbkNvdW50PjF8fHRoaXMucGFyZW50JiYhdGhpcy5wYXJlbnQuaXNKb2luUGVuZGluZygpKXJldHVybiB0aGlzLmFwcGx5Sm9pblBhdGNoKHIsdCxuKTswPT09ZGUuZmluZFBoeENoaWxkcmVuSW5GcmFnbWVudCh0LHRoaXMuaWQpLmZpbHRlcihmdW5jdGlvbihlKXt2YXIgdD1lLmlkJiZpLmVsLnF1ZXJ5U2VsZWN0b3IoXCIjXCIuY29uY2F0KGUuaWQpKSxuPXQmJnQuZ2V0QXR0cmlidXRlKFYpO3JldHVybiBuJiZlLnNldEF0dHJpYnV0ZShWLG4pLGkuam9pbkNoaWxkKGUpfSkubGVuZ3RoP3RoaXMucGFyZW50Pyh0aGlzLnJvb3QucGVuZGluZ0pvaW5PcHMucHVzaChbdGhpcyxmdW5jdGlvbigpe3JldHVybiBpLmFwcGx5Sm9pblBhdGNoKHIsdCxuKX1dKSx0aGlzLnBhcmVudC5hY2tKb2luKHRoaXMpKToodGhpcy5vbkFsbENoaWxkSm9pbnNDb21wbGV0ZSgpLHRoaXMuYXBwbHlKb2luUGF0Y2gocix0LG4pKTp0aGlzLnJvb3QucGVuZGluZ0pvaW5PcHMucHVzaChbdGhpcyxmdW5jdGlvbigpe3JldHVybiBpLmFwcGx5Sm9pblBhdGNoKHIsdCxuKX1dKX19LHtrZXk6XCJhdHRhY2hUcnVlRG9jRWxcIix2YWx1ZTpmdW5jdGlvbigpe3RoaXMuZWw9ZGUuYnlJZCh0aGlzLmlkKSx0aGlzLmVsLnNldEF0dHJpYnV0ZShcImRhdGEtcGh4LXJvb3QtaWRcIix0aGlzLnJvb3QuaWQpfX0se2tleTpcImRpc3BhdGNoRXZlbnRzXCIsdmFsdWU6ZnVuY3Rpb24oZSl7ZS5mb3JFYWNoKGZ1bmN0aW9uKGUpe3ZhciB0PXgoZSwyKSxuPXRbMF0saT10WzFdO3dpbmRvdy5kaXNwYXRjaEV2ZW50KG5ldyBDdXN0b21FdmVudChcInBoeDpob29rOlwiLmNvbmNhdChuKSx7ZGV0YWlsOml9KSl9KX19LHtrZXk6XCJhcHBseUpvaW5QYXRjaFwiLHZhbHVlOmZ1bmN0aW9uKGUsdCxuKXt2YXIgaT10aGlzO3RoaXMuYXR0YWNoVHJ1ZURvY0VsKCk7dmFyIHI9bmV3IGZlKHRoaXMsdGhpcy5lbCx0aGlzLmlkLHQsbnVsbCk7aWYoci5tYXJrUHJ1bmFibGVDb250ZW50Rm9yUmVtb3ZhbCgpLHRoaXMucGVyZm9ybVBhdGNoKHIsITEpLHRoaXMuam9pbk5ld0NoaWxkcmVuKCksZGUuYWxsKHRoaXMuZWwsXCJbXCIuY29uY2F0KHRoaXMuYmluZGluZyhcImhvb2tcIiksXCJdLCBbZGF0YS1waHgtXCIpLmNvbmNhdChcImhvb2tcIixcIl1cIiksZnVuY3Rpb24oZSl7dmFyIHQ9aS5hZGRIb29rKGUpO3QmJnQuX19tb3VudGVkKCl9KSx0aGlzLmpvaW5QZW5kaW5nPSExLHRoaXMuZGlzcGF0Y2hFdmVudHMobiksdGhpcy5hcHBseVBlbmRpbmdVcGRhdGVzKCksZSl7dmFyIG89ZS5raW5kLGE9ZS50bzt0aGlzLmxpdmVTb2NrZXQuaGlzdG9yeVBhdGNoKGEsbyl9dGhpcy5oaWRlTG9hZGVyKCksdGhpcy5qb2luQ291bnQ+MSYmdGhpcy50cmlnZ2VyUmVjb25uZWN0ZWQoKSx0aGlzLnN0b3BDYWxsYmFjaygpfX0se2tleTpcInRyaWdnZXJCZWZvcmVVcGRhdGVIb29rXCIsdmFsdWU6ZnVuY3Rpb24oZSx0KXt0aGlzLmxpdmVTb2NrZXQudHJpZ2dlckRPTShcIm9uQmVmb3JlRWxVcGRhdGVkXCIsW2UsdF0pO3ZhciBuPXRoaXMuZ2V0SG9vayhlKSxpPW4mJmRlLmlzSWdub3JlZChlLHRoaXMuYmluZGluZyhcInVwZGF0ZVwiKSk7aWYobiYmIWUuaXNFcXVhbE5vZGUodCkmJighaXx8IWZ1bmN0aW9uKGUsdCl7cmV0dXJuIEpTT04uc3RyaW5naWZ5KGUpPT09SlNPTi5zdHJpbmdpZnkodCl9KGUuZGF0YXNldCx0LmRhdGFzZXQpKSlyZXR1cm4gbi5fX2JlZm9yZVVwZGF0ZSgpLG59fSx7a2V5OlwicGVyZm9ybVBhdGNoXCIsdmFsdWU6ZnVuY3Rpb24oZSx0KXt2YXIgbj10aGlzLGk9W10scj0hMSxvPW5ldyBTZXQ7cmV0dXJuIGUuYWZ0ZXIoXCJhZGRlZFwiLGZ1bmN0aW9uKGUpe24ubGl2ZVNvY2tldC50cmlnZ2VyRE9NKFwib25Ob2RlQWRkZWRcIixbZV0pO3ZhciB0PW4uYWRkSG9vayhlKTt0JiZ0Ll9fbW91bnRlZCgpfSksZS5hZnRlcihcInBoeENoaWxkQWRkZWRcIixmdW5jdGlvbihlKXtyZXR1cm4gcj0hMH0pLGUuYmVmb3JlKFwidXBkYXRlZFwiLGZ1bmN0aW9uKGUsdCl7bi50cmlnZ2VyQmVmb3JlVXBkYXRlSG9vayhlLHQpJiZvLmFkZChlLmlkKX0pLGUuYWZ0ZXIoXCJ1cGRhdGVkXCIsZnVuY3Rpb24oZSl7by5oYXMoZS5pZCkmJm4uZ2V0SG9vayhlKS5fX3VwZGF0ZWQoKX0pLGUuYWZ0ZXIoXCJkaXNjYXJkZWRcIixmdW5jdGlvbihlKXt2YXIgdD1uLmNvbXBvbmVudElEKGUpO1wibnVtYmVyXCI9PXR5cGVvZiB0JiYtMT09PWkuaW5kZXhPZih0KSYmaS5wdXNoKHQpO3ZhciByPW4uZ2V0SG9vayhlKTtyJiZuLmRlc3Ryb3lIb29rKHIpfSksZS5wZXJmb3JtKCksdCYmdGhpcy5tYXliZVB1c2hDb21wb25lbnRzRGVzdHJveWVkKGkpLHJ9fSx7a2V5Olwiam9pbk5ld0NoaWxkcmVuXCIsdmFsdWU6ZnVuY3Rpb24oKXt2YXIgZT10aGlzO2RlLmZpbmRQaHhDaGlsZHJlbih0aGlzLmVsLHRoaXMuaWQpLmZvckVhY2goZnVuY3Rpb24odCl7cmV0dXJuIGUuam9pbkNoaWxkKHQpfSl9fSx7a2V5OlwiZ2V0Q2hpbGRCeUlkXCIsdmFsdWU6ZnVuY3Rpb24oZSl7cmV0dXJuIHRoaXMucm9vdC5jaGlsZHJlblt0aGlzLmlkXVtlXX19LHtrZXk6XCJnZXREZXNjZW5kZW50QnlFbFwiLHZhbHVlOmZ1bmN0aW9uKGUpe3JldHVybiBlLmlkPT09dGhpcy5pZD90aGlzOnRoaXMuY2hpbGRyZW5bZS5nZXRBdHRyaWJ1dGUoXCJkYXRhLXBoeC1wYXJlbnQtaWRcIildW2UuaWRdfX0se2tleTpcImRlc3Ryb3lEZXNjZW5kZW50XCIsdmFsdWU6ZnVuY3Rpb24oZSl7Zm9yKHZhciB0IGluIHRoaXMucm9vdC5jaGlsZHJlbilmb3IodmFyIG4gaW4gdGhpcy5yb290LmNoaWxkcmVuW3RdKWlmKG49PT1lKXJldHVybiB0aGlzLnJvb3QuY2hpbGRyZW5bdF1bbl0uZGVzdHJveSgpfX0se2tleTpcImpvaW5DaGlsZFwiLHZhbHVlOmZ1bmN0aW9uKHQpe2lmKCF0aGlzLmdldENoaWxkQnlJZCh0LmlkKSl7dmFyIG49bmV3IGUodCx0aGlzLmxpdmVTb2NrZXQsdGhpcyk7cmV0dXJuIHRoaXMucm9vdC5jaGlsZHJlblt0aGlzLmlkXVtuLmlkXT1uLG4uam9pbigpLHRoaXMuY2hpbGRKb2lucysrLCEwfX19LHtrZXk6XCJpc0pvaW5QZW5kaW5nXCIsdmFsdWU6ZnVuY3Rpb24oKXtyZXR1cm4gdGhpcy5qb2luUGVuZGluZ319LHtrZXk6XCJhY2tKb2luXCIsdmFsdWU6ZnVuY3Rpb24oZSl7dGhpcy5jaGlsZEpvaW5zLS0sMD09PXRoaXMuY2hpbGRKb2lucyYmKHRoaXMucGFyZW50P3RoaXMucGFyZW50LmFja0pvaW4odGhpcyk6dGhpcy5vbkFsbENoaWxkSm9pbnNDb21wbGV0ZSgpKX19LHtrZXk6XCJvbkFsbENoaWxkSm9pbnNDb21wbGV0ZVwiLHZhbHVlOmZ1bmN0aW9uKCl7dGhpcy5qb2luQ2FsbGJhY2soKSx0aGlzLnBlbmRpbmdKb2luT3BzLmZvckVhY2goZnVuY3Rpb24oZSl7dmFyIHQ9eChlLDIpLG49dFswXSxpPXRbMV07bi5pc0Rlc3Ryb3llZCgpfHxpKCl9KSx0aGlzLnBlbmRpbmdKb2luT3BzPVtdfX0se2tleTpcInVwZGF0ZVwiLHZhbHVlOmZ1bmN0aW9uKGUsdCl7dmFyIG49dGhpcztpZih0aGlzLmlzSm9pblBlbmRpbmcoKXx8dGhpcy5saXZlU29ja2V0Lmhhc1BlbmRpbmdMaW5rKCkpcmV0dXJuIHRoaXMucGVuZGluZ0RpZmZzLnB1c2goe2RpZmY6ZSxldmVudHM6dH0pO3RoaXMucmVuZGVyZWQubWVyZ2VEaWZmKGUpO3ZhciBpPSExO3RoaXMucmVuZGVyZWQuaXNDb21wb25lbnRPbmx5RGlmZihlKT90aGlzLmxpdmVTb2NrZXQudGltZShcImNvbXBvbmVudCBwYXRjaCBjb21wbGV0ZVwiLGZ1bmN0aW9uKCl7ZGUuZmluZFBhcmVudENJRHMobi5lbCxuLnJlbmRlcmVkLmNvbXBvbmVudENJRHMoZSkpLmZvckVhY2goZnVuY3Rpb24odCl7bi5jb21wb25lbnRQYXRjaChuLnJlbmRlcmVkLmdldENvbXBvbmVudChlLHQpLHQpJiYoaT0hMCl9KX0pOlooZSl8fHRoaXMubGl2ZVNvY2tldC50aW1lKFwiZnVsbCBwYXRjaCBjb21wbGV0ZVwiLGZ1bmN0aW9uKCl7dmFyIHQ9bi5yZW5kZXJDb250YWluZXIoZSxcInVwZGF0ZVwiKSxyPW5ldyBmZShuLG4uZWwsbi5pZCx0LG51bGwpO2k9bi5wZXJmb3JtUGF0Y2gociwhMCl9KSx0aGlzLmRpc3BhdGNoRXZlbnRzKHQpLGkmJnRoaXMuam9pbk5ld0NoaWxkcmVuKCl9fSx7a2V5OlwicmVuZGVyQ29udGFpbmVyXCIsdmFsdWU6ZnVuY3Rpb24oZSx0KXt2YXIgbj10aGlzO3JldHVybiB0aGlzLmxpdmVTb2NrZXQudGltZShcInRvU3RyaW5nIGRpZmYgKFwiLmNvbmNhdCh0LFwiKVwiKSxmdW5jdGlvbigpe3ZhciB0PW4uZWwudGFnTmFtZSxpPWU/bi5yZW5kZXJlZC5jb21wb25lbnRDSURzKGUpLmNvbmNhdChuLnBydW5pbmdDSURzKTpudWxsLHI9bi5yZW5kZXJlZC50b1N0cmluZyhpKTtyZXR1cm5cIjxcIi5jb25jYXQodCxcIj5cIikuY29uY2F0KHIsXCI8L1wiKS5jb25jYXQodCxcIj5cIil9KX19LHtrZXk6XCJjb21wb25lbnRQYXRjaFwiLHZhbHVlOmZ1bmN0aW9uKGUsdCl7aWYoWihlKSlyZXR1cm4hMTt2YXIgbj10aGlzLnJlbmRlcmVkLmNvbXBvbmVudFRvU3RyaW5nKHQpLGk9bmV3IGZlKHRoaXMsdGhpcy5lbCx0aGlzLmlkLG4sdCk7cmV0dXJuIHRoaXMucGVyZm9ybVBhdGNoKGksITApfX0se2tleTpcImdldEhvb2tcIix2YWx1ZTpmdW5jdGlvbihlKXtyZXR1cm4gdGhpcy52aWV3SG9va3NbZ2UuZWxlbWVudElEKGUpXX19LHtrZXk6XCJhZGRIb29rXCIsdmFsdWU6ZnVuY3Rpb24oZSl7aWYoIWdlLmVsZW1lbnRJRChlKSYmZS5nZXRBdHRyaWJ1dGUpe3ZhciB0PWUuZ2V0QXR0cmlidXRlKFwiZGF0YS1waHgtXCIuY29uY2F0KFwiaG9va1wiKSl8fGUuZ2V0QXR0cmlidXRlKHRoaXMuYmluZGluZyhcImhvb2tcIikpO2lmKCF0fHx0aGlzLm93bnNFbGVtZW50KGUpKXt2YXIgbj10aGlzLmxpdmVTb2NrZXQuZ2V0SG9va0NhbGxiYWNrcyh0KTtpZihuKXtlLmlkfHxLKCdubyBET00gSUQgZm9yIGhvb2sgXCInLmNvbmNhdCh0LCdcIi4gSG9va3MgcmVxdWlyZSBhIHVuaXF1ZSBJRCBvbiBlYWNoIGVsZW1lbnQuJyksZSk7dmFyIGk9bmV3IGdlKHRoaXMsZSxuKTtyZXR1cm4gdGhpcy52aWV3SG9va3NbZ2UuZWxlbWVudElEKGkuZWwpXT1pLGl9bnVsbCE9PXQmJksoJ3Vua25vd24gaG9vayBmb3VuZCBmb3IgXCInLmNvbmNhdCh0LCdcIicpLGUpfX19fSx7a2V5OlwiZGVzdHJveUhvb2tcIix2YWx1ZTpmdW5jdGlvbihlKXtlLl9fZGVzdHJveWVkKCksZS5fX2NsZWFudXBfXygpLGRlbGV0ZSB0aGlzLnZpZXdIb29rc1tnZS5lbGVtZW50SUQoZS5lbCldfX0se2tleTpcImFwcGx5UGVuZGluZ1VwZGF0ZXNcIix2YWx1ZTpmdW5jdGlvbigpe3ZhciBlPXRoaXM7dGhpcy5wZW5kaW5nRGlmZnMuZm9yRWFjaChmdW5jdGlvbih0KXt2YXIgbj10LmRpZmYsaT10LmV2ZW50cztyZXR1cm4gZS51cGRhdGUobixpKX0pLHRoaXMucGVuZGluZ0RpZmZzPVtdfX0se2tleTpcIm9uQ2hhbm5lbFwiLHZhbHVlOmZ1bmN0aW9uKGUsdCl7dmFyIG49dGhpczt0aGlzLmxpdmVTb2NrZXQub25DaGFubmVsKHRoaXMuY2hhbm5lbCxlLGZ1bmN0aW9uKGUpe24uaXNKb2luUGVuZGluZygpP24ucm9vdC5wZW5kaW5nSm9pbk9wcy5wdXNoKFtuLGZ1bmN0aW9uKCl7cmV0dXJuIHQoZSl9XSk6dChlKX0pfX0se2tleTpcImJpbmRDaGFubmVsXCIsdmFsdWU6ZnVuY3Rpb24oKXt2YXIgZT10aGlzO3RoaXMubGl2ZVNvY2tldC5vbkNoYW5uZWwodGhpcy5jaGFubmVsLFwiZGlmZlwiLGZ1bmN0aW9uKHQpe2UuYXBwbHlEaWZmKFwidXBkYXRlXCIsdCxmdW5jdGlvbih0KXt2YXIgbj10LmRpZmYsaT10LmV2ZW50cztyZXR1cm4gZS51cGRhdGUobixpKX0pfSksdGhpcy5vbkNoYW5uZWwoXCJyZWRpcmVjdFwiLGZ1bmN0aW9uKHQpe3ZhciBuPXQudG8saT10LmZsYXNoO3JldHVybiBlLm9uUmVkaXJlY3Qoe3RvOm4sZmxhc2g6aX0pfSksdGhpcy5vbkNoYW5uZWwoXCJsaXZlX3BhdGNoXCIsZnVuY3Rpb24odCl7cmV0dXJuIGUub25MaXZlUGF0Y2godCl9KSx0aGlzLm9uQ2hhbm5lbChcImxpdmVfcmVkaXJlY3RcIixmdW5jdGlvbih0KXtyZXR1cm4gZS5vbkxpdmVSZWRpcmVjdCh0KX0pLHRoaXMuY2hhbm5lbC5vbkVycm9yKGZ1bmN0aW9uKHQpe3JldHVybiBlLm9uRXJyb3IodCl9KSx0aGlzLmNoYW5uZWwub25DbG9zZShmdW5jdGlvbih0KXtyZXR1cm4gZS5vbkNsb3NlKHQpfSl9fSx7a2V5OlwiZGVzdHJveUFsbENoaWxkcmVuXCIsdmFsdWU6ZnVuY3Rpb24oKXtmb3IodmFyIGUgaW4gdGhpcy5yb290LmNoaWxkcmVuW3RoaXMuaWRdKXRoaXMuZ2V0Q2hpbGRCeUlkKGUpLmRlc3Ryb3koKX19LHtrZXk6XCJvbkxpdmVSZWRpcmVjdFwiLHZhbHVlOmZ1bmN0aW9uKGUpe3ZhciB0PWUudG8sbj1lLmtpbmQsaT1lLmZsYXNoLHI9dGhpcy5leHBhbmRVUkwodCk7dGhpcy5saXZlU29ja2V0Lmhpc3RvcnlSZWRpcmVjdChyLG4saSl9fSx7a2V5Olwib25MaXZlUGF0Y2hcIix2YWx1ZTpmdW5jdGlvbihlKXt2YXIgdD1lLnRvLG49ZS5raW5kO3RoaXMuaHJlZj10aGlzLmV4cGFuZFVSTCh0KSx0aGlzLmxpdmVTb2NrZXQuaGlzdG9yeVBhdGNoKHQsbil9fSx7a2V5OlwiZXhwYW5kVVJMXCIsdmFsdWU6ZnVuY3Rpb24oZSl7cmV0dXJuIGUuc3RhcnRzV2l0aChcIi9cIik/XCJcIi5jb25jYXQod2luZG93LmxvY2F0aW9uLnByb3RvY29sLFwiLy9cIikuY29uY2F0KHdpbmRvdy5sb2NhdGlvbi5ob3N0KS5jb25jYXQoZSk6ZX19LHtrZXk6XCJvblJlZGlyZWN0XCIsdmFsdWU6ZnVuY3Rpb24oZSl7dmFyIHQ9ZS50byxuPWUuZmxhc2g7dGhpcy5saXZlU29ja2V0LnJlZGlyZWN0KHQsbil9fSx7a2V5OlwiaXNEZXN0cm95ZWRcIix2YWx1ZTpmdW5jdGlvbigpe3JldHVybiB0aGlzLmRlc3Ryb3llZH19LHtrZXk6XCJqb2luXCIsdmFsdWU6ZnVuY3Rpb24oZSl7dmFyIHQ9dGhpczt0aGlzLnBhcmVudHx8KHRoaXMuc3RvcENhbGxiYWNrPXRoaXMubGl2ZVNvY2tldC53aXRoUGFnZUxvYWRpbmcoe3RvOnRoaXMuaHJlZixraW5kOlwiaW5pdGlhbFwifSkpLHRoaXMuam9pbkNhbGxiYWNrPWZ1bmN0aW9uKCl7cmV0dXJuIGUmJmUodCx0LmpvaW5Db3VudCl9LHRoaXMubGl2ZVNvY2tldC53cmFwUHVzaCh0aGlzLHt0aW1lb3V0OiExfSxmdW5jdGlvbigpe3JldHVybiB0LmNoYW5uZWwuam9pbigpLnJlY2VpdmUoXCJva1wiLGZ1bmN0aW9uKGUpe3JldHVybiB0Lm9uSm9pbihlKX0pLnJlY2VpdmUoXCJlcnJvclwiLGZ1bmN0aW9uKGUpe3JldHVybiB0Lm9uSm9pbkVycm9yKGUpfSkucmVjZWl2ZShcInRpbWVvdXRcIixmdW5jdGlvbigpe3JldHVybiB0Lm9uSm9pbkVycm9yKHtyZWFzb246XCJ0aW1lb3V0XCJ9KX0pfSl9fSx7a2V5Olwib25Kb2luRXJyb3JcIix2YWx1ZTpmdW5jdGlvbihlKXtyZXR1cm4oZS5yZWRpcmVjdHx8ZS5saXZlX3JlZGlyZWN0KSYmKHRoaXMuam9pblBlbmRpbmc9ITEsdGhpcy5jaGFubmVsLmxlYXZlKCkpLGUucmVkaXJlY3Q/dGhpcy5vblJlZGlyZWN0KGUucmVkaXJlY3QpOmUubGl2ZV9yZWRpcmVjdD90aGlzLm9uTGl2ZVJlZGlyZWN0KGUubGl2ZV9yZWRpcmVjdCk6KHRoaXMubG9nKFwiZXJyb3JcIixmdW5jdGlvbigpe3JldHVybltcInVuYWJsZSB0byBqb2luXCIsZV19KSx0aGlzLmxpdmVTb2NrZXQucmVsb2FkV2l0aEppdHRlcih0aGlzKSl9fSx7a2V5Olwib25DbG9zZVwiLHZhbHVlOmZ1bmN0aW9uKGUpe2lmKCF0aGlzLmlzRGVzdHJveWVkKCkpe2lmKHRoaXMuaXNKb2luUGVuZGluZygpfHx0aGlzLmxpdmVTb2NrZXQuaGFzUGVuZGluZ0xpbmsoKSYmXCJsZWF2ZVwiIT09ZSlyZXR1cm4gdGhpcy5saXZlU29ja2V0LnJlbG9hZFdpdGhKaXR0ZXIodGhpcyk7dGhpcy5kZXN0cm95QWxsQ2hpbGRyZW4oKSx0aGlzLmxpdmVTb2NrZXQuZHJvcEFjdGl2ZUVsZW1lbnQodGhpcyksZG9jdW1lbnQuYWN0aXZlRWxlbWVudCYmZG9jdW1lbnQuYWN0aXZlRWxlbWVudC5ibHVyKCksdGhpcy5saXZlU29ja2V0LmlzVW5sb2FkZWQoKSYmdGhpcy5zaG93TG9hZGVyKDIwMCl9fX0se2tleTpcIm9uRXJyb3JcIix2YWx1ZTpmdW5jdGlvbihlKXt0aGlzLm9uQ2xvc2UoZSksdGhpcy5sb2coXCJlcnJvclwiLGZ1bmN0aW9uKCl7cmV0dXJuW1widmlldyBjcmFzaGVkXCIsZV19KSx0aGlzLmxpdmVTb2NrZXQuaXNVbmxvYWRlZCgpfHx0aGlzLmRpc3BsYXlFcnJvcigpfX0se2tleTpcImRpc3BsYXlFcnJvclwiLHZhbHVlOmZ1bmN0aW9uKCl7dGhpcy5pc01haW4oKSYmZGUuZGlzcGF0Y2hFdmVudCh3aW5kb3csXCJwaHg6cGFnZS1sb2FkaW5nLXN0YXJ0XCIse3RvOnRoaXMuaHJlZixraW5kOlwiZXJyb3JcIn0pLHRoaXMuc2hvd0xvYWRlcigpLHRoaXMuc2V0Q29udGFpbmVyQ2xhc3NlcyhcInBoeC1kaXNjb25uZWN0ZWRcIixcInBoeC1lcnJvclwiKX19LHtrZXk6XCJwdXNoV2l0aFJlcGx5XCIsdmFsdWU6ZnVuY3Rpb24oZSx0LG4pe3ZhciBpPXRoaXMscj1hcmd1bWVudHMubGVuZ3RoPjMmJnZvaWQgMCE9PWFyZ3VtZW50c1szXT9hcmd1bWVudHNbM106ZnVuY3Rpb24oKXt9LG89eChlP2UoKTpbbnVsbCxbXV0sMiksYT1vWzBdLHU9eChvWzFdLDEpWzBdLHM9ZnVuY3Rpb24oKXt9O3JldHVybiB1JiZudWxsIT09dS5nZXRBdHRyaWJ1dGUodGhpcy5iaW5kaW5nKFwicGFnZS1sb2FkaW5nXCIpKSYmKHM9dGhpcy5saXZlU29ja2V0LndpdGhQYWdlTG9hZGluZyh7a2luZDpcImVsZW1lbnRcIix0YXJnZXQ6dX0pKSxcIm51bWJlclwiIT10eXBlb2Ygbi5jaWQmJmRlbGV0ZSBuLmNpZCx0aGlzLmxpdmVTb2NrZXQud3JhcFB1c2godGhpcyx7dGltZW91dDohMH0sZnVuY3Rpb24oKXtyZXR1cm4gaS5jaGFubmVsLnB1c2godCxuLDNlNCkucmVjZWl2ZShcIm9rXCIsZnVuY3Rpb24oZSl7dmFyIHQ9bnVsbDtudWxsIT09YSYmaS51bmRvUmVmcyhhKSxlLmRpZmYmJih0PWkuYXBwbHlEaWZmKFwidXBkYXRlXCIsZS5kaWZmLGZ1bmN0aW9uKGUpe3ZhciB0PWUuZGlmZixuPWUuZXZlbnRzO2kudXBkYXRlKHQsbil9KSksZS5yZWRpcmVjdCYmaS5vblJlZGlyZWN0KGUucmVkaXJlY3QpLGUubGl2ZV9wYXRjaCYmaS5vbkxpdmVQYXRjaChlLmxpdmVfcGF0Y2gpLGUubGl2ZV9yZWRpcmVjdCYmaS5vbkxpdmVSZWRpcmVjdChlLmxpdmVfcmVkaXJlY3QpLHMoKSxyKGUsdCl9KX0pfX0se2tleTpcInVuZG9SZWZzXCIsdmFsdWU6ZnVuY3Rpb24oZSl7dmFyIHQ9dGhpcztkZS5hbGwodGhpcy5lbCxcIltcIi5jb25jYXQoRiwnPVwiJykuY29uY2F0KGUsJ1wiXScpLGZ1bmN0aW9uKGUpe2UucmVtb3ZlQXR0cmlidXRlKEYpLG51bGwhPT1lLmdldEF0dHJpYnV0ZShcImRhdGEtcGh4LXJlYWRvbmx5XCIpJiYoZS5yZWFkT25seT0hMSxlLnJlbW92ZUF0dHJpYnV0ZShcImRhdGEtcGh4LXJlYWRvbmx5XCIpKSxudWxsIT09ZS5nZXRBdHRyaWJ1dGUoXCJkYXRhLXBoeC1kaXNhYmxlZFwiKSYmKGUuZGlzYWJsZWQ9ITEsZS5yZW1vdmVBdHRyaWJ1dGUoXCJkYXRhLXBoeC1kaXNhYmxlZFwiKSksai5mb3JFYWNoKGZ1bmN0aW9uKHQpe3JldHVybiBkZS5yZW1vdmVDbGFzcyhlLHQpfSk7dmFyIG49ZS5nZXRBdHRyaWJ1dGUoXCJkYXRhLXBoeC1kaXNhYmxlLXdpdGgtcmVzdG9yZVwiKTtudWxsIT09biYmKGUuaW5uZXJUZXh0PW4sZS5yZW1vdmVBdHRyaWJ1dGUoXCJkYXRhLXBoeC1kaXNhYmxlLXdpdGgtcmVzdG9yZVwiKSk7dmFyIGk9ZGUucHJpdmF0ZShlLEYpO2lmKGkpe3ZhciByPXQudHJpZ2dlckJlZm9yZVVwZGF0ZUhvb2soZSxpKTtmZS5wYXRjaEVsKGUsaSx0LmxpdmVTb2NrZXQuZ2V0QWN0aXZlRWxlbWVudCgpKSxyJiZyLl9fdXBkYXRlZCgpLGRlLmRlbGV0ZVByaXZhdGUoZSxGKX19KX19LHtrZXk6XCJwdXRSZWZcIix2YWx1ZTpmdW5jdGlvbihlLHQpe3ZhciBuPXRoaXMucmVmKyssaT10aGlzLmJpbmRpbmcoXCJkaXNhYmxlLXdpdGhcIik7cmV0dXJuIGUuZm9yRWFjaChmdW5jdGlvbihlKXtlLmNsYXNzTGlzdC5hZGQoXCJwaHgtXCIuY29uY2F0KHQsXCItbG9hZGluZ1wiKSksZS5zZXRBdHRyaWJ1dGUoRixuKTt2YXIgcj1lLmdldEF0dHJpYnV0ZShpKTtudWxsIT09ciYmKGUuZ2V0QXR0cmlidXRlKFwiZGF0YS1waHgtZGlzYWJsZS13aXRoLXJlc3RvcmVcIil8fGUuc2V0QXR0cmlidXRlKFwiZGF0YS1waHgtZGlzYWJsZS13aXRoLXJlc3RvcmVcIixlLmlubmVyVGV4dCksZS5pbm5lclRleHQ9cil9KSxbbixlXX19LHtrZXk6XCJjb21wb25lbnRJRFwiLHZhbHVlOmZ1bmN0aW9uKGUpe3ZhciB0PWUuZ2V0QXR0cmlidXRlJiZlLmdldEF0dHJpYnV0ZShIKTtyZXR1cm4gdD9wYXJzZUludCh0KTpudWxsfX0se2tleTpcInRhcmdldENvbXBvbmVudElEXCIsdmFsdWU6ZnVuY3Rpb24oZSx0KXtyZXR1cm4gZS5nZXRBdHRyaWJ1dGUodGhpcy5iaW5kaW5nKFwidGFyZ2V0XCIpKT90aGlzLmNsb3Nlc3RDb21wb25lbnRJRCh0KTpudWxsfX0se2tleTpcImNsb3Nlc3RDb21wb25lbnRJRFwiLHZhbHVlOmZ1bmN0aW9uKGUpe3ZhciB0PXRoaXM7cmV0dXJuIGU/ZWUoZS5jbG9zZXN0KFwiW1wiLmNvbmNhdChILFwiXVwiKSksZnVuY3Rpb24oZSl7cmV0dXJuIHQub3duc0VsZW1lbnQoZSkmJnQuY29tcG9uZW50SUQoZSl9KTpudWxsfX0se2tleTpcInB1c2hIb29rRXZlbnRcIix2YWx1ZTpmdW5jdGlvbihlLHQsbixpKXt2YXIgcj14KHRoaXMucHV0UmVmKFtdLFwiaG9va1wiKSwyKSxvPXJbMF0sYT1yWzFdO3JldHVybiB0aGlzLnB1c2hXaXRoUmVwbHkoZnVuY3Rpb24oKXtyZXR1cm5bbyxhXX0sXCJldmVudFwiLHt0eXBlOlwiaG9va1wiLGV2ZW50OnQsdmFsdWU6bixjaWQ6dGhpcy5jbG9zZXN0Q29tcG9uZW50SUQoZSl9LGZ1bmN0aW9uKGUsdCl7cmV0dXJuIGkodCxvKX0pLG99fSx7a2V5OlwiZXh0cmFjdE1ldGFcIix2YWx1ZTpmdW5jdGlvbihlLHQpe2Zvcih2YXIgbj10aGlzLmJpbmRpbmcoXCJ2YWx1ZS1cIiksaT0wO2k8ZS5hdHRyaWJ1dGVzLmxlbmd0aDtpKyspe3ZhciByPWUuYXR0cmlidXRlc1tpXS5uYW1lO3Iuc3RhcnRzV2l0aChuKSYmKHRbci5yZXBsYWNlKG4sXCJcIildPWUuZ2V0QXR0cmlidXRlKHIpKX1yZXR1cm4gdm9pZCAwIT09ZS52YWx1ZSYmKHQudmFsdWU9ZS52YWx1ZSxcIklOUFVUXCI9PT1lLnRhZ05hbWUmJkouaW5kZXhPZihlLnR5cGUpPj0wJiYhZS5jaGVja2VkJiZkZWxldGUgdC52YWx1ZSksdH19LHtrZXk6XCJwdXNoRXZlbnRcIix2YWx1ZTpmdW5jdGlvbihlLHQsbixpLHIpe3ZhciBvPXRoaXM7dGhpcy5wdXNoV2l0aFJlcGx5KGZ1bmN0aW9uKCl7cmV0dXJuIG8ucHV0UmVmKFt0XSxlKX0sXCJldmVudFwiLHt0eXBlOmUsZXZlbnQ6aSx2YWx1ZTp0aGlzLmV4dHJhY3RNZXRhKHQsciksY2lkOnRoaXMudGFyZ2V0Q29tcG9uZW50SUQodCxuKX0pfX0se2tleTpcInB1c2hLZXlcIix2YWx1ZTpmdW5jdGlvbihlLHQsbixpLHIpe3ZhciBvPXRoaXM7dGhpcy5wdXNoV2l0aFJlcGx5KGZ1bmN0aW9uKCl7cmV0dXJuIG8ucHV0UmVmKFtlXSxuKX0sXCJldmVudFwiLHt0eXBlOm4sZXZlbnQ6aSx2YWx1ZTp0aGlzLmV4dHJhY3RNZXRhKGUsciksY2lkOnRoaXMudGFyZ2V0Q29tcG9uZW50SUQoZSx0KX0pfX0se2tleTpcInB1c2hGaWxlUHJvZ3Jlc3NcIix2YWx1ZTpmdW5jdGlvbihlLHQsbil7dmFyIGk9YXJndW1lbnRzLmxlbmd0aD4zJiZ2b2lkIDAhPT1hcmd1bWVudHNbM10/YXJndW1lbnRzWzNdOmZ1bmN0aW9uKCl7fTt0aGlzLmxpdmVTb2NrZXQud2l0aGluT3duZXJzKGUuZm9ybSxmdW5jdGlvbihyLG8pe3IucHVzaFdpdGhSZXBseShudWxsLFwicHJvZ3Jlc3NcIix7ZXZlbnQ6ZS5nZXRBdHRyaWJ1dGUoci5iaW5kaW5nKFwicHJvZ3Jlc3NcIikpLHJlZjplLmdldEF0dHJpYnV0ZShNKSxlbnRyeV9yZWY6dCxwcm9ncmVzczpuLGNpZDpyLnRhcmdldENvbXBvbmVudElEKGUuZm9ybSxvKX0saSl9KX19LHtrZXk6XCJwdXNoSW5wdXRcIix2YWx1ZTpmdW5jdGlvbihlLHQsbixpLHIpe3ZhciBvPXRoaXMsYT10aGlzLnRhcmdldENvbXBvbmVudElEKGUuZm9ybSx0KSx1PWZ1bmN0aW9uKCl7cmV0dXJuIG8ucHV0UmVmKFtlLGUuZm9ybV0sXCJjaGFuZ2VcIil9LHM9dWUoZS5mb3JtLHtfdGFyZ2V0OmkubmFtZX0pO2UuZmlsZXMmJmUuZmlsZXMubGVuZ3RoPjAmJnJlLnRyYWNrRmlsZXMoZSxBcnJheS5mcm9tKGUuZmlsZXMpKTt2YXIgYz17dHlwZTpcImZvcm1cIixldmVudDpuLHZhbHVlOnMsdXBsb2FkczpyZS5zZXJpYWxpemVVcGxvYWRzKGUpLGNpZDphfTt0aGlzLnB1c2hXaXRoUmVwbHkodSxcImV2ZW50XCIsYyxmdW5jdGlvbihuKXtpZihkZS5zaG93RXJyb3IoZSxvLmxpdmVTb2NrZXQuYmluZGluZyhcImZlZWRiYWNrLWZvclwiKSksZGUuaXNVcGxvYWRJbnB1dChlKSYmbnVsbCE9PWUuZ2V0QXR0cmlidXRlKFwiZGF0YS1waHgtYXV0by11cGxvYWRcIikpe2lmKHJlLmZpbGVzQXdhaXRpbmdQcmVmbGlnaHQoZSkubGVuZ3RoPjApe3ZhciBpPXgodSgpLDIpLHM9aVswXTtpWzFdO28udXBsb2FkRmlsZXMoZS5mb3JtLHQscyxhLGZ1bmN0aW9uKHQpe3ImJnIobiksby50cmlnZ2VyQXdhaXRpbmdTdWJtaXQoZS5mb3JtKX0pfX1lbHNlIHImJnIobil9KX19LHtrZXk6XCJ0cmlnZ2VyQXdhaXRpbmdTdWJtaXRcIix2YWx1ZTpmdW5jdGlvbihlKXt2YXIgdD10aGlzLmdldFNjaGVkdWxlZFN1Ym1pdChlKTtpZih0KXt2YXIgbj14KHQsMyksaT0oblswXSxuWzFdLG5bMl0pO3RoaXMuY2FuY2VsU3VibWl0KGUpLGkoKX19fSx7a2V5OlwiZ2V0U2NoZWR1bGVkU3VibWl0XCIsdmFsdWU6ZnVuY3Rpb24oZSl7cmV0dXJuIHRoaXMuZm9ybVN1Ym1pdHMuZmluZChmdW5jdGlvbih0KXt2YXIgbj14KHQsMiksaT1uWzBdO25bMV07cmV0dXJuIGkuaXNTYW1lTm9kZShlKX0pfX0se2tleTpcInNjaGVkdWxlU3VibWl0XCIsdmFsdWU6ZnVuY3Rpb24oZSx0LG4pe2lmKHRoaXMuZ2V0U2NoZWR1bGVkU3VibWl0KGUpKXJldHVybiEwO3RoaXMuZm9ybVN1Ym1pdHMucHVzaChbZSx0LG5dKX19LHtrZXk6XCJjYW5jZWxTdWJtaXRcIix2YWx1ZTpmdW5jdGlvbihlKXt2YXIgdD10aGlzO3RoaXMuZm9ybVN1Ym1pdHM9dGhpcy5mb3JtU3VibWl0cy5maWx0ZXIoZnVuY3Rpb24obil7dmFyIGk9eChuLDMpLHI9aVswXSxvPWlbMV07aVsyXTtyZXR1cm4hci5pc1NhbWVOb2RlKGUpfHwodC51bmRvUmVmcyhvKSwhMSl9KX19LHtrZXk6XCJwdXNoRm9ybVN1Ym1pdFwiLHZhbHVlOmZ1bmN0aW9uKGUsdCxuLGkpe3ZhciByPXRoaXMsbz1mdW5jdGlvbihlKXtyZXR1cm4hKFkoZSxcIlwiLmNvbmNhdChyLmJpbmRpbmcoXCJ1cGRhdGVcIiksXCI9aWdub3JlXCIpLGUuZm9ybSl8fFkoZSxcImRhdGEtcGh4LXVwZGF0ZT1pZ25vcmVcIixlLmZvcm0pKX0sYT1mdW5jdGlvbihlKXtyZXR1cm4gZS5oYXNBdHRyaWJ1dGUoci5iaW5kaW5nKFwiZGlzYWJsZS13aXRoXCIpKX0sdT1mdW5jdGlvbihlKXtyZXR1cm5cIkJVVFRPTlwiPT1lLnRhZ05hbWV9LHM9ZnVuY3Rpb24oZSl7cmV0dXJuW1wiSU5QVVRcIixcIlRFWFRBUkVBXCIsXCJTRUxFQ1RcIl0uaW5jbHVkZXMoZS50YWdOYW1lKX0sYz1mdW5jdGlvbigpe3ZhciB0PUFycmF5LmZyb20oZS5lbGVtZW50cyksbj10LmZpbHRlcihhKSxpPXQuZmlsdGVyKHUpLmZpbHRlcihvKSxjPXQuZmlsdGVyKHMpLmZpbHRlcihvKTtyZXR1cm4gaS5mb3JFYWNoKGZ1bmN0aW9uKGUpe2Uuc2V0QXR0cmlidXRlKFwiZGF0YS1waHgtZGlzYWJsZWRcIixlLmRpc2FibGVkKSxlLmRpc2FibGVkPSEwfSksYy5mb3JFYWNoKGZ1bmN0aW9uKGUpe2Uuc2V0QXR0cmlidXRlKFwiZGF0YS1waHgtcmVhZG9ubHlcIixlLnJlYWRPbmx5KSxlLnJlYWRPbmx5PSEwLGUuZmlsZXMmJihlLnNldEF0dHJpYnV0ZShcImRhdGEtcGh4LWRpc2FibGVkXCIsZS5kaXNhYmxlZCksZS5kaXNhYmxlZD0hMCl9KSxlLnNldEF0dHJpYnV0ZShyLmJpbmRpbmcoXCJwYWdlLWxvYWRpbmdcIiksXCJcIiksci5wdXRSZWYoW2VdLmNvbmNhdChuKS5jb25jYXQoaSkuY29uY2F0KGMpLFwic3VibWl0XCIpfSxsPXRoaXMudGFyZ2V0Q29tcG9uZW50SUQoZSx0KTtpZihyZS5oYXNVcGxvYWRzSW5Qcm9ncmVzcyhlKSl7dmFyIGQ9eChjKCksMiksaD1kWzBdO2RbMV07cmV0dXJuIHRoaXMuc2NoZWR1bGVTdWJtaXQoZSxoLGZ1bmN0aW9uKCl7cmV0dXJuIHIucHVzaEZvcm1TdWJtaXQoZSx0LG4saSl9KX1pZihyZS5pbnB1dHNBd2FpdGluZ1ByZWZsaWdodChlKS5sZW5ndGg+MCl7dmFyIGY9eChjKCksMiksdj1mWzBdLHA9ZlsxXSxnPWZ1bmN0aW9uKCl7cmV0dXJuW3YscF19O3RoaXMudXBsb2FkRmlsZXMoZSx0LHYsbCxmdW5jdGlvbih0KXt2YXIgbz11ZShlLHt9KTtyLnB1c2hXaXRoUmVwbHkoZyxcImV2ZW50XCIse3R5cGU6XCJmb3JtXCIsZXZlbnQ6bix2YWx1ZTpvLGNpZDpsfSxpKX0pfWVsc2V7dmFyIG09dWUoZSk7dGhpcy5wdXNoV2l0aFJlcGx5KGMsXCJldmVudFwiLHt0eXBlOlwiZm9ybVwiLGV2ZW50Om4sdmFsdWU6bSxjaWQ6bH0saSl9fX0se2tleTpcInVwbG9hZEZpbGVzXCIsdmFsdWU6ZnVuY3Rpb24oZSx0LG4saSxyKXt2YXIgbz10aGlzLGE9dGhpcy5qb2luQ291bnQ7cmUuYWN0aXZlRmlsZUlucHV0cyhlKS5mb3JFYWNoKGZ1bmN0aW9uKGUpe3ZhciBpPW5ldyByZShlLG8scik7by51cGxvYWRlcnNbZV09aTt2YXIgdT1pLmVudHJpZXMoKS5tYXAoZnVuY3Rpb24oZSl7cmV0dXJuIGUudG9QcmVmbGlnaHRQYXlsb2FkKCl9KSxzPXtyZWY6ZS5nZXRBdHRyaWJ1dGUoTSksZW50cmllczp1LGNpZDpvLnRhcmdldENvbXBvbmVudElEKGUuZm9ybSx0KX07by5sb2coXCJ1cGxvYWRcIixmdW5jdGlvbigpe3JldHVybltcInNlbmRpbmcgcHJlZmxpZ2h0IHJlcXVlc3RcIixzXX0pLG8ucHVzaFdpdGhSZXBseShudWxsLFwiYWxsb3dfdXBsb2FkXCIscyxmdW5jdGlvbihlKXtpZihvLmxvZyhcInVwbG9hZFwiLGZ1bmN0aW9uKCl7cmV0dXJuW1wiZ290IHByZWZsaWdodCByZXNwb25zZVwiLGVdfSksZS5lcnJvcil7by51bmRvUmVmcyhuKTt2YXIgdD14KGUuZXJyb3IsMikscj10WzBdLHU9dFsxXTtvLmxvZyhcInVwbG9hZFwiLGZ1bmN0aW9uKCl7cmV0dXJuW1wiZXJyb3IgZm9yIGVudHJ5IFwiLmNvbmNhdChyKSx1XX0pfWVsc2V7aS5pbml0QWRhcHRlclVwbG9hZChlLGZ1bmN0aW9uKGUpe28uY2hhbm5lbC5vbkVycm9yKGZ1bmN0aW9uKCl7by5qb2luQ291bnQ9PT1hJiZlKCl9KX0sby5saXZlU29ja2V0KX19KX0pfX0se2tleTpcInB1c2hGb3JtUmVjb3ZlcnlcIix2YWx1ZTpmdW5jdGlvbihlLHQpe3ZhciBuPXRoaXM7dGhpcy5saXZlU29ja2V0LndpdGhpbk93bmVycyhlLGZ1bmN0aW9uKGkscil7dmFyIG89ZS5lbGVtZW50c1swXSxhPWUuZ2V0QXR0cmlidXRlKG4uYmluZGluZyhcImF1dG8tcmVjb3ZlclwiKSl8fGUuZ2V0QXR0cmlidXRlKG4uYmluZGluZyhcImNoYW5nZVwiKSk7aS5wdXNoSW5wdXQobyxyLGEsbyx0KX0pfX0se2tleTpcInB1c2hMaW5rUGF0Y2hcIix2YWx1ZTpmdW5jdGlvbihlLHQsbil7dmFyIGk9dGhpcyxyPXRoaXMubGl2ZVNvY2tldC5zZXRQZW5kaW5nTGluayhlKSxvPXQ/ZnVuY3Rpb24oKXtyZXR1cm4gaS5wdXRSZWYoW3RdLFwiY2xpY2tcIil9Om51bGw7dGhpcy5wdXNoV2l0aFJlcGx5KG8sXCJsaW5rXCIse3VybDplfSxmdW5jdGlvbih0KXt0LmxpbmtfcmVkaXJlY3Q/aS5saXZlU29ja2V0LnJlcGxhY2VNYWluKGUsbnVsbCxuLHIpOmkubGl2ZVNvY2tldC5jb21taXRQZW5kaW5nTGluayhyKSYmKGkuaHJlZj1lLGkuYXBwbHlQZW5kaW5nVXBkYXRlcygpLG4mJm4oKSl9KS5yZWNlaXZlKFwidGltZW91dFwiLGZ1bmN0aW9uKCl7cmV0dXJuIGkubGl2ZVNvY2tldC5yZWRpcmVjdCh3aW5kb3cubG9jYXRpb24uaHJlZil9KX19LHtrZXk6XCJmb3Jtc0ZvclJlY292ZXJ5XCIsdmFsdWU6ZnVuY3Rpb24oZSl7dmFyIHQ9dGhpcztpZigwPT09dGhpcy5qb2luQ291bnQpcmV0dXJuW107dmFyIG49dGhpcy5iaW5kaW5nKFwiY2hhbmdlXCIpLGk9ZG9jdW1lbnQuY3JlYXRlRWxlbWVudChcInRlbXBsYXRlXCIpO3JldHVybiBpLmlubmVySFRNTD1lLGRlLmFsbCh0aGlzLmVsLFwiZm9ybVtcIi5jb25jYXQobixcIl1cIikpLmZpbHRlcihmdW5jdGlvbihlKXtyZXR1cm4gdC5vd25zRWxlbWVudChlKX0pLmZpbHRlcihmdW5jdGlvbihlKXtyZXR1cm4gZS5lbGVtZW50cy5sZW5ndGg+MH0pLmZpbHRlcihmdW5jdGlvbihlKXtyZXR1cm5cImlnbm9yZVwiIT09ZS5nZXRBdHRyaWJ1dGUodC5iaW5kaW5nKFwiYXV0by1yZWNvdmVyXCIpKX0pLmZpbHRlcihmdW5jdGlvbihlKXtyZXR1cm4gaS5jb250ZW50LnF1ZXJ5U2VsZWN0b3IoXCJmb3JtW1wiLmNvbmNhdChuLCc9XCInKS5jb25jYXQoZS5nZXRBdHRyaWJ1dGUobiksJ1wiXScpKX0pfX0se2tleTpcIm1heWJlUHVzaENvbXBvbmVudHNEZXN0cm95ZWRcIix2YWx1ZTpmdW5jdGlvbihlKXt2YXIgdCxuPXRoaXMsaT1lLmZpbHRlcihmdW5jdGlvbihlKXtyZXR1cm4gMD09PWRlLmZpbmRDb21wb25lbnROb2RlTGlzdChuLmVsLGUpLmxlbmd0aH0pO2kubGVuZ3RoPjAmJigodD10aGlzLnBydW5pbmdDSURzKS5wdXNoLmFwcGx5KHQsQShpKSksdGhpcy5wdXNoV2l0aFJlcGx5KG51bGwsXCJjaWRzX3dpbGxfZGVzdHJveVwiLHtjaWRzOml9LGZ1bmN0aW9uKCl7bi5wcnVuaW5nQ0lEcz1uLnBydW5pbmdDSURzLmZpbHRlcihmdW5jdGlvbihlKXtyZXR1cm4tMSE9PWkuaW5kZXhPZihlKX0pO3ZhciBlPWkuZmlsdGVyKGZ1bmN0aW9uKGUpe3JldHVybiAwPT09ZGUuZmluZENvbXBvbmVudE5vZGVMaXN0KG4uZWwsZSkubGVuZ3RofSk7ZS5sZW5ndGg+MCYmbi5wdXNoV2l0aFJlcGx5KG51bGwsXCJjaWRzX2Rlc3Ryb3llZFwiLHtjaWRzOmV9LGZ1bmN0aW9uKGUpe24ucmVuZGVyZWQucHJ1bmVDSURzKGUuY2lkcyl9KX0pKX19LHtrZXk6XCJvd25zRWxlbWVudFwiLHZhbHVlOmZ1bmN0aW9uKGUpe3JldHVybiBlLmdldEF0dHJpYnV0ZShcImRhdGEtcGh4LXBhcmVudC1pZFwiKT09PXRoaXMuaWR8fGVlKGUuY2xvc2VzdChVKSxmdW5jdGlvbihlKXtyZXR1cm4gZS5pZH0pPT09dGhpcy5pZH19LHtrZXk6XCJzdWJtaXRGb3JtXCIsdmFsdWU6ZnVuY3Rpb24oZSx0LG4pe3ZhciBpPXRoaXM7ZGUucHV0UHJpdmF0ZShlLFwicGh4LWhhcy1zdWJtaXR0ZWRcIiwhMCksdGhpcy5saXZlU29ja2V0LmJsdXJBY3RpdmVFbGVtZW50KHRoaXMpLHRoaXMucHVzaEZvcm1TdWJtaXQoZSx0LG4sZnVuY3Rpb24oKXtpLmxpdmVTb2NrZXQucmVzdG9yZVByZXZpb3VzbHlBY3RpdmVGb2N1cygpfSl9fSx7a2V5OlwiYmluZGluZ1wiLHZhbHVlOmZ1bmN0aW9uKGUpe3JldHVybiB0aGlzLmxpdmVTb2NrZXQuYmluZGluZyhlKX19XSksZX0oKSxwZT0xLGdlPWZ1bmN0aW9uKCl7ZnVuY3Rpb24gZSh0LG4saSl7Zm9yKHZhciByIGluIFQodGhpcyxlKSx0aGlzLl9fdmlldz10LHRoaXMuX19saXZlU29ja2V0PXQubGl2ZVNvY2tldCx0aGlzLl9fY2FsbGJhY2tzPWksdGhpcy5fX2xpc3RlbmVycz1uZXcgU2V0LHRoaXMuX19pc0Rpc2Nvbm5lY3RlZD0hMSx0aGlzLmVsPW4sdGhpcy52aWV3TmFtZT10Lm5hbWUoKSx0aGlzLmVsLnBoeEhvb2tJZD10aGlzLmNvbnN0cnVjdG9yLm1ha2VJRCgpLHRoaXMuX19jYWxsYmFja3MpdGhpc1tyXT10aGlzLl9fY2FsbGJhY2tzW3JdfXJldHVybiBfKGUsbnVsbCxbe2tleTpcIm1ha2VJRFwiLHZhbHVlOmZ1bmN0aW9uKCl7cmV0dXJuIHBlKyt9fSx7a2V5OlwiZWxlbWVudElEXCIsdmFsdWU6ZnVuY3Rpb24oZSl7cmV0dXJuIGUucGh4SG9va0lkfX1dKSxfKGUsW3trZXk6XCJfX21vdW50ZWRcIix2YWx1ZTpmdW5jdGlvbigpe3RoaXMubW91bnRlZCYmdGhpcy5tb3VudGVkKCl9fSx7a2V5OlwiX191cGRhdGVkXCIsdmFsdWU6ZnVuY3Rpb24oKXt0aGlzLnVwZGF0ZWQmJnRoaXMudXBkYXRlZCgpfX0se2tleTpcIl9fYmVmb3JlVXBkYXRlXCIsdmFsdWU6ZnVuY3Rpb24oKXt0aGlzLmJlZm9yZVVwZGF0ZSYmdGhpcy5iZWZvcmVVcGRhdGUoKX19LHtrZXk6XCJfX2Rlc3Ryb3llZFwiLHZhbHVlOmZ1bmN0aW9uKCl7dGhpcy5kZXN0cm95ZWQmJnRoaXMuZGVzdHJveWVkKCl9fSx7a2V5OlwiX19yZWNvbm5lY3RlZFwiLHZhbHVlOmZ1bmN0aW9uKCl7dGhpcy5fX2lzRGlzY29ubmVjdGVkJiYodGhpcy5fX2lzRGlzY29ubmVjdGVkPSExLHRoaXMucmVjb25uZWN0ZWQmJnRoaXMucmVjb25uZWN0ZWQoKSl9fSx7a2V5OlwiX19kaXNjb25uZWN0ZWRcIix2YWx1ZTpmdW5jdGlvbigpe3RoaXMuX19pc0Rpc2Nvbm5lY3RlZD0hMCx0aGlzLmRpc2Nvbm5lY3RlZCYmdGhpcy5kaXNjb25uZWN0ZWQoKX19LHtrZXk6XCJwdXNoRXZlbnRcIix2YWx1ZTpmdW5jdGlvbihlKXt2YXIgdD1hcmd1bWVudHMubGVuZ3RoPjEmJnZvaWQgMCE9PWFyZ3VtZW50c1sxXT9hcmd1bWVudHNbMV06e30sbj1hcmd1bWVudHMubGVuZ3RoPjImJnZvaWQgMCE9PWFyZ3VtZW50c1syXT9hcmd1bWVudHNbMl06ZnVuY3Rpb24oKXt9O3JldHVybiB0aGlzLl9fdmlldy5wdXNoSG9va0V2ZW50KG51bGwsZSx0LG4pfX0se2tleTpcInB1c2hFdmVudFRvXCIsdmFsdWU6ZnVuY3Rpb24oZSx0KXt2YXIgbj1hcmd1bWVudHMubGVuZ3RoPjImJnZvaWQgMCE9PWFyZ3VtZW50c1syXT9hcmd1bWVudHNbMl06e30saT1hcmd1bWVudHMubGVuZ3RoPjMmJnZvaWQgMCE9PWFyZ3VtZW50c1szXT9hcmd1bWVudHNbM106ZnVuY3Rpb24oKXt9O3JldHVybiB0aGlzLl9fdmlldy53aXRoaW5UYXJnZXRzKGUsZnVuY3Rpb24oZSxyKXtyZXR1cm4gZS5wdXNoSG9va0V2ZW50KHIsdCxuLGkpfSl9fSx7a2V5OlwiaGFuZGxlRXZlbnRcIix2YWx1ZTpmdW5jdGlvbihlLHQpe3ZhciBuPWZ1bmN0aW9uKG4saSl7cmV0dXJuIGk/ZTp0KG4uZGV0YWlsKX07cmV0dXJuIHdpbmRvdy5hZGRFdmVudExpc3RlbmVyKFwicGh4Omhvb2s6XCIuY29uY2F0KGUpLG4pLHRoaXMuX19saXN0ZW5lcnMuYWRkKG4pLG59fSx7a2V5OlwicmVtb3ZlSGFuZGxlRXZlbnRcIix2YWx1ZTpmdW5jdGlvbihlKXt2YXIgdD1lKG51bGwsITApO3dpbmRvdy5yZW1vdmVFdmVudExpc3RlbmVyKFwicGh4Omhvb2s6XCIuY29uY2F0KHQpLGUpLHRoaXMuX19saXN0ZW5lcnMuZGVsZXRlKGUpfX0se2tleTpcIl9fY2xlYW51cF9fXCIsdmFsdWU6ZnVuY3Rpb24oKXt2YXIgZT10aGlzO3RoaXMuX19saXN0ZW5lcnMuZm9yRWFjaChmdW5jdGlvbih0KXtyZXR1cm4gZS5yZW1vdmVIYW5kbGVFdmVudCh0KX0pfX1dKSxlfSgpO3QuZGVmYXVsdD1jZX0sZnVuY3Rpb24oZSx0KXt2YXIgbjtuPWZ1bmN0aW9uKCl7cmV0dXJuIHRoaXN9KCk7dHJ5e249bnx8RnVuY3Rpb24oXCJyZXR1cm4gdGhpc1wiKSgpfHwoMCxldmFsKShcInRoaXNcIil9Y2F0Y2goZSl7XCJvYmplY3RcIj09dHlwZW9mIHdpbmRvdyYmKG49d2luZG93KX1lLmV4cG9ydHM9bn0sZnVuY3Rpb24oZSx0LG4peyhmdW5jdGlvbih0KXt0LlBob2VuaXh8fCh0LlBob2VuaXg9e30pLGUuZXhwb3J0cz10LlBob2VuaXguTGl2ZVZpZXc9bigwKX0pLmNhbGwodGhpcyxuKDEpKX1dKX0pOyJdLCJtYXBwaW5ncyI6IkFBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUNBO0FBREE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQ0E7QUFEQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUNBO0FBREE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFDQTtBQURBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFDQTtBQURBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQ0E7QUFEQTtBQUFBO0FBQUE7QUFDQTtBQURBO0FBQUE7QUFBQTtBQUNBO0FBREE7QUFBQTtBQUFBO0FBQUE7QUFDQTtBQURBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUNBO0FBREE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUNBO0FBREE7QUFBQTtBQUNBO0FBREE7QUFBQTtBQUNBO0FBREE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUNBO0FBREE7QUFDQTtBQURBO0FBQUE7QUFBQTtBQUNBO0FBREE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUNBO0FBREE7QUFBQTtBQUFBO0FBQ0E7QUFEQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQ0E7QUFEQTtBQUFBO0FBQ0E7QUFEQTtBQUFBO0FBQUE7QUFDQTtBQURBO0FBQUE7QUFBQTtBQUFBO0FBQ0E7QUFEQTtBQUFBO0FBQUE7QUFDQTtBQURBO0FBQUE7QUFDQTtBQURBO0FBQUE7QUFBQTtBQUFBO0FBQ0E7QUFEQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQ0E7QUFEQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFDQTtBQURBO0FBQUE7QUFBQTtBQUNBO0FBREE7QUFBQTtBQUFBO0FBQUE7QUFDQTtBQURBO0FBQUE7QUFDQTtBQURBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUNBO0FBREE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFDQTtBQURBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFDQTtBQURBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQ0E7QUFEQTtBQUFBO0FBQ0E7QUFEQTtBQUFBO0FBQUE7QUFDQTtBQURBO0FBQUE7QUFBQTtBQUNBO0FBREE7QUFBQTtBQUNBO0FBREE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQ0E7QUFEQTtBQUFBO0FBQ0E7QUFEQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQ0E7QUFEQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUNBO0FBREE7QUFBQTtBQUFBO0FBQ0E7QUFEQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUNBO0FBREE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUNBO0FBREE7QUFBQTtBQUFBO0FBQ0E7QUFEQTtBQUFBO0FBQUE7QUFDQTtBQURBO0FBQUE7QUFBQTtBQUNBO0FBREE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFDQTtBQURBO0FBQUE7QUFDQTtBQURBO0FBQ0E7QUFEQTtBQUFBO0FBQ0E7QUFEQTtBQUFBO0FBQUE7QUFDQTtBQURBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUNBO0FBREE7QUFBQTtBQUFBO0FBQ0E7QUFEQTtBQUFBO0FBQ0E7QUFEQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFDQTtBQURBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQ0E7QUFEQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUNBO0FBREE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFDQTtBQURBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQ0E7QUFEQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFDQTtBQURBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQ0E7QUFEQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQ0E7QUFEQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQ0E7QUFEQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUNBO0FBREE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFDQTtBQURBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQ0E7QUFEQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFDQTtBQURBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUNBO0FBREE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUNBO0FBREE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFDQTtBQURBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQ0E7QUFEQTtBQUFBO0FBQ0E7QUFEQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFDQTtBQURBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUNBO0FBREE7QUFDQTtBQURBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUNBO0FBREE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUNBO0FBREE7QUFBQTtBQUNBO0FBREE7QUFBQTtBQUFBO0FBQUE7QUFDQTtBQURBO0FBQUE7QUFDQTtBQURBO0FBQUE7QUFDQTtBQURBO0FBQUE7QUFDQTtBQURBO0FBQ0E7QUFEQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUNBO0FBREE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUNBO0FBREE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFDQTtBQURBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQ0E7QUFEQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQ0E7QUFEQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQ0E7QUFEQTtBQUFBO0FBQ0E7QUFEQTtBQUFBO0FBQUE7QUFDQTtBQURBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFDQTtBQURBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUNBO0FBREE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFDQTtBQURBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFDQTtBQURBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUNBO0FBREE7QUFBQTtBQUFBO0FBQ0E7QUFEQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUNBO0FBREE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUNBO0FBREE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUNBO0FBREE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQ0E7QUFEQTtBQUFBO0FBQUE7QUFDQTtBQURBO0FBQUE7QUFBQTtBQUFBO0FBQ0E7QUFEQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFDQTtBQURBO0FBQUE7QUFDQTtBQURBO0FBQUE7QUFBQTtBQUFBO0FBQ0E7QUFEQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUNBO0FBREE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFDQTtBQURBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQ0E7QUFEQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQ0E7QUFEQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFDQTtBQURBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFDQTtBQURBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFDQTtBQURBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFDQTtBQURBO0FBQUE7QUFBQTtBQUFBO0FBQ0E7QUFEQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFDQTtBQURBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQ0E7QUFEQTtBQUFBO0FBQUE7QUFDQTtBQURBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFDQTtBQURBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQ0E7QUFEQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUNBO0FBREE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUNBO0FBREE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFDQTtBQURBO0FBQUE7QUFDQTtBQURBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFDQTtBQURBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUNBO0FBREE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQ0E7QUFEQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFDQTtBQURBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFDQTtBQURBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQ0E7QUFEQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQ0E7QUFEQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQ0E7QUFEQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUNBO0FBREE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUNBO0FBREE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUNBO0FBREE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQ0E7QUFEQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFDQTtBQURBO0FBQUE7QUFBQTtBQUNBO0FBREE7QUFBQTtBQUFBO0FBQ0E7QUFEQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQ0E7QUFEQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQSIsInNvdXJjZVJvb3QiOiIifQ==\n//# sourceURL=webpack-internal:///../deps/phoenix_live_view/priv/static/phoenix_live_view.js\n"); + + /***/ }), + + /***/ "./css/app.scss": + /*!**********************!*\ + !*** ./css/app.scss ***! + \**********************/ + /*! no static exports found */ + /***/ (function(module, exports, __webpack_require__) { + + eval("// extracted by mini-css-extract-plugin//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiLi9jc3MvYXBwLnNjc3MuanMiLCJzb3VyY2VzIjpbIndlYnBhY2s6Ly8vLi9jc3MvYXBwLnNjc3M/MjQyYyJdLCJzb3VyY2VzQ29udGVudCI6WyIvLyBleHRyYWN0ZWQgYnkgbWluaS1jc3MtZXh0cmFjdC1wbHVnaW4iXSwibWFwcGluZ3MiOiJBQUFBIiwic291cmNlUm9vdCI6IiJ9\n//# sourceURL=webpack-internal:///./css/app.scss\n"); + + /***/ }), + + /***/ "./js/app.js": + /*!*******************!*\ + !*** ./js/app.js ***! + \*******************/ + /*! no exports provided */ + /***/ (function(module, __webpack_exports__, __webpack_require__) { + + "use strict"; + eval("__webpack_require__.r(__webpack_exports__);\n/* harmony import */ var _css_app_scss__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ../css/app.scss */ \"./css/app.scss\");\n/* harmony import */ var _css_app_scss__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(_css_app_scss__WEBPACK_IMPORTED_MODULE_0__);\n/* harmony import */ var phoenix_html__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! phoenix_html */ \"../deps/phoenix_html/priv/static/phoenix_html.js\");\n/* harmony import */ var phoenix_html__WEBPACK_IMPORTED_MODULE_1___default = /*#__PURE__*/__webpack_require__.n(phoenix_html__WEBPACK_IMPORTED_MODULE_1__);\n/* harmony import */ var phoenix__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! phoenix */ \"../deps/phoenix/priv/static/phoenix.js\");\n/* harmony import */ var phoenix__WEBPACK_IMPORTED_MODULE_2___default = /*#__PURE__*/__webpack_require__.n(phoenix__WEBPACK_IMPORTED_MODULE_2__);\n/* harmony import */ var nprogress__WEBPACK_IMPORTED_MODULE_3__ = __webpack_require__(/*! nprogress */ \"./node_modules/nprogress/nprogress.js\");\n/* harmony import */ var nprogress__WEBPACK_IMPORTED_MODULE_3___default = /*#__PURE__*/__webpack_require__.n(nprogress__WEBPACK_IMPORTED_MODULE_3__);\n/* harmony import */ var phoenix_live_view__WEBPACK_IMPORTED_MODULE_4__ = __webpack_require__(/*! phoenix_live_view */ \"../deps/phoenix_live_view/priv/static/phoenix_live_view.js\");\n/* harmony import */ var phoenix_live_view__WEBPACK_IMPORTED_MODULE_4___default = /*#__PURE__*/__webpack_require__.n(phoenix_live_view__WEBPACK_IMPORTED_MODULE_4__);\n// We need to import the CSS so that webpack will load it.\n// The MiniCssExtractPlugin is used to separate it out into\n// its own CSS file.\n // webpack automatically bundles all modules in your\n// entry points. Those entry points can be configured\n// in \"webpack.config.js\".\n//\n// Import deps with the dep name or local files with a relative path, for example:\n//\n// import {Socket} from \"phoenix\"\n// import socket from \"./socket\"\n//\n\n\n\n\n\nvar csrfToken = document.querySelector(\"meta[name='csrf-token']\").getAttribute(\"content\");\nvar liveSocket = new phoenix_live_view__WEBPACK_IMPORTED_MODULE_4__[\"LiveSocket\"](\"/live\", phoenix__WEBPACK_IMPORTED_MODULE_2__[\"Socket\"], {\n params: {\n _csrf_token: csrfToken\n }\n}); // Show progress bar on live navigation and form submits\n\nwindow.addEventListener(\"phx:page-loading-start\", function (info) {\n return nprogress__WEBPACK_IMPORTED_MODULE_3___default.a.start();\n});\nwindow.addEventListener(\"phx:page-loading-stop\", function (info) {\n return nprogress__WEBPACK_IMPORTED_MODULE_3___default.a.done();\n}); // connect if there are any LiveViews on the page\n\nliveSocket.connect(); // expose liveSocket on window for web console debug logs and latency simulation:\n// >> liveSocket.enableDebug()\n// >> liveSocket.enableLatencySim(1000) // enabled for duration of browser session\n// >> liveSocket.disableLatencySim()\n\nwindow.liveSocket = liveSocket;//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiLi9qcy9hcHAuanMuanMiLCJzb3VyY2VzIjpbIndlYnBhY2s6Ly8vLi9qcy9hcHAuanM/NzQ3MyJdLCJzb3VyY2VzQ29udGVudCI6WyIvLyBXZSBuZWVkIHRvIGltcG9ydCB0aGUgQ1NTIHNvIHRoYXQgd2VicGFjayB3aWxsIGxvYWQgaXQuXG4vLyBUaGUgTWluaUNzc0V4dHJhY3RQbHVnaW4gaXMgdXNlZCB0byBzZXBhcmF0ZSBpdCBvdXQgaW50b1xuLy8gaXRzIG93biBDU1MgZmlsZS5cbmltcG9ydCBcIi4uL2Nzcy9hcHAuc2Nzc1wiXG5cbi8vIHdlYnBhY2sgYXV0b21hdGljYWxseSBidW5kbGVzIGFsbCBtb2R1bGVzIGluIHlvdXJcbi8vIGVudHJ5IHBvaW50cy4gVGhvc2UgZW50cnkgcG9pbnRzIGNhbiBiZSBjb25maWd1cmVkXG4vLyBpbiBcIndlYnBhY2suY29uZmlnLmpzXCIuXG4vL1xuLy8gSW1wb3J0IGRlcHMgd2l0aCB0aGUgZGVwIG5hbWUgb3IgbG9jYWwgZmlsZXMgd2l0aCBhIHJlbGF0aXZlIHBhdGgsIGZvciBleGFtcGxlOlxuLy9cbi8vICAgICBpbXBvcnQge1NvY2tldH0gZnJvbSBcInBob2VuaXhcIlxuLy8gICAgIGltcG9ydCBzb2NrZXQgZnJvbSBcIi4vc29ja2V0XCJcbi8vXG5pbXBvcnQgXCJwaG9lbml4X2h0bWxcIlxuaW1wb3J0IHtTb2NrZXR9IGZyb20gXCJwaG9lbml4XCJcbmltcG9ydCBOUHJvZ3Jlc3MgZnJvbSBcIm5wcm9ncmVzc1wiXG5pbXBvcnQge0xpdmVTb2NrZXR9IGZyb20gXCJwaG9lbml4X2xpdmVfdmlld1wiXG5cbmxldCBjc3JmVG9rZW4gPSBkb2N1bWVudC5xdWVyeVNlbGVjdG9yKFwibWV0YVtuYW1lPSdjc3JmLXRva2VuJ11cIikuZ2V0QXR0cmlidXRlKFwiY29udGVudFwiKVxubGV0IGxpdmVTb2NrZXQgPSBuZXcgTGl2ZVNvY2tldChcIi9saXZlXCIsIFNvY2tldCwge3BhcmFtczoge19jc3JmX3Rva2VuOiBjc3JmVG9rZW59fSlcblxuLy8gU2hvdyBwcm9ncmVzcyBiYXIgb24gbGl2ZSBuYXZpZ2F0aW9uIGFuZCBmb3JtIHN1Ym1pdHNcbndpbmRvdy5hZGRFdmVudExpc3RlbmVyKFwicGh4OnBhZ2UtbG9hZGluZy1zdGFydFwiLCBpbmZvID0+IE5Qcm9ncmVzcy5zdGFydCgpKVxud2luZG93LmFkZEV2ZW50TGlzdGVuZXIoXCJwaHg6cGFnZS1sb2FkaW5nLXN0b3BcIiwgaW5mbyA9PiBOUHJvZ3Jlc3MuZG9uZSgpKVxuXG4vLyBjb25uZWN0IGlmIHRoZXJlIGFyZSBhbnkgTGl2ZVZpZXdzIG9uIHRoZSBwYWdlXG5saXZlU29ja2V0LmNvbm5lY3QoKVxuXG4vLyBleHBvc2UgbGl2ZVNvY2tldCBvbiB3aW5kb3cgZm9yIHdlYiBjb25zb2xlIGRlYnVnIGxvZ3MgYW5kIGxhdGVuY3kgc2ltdWxhdGlvbjpcbi8vID4+IGxpdmVTb2NrZXQuZW5hYmxlRGVidWcoKVxuLy8gPj4gbGl2ZVNvY2tldC5lbmFibGVMYXRlbmN5U2ltKDEwMDApICAvLyBlbmFibGVkIGZvciBkdXJhdGlvbiBvZiBicm93c2VyIHNlc3Npb25cbi8vID4+IGxpdmVTb2NrZXQuZGlzYWJsZUxhdGVuY3lTaW0oKVxud2luZG93LmxpdmVTb2NrZXQgPSBsaXZlU29ja2V0XG5cbiJdLCJtYXBwaW5ncyI6IkFBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQ0E7QUFDQTtBQUNBO0FBR0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQUE7QUFDQTtBQUNBO0FBQ0E7QUFFQTtBQUNBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFDQTtBQUVBO0FBQUE7QUFBQTtBQUNBO0FBQUE7QUFBQTtBQUNBO0FBRUE7QUFHQTtBQUNBO0FBQ0E7QUFDQTtBQUFBIiwic291cmNlUm9vdCI6IiJ9\n//# sourceURL=webpack-internal:///./js/app.js\n"); + + /***/ }), + + /***/ "./node_modules/nprogress/nprogress.js": + /*!*********************************************!*\ + !*** ./node_modules/nprogress/nprogress.js ***! + \*********************************************/ + /*! no static exports found */ + /***/ (function(module, exports, __webpack_require__) { + + eval("var __WEBPACK_AMD_DEFINE_FACTORY__, __WEBPACK_AMD_DEFINE_RESULT__;/* NProgress, (c) 2013, 2014 Rico Sta. Cruz - http://ricostacruz.com/nprogress\n * @license MIT */\n\n;(function(root, factory) {\n\n if (true) {\n !(__WEBPACK_AMD_DEFINE_FACTORY__ = (factory),\n\t\t\t\t__WEBPACK_AMD_DEFINE_RESULT__ = (typeof __WEBPACK_AMD_DEFINE_FACTORY__ === 'function' ?\n\t\t\t\t(__WEBPACK_AMD_DEFINE_FACTORY__.call(exports, __webpack_require__, exports, module)) :\n\t\t\t\t__WEBPACK_AMD_DEFINE_FACTORY__),\n\t\t\t\t__WEBPACK_AMD_DEFINE_RESULT__ !== undefined && (module.exports = __WEBPACK_AMD_DEFINE_RESULT__));\n } else {}\n\n})(this, function() {\n var NProgress = {};\n\n NProgress.version = '0.2.0';\n\n var Settings = NProgress.settings = {\n minimum: 0.08,\n easing: 'ease',\n positionUsing: '',\n speed: 200,\n trickle: true,\n trickleRate: 0.02,\n trickleSpeed: 800,\n showSpinner: true,\n barSelector: '[role=\"bar\"]',\n spinnerSelector: '[role=\"spinner\"]',\n parent: 'body',\n template: '
'\n };\n\n /**\n * Updates configuration.\n *\n * NProgress.configure({\n * minimum: 0.1\n * });\n */\n NProgress.configure = function(options) {\n var key, value;\n for (key in options) {\n value = options[key];\n if (value !== undefined && options.hasOwnProperty(key)) Settings[key] = value;\n }\n\n return this;\n };\n\n /**\n * Last number.\n */\n\n NProgress.status = null;\n\n /**\n * Sets the progress bar status, where `n` is a number from `0.0` to `1.0`.\n *\n * NProgress.set(0.4);\n * NProgress.set(1.0);\n */\n\n NProgress.set = function(n) {\n var started = NProgress.isStarted();\n\n n = clamp(n, Settings.minimum, 1);\n NProgress.status = (n === 1 ? null : n);\n\n var progress = NProgress.render(!started),\n bar = progress.querySelector(Settings.barSelector),\n speed = Settings.speed,\n ease = Settings.easing;\n\n progress.offsetWidth; /* Repaint */\n\n queue(function(next) {\n // Set positionUsing if it hasn't already been set\n if (Settings.positionUsing === '') Settings.positionUsing = NProgress.getPositioningCSS();\n\n // Add transition\n css(bar, barPositionCSS(n, speed, ease));\n\n if (n === 1) {\n // Fade out\n css(progress, { \n transition: 'none', \n opacity: 1 \n });\n progress.offsetWidth; /* Repaint */\n\n setTimeout(function() {\n css(progress, { \n transition: 'all ' + speed + 'ms linear', \n opacity: 0 \n });\n setTimeout(function() {\n NProgress.remove();\n next();\n }, speed);\n }, speed);\n } else {\n setTimeout(next, speed);\n }\n });\n\n return this;\n };\n\n NProgress.isStarted = function() {\n return typeof NProgress.status === 'number';\n };\n\n /**\n * Shows the progress bar.\n * This is the same as setting the status to 0%, except that it doesn't go backwards.\n *\n * NProgress.start();\n *\n */\n NProgress.start = function() {\n if (!NProgress.status) NProgress.set(0);\n\n var work = function() {\n setTimeout(function() {\n if (!NProgress.status) return;\n NProgress.trickle();\n work();\n }, Settings.trickleSpeed);\n };\n\n if (Settings.trickle) work();\n\n return this;\n };\n\n /**\n * Hides the progress bar.\n * This is the *sort of* the same as setting the status to 100%, with the\n * difference being `done()` makes some placebo effect of some realistic motion.\n *\n * NProgress.done();\n *\n * If `true` is passed, it will show the progress bar even if its hidden.\n *\n * NProgress.done(true);\n */\n\n NProgress.done = function(force) {\n if (!force && !NProgress.status) return this;\n\n return NProgress.inc(0.3 + 0.5 * Math.random()).set(1);\n };\n\n /**\n * Increments by a random amount.\n */\n\n NProgress.inc = function(amount) {\n var n = NProgress.status;\n\n if (!n) {\n return NProgress.start();\n } else {\n if (typeof amount !== 'number') {\n amount = (1 - n) * clamp(Math.random() * n, 0.1, 0.95);\n }\n\n n = clamp(n + amount, 0, 0.994);\n return NProgress.set(n);\n }\n };\n\n NProgress.trickle = function() {\n return NProgress.inc(Math.random() * Settings.trickleRate);\n };\n\n /**\n * Waits for all supplied jQuery promises and\n * increases the progress as the promises resolve.\n *\n * @param $promise jQUery Promise\n */\n (function() {\n var initial = 0, current = 0;\n\n NProgress.promise = function($promise) {\n if (!$promise || $promise.state() === \"resolved\") {\n return this;\n }\n\n if (current === 0) {\n NProgress.start();\n }\n\n initial++;\n current++;\n\n $promise.always(function() {\n current--;\n if (current === 0) {\n initial = 0;\n NProgress.done();\n } else {\n NProgress.set((initial - current) / initial);\n }\n });\n\n return this;\n };\n\n })();\n\n /**\n * (Internal) renders the progress bar markup based on the `template`\n * setting.\n */\n\n NProgress.render = function(fromStart) {\n if (NProgress.isRendered()) return document.getElementById('nprogress');\n\n addClass(document.documentElement, 'nprogress-busy');\n \n var progress = document.createElement('div');\n progress.id = 'nprogress';\n progress.innerHTML = Settings.template;\n\n var bar = progress.querySelector(Settings.barSelector),\n perc = fromStart ? '-100' : toBarPerc(NProgress.status || 0),\n parent = document.querySelector(Settings.parent),\n spinner;\n \n css(bar, {\n transition: 'all 0 linear',\n transform: 'translate3d(' + perc + '%,0,0)'\n });\n\n if (!Settings.showSpinner) {\n spinner = progress.querySelector(Settings.spinnerSelector);\n spinner && removeElement(spinner);\n }\n\n if (parent != document.body) {\n addClass(parent, 'nprogress-custom-parent');\n }\n\n parent.appendChild(progress);\n return progress;\n };\n\n /**\n * Removes the element. Opposite of render().\n */\n\n NProgress.remove = function() {\n removeClass(document.documentElement, 'nprogress-busy');\n removeClass(document.querySelector(Settings.parent), 'nprogress-custom-parent');\n var progress = document.getElementById('nprogress');\n progress && removeElement(progress);\n };\n\n /**\n * Checks if the progress bar is rendered.\n */\n\n NProgress.isRendered = function() {\n return !!document.getElementById('nprogress');\n };\n\n /**\n * Determine which positioning CSS rule to use.\n */\n\n NProgress.getPositioningCSS = function() {\n // Sniff on document.body.style\n var bodyStyle = document.body.style;\n\n // Sniff prefixes\n var vendorPrefix = ('WebkitTransform' in bodyStyle) ? 'Webkit' :\n ('MozTransform' in bodyStyle) ? 'Moz' :\n ('msTransform' in bodyStyle) ? 'ms' :\n ('OTransform' in bodyStyle) ? 'O' : '';\n\n if (vendorPrefix + 'Perspective' in bodyStyle) {\n // Modern browsers with 3D support, e.g. Webkit, IE10\n return 'translate3d';\n } else if (vendorPrefix + 'Transform' in bodyStyle) {\n // Browsers without 3D support, e.g. IE9\n return 'translate';\n } else {\n // Browsers without translate() support, e.g. IE7-8\n return 'margin';\n }\n };\n\n /**\n * Helpers\n */\n\n function clamp(n, min, max) {\n if (n < min) return min;\n if (n > max) return max;\n return n;\n }\n\n /**\n * (Internal) converts a percentage (`0..1`) to a bar translateX\n * percentage (`-100%..0%`).\n */\n\n function toBarPerc(n) {\n return (-1 + n) * 100;\n }\n\n\n /**\n * (Internal) returns the correct CSS for changing the bar's\n * position given an n percentage, and speed and ease from Settings\n */\n\n function barPositionCSS(n, speed, ease) {\n var barCSS;\n\n if (Settings.positionUsing === 'translate3d') {\n barCSS = { transform: 'translate3d('+toBarPerc(n)+'%,0,0)' };\n } else if (Settings.positionUsing === 'translate') {\n barCSS = { transform: 'translate('+toBarPerc(n)+'%,0)' };\n } else {\n barCSS = { 'margin-left': toBarPerc(n)+'%' };\n }\n\n barCSS.transition = 'all '+speed+'ms '+ease;\n\n return barCSS;\n }\n\n /**\n * (Internal) Queues a function to be executed.\n */\n\n var queue = (function() {\n var pending = [];\n \n function next() {\n var fn = pending.shift();\n if (fn) {\n fn(next);\n }\n }\n\n return function(fn) {\n pending.push(fn);\n if (pending.length == 1) next();\n };\n })();\n\n /**\n * (Internal) Applies css properties to an element, similar to the jQuery \n * css method.\n *\n * While this helper does assist with vendor prefixed property names, it \n * does not perform any manipulation of values prior to setting styles.\n */\n\n var css = (function() {\n var cssPrefixes = [ 'Webkit', 'O', 'Moz', 'ms' ],\n cssProps = {};\n\n function camelCase(string) {\n return string.replace(/^-ms-/, 'ms-').replace(/-([\\da-z])/gi, function(match, letter) {\n return letter.toUpperCase();\n });\n }\n\n function getVendorProp(name) {\n var style = document.body.style;\n if (name in style) return name;\n\n var i = cssPrefixes.length,\n capName = name.charAt(0).toUpperCase() + name.slice(1),\n vendorName;\n while (i--) {\n vendorName = cssPrefixes[i] + capName;\n if (vendorName in style) return vendorName;\n }\n\n return name;\n }\n\n function getStyleProp(name) {\n name = camelCase(name);\n return cssProps[name] || (cssProps[name] = getVendorProp(name));\n }\n\n function applyCss(element, prop, value) {\n prop = getStyleProp(prop);\n element.style[prop] = value;\n }\n\n return function(element, properties) {\n var args = arguments,\n prop, \n value;\n\n if (args.length == 2) {\n for (prop in properties) {\n value = properties[prop];\n if (value !== undefined && properties.hasOwnProperty(prop)) applyCss(element, prop, value);\n }\n } else {\n applyCss(element, args[1], args[2]);\n }\n }\n })();\n\n /**\n * (Internal) Determines if an element or space separated list of class names contains a class name.\n */\n\n function hasClass(element, name) {\n var list = typeof element == 'string' ? element : classList(element);\n return list.indexOf(' ' + name + ' ') >= 0;\n }\n\n /**\n * (Internal) Adds a class to an element.\n */\n\n function addClass(element, name) {\n var oldList = classList(element),\n newList = oldList + name;\n\n if (hasClass(oldList, name)) return; \n\n // Trim the opening space.\n element.className = newList.substring(1);\n }\n\n /**\n * (Internal) Removes a class from an element.\n */\n\n function removeClass(element, name) {\n var oldList = classList(element),\n newList;\n\n if (!hasClass(element, name)) return;\n\n // Replace the class name.\n newList = oldList.replace(' ' + name + ' ', ' ');\n\n // Trim the opening and closing spaces.\n element.className = newList.substring(1, newList.length - 1);\n }\n\n /**\n * (Internal) Gets a space separated list of the class names on the element. \n * The list is wrapped with a single space on each end to facilitate finding \n * matches within the list.\n */\n\n function classList(element) {\n return (' ' + (element.className || '') + ' ').replace(/\\s+/gi, ' ');\n }\n\n /**\n * (Internal) Removes an element from the DOM.\n */\n\n function removeElement(element) {\n element && element.parentNode && element.parentNode.removeChild(element);\n }\n\n return NProgress;\n});\n\n//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiLi9ub2RlX21vZHVsZXMvbnByb2dyZXNzL25wcm9ncmVzcy5qcy5qcyIsInNvdXJjZXMiOlsid2VicGFjazovLy8uL25vZGVfbW9kdWxlcy9ucHJvZ3Jlc3MvbnByb2dyZXNzLmpzPzMyM2UiXSwic291cmNlc0NvbnRlbnQiOlsiLyogTlByb2dyZXNzLCAoYykgMjAxMywgMjAxNCBSaWNvIFN0YS4gQ3J1eiAtIGh0dHA6Ly9yaWNvc3RhY3J1ei5jb20vbnByb2dyZXNzXG4gKiBAbGljZW5zZSBNSVQgKi9cblxuOyhmdW5jdGlvbihyb290LCBmYWN0b3J5KSB7XG5cbiAgaWYgKHR5cGVvZiBkZWZpbmUgPT09ICdmdW5jdGlvbicgJiYgZGVmaW5lLmFtZCkge1xuICAgIGRlZmluZShmYWN0b3J5KTtcbiAgfSBlbHNlIGlmICh0eXBlb2YgZXhwb3J0cyA9PT0gJ29iamVjdCcpIHtcbiAgICBtb2R1bGUuZXhwb3J0cyA9IGZhY3RvcnkoKTtcbiAgfSBlbHNlIHtcbiAgICByb290Lk5Qcm9ncmVzcyA9IGZhY3RvcnkoKTtcbiAgfVxuXG59KSh0aGlzLCBmdW5jdGlvbigpIHtcbiAgdmFyIE5Qcm9ncmVzcyA9IHt9O1xuXG4gIE5Qcm9ncmVzcy52ZXJzaW9uID0gJzAuMi4wJztcblxuICB2YXIgU2V0dGluZ3MgPSBOUHJvZ3Jlc3Muc2V0dGluZ3MgPSB7XG4gICAgbWluaW11bTogMC4wOCxcbiAgICBlYXNpbmc6ICdlYXNlJyxcbiAgICBwb3NpdGlvblVzaW5nOiAnJyxcbiAgICBzcGVlZDogMjAwLFxuICAgIHRyaWNrbGU6IHRydWUsXG4gICAgdHJpY2tsZVJhdGU6IDAuMDIsXG4gICAgdHJpY2tsZVNwZWVkOiA4MDAsXG4gICAgc2hvd1NwaW5uZXI6IHRydWUsXG4gICAgYmFyU2VsZWN0b3I6ICdbcm9sZT1cImJhclwiXScsXG4gICAgc3Bpbm5lclNlbGVjdG9yOiAnW3JvbGU9XCJzcGlubmVyXCJdJyxcbiAgICBwYXJlbnQ6ICdib2R5JyxcbiAgICB0ZW1wbGF0ZTogJzxkaXYgY2xhc3M9XCJiYXJcIiByb2xlPVwiYmFyXCI+PGRpdiBjbGFzcz1cInBlZ1wiPjwvZGl2PjwvZGl2PjxkaXYgY2xhc3M9XCJzcGlubmVyXCIgcm9sZT1cInNwaW5uZXJcIj48ZGl2IGNsYXNzPVwic3Bpbm5lci1pY29uXCI+PC9kaXY+PC9kaXY+J1xuICB9O1xuXG4gIC8qKlxuICAgKiBVcGRhdGVzIGNvbmZpZ3VyYXRpb24uXG4gICAqXG4gICAqICAgICBOUHJvZ3Jlc3MuY29uZmlndXJlKHtcbiAgICogICAgICAgbWluaW11bTogMC4xXG4gICAqICAgICB9KTtcbiAgICovXG4gIE5Qcm9ncmVzcy5jb25maWd1cmUgPSBmdW5jdGlvbihvcHRpb25zKSB7XG4gICAgdmFyIGtleSwgdmFsdWU7XG4gICAgZm9yIChrZXkgaW4gb3B0aW9ucykge1xuICAgICAgdmFsdWUgPSBvcHRpb25zW2tleV07XG4gICAgICBpZiAodmFsdWUgIT09IHVuZGVmaW5lZCAmJiBvcHRpb25zLmhhc093blByb3BlcnR5KGtleSkpIFNldHRpbmdzW2tleV0gPSB2YWx1ZTtcbiAgICB9XG5cbiAgICByZXR1cm4gdGhpcztcbiAgfTtcblxuICAvKipcbiAgICogTGFzdCBudW1iZXIuXG4gICAqL1xuXG4gIE5Qcm9ncmVzcy5zdGF0dXMgPSBudWxsO1xuXG4gIC8qKlxuICAgKiBTZXRzIHRoZSBwcm9ncmVzcyBiYXIgc3RhdHVzLCB3aGVyZSBgbmAgaXMgYSBudW1iZXIgZnJvbSBgMC4wYCB0byBgMS4wYC5cbiAgICpcbiAgICogICAgIE5Qcm9ncmVzcy5zZXQoMC40KTtcbiAgICogICAgIE5Qcm9ncmVzcy5zZXQoMS4wKTtcbiAgICovXG5cbiAgTlByb2dyZXNzLnNldCA9IGZ1bmN0aW9uKG4pIHtcbiAgICB2YXIgc3RhcnRlZCA9IE5Qcm9ncmVzcy5pc1N0YXJ0ZWQoKTtcblxuICAgIG4gPSBjbGFtcChuLCBTZXR0aW5ncy5taW5pbXVtLCAxKTtcbiAgICBOUHJvZ3Jlc3Muc3RhdHVzID0gKG4gPT09IDEgPyBudWxsIDogbik7XG5cbiAgICB2YXIgcHJvZ3Jlc3MgPSBOUHJvZ3Jlc3MucmVuZGVyKCFzdGFydGVkKSxcbiAgICAgICAgYmFyICAgICAgPSBwcm9ncmVzcy5xdWVyeVNlbGVjdG9yKFNldHRpbmdzLmJhclNlbGVjdG9yKSxcbiAgICAgICAgc3BlZWQgICAgPSBTZXR0aW5ncy5zcGVlZCxcbiAgICAgICAgZWFzZSAgICAgPSBTZXR0aW5ncy5lYXNpbmc7XG5cbiAgICBwcm9ncmVzcy5vZmZzZXRXaWR0aDsgLyogUmVwYWludCAqL1xuXG4gICAgcXVldWUoZnVuY3Rpb24obmV4dCkge1xuICAgICAgLy8gU2V0IHBvc2l0aW9uVXNpbmcgaWYgaXQgaGFzbid0IGFscmVhZHkgYmVlbiBzZXRcbiAgICAgIGlmIChTZXR0aW5ncy5wb3NpdGlvblVzaW5nID09PSAnJykgU2V0dGluZ3MucG9zaXRpb25Vc2luZyA9IE5Qcm9ncmVzcy5nZXRQb3NpdGlvbmluZ0NTUygpO1xuXG4gICAgICAvLyBBZGQgdHJhbnNpdGlvblxuICAgICAgY3NzKGJhciwgYmFyUG9zaXRpb25DU1Mobiwgc3BlZWQsIGVhc2UpKTtcblxuICAgICAgaWYgKG4gPT09IDEpIHtcbiAgICAgICAgLy8gRmFkZSBvdXRcbiAgICAgICAgY3NzKHByb2dyZXNzLCB7IFxuICAgICAgICAgIHRyYW5zaXRpb246ICdub25lJywgXG4gICAgICAgICAgb3BhY2l0eTogMSBcbiAgICAgICAgfSk7XG4gICAgICAgIHByb2dyZXNzLm9mZnNldFdpZHRoOyAvKiBSZXBhaW50ICovXG5cbiAgICAgICAgc2V0VGltZW91dChmdW5jdGlvbigpIHtcbiAgICAgICAgICBjc3MocHJvZ3Jlc3MsIHsgXG4gICAgICAgICAgICB0cmFuc2l0aW9uOiAnYWxsICcgKyBzcGVlZCArICdtcyBsaW5lYXInLCBcbiAgICAgICAgICAgIG9wYWNpdHk6IDAgXG4gICAgICAgICAgfSk7XG4gICAgICAgICAgc2V0VGltZW91dChmdW5jdGlvbigpIHtcbiAgICAgICAgICAgIE5Qcm9ncmVzcy5yZW1vdmUoKTtcbiAgICAgICAgICAgIG5leHQoKTtcbiAgICAgICAgICB9LCBzcGVlZCk7XG4gICAgICAgIH0sIHNwZWVkKTtcbiAgICAgIH0gZWxzZSB7XG4gICAgICAgIHNldFRpbWVvdXQobmV4dCwgc3BlZWQpO1xuICAgICAgfVxuICAgIH0pO1xuXG4gICAgcmV0dXJuIHRoaXM7XG4gIH07XG5cbiAgTlByb2dyZXNzLmlzU3RhcnRlZCA9IGZ1bmN0aW9uKCkge1xuICAgIHJldHVybiB0eXBlb2YgTlByb2dyZXNzLnN0YXR1cyA9PT0gJ251bWJlcic7XG4gIH07XG5cbiAgLyoqXG4gICAqIFNob3dzIHRoZSBwcm9ncmVzcyBiYXIuXG4gICAqIFRoaXMgaXMgdGhlIHNhbWUgYXMgc2V0dGluZyB0aGUgc3RhdHVzIHRvIDAlLCBleGNlcHQgdGhhdCBpdCBkb2Vzbid0IGdvIGJhY2t3YXJkcy5cbiAgICpcbiAgICogICAgIE5Qcm9ncmVzcy5zdGFydCgpO1xuICAgKlxuICAgKi9cbiAgTlByb2dyZXNzLnN0YXJ0ID0gZnVuY3Rpb24oKSB7XG4gICAgaWYgKCFOUHJvZ3Jlc3Muc3RhdHVzKSBOUHJvZ3Jlc3Muc2V0KDApO1xuXG4gICAgdmFyIHdvcmsgPSBmdW5jdGlvbigpIHtcbiAgICAgIHNldFRpbWVvdXQoZnVuY3Rpb24oKSB7XG4gICAgICAgIGlmICghTlByb2dyZXNzLnN0YXR1cykgcmV0dXJuO1xuICAgICAgICBOUHJvZ3Jlc3MudHJpY2tsZSgpO1xuICAgICAgICB3b3JrKCk7XG4gICAgICB9LCBTZXR0aW5ncy50cmlja2xlU3BlZWQpO1xuICAgIH07XG5cbiAgICBpZiAoU2V0dGluZ3MudHJpY2tsZSkgd29yaygpO1xuXG4gICAgcmV0dXJuIHRoaXM7XG4gIH07XG5cbiAgLyoqXG4gICAqIEhpZGVzIHRoZSBwcm9ncmVzcyBiYXIuXG4gICAqIFRoaXMgaXMgdGhlICpzb3J0IG9mKiB0aGUgc2FtZSBhcyBzZXR0aW5nIHRoZSBzdGF0dXMgdG8gMTAwJSwgd2l0aCB0aGVcbiAgICogZGlmZmVyZW5jZSBiZWluZyBgZG9uZSgpYCBtYWtlcyBzb21lIHBsYWNlYm8gZWZmZWN0IG9mIHNvbWUgcmVhbGlzdGljIG1vdGlvbi5cbiAgICpcbiAgICogICAgIE5Qcm9ncmVzcy5kb25lKCk7XG4gICAqXG4gICAqIElmIGB0cnVlYCBpcyBwYXNzZWQsIGl0IHdpbGwgc2hvdyB0aGUgcHJvZ3Jlc3MgYmFyIGV2ZW4gaWYgaXRzIGhpZGRlbi5cbiAgICpcbiAgICogICAgIE5Qcm9ncmVzcy5kb25lKHRydWUpO1xuICAgKi9cblxuICBOUHJvZ3Jlc3MuZG9uZSA9IGZ1bmN0aW9uKGZvcmNlKSB7XG4gICAgaWYgKCFmb3JjZSAmJiAhTlByb2dyZXNzLnN0YXR1cykgcmV0dXJuIHRoaXM7XG5cbiAgICByZXR1cm4gTlByb2dyZXNzLmluYygwLjMgKyAwLjUgKiBNYXRoLnJhbmRvbSgpKS5zZXQoMSk7XG4gIH07XG5cbiAgLyoqXG4gICAqIEluY3JlbWVudHMgYnkgYSByYW5kb20gYW1vdW50LlxuICAgKi9cblxuICBOUHJvZ3Jlc3MuaW5jID0gZnVuY3Rpb24oYW1vdW50KSB7XG4gICAgdmFyIG4gPSBOUHJvZ3Jlc3Muc3RhdHVzO1xuXG4gICAgaWYgKCFuKSB7XG4gICAgICByZXR1cm4gTlByb2dyZXNzLnN0YXJ0KCk7XG4gICAgfSBlbHNlIHtcbiAgICAgIGlmICh0eXBlb2YgYW1vdW50ICE9PSAnbnVtYmVyJykge1xuICAgICAgICBhbW91bnQgPSAoMSAtIG4pICogY2xhbXAoTWF0aC5yYW5kb20oKSAqIG4sIDAuMSwgMC45NSk7XG4gICAgICB9XG5cbiAgICAgIG4gPSBjbGFtcChuICsgYW1vdW50LCAwLCAwLjk5NCk7XG4gICAgICByZXR1cm4gTlByb2dyZXNzLnNldChuKTtcbiAgICB9XG4gIH07XG5cbiAgTlByb2dyZXNzLnRyaWNrbGUgPSBmdW5jdGlvbigpIHtcbiAgICByZXR1cm4gTlByb2dyZXNzLmluYyhNYXRoLnJhbmRvbSgpICogU2V0dGluZ3MudHJpY2tsZVJhdGUpO1xuICB9O1xuXG4gIC8qKlxuICAgKiBXYWl0cyBmb3IgYWxsIHN1cHBsaWVkIGpRdWVyeSBwcm9taXNlcyBhbmRcbiAgICogaW5jcmVhc2VzIHRoZSBwcm9ncmVzcyBhcyB0aGUgcHJvbWlzZXMgcmVzb2x2ZS5cbiAgICpcbiAgICogQHBhcmFtICRwcm9taXNlIGpRVWVyeSBQcm9taXNlXG4gICAqL1xuICAoZnVuY3Rpb24oKSB7XG4gICAgdmFyIGluaXRpYWwgPSAwLCBjdXJyZW50ID0gMDtcblxuICAgIE5Qcm9ncmVzcy5wcm9taXNlID0gZnVuY3Rpb24oJHByb21pc2UpIHtcbiAgICAgIGlmICghJHByb21pc2UgfHwgJHByb21pc2Uuc3RhdGUoKSA9PT0gXCJyZXNvbHZlZFwiKSB7XG4gICAgICAgIHJldHVybiB0aGlzO1xuICAgICAgfVxuXG4gICAgICBpZiAoY3VycmVudCA9PT0gMCkge1xuICAgICAgICBOUHJvZ3Jlc3Muc3RhcnQoKTtcbiAgICAgIH1cblxuICAgICAgaW5pdGlhbCsrO1xuICAgICAgY3VycmVudCsrO1xuXG4gICAgICAkcHJvbWlzZS5hbHdheXMoZnVuY3Rpb24oKSB7XG4gICAgICAgIGN1cnJlbnQtLTtcbiAgICAgICAgaWYgKGN1cnJlbnQgPT09IDApIHtcbiAgICAgICAgICAgIGluaXRpYWwgPSAwO1xuICAgICAgICAgICAgTlByb2dyZXNzLmRvbmUoKTtcbiAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAgIE5Qcm9ncmVzcy5zZXQoKGluaXRpYWwgLSBjdXJyZW50KSAvIGluaXRpYWwpO1xuICAgICAgICB9XG4gICAgICB9KTtcblxuICAgICAgcmV0dXJuIHRoaXM7XG4gICAgfTtcblxuICB9KSgpO1xuXG4gIC8qKlxuICAgKiAoSW50ZXJuYWwpIHJlbmRlcnMgdGhlIHByb2dyZXNzIGJhciBtYXJrdXAgYmFzZWQgb24gdGhlIGB0ZW1wbGF0ZWBcbiAgICogc2V0dGluZy5cbiAgICovXG5cbiAgTlByb2dyZXNzLnJlbmRlciA9IGZ1bmN0aW9uKGZyb21TdGFydCkge1xuICAgIGlmIChOUHJvZ3Jlc3MuaXNSZW5kZXJlZCgpKSByZXR1cm4gZG9jdW1lbnQuZ2V0RWxlbWVudEJ5SWQoJ25wcm9ncmVzcycpO1xuXG4gICAgYWRkQ2xhc3MoZG9jdW1lbnQuZG9jdW1lbnRFbGVtZW50LCAnbnByb2dyZXNzLWJ1c3knKTtcbiAgICBcbiAgICB2YXIgcHJvZ3Jlc3MgPSBkb2N1bWVudC5jcmVhdGVFbGVtZW50KCdkaXYnKTtcbiAgICBwcm9ncmVzcy5pZCA9ICducHJvZ3Jlc3MnO1xuICAgIHByb2dyZXNzLmlubmVySFRNTCA9IFNldHRpbmdzLnRlbXBsYXRlO1xuXG4gICAgdmFyIGJhciAgICAgID0gcHJvZ3Jlc3MucXVlcnlTZWxlY3RvcihTZXR0aW5ncy5iYXJTZWxlY3RvciksXG4gICAgICAgIHBlcmMgICAgID0gZnJvbVN0YXJ0ID8gJy0xMDAnIDogdG9CYXJQZXJjKE5Qcm9ncmVzcy5zdGF0dXMgfHwgMCksXG4gICAgICAgIHBhcmVudCAgID0gZG9jdW1lbnQucXVlcnlTZWxlY3RvcihTZXR0aW5ncy5wYXJlbnQpLFxuICAgICAgICBzcGlubmVyO1xuICAgIFxuICAgIGNzcyhiYXIsIHtcbiAgICAgIHRyYW5zaXRpb246ICdhbGwgMCBsaW5lYXInLFxuICAgICAgdHJhbnNmb3JtOiAndHJhbnNsYXRlM2QoJyArIHBlcmMgKyAnJSwwLDApJ1xuICAgIH0pO1xuXG4gICAgaWYgKCFTZXR0aW5ncy5zaG93U3Bpbm5lcikge1xuICAgICAgc3Bpbm5lciA9IHByb2dyZXNzLnF1ZXJ5U2VsZWN0b3IoU2V0dGluZ3Muc3Bpbm5lclNlbGVjdG9yKTtcbiAgICAgIHNwaW5uZXIgJiYgcmVtb3ZlRWxlbWVudChzcGlubmVyKTtcbiAgICB9XG5cbiAgICBpZiAocGFyZW50ICE9IGRvY3VtZW50LmJvZHkpIHtcbiAgICAgIGFkZENsYXNzKHBhcmVudCwgJ25wcm9ncmVzcy1jdXN0b20tcGFyZW50Jyk7XG4gICAgfVxuXG4gICAgcGFyZW50LmFwcGVuZENoaWxkKHByb2dyZXNzKTtcbiAgICByZXR1cm4gcHJvZ3Jlc3M7XG4gIH07XG5cbiAgLyoqXG4gICAqIFJlbW92ZXMgdGhlIGVsZW1lbnQuIE9wcG9zaXRlIG9mIHJlbmRlcigpLlxuICAgKi9cblxuICBOUHJvZ3Jlc3MucmVtb3ZlID0gZnVuY3Rpb24oKSB7XG4gICAgcmVtb3ZlQ2xhc3MoZG9jdW1lbnQuZG9jdW1lbnRFbGVtZW50LCAnbnByb2dyZXNzLWJ1c3knKTtcbiAgICByZW1vdmVDbGFzcyhkb2N1bWVudC5xdWVyeVNlbGVjdG9yKFNldHRpbmdzLnBhcmVudCksICducHJvZ3Jlc3MtY3VzdG9tLXBhcmVudCcpO1xuICAgIHZhciBwcm9ncmVzcyA9IGRvY3VtZW50LmdldEVsZW1lbnRCeUlkKCducHJvZ3Jlc3MnKTtcbiAgICBwcm9ncmVzcyAmJiByZW1vdmVFbGVtZW50KHByb2dyZXNzKTtcbiAgfTtcblxuICAvKipcbiAgICogQ2hlY2tzIGlmIHRoZSBwcm9ncmVzcyBiYXIgaXMgcmVuZGVyZWQuXG4gICAqL1xuXG4gIE5Qcm9ncmVzcy5pc1JlbmRlcmVkID0gZnVuY3Rpb24oKSB7XG4gICAgcmV0dXJuICEhZG9jdW1lbnQuZ2V0RWxlbWVudEJ5SWQoJ25wcm9ncmVzcycpO1xuICB9O1xuXG4gIC8qKlxuICAgKiBEZXRlcm1pbmUgd2hpY2ggcG9zaXRpb25pbmcgQ1NTIHJ1bGUgdG8gdXNlLlxuICAgKi9cblxuICBOUHJvZ3Jlc3MuZ2V0UG9zaXRpb25pbmdDU1MgPSBmdW5jdGlvbigpIHtcbiAgICAvLyBTbmlmZiBvbiBkb2N1bWVudC5ib2R5LnN0eWxlXG4gICAgdmFyIGJvZHlTdHlsZSA9IGRvY3VtZW50LmJvZHkuc3R5bGU7XG5cbiAgICAvLyBTbmlmZiBwcmVmaXhlc1xuICAgIHZhciB2ZW5kb3JQcmVmaXggPSAoJ1dlYmtpdFRyYW5zZm9ybScgaW4gYm9keVN0eWxlKSA/ICdXZWJraXQnIDpcbiAgICAgICAgICAgICAgICAgICAgICAgKCdNb3pUcmFuc2Zvcm0nIGluIGJvZHlTdHlsZSkgPyAnTW96JyA6XG4gICAgICAgICAgICAgICAgICAgICAgICgnbXNUcmFuc2Zvcm0nIGluIGJvZHlTdHlsZSkgPyAnbXMnIDpcbiAgICAgICAgICAgICAgICAgICAgICAgKCdPVHJhbnNmb3JtJyBpbiBib2R5U3R5bGUpID8gJ08nIDogJyc7XG5cbiAgICBpZiAodmVuZG9yUHJlZml4ICsgJ1BlcnNwZWN0aXZlJyBpbiBib2R5U3R5bGUpIHtcbiAgICAgIC8vIE1vZGVybiBicm93c2VycyB3aXRoIDNEIHN1cHBvcnQsIGUuZy4gV2Via2l0LCBJRTEwXG4gICAgICByZXR1cm4gJ3RyYW5zbGF0ZTNkJztcbiAgICB9IGVsc2UgaWYgKHZlbmRvclByZWZpeCArICdUcmFuc2Zvcm0nIGluIGJvZHlTdHlsZSkge1xuICAgICAgLy8gQnJvd3NlcnMgd2l0aG91dCAzRCBzdXBwb3J0LCBlLmcuIElFOVxuICAgICAgcmV0dXJuICd0cmFuc2xhdGUnO1xuICAgIH0gZWxzZSB7XG4gICAgICAvLyBCcm93c2VycyB3aXRob3V0IHRyYW5zbGF0ZSgpIHN1cHBvcnQsIGUuZy4gSUU3LThcbiAgICAgIHJldHVybiAnbWFyZ2luJztcbiAgICB9XG4gIH07XG5cbiAgLyoqXG4gICAqIEhlbHBlcnNcbiAgICovXG5cbiAgZnVuY3Rpb24gY2xhbXAobiwgbWluLCBtYXgpIHtcbiAgICBpZiAobiA8IG1pbikgcmV0dXJuIG1pbjtcbiAgICBpZiAobiA+IG1heCkgcmV0dXJuIG1heDtcbiAgICByZXR1cm4gbjtcbiAgfVxuXG4gIC8qKlxuICAgKiAoSW50ZXJuYWwpIGNvbnZlcnRzIGEgcGVyY2VudGFnZSAoYDAuLjFgKSB0byBhIGJhciB0cmFuc2xhdGVYXG4gICAqIHBlcmNlbnRhZ2UgKGAtMTAwJS4uMCVgKS5cbiAgICovXG5cbiAgZnVuY3Rpb24gdG9CYXJQZXJjKG4pIHtcbiAgICByZXR1cm4gKC0xICsgbikgKiAxMDA7XG4gIH1cblxuXG4gIC8qKlxuICAgKiAoSW50ZXJuYWwpIHJldHVybnMgdGhlIGNvcnJlY3QgQ1NTIGZvciBjaGFuZ2luZyB0aGUgYmFyJ3NcbiAgICogcG9zaXRpb24gZ2l2ZW4gYW4gbiBwZXJjZW50YWdlLCBhbmQgc3BlZWQgYW5kIGVhc2UgZnJvbSBTZXR0aW5nc1xuICAgKi9cblxuICBmdW5jdGlvbiBiYXJQb3NpdGlvbkNTUyhuLCBzcGVlZCwgZWFzZSkge1xuICAgIHZhciBiYXJDU1M7XG5cbiAgICBpZiAoU2V0dGluZ3MucG9zaXRpb25Vc2luZyA9PT0gJ3RyYW5zbGF0ZTNkJykge1xuICAgICAgYmFyQ1NTID0geyB0cmFuc2Zvcm06ICd0cmFuc2xhdGUzZCgnK3RvQmFyUGVyYyhuKSsnJSwwLDApJyB9O1xuICAgIH0gZWxzZSBpZiAoU2V0dGluZ3MucG9zaXRpb25Vc2luZyA9PT0gJ3RyYW5zbGF0ZScpIHtcbiAgICAgIGJhckNTUyA9IHsgdHJhbnNmb3JtOiAndHJhbnNsYXRlKCcrdG9CYXJQZXJjKG4pKyclLDApJyB9O1xuICAgIH0gZWxzZSB7XG4gICAgICBiYXJDU1MgPSB7ICdtYXJnaW4tbGVmdCc6IHRvQmFyUGVyYyhuKSsnJScgfTtcbiAgICB9XG5cbiAgICBiYXJDU1MudHJhbnNpdGlvbiA9ICdhbGwgJytzcGVlZCsnbXMgJytlYXNlO1xuXG4gICAgcmV0dXJuIGJhckNTUztcbiAgfVxuXG4gIC8qKlxuICAgKiAoSW50ZXJuYWwpIFF1ZXVlcyBhIGZ1bmN0aW9uIHRvIGJlIGV4ZWN1dGVkLlxuICAgKi9cblxuICB2YXIgcXVldWUgPSAoZnVuY3Rpb24oKSB7XG4gICAgdmFyIHBlbmRpbmcgPSBbXTtcbiAgICBcbiAgICBmdW5jdGlvbiBuZXh0KCkge1xuICAgICAgdmFyIGZuID0gcGVuZGluZy5zaGlmdCgpO1xuICAgICAgaWYgKGZuKSB7XG4gICAgICAgIGZuKG5leHQpO1xuICAgICAgfVxuICAgIH1cblxuICAgIHJldHVybiBmdW5jdGlvbihmbikge1xuICAgICAgcGVuZGluZy5wdXNoKGZuKTtcbiAgICAgIGlmIChwZW5kaW5nLmxlbmd0aCA9PSAxKSBuZXh0KCk7XG4gICAgfTtcbiAgfSkoKTtcblxuICAvKipcbiAgICogKEludGVybmFsKSBBcHBsaWVzIGNzcyBwcm9wZXJ0aWVzIHRvIGFuIGVsZW1lbnQsIHNpbWlsYXIgdG8gdGhlIGpRdWVyeSBcbiAgICogY3NzIG1ldGhvZC5cbiAgICpcbiAgICogV2hpbGUgdGhpcyBoZWxwZXIgZG9lcyBhc3Npc3Qgd2l0aCB2ZW5kb3IgcHJlZml4ZWQgcHJvcGVydHkgbmFtZXMsIGl0IFxuICAgKiBkb2VzIG5vdCBwZXJmb3JtIGFueSBtYW5pcHVsYXRpb24gb2YgdmFsdWVzIHByaW9yIHRvIHNldHRpbmcgc3R5bGVzLlxuICAgKi9cblxuICB2YXIgY3NzID0gKGZ1bmN0aW9uKCkge1xuICAgIHZhciBjc3NQcmVmaXhlcyA9IFsgJ1dlYmtpdCcsICdPJywgJ01veicsICdtcycgXSxcbiAgICAgICAgY3NzUHJvcHMgICAgPSB7fTtcblxuICAgIGZ1bmN0aW9uIGNhbWVsQ2FzZShzdHJpbmcpIHtcbiAgICAgIHJldHVybiBzdHJpbmcucmVwbGFjZSgvXi1tcy0vLCAnbXMtJykucmVwbGFjZSgvLShbXFxkYS16XSkvZ2ksIGZ1bmN0aW9uKG1hdGNoLCBsZXR0ZXIpIHtcbiAgICAgICAgcmV0dXJuIGxldHRlci50b1VwcGVyQ2FzZSgpO1xuICAgICAgfSk7XG4gICAgfVxuXG4gICAgZnVuY3Rpb24gZ2V0VmVuZG9yUHJvcChuYW1lKSB7XG4gICAgICB2YXIgc3R5bGUgPSBkb2N1bWVudC5ib2R5LnN0eWxlO1xuICAgICAgaWYgKG5hbWUgaW4gc3R5bGUpIHJldHVybiBuYW1lO1xuXG4gICAgICB2YXIgaSA9IGNzc1ByZWZpeGVzLmxlbmd0aCxcbiAgICAgICAgICBjYXBOYW1lID0gbmFtZS5jaGFyQXQoMCkudG9VcHBlckNhc2UoKSArIG5hbWUuc2xpY2UoMSksXG4gICAgICAgICAgdmVuZG9yTmFtZTtcbiAgICAgIHdoaWxlIChpLS0pIHtcbiAgICAgICAgdmVuZG9yTmFtZSA9IGNzc1ByZWZpeGVzW2ldICsgY2FwTmFtZTtcbiAgICAgICAgaWYgKHZlbmRvck5hbWUgaW4gc3R5bGUpIHJldHVybiB2ZW5kb3JOYW1lO1xuICAgICAgfVxuXG4gICAgICByZXR1cm4gbmFtZTtcbiAgICB9XG5cbiAgICBmdW5jdGlvbiBnZXRTdHlsZVByb3AobmFtZSkge1xuICAgICAgbmFtZSA9IGNhbWVsQ2FzZShuYW1lKTtcbiAgICAgIHJldHVybiBjc3NQcm9wc1tuYW1lXSB8fCAoY3NzUHJvcHNbbmFtZV0gPSBnZXRWZW5kb3JQcm9wKG5hbWUpKTtcbiAgICB9XG5cbiAgICBmdW5jdGlvbiBhcHBseUNzcyhlbGVtZW50LCBwcm9wLCB2YWx1ZSkge1xuICAgICAgcHJvcCA9IGdldFN0eWxlUHJvcChwcm9wKTtcbiAgICAgIGVsZW1lbnQuc3R5bGVbcHJvcF0gPSB2YWx1ZTtcbiAgICB9XG5cbiAgICByZXR1cm4gZnVuY3Rpb24oZWxlbWVudCwgcHJvcGVydGllcykge1xuICAgICAgdmFyIGFyZ3MgPSBhcmd1bWVudHMsXG4gICAgICAgICAgcHJvcCwgXG4gICAgICAgICAgdmFsdWU7XG5cbiAgICAgIGlmIChhcmdzLmxlbmd0aCA9PSAyKSB7XG4gICAgICAgIGZvciAocHJvcCBpbiBwcm9wZXJ0aWVzKSB7XG4gICAgICAgICAgdmFsdWUgPSBwcm9wZXJ0aWVzW3Byb3BdO1xuICAgICAgICAgIGlmICh2YWx1ZSAhPT0gdW5kZWZpbmVkICYmIHByb3BlcnRpZXMuaGFzT3duUHJvcGVydHkocHJvcCkpIGFwcGx5Q3NzKGVsZW1lbnQsIHByb3AsIHZhbHVlKTtcbiAgICAgICAgfVxuICAgICAgfSBlbHNlIHtcbiAgICAgICAgYXBwbHlDc3MoZWxlbWVudCwgYXJnc1sxXSwgYXJnc1syXSk7XG4gICAgICB9XG4gICAgfVxuICB9KSgpO1xuXG4gIC8qKlxuICAgKiAoSW50ZXJuYWwpIERldGVybWluZXMgaWYgYW4gZWxlbWVudCBvciBzcGFjZSBzZXBhcmF0ZWQgbGlzdCBvZiBjbGFzcyBuYW1lcyBjb250YWlucyBhIGNsYXNzIG5hbWUuXG4gICAqL1xuXG4gIGZ1bmN0aW9uIGhhc0NsYXNzKGVsZW1lbnQsIG5hbWUpIHtcbiAgICB2YXIgbGlzdCA9IHR5cGVvZiBlbGVtZW50ID09ICdzdHJpbmcnID8gZWxlbWVudCA6IGNsYXNzTGlzdChlbGVtZW50KTtcbiAgICByZXR1cm4gbGlzdC5pbmRleE9mKCcgJyArIG5hbWUgKyAnICcpID49IDA7XG4gIH1cblxuICAvKipcbiAgICogKEludGVybmFsKSBBZGRzIGEgY2xhc3MgdG8gYW4gZWxlbWVudC5cbiAgICovXG5cbiAgZnVuY3Rpb24gYWRkQ2xhc3MoZWxlbWVudCwgbmFtZSkge1xuICAgIHZhciBvbGRMaXN0ID0gY2xhc3NMaXN0KGVsZW1lbnQpLFxuICAgICAgICBuZXdMaXN0ID0gb2xkTGlzdCArIG5hbWU7XG5cbiAgICBpZiAoaGFzQ2xhc3Mob2xkTGlzdCwgbmFtZSkpIHJldHVybjsgXG5cbiAgICAvLyBUcmltIHRoZSBvcGVuaW5nIHNwYWNlLlxuICAgIGVsZW1lbnQuY2xhc3NOYW1lID0gbmV3TGlzdC5zdWJzdHJpbmcoMSk7XG4gIH1cblxuICAvKipcbiAgICogKEludGVybmFsKSBSZW1vdmVzIGEgY2xhc3MgZnJvbSBhbiBlbGVtZW50LlxuICAgKi9cblxuICBmdW5jdGlvbiByZW1vdmVDbGFzcyhlbGVtZW50LCBuYW1lKSB7XG4gICAgdmFyIG9sZExpc3QgPSBjbGFzc0xpc3QoZWxlbWVudCksXG4gICAgICAgIG5ld0xpc3Q7XG5cbiAgICBpZiAoIWhhc0NsYXNzKGVsZW1lbnQsIG5hbWUpKSByZXR1cm47XG5cbiAgICAvLyBSZXBsYWNlIHRoZSBjbGFzcyBuYW1lLlxuICAgIG5ld0xpc3QgPSBvbGRMaXN0LnJlcGxhY2UoJyAnICsgbmFtZSArICcgJywgJyAnKTtcblxuICAgIC8vIFRyaW0gdGhlIG9wZW5pbmcgYW5kIGNsb3Npbmcgc3BhY2VzLlxuICAgIGVsZW1lbnQuY2xhc3NOYW1lID0gbmV3TGlzdC5zdWJzdHJpbmcoMSwgbmV3TGlzdC5sZW5ndGggLSAxKTtcbiAgfVxuXG4gIC8qKlxuICAgKiAoSW50ZXJuYWwpIEdldHMgYSBzcGFjZSBzZXBhcmF0ZWQgbGlzdCBvZiB0aGUgY2xhc3MgbmFtZXMgb24gdGhlIGVsZW1lbnQuIFxuICAgKiBUaGUgbGlzdCBpcyB3cmFwcGVkIHdpdGggYSBzaW5nbGUgc3BhY2Ugb24gZWFjaCBlbmQgdG8gZmFjaWxpdGF0ZSBmaW5kaW5nIFxuICAgKiBtYXRjaGVzIHdpdGhpbiB0aGUgbGlzdC5cbiAgICovXG5cbiAgZnVuY3Rpb24gY2xhc3NMaXN0KGVsZW1lbnQpIHtcbiAgICByZXR1cm4gKCcgJyArIChlbGVtZW50LmNsYXNzTmFtZSB8fCAnJykgKyAnICcpLnJlcGxhY2UoL1xccysvZ2ksICcgJyk7XG4gIH1cblxuICAvKipcbiAgICogKEludGVybmFsKSBSZW1vdmVzIGFuIGVsZW1lbnQgZnJvbSB0aGUgRE9NLlxuICAgKi9cblxuICBmdW5jdGlvbiByZW1vdmVFbGVtZW50KGVsZW1lbnQpIHtcbiAgICBlbGVtZW50ICYmIGVsZW1lbnQucGFyZW50Tm9kZSAmJiBlbGVtZW50LnBhcmVudE5vZGUucmVtb3ZlQ2hpbGQoZWxlbWVudCk7XG4gIH1cblxuICByZXR1cm4gTlByb2dyZXNzO1xufSk7XG5cbiJdLCJtYXBwaW5ncyI6IkFBQUE7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUNBLFdBSUE7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBOyIsInNvdXJjZVJvb3QiOiIifQ==\n//# sourceURL=webpack-internal:///./node_modules/nprogress/nprogress.js\n"); + + /***/ }), + + /***/ 0: + /*!*************************!*\ + !*** multi ./js/app.js ***! + \*************************/ + /*! no static exports found */ + /***/ (function(module, exports, __webpack_require__) { + + module.exports = __webpack_require__(/*! ./js/app.js */"./js/app.js"); + + + /***/ }) + + /******/ }); \ No newline at end of file diff --git a/priv/static/robots.txt b/priv/static/robots.txt new file mode 100644 index 0000000..3c9c7c0 --- /dev/null +++ b/priv/static/robots.txt @@ -0,0 +1,5 @@ +# See http://www.robotstxt.org/robotstxt.html for documentation on how to use the robots.txt file +# +# To ban all spiders from the entire site uncomment the next two lines: +# User-agent: * +# Disallow: / diff --git a/test/fridge_tracker_web/live/page_live_test.exs b/test/fridge_tracker_web/live/page_live_test.exs new file mode 100644 index 0000000..dda45b4 --- /dev/null +++ b/test/fridge_tracker_web/live/page_live_test.exs @@ -0,0 +1,11 @@ +defmodule TrackerAppWeb.PageLiveTest do + use TrackerAppWeb.ConnCase + + import Phoenix.LiveViewTest + + test "disconnected and connected render", %{conn: conn} do + {:ok, page_live, disconnected_html} = live(conn, "/") + assert disconnected_html =~ "Welcome to Phoenix!" + assert render(page_live) =~ "Welcome to Phoenix!" + end +end diff --git a/test/fridge_tracker_web/views/error_view_test.exs b/test/fridge_tracker_web/views/error_view_test.exs new file mode 100644 index 0000000..723bc06 --- /dev/null +++ b/test/fridge_tracker_web/views/error_view_test.exs @@ -0,0 +1,14 @@ +defmodule TrackerAppWeb.ErrorViewTest do + use TrackerAppWeb.ConnCase, async: true + + # Bring render/3 and render_to_string/3 for testing custom views + import Phoenix.View + + test "renders 404.html" do + assert render_to_string(TrackerAppWeb.ErrorView, "404.html", []) == "Not Found" + end + + test "renders 500.html" do + assert render_to_string(TrackerAppWeb.ErrorView, "500.html", []) == "Internal Server Error" + end +end diff --git a/test/fridge_tracker_web/views/layout_view_test.exs b/test/fridge_tracker_web/views/layout_view_test.exs new file mode 100644 index 0000000..84389c2 --- /dev/null +++ b/test/fridge_tracker_web/views/layout_view_test.exs @@ -0,0 +1,8 @@ +defmodule TrackerAppWeb.LayoutViewTest do + use TrackerAppWeb.ConnCase, async: true + + # When testing helpers, you may want to import Phoenix.HTML and + # use functions such as safe_to_string() to convert the helper + # result into an HTML string. + # import Phoenix.HTML +end diff --git a/test/support/channel_case.ex b/test/support/channel_case.ex new file mode 100644 index 0000000..37b2a6a --- /dev/null +++ b/test/support/channel_case.ex @@ -0,0 +1,34 @@ +defmodule TrackerAppWeb.ChannelCase do + @moduledoc """ + This module defines the test case to be used by + channel tests. + + Such tests rely on `Phoenix.ChannelTest` and also + import other functionality to make it easier + to build common data structures and query the data layer. + + Finally, if the test case interacts with the database, + we enable the SQL sandbox, so changes done to the database + are reverted at the end of every test. If you are using + PostgreSQL, you can even run database tests asynchronously + by setting `use TrackerAppWeb.ChannelCase, async: true`, although + this option is not recommended for other databases. + """ + + use ExUnit.CaseTemplate + + using do + quote do + # Import conveniences for testing with channels + import Phoenix.ChannelTest + import TrackerAppWeb.ChannelCase + + # The default endpoint for testing + @endpoint TrackerAppWeb.Endpoint + end + end + + setup _tags do + :ok + end +end diff --git a/test/support/conn_case.ex b/test/support/conn_case.ex new file mode 100644 index 0000000..6c511bb --- /dev/null +++ b/test/support/conn_case.ex @@ -0,0 +1,37 @@ +defmodule TrackerAppWeb.ConnCase do + @moduledoc """ + This module defines the test case to be used by + tests that require setting up a connection. + + Such tests rely on `Phoenix.ConnTest` and also + import other functionality to make it easier + to build common data structures and query the data layer. + + Finally, if the test case interacts with the database, + we enable the SQL sandbox, so changes done to the database + are reverted at the end of every test. If you are using + PostgreSQL, you can even run database tests asynchronously + by setting `use TrackerAppWeb.ConnCase, async: true`, although + this option is not recommended for other databases. + """ + + use ExUnit.CaseTemplate + + using do + quote do + # Import conveniences for testing with connections + import Plug.Conn + import Phoenix.ConnTest + import TrackerAppWeb.ConnCase + + alias TrackerAppWeb.Router.Helpers, as: Routes + + # The default endpoint for testing + @endpoint TrackerAppWeb.Endpoint + end + end + + setup _tags do + {:ok, conn: Phoenix.ConnTest.build_conn()} + end +end diff --git a/test/test_helper.exs b/test/test_helper.exs new file mode 100644 index 0000000..869559e --- /dev/null +++ b/test/test_helper.exs @@ -0,0 +1 @@ +ExUnit.start()