Skip to content

Commit

Permalink
Add test app
Browse files Browse the repository at this point in the history
  • Loading branch information
Serabe committed Oct 6, 2023
1 parent 72150a8 commit 644403a
Show file tree
Hide file tree
Showing 40 changed files with 3,047 additions and 4 deletions.
38 changes: 34 additions & 4 deletions lib/live_view_events/send.ex → lib/live_view_events/notify.ex
Original file line number Diff line number Diff line change
Expand Up @@ -5,6 +5,31 @@ defmodule LiveViewEvents.Notify do

@doc """
Use this macro instead of the default `assigns(socket, assign)` in
`c:Phoenix.LiveComponent.update/2`.
This will detect if `c:Phoenix.LiveComponent.update/2` is being called
because of an event send with either `notify_to/2` or `notify_to/3`
and handle it with `c:Phoenix.LiveView.handle_info/2`. Otherwise,
it will assign the given `assigns` to `socket`.
If there is no `c:Phoenix.LiveView.handle_info/2` defined in the
component, sending an event won't raise an error but if there is one
defined and it cannot handle the received message, it will.
Furthermore, if the handler returns anything that is not a tuple
`{:noreply, socket}`, it'll raise an exception too.
## Why using `c:Phoenix.LiveView.handle_info/2` in components?
In one word: consistency. Messages coming from the client are
handled by `c:Phoenix.LiveView.handle_event/3` or by
`c:Phoenix.LiveComponent.handle_event/3`. For messages sent from
the server are currently being handled by `c:Phoenix.LiveView.handle_info/2`
in live views, with not official way to do this but the hack this
library is based on.
The hack is basically send an update with `Phoenix.LiveView.send_update/3`
and handle it in `c:Phoenix.LiveComponent.update/2`
"""
defmacro handle_info_or_assign(socket, assigns) do
quote do
Expand Down Expand Up @@ -47,8 +72,9 @@ defmodule LiveViewEvents.Notify do
def notify_to(:self, message), do: notify_to(self(), message)
def notify_to(pid, message) when is_pid(pid), do: send(pid, message)

def notify_to({module, id}, message) do
Phoenix.LiveView.send_update(self(), module, %{:id => id, @assign_name_for_event => message})
def notify_to(target, message) when is_tuple(target) do
{pid, module, id} = process_tuple(target)
Phoenix.LiveView.send_update(pid, module, %{:id => id, @assign_name_for_event => message})
end

@doc """
Expand All @@ -59,10 +85,14 @@ defmodule LiveViewEvents.Notify do
def notify_to(:self, message, params), do: notify_to(self(), message, params)
def notify_to(pid, message, params) when is_pid(pid), do: send(pid, {message, params})

def notify_to({module, id}, message, params) do
Phoenix.LiveView.send_update(self(), module, %{
def notify_to(target, message, params) when is_tuple(target) do
{pid, module, id}=process_tuple(target)
Phoenix.LiveView.send_update(pid, module, %{
:id => id,
@assign_name_for_event => {message, params}
})
end

defp process_tuple({module, id}), do: {self(), module, id}
defp process_tuple({pid, _module, _id}=target) when is_pid(pid), do: target
end
5 changes: 5 additions & 0 deletions test_app/.formatter.exs
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
[
import_deps: [:phoenix],
plugins: [Phoenix.LiveView.HTMLFormatter],
inputs: ["*.{heex,ex,exs}", "{config,lib,test}/**/*.{heex,ex,exs}"]
]
27 changes: 27 additions & 0 deletions test_app/.gitignore
Original file line number Diff line number Diff line change
@@ -0,0 +1,27 @@
# 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

# Temporary files, for example, from tests.
/tmp/

# Ignore package tarball (built via "mix hex.build").
test_app-*.tar

18 changes: 18 additions & 0 deletions test_app/README.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,18 @@
# TestApp

To start your Phoenix server:

* Run `mix setup` to install and setup dependencies
* Start Phoenix endpoint with `mix phx.server` or inside IEx with `iex -S mix phx.server`

Now you can visit [`localhost:4000`](http://localhost:4000) 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
30 changes: 30 additions & 0 deletions test_app/config/config.exs
Original file line number Diff line number Diff line change
@@ -0,0 +1,30 @@
# This file is responsible for configuring your application
# and its dependencies with the aid of the Config module.
#
# This configuration file is loaded before any dependency and
# is restricted to this project.

# General application configuration
import Config

# Configures the endpoint
config :test_app, TestAppWeb.Endpoint,
url: [host: "localhost"],
render_errors: [
formats: [html: TestAppWeb.ErrorHTML, json: TestAppWeb.ErrorJSON],
layout: false
],
pubsub_server: TestApp.PubSub,
live_view: [signing_salt: "Yv51WWUs"]

# 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 "#{config_env()}.exs"
62 changes: 62 additions & 0 deletions test_app/config/dev.exs
Original file line number Diff line number Diff line change
@@ -0,0 +1,62 @@
import 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 can use it
# to bundle .js and .css sources.
config :test_app, TestAppWeb.Endpoint,
# Binding to loopback ipv4 address prevents access from other machines.
# Change to `ip: {0, 0, 0, 0}` to allow access from other machines.
http: [ip: {127, 0, 0, 1}, port: 4000],
check_origin: false,
code_reloader: true,
debug_errors: true,
secret_key_base: "/OOtixijCMkathudAQ6iWMDxAguNzdKOUBFOLZo/2h9IjzdYcX63RdVs4uvmdB3f",
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
#
# 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 :test_app, TestAppWeb.Endpoint,
live_reload: [
patterns: [
~r"priv/static/.*(js|css|png|jpeg|jpg|gif|svg)$",
~r"lib/test_app_web/(controllers|live|components)/.*(ex|heex)$"
]
]

# Enable dev routes for dashboard and mailbox
config :test_app, dev_routes: true

# 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
7 changes: 7 additions & 0 deletions test_app/config/prod.exs
Original file line number Diff line number Diff line change
@@ -0,0 +1,7 @@
import Config

# Do not print debug messages in production
config :logger, level: :info

# Runtime production configuration, including reading
# of environment variables, is done on config/runtime.exs.
82 changes: 82 additions & 0 deletions test_app/config/runtime.exs
Original file line number Diff line number Diff line change
@@ -0,0 +1,82 @@
import Config

# config/runtime.exs is executed for all environments, including
# during releases. It is executed after compilation and before the
# system starts, so it is typically used to load production configuration
# and secrets from environment variables or elsewhere. Do not define
# any compile-time configuration in here, as it won't be applied.
# The block below contains prod specific runtime configuration.

# ## Using releases
#
# If you use `mix release`, you need to explicitly enable the server
# by passing the PHX_SERVER=true when you start it:
#
# PHX_SERVER=true bin/test_app start
#
# Alternatively, you can use `mix phx.gen.release` to generate a `bin/server`
# script that automatically sets the env var above.
if System.get_env("PHX_SERVER") do
config :test_app, TestAppWeb.Endpoint, server: true
end

if config_env() == :prod do
# The secret key base is used to sign/encrypt cookies and other secrets.
# A default value is used in config/dev.exs and config/test.exs but you
# want to use a different value for prod and you most likely don't want
# to check this value into version control, so we use an environment
# variable instead.
secret_key_base =
System.get_env("SECRET_KEY_BASE") ||
raise """
environment variable SECRET_KEY_BASE is missing.
You can generate one by calling: mix phx.gen.secret
"""

host = System.get_env("PHX_HOST") || "example.com"
port = String.to_integer(System.get_env("PORT") || "4000")

config :test_app, TestAppWeb.Endpoint,
url: [host: host, port: 443, scheme: "https"],
http: [
# Enable IPv6 and bind on all interfaces.
# Set it to {0, 0, 0, 0, 0, 0, 0, 1} for local network only access.
# See the documentation on https://hexdocs.pm/plug_cowboy/Plug.Cowboy.html
# for details about using IPv6 vs IPv4 and loopback vs public addresses.
ip: {0, 0, 0, 0, 0, 0, 0, 0},
port: port
],
secret_key_base: secret_key_base

# ## SSL Support
#
# To get SSL working, you will need to add the `https` key
# to your endpoint configuration:
#
# config :test_app, TestAppWeb.Endpoint,
# https: [
# ...,
# port: 443,
# cipher_suite: :strong,
# keyfile: System.get_env("SOME_APP_SSL_KEY_PATH"),
# certfile: System.get_env("SOME_APP_SSL_CERT_PATH")
# ]
#
# 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 :test_app, TestAppWeb.Endpoint,
# force_ssl: [hsts: true]
#
# Check `Plug.SSL` for all available options in `force_ssl`.
end
14 changes: 14 additions & 0 deletions test_app/config/test.exs
Original file line number Diff line number Diff line change
@@ -0,0 +1,14 @@
import Config

# We don't run a server during test. If one is required,
# you can enable the server option below.
config :test_app, TestAppWeb.Endpoint,
http: [ip: {127, 0, 0, 1}, port: 4002],
secret_key_base: "EdKwrnAZO/we479ohsBsJbnpaK+0LV2SONLY83a+FNvycfw2Rfp7C7WtVWSbrAZS",
server: false

# Print only warnings and errors during test
config :logger, level: :warning

# Initialize plugs at runtime for faster test compilation
config :phoenix, :plug_init_mode, :runtime
9 changes: 9 additions & 0 deletions test_app/lib/test_app.ex
Original file line number Diff line number Diff line change
@@ -0,0 +1,9 @@
defmodule TestApp do
@moduledoc """
TestApp keeps the contexts that define your domain
and business logic.
Contexts are also responsible for managing your data, regardless
if it comes from the database, an external API or others.
"""
end
34 changes: 34 additions & 0 deletions test_app/lib/test_app/application.ex
Original file line number Diff line number Diff line change
@@ -0,0 +1,34 @@
defmodule TestApp.Application do
# See https://hexdocs.pm/elixir/Application.html
# for more information on OTP Applications
@moduledoc false

use Application

@impl true
def start(_type, _args) do
children = [
# Start the Telemetry supervisor
TestAppWeb.Telemetry,
# Start the PubSub system
{Phoenix.PubSub, name: TestApp.PubSub},
# Start the Endpoint (http/https)
TestAppWeb.Endpoint
# Start a worker by calling: TestApp.Worker.start_link(arg)
# {TestApp.Worker, arg}
]

# See https://hexdocs.pm/elixir/Supervisor.html
# for other strategies and supported options
opts = [strategy: :one_for_one, name: TestApp.Supervisor]
Supervisor.start_link(children, opts)
end

# Tell Phoenix to update the endpoint configuration
# whenever the application is updated.
@impl true
def config_change(changed, _new, removed) do
TestAppWeb.Endpoint.config_change(changed, removed)
:ok
end
end
Loading

0 comments on commit 644403a

Please sign in to comment.