mirror of
https://github.com/fergalmoran/malarkey.git
synced 2025-12-22 09:48:46 +00:00
Out with the old
This commit is contained in:
@@ -1,47 +0,0 @@
|
|||||||
# This file excludes paths from the Docker build context.
|
|
||||||
#
|
|
||||||
# By default, Docker's build context includes all files (and folders) in the
|
|
||||||
# current directory. Even if a file isn't copied into the container it is still sent to
|
|
||||||
# the Docker daemon.
|
|
||||||
#
|
|
||||||
# There are multiple reasons to exclude files from the build context:
|
|
||||||
#
|
|
||||||
# 1. Prevent nested folders from being copied into the container (ex: exclude
|
|
||||||
# /assets/node_modules when copying /assets)
|
|
||||||
# 2. Reduce the size of the build context and improve build time (ex. /build, /deps, /doc)
|
|
||||||
# 3. Avoid sending files containing sensitive information
|
|
||||||
#
|
|
||||||
# More information on using .dockerignore is available here:
|
|
||||||
# https://docs.docker.com/engine/reference/builder/#dockerignore-file
|
|
||||||
|
|
||||||
.dockerignore
|
|
||||||
|
|
||||||
# Ignore git, but keep git HEAD and refs to access current commit hash if needed:
|
|
||||||
#
|
|
||||||
# $ cat .git/HEAD | awk '{print ".git/"$2}' | xargs cat
|
|
||||||
# d0b8727759e1e0e7aa3d41707d12376e373d5ecc
|
|
||||||
.git
|
|
||||||
!.git/HEAD
|
|
||||||
!.git/refs
|
|
||||||
|
|
||||||
# Common development/test artifacts
|
|
||||||
/cover/
|
|
||||||
/doc/
|
|
||||||
/test/
|
|
||||||
/tmp/
|
|
||||||
.elixir_ls
|
|
||||||
|
|
||||||
# Mix artifacts
|
|
||||||
/_build/
|
|
||||||
/deps/
|
|
||||||
*.ez
|
|
||||||
|
|
||||||
# Generated on crash by the VM
|
|
||||||
erl_crash.dump
|
|
||||||
|
|
||||||
# Static artifacts - These should be fetched and built inside the Docker image
|
|
||||||
/assets/node_modules/
|
|
||||||
/priv/static/assets/
|
|
||||||
/priv/static/cache_manifest.json
|
|
||||||
|
|
||||||
.prv/
|
|
||||||
@@ -1,6 +0,0 @@
|
|||||||
[
|
|
||||||
import_deps: [:ecto, :ecto_sql, :phoenix],
|
|
||||||
subdirectories: ["priv/*/migrations"],
|
|
||||||
plugins: [Phoenix.LiveView.HTMLFormatter],
|
|
||||||
inputs: ["*.{heex,ex,exs}", "{config,lib,test}/**/*.{heex,ex,exs}", "priv/*/seeds.exs"]
|
|
||||||
]
|
|
||||||
15
.github/workflows/fly.yml
vendored
15
.github/workflows/fly.yml
vendored
@@ -1,15 +0,0 @@
|
|||||||
name: Fly Deploy
|
|
||||||
on:
|
|
||||||
push:
|
|
||||||
branches:
|
|
||||||
- develop
|
|
||||||
env:
|
|
||||||
FLY_API_TOKEN: ${{ secrets.FLY_API_TOKEN }}
|
|
||||||
jobs:
|
|
||||||
deploy:
|
|
||||||
name: Deploy app
|
|
||||||
runs-on: ubuntu-latest
|
|
||||||
steps:
|
|
||||||
- uses: actions/checkout@v3
|
|
||||||
- uses: superfly/flyctl-actions/setup-flyctl@master
|
|
||||||
- run: flyctl deploy --remote-only
|
|
||||||
36
.gitignore
vendored
36
.gitignore
vendored
@@ -1,36 +0,0 @@
|
|||||||
# 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").
|
|
||||||
malarkey-*.tar
|
|
||||||
|
|
||||||
# Ignore assets that are produced by build tools.
|
|
||||||
/priv/static/assets/
|
|
||||||
|
|
||||||
# Ignore digested assets cache.
|
|
||||||
/priv/static/cache_manifest.json
|
|
||||||
|
|
||||||
# In case you use Node.js/npm, you want to ignore these.
|
|
||||||
npm-debug.log
|
|
||||||
/assets/node_modules/
|
|
||||||
|
|
||||||
|
|
||||||
.prv/
|
|
||||||
5
.idea/.gitignore
generated
vendored
5
.idea/.gitignore
generated
vendored
@@ -1,5 +0,0 @@
|
|||||||
# Default ignored files
|
|
||||||
/shelf/
|
|
||||||
/workspace.xml
|
|
||||||
# Editor-based HTTP Client requests
|
|
||||||
/httpRequests/
|
|
||||||
12
.idea/malarkey.iml
generated
12
.idea/malarkey.iml
generated
@@ -1,12 +0,0 @@
|
|||||||
<?xml version="1.0" encoding="UTF-8"?>
|
|
||||||
<module type="WEB_MODULE" version="4">
|
|
||||||
<component name="NewModuleRootManager">
|
|
||||||
<content url="file://$MODULE_DIR$">
|
|
||||||
<excludeFolder url="file://$MODULE_DIR$/temp" />
|
|
||||||
<excludeFolder url="file://$MODULE_DIR$/.tmp" />
|
|
||||||
<excludeFolder url="file://$MODULE_DIR$/tmp" />
|
|
||||||
</content>
|
|
||||||
<orderEntry type="inheritedJdk" />
|
|
||||||
<orderEntry type="sourceFolder" forTests="false" />
|
|
||||||
</component>
|
|
||||||
</module>
|
|
||||||
8
.idea/modules.xml
generated
8
.idea/modules.xml
generated
@@ -1,8 +0,0 @@
|
|||||||
<?xml version="1.0" encoding="UTF-8"?>
|
|
||||||
<project version="4">
|
|
||||||
<component name="ProjectModuleManager">
|
|
||||||
<modules>
|
|
||||||
<module fileurl="file://$PROJECT_DIR$/.idea/malarkey.iml" filepath="$PROJECT_DIR$/.idea/malarkey.iml" />
|
|
||||||
</modules>
|
|
||||||
</component>
|
|
||||||
</project>
|
|
||||||
7
.idea/vcs.xml
generated
7
.idea/vcs.xml
generated
@@ -1,7 +0,0 @@
|
|||||||
<?xml version="1.0" encoding="UTF-8"?>
|
|
||||||
<project version="4">
|
|
||||||
<component name="VcsDirectoryMappings">
|
|
||||||
<mapping directory="" vcs="Git" />
|
|
||||||
<mapping directory="$PROJECT_DIR$/deps/pow" vcs="Git" />
|
|
||||||
</component>
|
|
||||||
</project>
|
|
||||||
22
.vscode/settings.json
vendored
22
.vscode/settings.json
vendored
@@ -1,22 +0,0 @@
|
|||||||
{
|
|
||||||
"workbench.colorCustomizations": {
|
|
||||||
"activityBar.activeBackground": "#e76105",
|
|
||||||
"activityBar.background": "#e76105",
|
|
||||||
"activityBar.foreground": "#e7e7e7",
|
|
||||||
"activityBar.inactiveForeground": "#e7e7e799",
|
|
||||||
"activityBarBadge.background": "#025e27",
|
|
||||||
"activityBarBadge.foreground": "#e7e7e7",
|
|
||||||
"commandCenter.border": "#e7e7e799",
|
|
||||||
"sash.hoverBorder": "#e76105",
|
|
||||||
"statusBar.background": "#b54c04",
|
|
||||||
"statusBar.foreground": "#e7e7e7",
|
|
||||||
"statusBarItem.hoverBackground": "#e76105",
|
|
||||||
"statusBarItem.remoteBackground": "#b54c04",
|
|
||||||
"statusBarItem.remoteForeground": "#e7e7e7",
|
|
||||||
"titleBar.activeBackground": "#b54c04",
|
|
||||||
"titleBar.activeForeground": "#e7e7e7",
|
|
||||||
"titleBar.inactiveBackground": "#b54c0499",
|
|
||||||
"titleBar.inactiveForeground": "#e7e7e799"
|
|
||||||
},
|
|
||||||
"peacock.color": "#b54c04"
|
|
||||||
}
|
|
||||||
99
Dockerfile
99
Dockerfile
@@ -1,99 +0,0 @@
|
|||||||
# Find eligible builder and runner images on Docker Hub. We use Ubuntu/Debian
|
|
||||||
# instead of Alpine to avoid DNS resolution issues in production.
|
|
||||||
#
|
|
||||||
# https://hub.docker.com/r/hexpm/elixir/tags?page=1&name=ubuntu
|
|
||||||
# https://hub.docker.com/_/ubuntu?tab=tags
|
|
||||||
#
|
|
||||||
# This file is based on these images:
|
|
||||||
#
|
|
||||||
# - https://hub.docker.com/r/hexpm/elixir/tags - for the build image
|
|
||||||
# - https://hub.docker.com/_/debian?tab=tags&page=1&name=bullseye-20221004-slim - for the release image
|
|
||||||
# - https://pkgs.org/ - resource for finding needed packages
|
|
||||||
# - Ex: hexpm/elixir:1.14.0-erlang-25.1.2-debian-bullseye-20221004-slim
|
|
||||||
#
|
|
||||||
ARG ELIXIR_VERSION=1.14.0
|
|
||||||
ARG OTP_VERSION=25.1.2
|
|
||||||
ARG DEBIAN_VERSION=bullseye-20221004-slim
|
|
||||||
|
|
||||||
ARG BUILDER_IMAGE="hexpm/elixir:${ELIXIR_VERSION}-erlang-${OTP_VERSION}-debian-${DEBIAN_VERSION}"
|
|
||||||
ARG RUNNER_IMAGE="debian:${DEBIAN_VERSION}"
|
|
||||||
|
|
||||||
FROM ${BUILDER_IMAGE} as builder
|
|
||||||
|
|
||||||
# install build dependencies
|
|
||||||
RUN apt-get update -y && apt-get install -y build-essential git \
|
|
||||||
&& apt-get clean && rm -f /var/lib/apt/lists/*_*
|
|
||||||
|
|
||||||
# prepare build dir
|
|
||||||
WORKDIR /app
|
|
||||||
|
|
||||||
# install hex + rebar
|
|
||||||
RUN mix local.hex --force && \
|
|
||||||
mix local.rebar --force
|
|
||||||
|
|
||||||
# set build ENV
|
|
||||||
ENV MIX_ENV="prod"
|
|
||||||
|
|
||||||
# install mix dependencies
|
|
||||||
COPY mix.exs mix.lock ./
|
|
||||||
RUN mix deps.get --only $MIX_ENV
|
|
||||||
RUN mkdir config
|
|
||||||
|
|
||||||
# copy compile-time config files before we compile dependencies
|
|
||||||
# to ensure any relevant config change will trigger the dependencies
|
|
||||||
# to be re-compiled.
|
|
||||||
COPY config/config.exs config/${MIX_ENV}.exs config/
|
|
||||||
RUN mix deps.compile
|
|
||||||
|
|
||||||
COPY priv priv
|
|
||||||
|
|
||||||
COPY lib lib
|
|
||||||
|
|
||||||
COPY assets assets
|
|
||||||
|
|
||||||
# compile assets
|
|
||||||
RUN mix assets.deploy
|
|
||||||
|
|
||||||
# Compile the release
|
|
||||||
RUN mix compile
|
|
||||||
|
|
||||||
# Changes to config/runtime.exs don't require recompiling the code
|
|
||||||
COPY config/runtime.exs config/
|
|
||||||
|
|
||||||
COPY rel rel
|
|
||||||
RUN mix release
|
|
||||||
|
|
||||||
# start a new build stage so that the final image will only contain
|
|
||||||
# the compiled release and other runtime necessities
|
|
||||||
FROM ${RUNNER_IMAGE}
|
|
||||||
|
|
||||||
RUN apt-get update -y && apt-get install -y libstdc++6 openssl libncurses5 locales \
|
|
||||||
&& apt-get clean && rm -f /var/lib/apt/lists/*_*
|
|
||||||
|
|
||||||
# Set the locale
|
|
||||||
RUN sed -i '/en_US.UTF-8/s/^# //g' /etc/locale.gen && locale-gen
|
|
||||||
|
|
||||||
ENV LANG en_US.UTF-8
|
|
||||||
ENV LANGUAGE en_US:en
|
|
||||||
ENV LC_ALL en_US.UTF-8
|
|
||||||
|
|
||||||
WORKDIR "/app"
|
|
||||||
RUN chown nobody /app
|
|
||||||
|
|
||||||
# set runner ENV
|
|
||||||
ENV MIX_ENV="prod"
|
|
||||||
|
|
||||||
# Only copy the final release from the build stage
|
|
||||||
COPY --from=builder --chown=nobody:root /app/_build/${MIX_ENV}/rel/malarkey ./
|
|
||||||
|
|
||||||
USER nobody
|
|
||||||
|
|
||||||
CMD ["/app/bin/server"]
|
|
||||||
|
|
||||||
# Appended by flyctl
|
|
||||||
ENV ECTO_IPV6 true
|
|
||||||
ENV ERL_AFLAGS "-proto_dist inet6_tcp"
|
|
||||||
|
|
||||||
# Appended by flyctl
|
|
||||||
ENV ECTO_IPV6 true
|
|
||||||
ENV ERL_AFLAGS "-proto_dist inet6_tcp"
|
|
||||||
21
LICENSE
21
LICENSE
@@ -1,21 +0,0 @@
|
|||||||
MIT License
|
|
||||||
|
|
||||||
Copyright (c) 2022 Fergal Moran
|
|
||||||
|
|
||||||
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
||||||
of this software and associated documentation files (the "Software"), to deal
|
|
||||||
in the Software without restriction, including without limitation the rights
|
|
||||||
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
||||||
copies of the Software, and to permit persons to whom the Software is
|
|
||||||
furnished to do so, subject to the following conditions:
|
|
||||||
|
|
||||||
The above copyright notice and this permission notice shall be included in all
|
|
||||||
copies or substantial portions of the Software.
|
|
||||||
|
|
||||||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
||||||
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
||||||
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
||||||
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
||||||
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
||||||
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
||||||
SOFTWARE.
|
|
||||||
@@ -1,5 +0,0 @@
|
|||||||
@import "tailwindcss/base";
|
|
||||||
@import "tailwindcss/components";
|
|
||||||
@import "tailwindcss/utilities";
|
|
||||||
|
|
||||||
/* This file is for your main application CSS */
|
|
||||||
@@ -1,41 +0,0 @@
|
|||||||
// If you want to use Phoenix channels, run `mix help phx.gen.channel`
|
|
||||||
// to get started and then uncomment the line below.
|
|
||||||
// import "./user_socket.js"
|
|
||||||
|
|
||||||
// You can include dependencies in two ways.
|
|
||||||
//
|
|
||||||
// The simplest option is to put them in assets/vendor and
|
|
||||||
// import them using relative paths:
|
|
||||||
//
|
|
||||||
// import "../vendor/some-package.js"
|
|
||||||
//
|
|
||||||
// Alternatively, you can `npm install some-package --prefix assets` and import
|
|
||||||
// them using a path starting with the package name:
|
|
||||||
//
|
|
||||||
// import "some-package"
|
|
||||||
//
|
|
||||||
|
|
||||||
// Include phoenix_html to handle method=PUT/DELETE in forms and buttons.
|
|
||||||
import "phoenix_html"
|
|
||||||
// Establish Phoenix Socket and LiveView configuration.
|
|
||||||
import {Socket} from "phoenix"
|
|
||||||
import {LiveSocket} from "phoenix_live_view"
|
|
||||||
import topbar from "../vendor/topbar"
|
|
||||||
|
|
||||||
let csrfToken = document.querySelector("meta[name='csrf-token']").getAttribute("content")
|
|
||||||
let liveSocket = new LiveSocket("/live", Socket, {params: {_csrf_token: csrfToken}})
|
|
||||||
|
|
||||||
// Show progress bar on live navigation and form submits
|
|
||||||
topbar.config({barColors: {0: "#29d"}, shadowColor: "rgba(0, 0, 0, .3)"})
|
|
||||||
window.addEventListener("phx:page-loading-start", info => topbar.delayedShow(200))
|
|
||||||
window.addEventListener("phx:page-loading-stop", info => topbar.hide())
|
|
||||||
|
|
||||||
// connect if there are any LiveViews on the page
|
|
||||||
liveSocket.connect()
|
|
||||||
|
|
||||||
// expose liveSocket on window for web console debug logs and latency simulation:
|
|
||||||
// >> liveSocket.enableDebug()
|
|
||||||
// >> liveSocket.enableLatencySim(1000) // enabled for duration of browser session
|
|
||||||
// >> liveSocket.disableLatencySim()
|
|
||||||
window.liveSocket = liveSocket
|
|
||||||
|
|
||||||
@@ -1,26 +0,0 @@
|
|||||||
// See the Tailwind configuration guide for advanced usage
|
|
||||||
// https://tailwindcss.com/docs/configuration
|
|
||||||
|
|
||||||
const plugin = require("tailwindcss/plugin")
|
|
||||||
|
|
||||||
module.exports = {
|
|
||||||
content: [
|
|
||||||
"./js/**/*.js",
|
|
||||||
"../lib/*_web.ex",
|
|
||||||
"../lib/*_web/**/*.*ex"
|
|
||||||
],
|
|
||||||
theme: {
|
|
||||||
extend: {
|
|
||||||
colors: {
|
|
||||||
brand: "#FD4F00",
|
|
||||||
}
|
|
||||||
},
|
|
||||||
},
|
|
||||||
plugins: [
|
|
||||||
require("@tailwindcss/forms"),
|
|
||||||
plugin(({addVariant}) => addVariant("phx-no-feedback", [".phx-no-feedback&", ".phx-no-feedback &"])),
|
|
||||||
plugin(({addVariant}) => addVariant("phx-click-loading", [".phx-click-loading&", ".phx-click-loading &"])),
|
|
||||||
plugin(({addVariant}) => addVariant("phx-submit-loading", [".phx-submit-loading&", ".phx-submit-loading &"])),
|
|
||||||
plugin(({addVariant}) => addVariant("phx-change-loading", [".phx-change-loading&", ".phx-change-loading &"]))
|
|
||||||
]
|
|
||||||
}
|
|
||||||
167
assets/vendor/topbar.js
vendored
167
assets/vendor/topbar.js
vendored
@@ -1,167 +0,0 @@
|
|||||||
/**
|
|
||||||
* @license MIT
|
|
||||||
* topbar 1.0.0, 2021-01-06
|
|
||||||
* Modifications:
|
|
||||||
* - add delayedShow(time) (2022-09-21)
|
|
||||||
* http://buunguyen.github.io/topbar
|
|
||||||
* Copyright (c) 2021 Buu Nguyen
|
|
||||||
*/
|
|
||||||
(function (window, document) {
|
|
||||||
"use strict";
|
|
||||||
|
|
||||||
// https://gist.github.com/paulirish/1579671
|
|
||||||
(function () {
|
|
||||||
var lastTime = 0;
|
|
||||||
var vendors = ["ms", "moz", "webkit", "o"];
|
|
||||||
for (var x = 0; x < vendors.length && !window.requestAnimationFrame; ++x) {
|
|
||||||
window.requestAnimationFrame =
|
|
||||||
window[vendors[x] + "RequestAnimationFrame"];
|
|
||||||
window.cancelAnimationFrame =
|
|
||||||
window[vendors[x] + "CancelAnimationFrame"] ||
|
|
||||||
window[vendors[x] + "CancelRequestAnimationFrame"];
|
|
||||||
}
|
|
||||||
if (!window.requestAnimationFrame)
|
|
||||||
window.requestAnimationFrame = function (callback, element) {
|
|
||||||
var currTime = new Date().getTime();
|
|
||||||
var timeToCall = Math.max(0, 16 - (currTime - lastTime));
|
|
||||||
var id = window.setTimeout(function () {
|
|
||||||
callback(currTime + timeToCall);
|
|
||||||
}, timeToCall);
|
|
||||||
lastTime = currTime + timeToCall;
|
|
||||||
return id;
|
|
||||||
};
|
|
||||||
if (!window.cancelAnimationFrame)
|
|
||||||
window.cancelAnimationFrame = function (id) {
|
|
||||||
clearTimeout(id);
|
|
||||||
};
|
|
||||||
})();
|
|
||||||
|
|
||||||
var canvas,
|
|
||||||
currentProgress,
|
|
||||||
showing,
|
|
||||||
progressTimerId = null,
|
|
||||||
fadeTimerId = null,
|
|
||||||
delayTimerId = null,
|
|
||||||
addEvent = function (elem, type, handler) {
|
|
||||||
if (elem.addEventListener) elem.addEventListener(type, handler, false);
|
|
||||||
else if (elem.attachEvent) elem.attachEvent("on" + type, handler);
|
|
||||||
else elem["on" + type] = handler;
|
|
||||||
},
|
|
||||||
options = {
|
|
||||||
autoRun: true,
|
|
||||||
barThickness: 3,
|
|
||||||
barColors: {
|
|
||||||
0: "rgba(26, 188, 156, .9)",
|
|
||||||
".25": "rgba(52, 152, 219, .9)",
|
|
||||||
".50": "rgba(241, 196, 15, .9)",
|
|
||||||
".75": "rgba(230, 126, 34, .9)",
|
|
||||||
"1.0": "rgba(211, 84, 0, .9)",
|
|
||||||
},
|
|
||||||
shadowBlur: 10,
|
|
||||||
shadowColor: "rgba(0, 0, 0, .6)",
|
|
||||||
className: null,
|
|
||||||
},
|
|
||||||
repaint = function () {
|
|
||||||
canvas.width = window.innerWidth;
|
|
||||||
canvas.height = options.barThickness * 5; // need space for shadow
|
|
||||||
|
|
||||||
var ctx = canvas.getContext("2d");
|
|
||||||
ctx.shadowBlur = options.shadowBlur;
|
|
||||||
ctx.shadowColor = options.shadowColor;
|
|
||||||
|
|
||||||
var lineGradient = ctx.createLinearGradient(0, 0, canvas.width, 0);
|
|
||||||
for (var stop in options.barColors)
|
|
||||||
lineGradient.addColorStop(stop, options.barColors[stop]);
|
|
||||||
ctx.lineWidth = options.barThickness;
|
|
||||||
ctx.beginPath();
|
|
||||||
ctx.moveTo(0, options.barThickness / 2);
|
|
||||||
ctx.lineTo(
|
|
||||||
Math.ceil(currentProgress * canvas.width),
|
|
||||||
options.barThickness / 2
|
|
||||||
);
|
|
||||||
ctx.strokeStyle = lineGradient;
|
|
||||||
ctx.stroke();
|
|
||||||
},
|
|
||||||
createCanvas = function () {
|
|
||||||
canvas = document.createElement("canvas");
|
|
||||||
var style = canvas.style;
|
|
||||||
style.position = "fixed";
|
|
||||||
style.top = style.left = style.right = style.margin = style.padding = 0;
|
|
||||||
style.zIndex = 100001;
|
|
||||||
style.display = "none";
|
|
||||||
if (options.className) canvas.classList.add(options.className);
|
|
||||||
document.body.appendChild(canvas);
|
|
||||||
addEvent(window, "resize", repaint);
|
|
||||||
},
|
|
||||||
topbar = {
|
|
||||||
config: function (opts) {
|
|
||||||
for (var key in opts)
|
|
||||||
if (options.hasOwnProperty(key)) options[key] = opts[key];
|
|
||||||
},
|
|
||||||
delayedShow: function(time) {
|
|
||||||
if (showing) return;
|
|
||||||
if (delayTimerId) return;
|
|
||||||
delayTimerId = setTimeout(() => topbar.show(), time);
|
|
||||||
},
|
|
||||||
show: function () {
|
|
||||||
if (showing) return;
|
|
||||||
showing = true;
|
|
||||||
if (fadeTimerId !== null) window.cancelAnimationFrame(fadeTimerId);
|
|
||||||
if (!canvas) createCanvas();
|
|
||||||
canvas.style.opacity = 1;
|
|
||||||
canvas.style.display = "block";
|
|
||||||
topbar.progress(0);
|
|
||||||
if (options.autoRun) {
|
|
||||||
(function loop() {
|
|
||||||
progressTimerId = window.requestAnimationFrame(loop);
|
|
||||||
topbar.progress(
|
|
||||||
"+" + 0.05 * Math.pow(1 - Math.sqrt(currentProgress), 2)
|
|
||||||
);
|
|
||||||
})();
|
|
||||||
}
|
|
||||||
},
|
|
||||||
progress: function (to) {
|
|
||||||
if (typeof to === "undefined") return currentProgress;
|
|
||||||
if (typeof to === "string") {
|
|
||||||
to =
|
|
||||||
(to.indexOf("+") >= 0 || to.indexOf("-") >= 0
|
|
||||||
? currentProgress
|
|
||||||
: 0) + parseFloat(to);
|
|
||||||
}
|
|
||||||
currentProgress = to > 1 ? 1 : to;
|
|
||||||
repaint();
|
|
||||||
return currentProgress;
|
|
||||||
},
|
|
||||||
hide: function () {
|
|
||||||
clearTimeout(delayTimerId);
|
|
||||||
delayTimerId = null;
|
|
||||||
if (!showing) return;
|
|
||||||
showing = false;
|
|
||||||
if (progressTimerId != null) {
|
|
||||||
window.cancelAnimationFrame(progressTimerId);
|
|
||||||
progressTimerId = null;
|
|
||||||
}
|
|
||||||
(function loop() {
|
|
||||||
if (topbar.progress("+.1") >= 1) {
|
|
||||||
canvas.style.opacity -= 0.05;
|
|
||||||
if (canvas.style.opacity <= 0.05) {
|
|
||||||
canvas.style.display = "none";
|
|
||||||
fadeTimerId = null;
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
fadeTimerId = window.requestAnimationFrame(loop);
|
|
||||||
})();
|
|
||||||
},
|
|
||||||
};
|
|
||||||
|
|
||||||
if (typeof module === "object" && typeof module.exports === "object") {
|
|
||||||
module.exports = topbar;
|
|
||||||
} else if (typeof define === "function" && define.amd) {
|
|
||||||
define(function () {
|
|
||||||
return topbar;
|
|
||||||
});
|
|
||||||
} else {
|
|
||||||
this.topbar = topbar;
|
|
||||||
}
|
|
||||||
}.call(this, window, document));
|
|
||||||
@@ -1,65 +0,0 @@
|
|||||||
# 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
|
|
||||||
|
|
||||||
config :malarkey,
|
|
||||||
ecto_repos: [Malarkey.Repo]
|
|
||||||
|
|
||||||
# Configures the endpoint
|
|
||||||
config :malarkey, MalarkeyWeb.Endpoint,
|
|
||||||
url: [host: "localhost"],
|
|
||||||
render_errors: [
|
|
||||||
formats: [html: MalarkeyWeb.ErrorHTML, json: MalarkeyWeb.ErrorJSON],
|
|
||||||
layout: false
|
|
||||||
],
|
|
||||||
pubsub_server: Malarkey.PubSub,
|
|
||||||
live_view: [signing_salt: "UYdloyCB"],
|
|
||||||
check_origin: ["https://malarkey.fergl.ie/"]
|
|
||||||
|
|
||||||
# Configures the mailer
|
|
||||||
#
|
|
||||||
# By default it uses the "Local" adapter which stores the emails
|
|
||||||
# locally. You can see the emails in your browser, at "/dev/mailbox".
|
|
||||||
#
|
|
||||||
# For production it's recommended to configure a different adapter
|
|
||||||
# at the `config/runtime.exs`.
|
|
||||||
config :malarkey, Malarkey.Mailer, adapter: Swoosh.Adapters.Local
|
|
||||||
|
|
||||||
# Configure esbuild (the version is required)
|
|
||||||
config :esbuild,
|
|
||||||
version: "0.14.41",
|
|
||||||
default: [
|
|
||||||
args:
|
|
||||||
~w(js/app.js --bundle --target=es2017 --outdir=../priv/static/assets --external:/fonts/* --external:/images/*),
|
|
||||||
cd: Path.expand("../assets", __DIR__),
|
|
||||||
env: %{"NODE_PATH" => Path.expand("../deps", __DIR__)}
|
|
||||||
]
|
|
||||||
|
|
||||||
# Configure tailwind (the version is required)
|
|
||||||
config :tailwind,
|
|
||||||
version: "3.1.8",
|
|
||||||
default: [
|
|
||||||
args: ~w(
|
|
||||||
--config=tailwind.config.js
|
|
||||||
--input=css/app.css
|
|
||||||
--output=../priv/static/assets/app.css
|
|
||||||
),
|
|
||||||
cd: Path.expand("../assets", __DIR__)
|
|
||||||
]
|
|
||||||
|
|
||||||
# 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"
|
|
||||||
@@ -1,80 +0,0 @@
|
|||||||
import Config
|
|
||||||
|
|
||||||
# Configure your database
|
|
||||||
config :malarkey, Malarkey.Repo,
|
|
||||||
username: "postgres",
|
|
||||||
password: "hackme",
|
|
||||||
hostname: "localhost",
|
|
||||||
database: "malarkey_dev",
|
|
||||||
stacktrace: true,
|
|
||||||
show_sensitive_data_on_connection_error: true,
|
|
||||||
pool_size: 10
|
|
||||||
|
|
||||||
# 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 esbuild to bundle .js and .css sources.
|
|
||||||
config :malarkey, MalarkeyWeb.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: "7J/3rVbYDSqC9cp4x/IxzhIr3OBHaQXso8CbFf0j4GfyfwCIpvc5ghgO7vPuBCwP",
|
|
||||||
watchers: [
|
|
||||||
esbuild: {Esbuild, :install_and_run, [:default, ~w(--sourcemap=inline --watch)]},
|
|
||||||
tailwind: {Tailwind, :install_and_run, [:default, ~w(--watch)]}
|
|
||||||
]
|
|
||||||
|
|
||||||
# ## 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 :malarkey, MalarkeyWeb.Endpoint,
|
|
||||||
live_reload: [
|
|
||||||
patterns: [
|
|
||||||
~r"priv/static/.*(js|css|png|jpeg|jpg|gif|svg)$",
|
|
||||||
~r"priv/gettext/.*(po)$",
|
|
||||||
~r"lib/malarkey_web/(live|views)/.*(ex)$",
|
|
||||||
~r"lib/malarkey_web/templates/.*(eex)$"
|
|
||||||
]
|
|
||||||
]
|
|
||||||
|
|
||||||
# Enable dev routes for dashboard and mailbox
|
|
||||||
config :malarkey, 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
|
|
||||||
|
|
||||||
# Disable swoosh api client as it is only required for production adapters.
|
|
||||||
config :swoosh, :api_client, false
|
|
||||||
@@ -1,21 +0,0 @@
|
|||||||
import 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 :malarkey, MalarkeyWeb.Endpoint, cache_static_manifest: "priv/static/cache_manifest.json"
|
|
||||||
|
|
||||||
# Configures Swoosh API Client
|
|
||||||
config :swoosh, :api_client, Malarkey.Finch
|
|
||||||
|
|
||||||
# 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.
|
|
||||||
@@ -1,115 +0,0 @@
|
|||||||
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/malarkey 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 :malarkey, MalarkeyWeb.Endpoint, server: true
|
|
||||||
end
|
|
||||||
|
|
||||||
if config_env() == :prod do
|
|
||||||
database_url =
|
|
||||||
System.get_env("DATABASE_URL") ||
|
|
||||||
raise """
|
|
||||||
environment variable DATABASE_URL is missing.
|
|
||||||
For example: ecto://USER:PASS@HOST/DATABASE
|
|
||||||
"""
|
|
||||||
|
|
||||||
maybe_ipv6 = if System.get_env("ECTO_IPV6"), do: [:inet6], else: []
|
|
||||||
|
|
||||||
config :malarkey, Malarkey.Repo,
|
|
||||||
# ssl: true,
|
|
||||||
url: database_url,
|
|
||||||
pool_size: String.to_integer(System.get_env("POOL_SIZE") || "10"),
|
|
||||||
socket_options: maybe_ipv6
|
|
||||||
|
|
||||||
# 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 :malarkey, MalarkeyWeb.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 :malarkey, MalarkeyWeb.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 :malarkey, MalarkeyWeb.Endpoint,
|
|
||||||
# force_ssl: [hsts: true]
|
|
||||||
#
|
|
||||||
# Check `Plug.SSL` for all available options in `force_ssl`.
|
|
||||||
|
|
||||||
# ## Configuring the mailer
|
|
||||||
#
|
|
||||||
# In production you need to configure the mailer to use a different adapter.
|
|
||||||
# Also, you may need to configure the Swoosh API client of your choice if you
|
|
||||||
# are not using SMTP. Here is an example of the configuration:
|
|
||||||
#
|
|
||||||
# config :malarkey, Malarkey.Mailer,
|
|
||||||
# adapter: Swoosh.Adapters.Mailgun,
|
|
||||||
# api_key: System.get_env("MAILGUN_API_KEY"),
|
|
||||||
# domain: System.get_env("MAILGUN_DOMAIN")
|
|
||||||
#
|
|
||||||
# For this example you need include a HTTP client required by Swoosh API client.
|
|
||||||
# Swoosh supports Hackney and Finch out of the box:
|
|
||||||
#
|
|
||||||
# config :swoosh, :api_client, Swoosh.ApiClient.Hackney
|
|
||||||
#
|
|
||||||
# See https://hexdocs.pm/swoosh/Swoosh.html#module-installation for details.
|
|
||||||
end
|
|
||||||
@@ -1,36 +0,0 @@
|
|||||||
import Config
|
|
||||||
|
|
||||||
# Only in tests, remove the complexity from the password hashing algorithm
|
|
||||||
config :bcrypt_elixir, :log_rounds, 1
|
|
||||||
|
|
||||||
# Configure your database
|
|
||||||
#
|
|
||||||
# The MIX_TEST_PARTITION environment variable can be used
|
|
||||||
# to provide built-in test partitioning in CI environment.
|
|
||||||
# Run `mix help test` for more information.
|
|
||||||
config :malarkey, Malarkey.Repo,
|
|
||||||
username: "postgres",
|
|
||||||
password: "hackme",
|
|
||||||
hostname: "localhost",
|
|
||||||
database: "malarkey_test#{System.get_env("MIX_TEST_PARTITION")}",
|
|
||||||
pool: Ecto.Adapters.SQL.Sandbox,
|
|
||||||
pool_size: 10
|
|
||||||
|
|
||||||
# We don't run a server during test. If one is required,
|
|
||||||
# you can enable the server option below.
|
|
||||||
config :malarkey, MalarkeyWeb.Endpoint,
|
|
||||||
http: [ip: {127, 0, 0, 1}, port: 4002],
|
|
||||||
secret_key_base: "2/EjqngGC/YAYuEnyFk4BIRyMhBs29VtzndNLvvdYeiegOFctblOTc4i5m8z4GvS",
|
|
||||||
server: false
|
|
||||||
|
|
||||||
# In test we don't send emails.
|
|
||||||
config :malarkey, Malarkey.Mailer, adapter: Swoosh.Adapters.Test
|
|
||||||
|
|
||||||
# Disable swoosh api client as it is only required for production adapters.
|
|
||||||
config :swoosh, :api_client, 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
|
|
||||||
43
fly.toml
43
fly.toml
@@ -1,43 +0,0 @@
|
|||||||
# fly.toml file generated for malarkey on 2022-11-25T18:51:01Z
|
|
||||||
|
|
||||||
app = "malarkey"
|
|
||||||
kill_signal = "SIGTERM"
|
|
||||||
kill_timeout = 5
|
|
||||||
processes = []
|
|
||||||
|
|
||||||
[deploy]
|
|
||||||
release_command = "/app/bin/migrate"
|
|
||||||
|
|
||||||
[env]
|
|
||||||
PHX_HOST = "malarkey.fly.dev"
|
|
||||||
PORT = "8080"
|
|
||||||
|
|
||||||
[experimental]
|
|
||||||
allowed_public_ports = []
|
|
||||||
auto_rollback = true
|
|
||||||
|
|
||||||
[[services]]
|
|
||||||
http_checks = []
|
|
||||||
internal_port = 8080
|
|
||||||
processes = ["app"]
|
|
||||||
protocol = "tcp"
|
|
||||||
script_checks = []
|
|
||||||
[services.concurrency]
|
|
||||||
hard_limit = 25
|
|
||||||
soft_limit = 20
|
|
||||||
type = "connections"
|
|
||||||
|
|
||||||
[[services.ports]]
|
|
||||||
force_https = true
|
|
||||||
handlers = ["http"]
|
|
||||||
port = 80
|
|
||||||
|
|
||||||
[[services.ports]]
|
|
||||||
handlers = ["tls", "http"]
|
|
||||||
port = 443
|
|
||||||
|
|
||||||
[[services.tcp_checks]]
|
|
||||||
grace_period = "1s"
|
|
||||||
interval = "15s"
|
|
||||||
restart_limit = 0
|
|
||||||
timeout = "2s"
|
|
||||||
@@ -1,9 +0,0 @@
|
|||||||
defmodule Malarkey do
|
|
||||||
@moduledoc """
|
|
||||||
Malarkey 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
|
|
||||||
@@ -1,353 +0,0 @@
|
|||||||
defmodule Malarkey.Accounts do
|
|
||||||
@moduledoc """
|
|
||||||
The Accounts context.
|
|
||||||
"""
|
|
||||||
|
|
||||||
import Ecto.Query, warn: false
|
|
||||||
alias Malarkey.Repo
|
|
||||||
|
|
||||||
alias Malarkey.Accounts.{User, UserToken, UserNotifier}
|
|
||||||
|
|
||||||
## Database getters
|
|
||||||
|
|
||||||
@doc """
|
|
||||||
Gets a user by email.
|
|
||||||
|
|
||||||
## Examples
|
|
||||||
|
|
||||||
iex> get_user_by_email("foo@example.com")
|
|
||||||
%User{}
|
|
||||||
|
|
||||||
iex> get_user_by_email("unknown@example.com")
|
|
||||||
nil
|
|
||||||
|
|
||||||
"""
|
|
||||||
def get_user_by_email(email) when is_binary(email) do
|
|
||||||
Repo.get_by(User, email: email)
|
|
||||||
end
|
|
||||||
|
|
||||||
@doc """
|
|
||||||
Gets a user by email and password.
|
|
||||||
|
|
||||||
## Examples
|
|
||||||
|
|
||||||
iex> get_user_by_email_and_password("foo@example.com", "correct_password")
|
|
||||||
%User{}
|
|
||||||
|
|
||||||
iex> get_user_by_email_and_password("foo@example.com", "invalid_password")
|
|
||||||
nil
|
|
||||||
|
|
||||||
"""
|
|
||||||
def get_user_by_email_and_password(email, password)
|
|
||||||
when is_binary(email) and is_binary(password) do
|
|
||||||
user = Repo.get_by(User, email: email)
|
|
||||||
if User.valid_password?(user, password), do: user
|
|
||||||
end
|
|
||||||
|
|
||||||
@doc """
|
|
||||||
Gets a single user.
|
|
||||||
|
|
||||||
Raises `Ecto.NoResultsError` if the User does not exist.
|
|
||||||
|
|
||||||
## Examples
|
|
||||||
|
|
||||||
iex> get_user!(123)
|
|
||||||
%User{}
|
|
||||||
|
|
||||||
iex> get_user!(456)
|
|
||||||
** (Ecto.NoResultsError)
|
|
||||||
|
|
||||||
"""
|
|
||||||
def get_user!(id), do: Repo.get!(User, id)
|
|
||||||
|
|
||||||
## User registration
|
|
||||||
|
|
||||||
@doc """
|
|
||||||
Registers a user.
|
|
||||||
|
|
||||||
## Examples
|
|
||||||
|
|
||||||
iex> register_user(%{field: value})
|
|
||||||
{:ok, %User{}}
|
|
||||||
|
|
||||||
iex> register_user(%{field: bad_value})
|
|
||||||
{:error, %Ecto.Changeset{}}
|
|
||||||
|
|
||||||
"""
|
|
||||||
def register_user(attrs) do
|
|
||||||
%User{}
|
|
||||||
|> User.registration_changeset(attrs)
|
|
||||||
|> Repo.insert()
|
|
||||||
end
|
|
||||||
|
|
||||||
@doc """
|
|
||||||
Returns an `%Ecto.Changeset{}` for tracking user changes.
|
|
||||||
|
|
||||||
## Examples
|
|
||||||
|
|
||||||
iex> change_user_registration(user)
|
|
||||||
%Ecto.Changeset{data: %User{}}
|
|
||||||
|
|
||||||
"""
|
|
||||||
def change_user_registration(%User{} = user, attrs \\ %{}) do
|
|
||||||
User.registration_changeset(user, attrs, hash_password: false, validate_email: false)
|
|
||||||
end
|
|
||||||
|
|
||||||
## Settings
|
|
||||||
|
|
||||||
@doc """
|
|
||||||
Returns an `%Ecto.Changeset{}` for changing the user email.
|
|
||||||
|
|
||||||
## Examples
|
|
||||||
|
|
||||||
iex> change_user_email(user)
|
|
||||||
%Ecto.Changeset{data: %User{}}
|
|
||||||
|
|
||||||
"""
|
|
||||||
def change_user_email(user, attrs \\ %{}) do
|
|
||||||
User.email_changeset(user, attrs, validate_email: false)
|
|
||||||
end
|
|
||||||
|
|
||||||
@doc """
|
|
||||||
Emulates that the email will change without actually changing
|
|
||||||
it in the database.
|
|
||||||
|
|
||||||
## Examples
|
|
||||||
|
|
||||||
iex> apply_user_email(user, "valid password", %{email: ...})
|
|
||||||
{:ok, %User{}}
|
|
||||||
|
|
||||||
iex> apply_user_email(user, "invalid password", %{email: ...})
|
|
||||||
{:error, %Ecto.Changeset{}}
|
|
||||||
|
|
||||||
"""
|
|
||||||
def apply_user_email(user, password, attrs) do
|
|
||||||
user
|
|
||||||
|> User.email_changeset(attrs)
|
|
||||||
|> User.validate_current_password(password)
|
|
||||||
|> Ecto.Changeset.apply_action(:update)
|
|
||||||
end
|
|
||||||
|
|
||||||
@doc """
|
|
||||||
Updates the user email using the given token.
|
|
||||||
|
|
||||||
If the token matches, the user email is updated and the token is deleted.
|
|
||||||
The confirmed_at date is also updated to the current time.
|
|
||||||
"""
|
|
||||||
def update_user_email(user, token) do
|
|
||||||
context = "change:#{user.email}"
|
|
||||||
|
|
||||||
with {:ok, query} <- UserToken.verify_change_email_token_query(token, context),
|
|
||||||
%UserToken{sent_to: email} <- Repo.one(query),
|
|
||||||
{:ok, _} <- Repo.transaction(user_email_multi(user, email, context)) do
|
|
||||||
:ok
|
|
||||||
else
|
|
||||||
_ -> :error
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
defp user_email_multi(user, email, context) do
|
|
||||||
changeset =
|
|
||||||
user
|
|
||||||
|> User.email_changeset(%{email: email})
|
|
||||||
|> User.confirm_changeset()
|
|
||||||
|
|
||||||
Ecto.Multi.new()
|
|
||||||
|> Ecto.Multi.update(:user, changeset)
|
|
||||||
|> Ecto.Multi.delete_all(:tokens, UserToken.user_and_contexts_query(user, [context]))
|
|
||||||
end
|
|
||||||
|
|
||||||
@doc ~S"""
|
|
||||||
Delivers the update email instructions to the given user.
|
|
||||||
|
|
||||||
## Examples
|
|
||||||
|
|
||||||
iex> deliver_user_update_email_instructions(user, current_email, &url(~p"/users/settings/confirm_email/#{&1})")
|
|
||||||
{:ok, %{to: ..., body: ...}}
|
|
||||||
|
|
||||||
"""
|
|
||||||
def deliver_user_update_email_instructions(%User{} = user, current_email, update_email_url_fun)
|
|
||||||
when is_function(update_email_url_fun, 1) do
|
|
||||||
{encoded_token, user_token} = UserToken.build_email_token(user, "change:#{current_email}")
|
|
||||||
|
|
||||||
Repo.insert!(user_token)
|
|
||||||
UserNotifier.deliver_update_email_instructions(user, update_email_url_fun.(encoded_token))
|
|
||||||
end
|
|
||||||
|
|
||||||
@doc """
|
|
||||||
Returns an `%Ecto.Changeset{}` for changing the user password.
|
|
||||||
|
|
||||||
## Examples
|
|
||||||
|
|
||||||
iex> change_user_password(user)
|
|
||||||
%Ecto.Changeset{data: %User{}}
|
|
||||||
|
|
||||||
"""
|
|
||||||
def change_user_password(user, attrs \\ %{}) do
|
|
||||||
User.password_changeset(user, attrs, hash_password: false)
|
|
||||||
end
|
|
||||||
|
|
||||||
@doc """
|
|
||||||
Updates the user password.
|
|
||||||
|
|
||||||
## Examples
|
|
||||||
|
|
||||||
iex> update_user_password(user, "valid password", %{password: ...})
|
|
||||||
{:ok, %User{}}
|
|
||||||
|
|
||||||
iex> update_user_password(user, "invalid password", %{password: ...})
|
|
||||||
{:error, %Ecto.Changeset{}}
|
|
||||||
|
|
||||||
"""
|
|
||||||
def update_user_password(user, password, attrs) do
|
|
||||||
changeset =
|
|
||||||
user
|
|
||||||
|> User.password_changeset(attrs)
|
|
||||||
|> User.validate_current_password(password)
|
|
||||||
|
|
||||||
Ecto.Multi.new()
|
|
||||||
|> Ecto.Multi.update(:user, changeset)
|
|
||||||
|> Ecto.Multi.delete_all(:tokens, UserToken.user_and_contexts_query(user, :all))
|
|
||||||
|> Repo.transaction()
|
|
||||||
|> case do
|
|
||||||
{:ok, %{user: user}} -> {:ok, user}
|
|
||||||
{:error, :user, changeset, _} -> {:error, changeset}
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
## Session
|
|
||||||
|
|
||||||
@doc """
|
|
||||||
Generates a session token.
|
|
||||||
"""
|
|
||||||
def generate_user_session_token(user) do
|
|
||||||
{token, user_token} = UserToken.build_session_token(user)
|
|
||||||
Repo.insert!(user_token)
|
|
||||||
token
|
|
||||||
end
|
|
||||||
|
|
||||||
@doc """
|
|
||||||
Gets the user with the given signed token.
|
|
||||||
"""
|
|
||||||
def get_user_by_session_token(token) do
|
|
||||||
{:ok, query} = UserToken.verify_session_token_query(token)
|
|
||||||
Repo.one(query)
|
|
||||||
end
|
|
||||||
|
|
||||||
@doc """
|
|
||||||
Deletes the signed token with the given context.
|
|
||||||
"""
|
|
||||||
def delete_user_session_token(token) do
|
|
||||||
Repo.delete_all(UserToken.token_and_context_query(token, "session"))
|
|
||||||
:ok
|
|
||||||
end
|
|
||||||
|
|
||||||
## Confirmation
|
|
||||||
|
|
||||||
@doc ~S"""
|
|
||||||
Delivers the confirmation email instructions to the given user.
|
|
||||||
|
|
||||||
## Examples
|
|
||||||
|
|
||||||
iex> deliver_user_confirmation_instructions(user, &url(~p"/users/confirm/#{&1}"))
|
|
||||||
{:ok, %{to: ..., body: ...}}
|
|
||||||
|
|
||||||
iex> deliver_user_confirmation_instructions(confirmed_user, &url(~p"/users/confirm/#{&1}"))
|
|
||||||
{:error, :already_confirmed}
|
|
||||||
|
|
||||||
"""
|
|
||||||
def deliver_user_confirmation_instructions(%User{} = user, confirmation_url_fun)
|
|
||||||
when is_function(confirmation_url_fun, 1) do
|
|
||||||
if user.confirmed_at do
|
|
||||||
{:error, :already_confirmed}
|
|
||||||
else
|
|
||||||
{encoded_token, user_token} = UserToken.build_email_token(user, "confirm")
|
|
||||||
Repo.insert!(user_token)
|
|
||||||
UserNotifier.deliver_confirmation_instructions(user, confirmation_url_fun.(encoded_token))
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
@doc """
|
|
||||||
Confirms a user by the given token.
|
|
||||||
|
|
||||||
If the token matches, the user account is marked as confirmed
|
|
||||||
and the token is deleted.
|
|
||||||
"""
|
|
||||||
def confirm_user(token) do
|
|
||||||
with {:ok, query} <- UserToken.verify_email_token_query(token, "confirm"),
|
|
||||||
%User{} = user <- Repo.one(query),
|
|
||||||
{:ok, %{user: user}} <- Repo.transaction(confirm_user_multi(user)) do
|
|
||||||
{:ok, user}
|
|
||||||
else
|
|
||||||
_ -> :error
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
defp confirm_user_multi(user) do
|
|
||||||
Ecto.Multi.new()
|
|
||||||
|> Ecto.Multi.update(:user, User.confirm_changeset(user))
|
|
||||||
|> Ecto.Multi.delete_all(:tokens, UserToken.user_and_contexts_query(user, ["confirm"]))
|
|
||||||
end
|
|
||||||
|
|
||||||
## Reset password
|
|
||||||
|
|
||||||
@doc ~S"""
|
|
||||||
Delivers the reset password email to the given user.
|
|
||||||
|
|
||||||
## Examples
|
|
||||||
|
|
||||||
iex> deliver_user_reset_password_instructions(user, &url(~p"/users/reset_password/#{&1}"))
|
|
||||||
{:ok, %{to: ..., body: ...}}
|
|
||||||
|
|
||||||
"""
|
|
||||||
def deliver_user_reset_password_instructions(%User{} = user, reset_password_url_fun)
|
|
||||||
when is_function(reset_password_url_fun, 1) do
|
|
||||||
{encoded_token, user_token} = UserToken.build_email_token(user, "reset_password")
|
|
||||||
Repo.insert!(user_token)
|
|
||||||
UserNotifier.deliver_reset_password_instructions(user, reset_password_url_fun.(encoded_token))
|
|
||||||
end
|
|
||||||
|
|
||||||
@doc """
|
|
||||||
Gets the user by reset password token.
|
|
||||||
|
|
||||||
## Examples
|
|
||||||
|
|
||||||
iex> get_user_by_reset_password_token("validtoken")
|
|
||||||
%User{}
|
|
||||||
|
|
||||||
iex> get_user_by_reset_password_token("invalidtoken")
|
|
||||||
nil
|
|
||||||
|
|
||||||
"""
|
|
||||||
def get_user_by_reset_password_token(token) do
|
|
||||||
with {:ok, query} <- UserToken.verify_email_token_query(token, "reset_password"),
|
|
||||||
%User{} = user <- Repo.one(query) do
|
|
||||||
user
|
|
||||||
else
|
|
||||||
_ -> nil
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
@doc """
|
|
||||||
Resets the user password.
|
|
||||||
|
|
||||||
## Examples
|
|
||||||
|
|
||||||
iex> reset_user_password(user, %{password: "new long password", password_confirmation: "new long password"})
|
|
||||||
{:ok, %User{}}
|
|
||||||
|
|
||||||
iex> reset_user_password(user, %{password: "valid", password_confirmation: "not the same"})
|
|
||||||
{:error, %Ecto.Changeset{}}
|
|
||||||
|
|
||||||
"""
|
|
||||||
def reset_user_password(user, attrs) do
|
|
||||||
Ecto.Multi.new()
|
|
||||||
|> Ecto.Multi.update(:user, User.password_changeset(user, attrs))
|
|
||||||
|> Ecto.Multi.delete_all(:tokens, UserToken.user_and_contexts_query(user, :all))
|
|
||||||
|> Repo.transaction()
|
|
||||||
|> case do
|
|
||||||
{:ok, %{user: user}} -> {:ok, user}
|
|
||||||
{:error, :user, changeset, _} -> {:error, changeset}
|
|
||||||
end
|
|
||||||
end
|
|
||||||
end
|
|
||||||
@@ -1,179 +0,0 @@
|
|||||||
defmodule Malarkey.Accounts.User do
|
|
||||||
use Ecto.Schema
|
|
||||||
import Ecto.Changeset
|
|
||||||
alias Malarkey.Timeline.Post
|
|
||||||
|
|
||||||
schema "users" do
|
|
||||||
field :email, :string
|
|
||||||
field :password, :string, virtual: true, redact: true
|
|
||||||
field :hashed_password, :string, redact: true
|
|
||||||
field :confirmed_at, :naive_datetime
|
|
||||||
field :username, :string
|
|
||||||
field :fullname, :string
|
|
||||||
|
|
||||||
many_to_many(
|
|
||||||
:likes,
|
|
||||||
Post,
|
|
||||||
join_through: Malarkey.Timeline.PostUserLike,
|
|
||||||
on_replace: :delete
|
|
||||||
)
|
|
||||||
|
|
||||||
many_to_many(
|
|
||||||
:dislikes,
|
|
||||||
Post,
|
|
||||||
join_through: Malarkey.Timeline.PostUserDislike,
|
|
||||||
on_replace: :delete
|
|
||||||
)
|
|
||||||
|
|
||||||
many_to_many(
|
|
||||||
:reposts,
|
|
||||||
Post,
|
|
||||||
join_through: Malarkey.Timeline.PostUserReposts,
|
|
||||||
on_replace: :delete
|
|
||||||
)
|
|
||||||
|
|
||||||
timestamps()
|
|
||||||
end
|
|
||||||
|
|
||||||
@doc """
|
|
||||||
A user changeset for registration.
|
|
||||||
|
|
||||||
It is important to validate the length of both email and password.
|
|
||||||
Otherwise databases may truncate the email without warnings, which
|
|
||||||
could lead to unpredictable or insecure behaviour. Long passwords may
|
|
||||||
also be very expensive to hash for certain algorithms.
|
|
||||||
|
|
||||||
## Options
|
|
||||||
|
|
||||||
* `:hash_password` - Hashes the password so it can be stored securely
|
|
||||||
in the database and ensures the password field is cleared to prevent
|
|
||||||
leaks in the logs. If password hashing is not needed and clearing the
|
|
||||||
password field is not desired (like when using this changeset for
|
|
||||||
validations on a LiveView form), this option can be set to `false`.
|
|
||||||
Defaults to `true`.
|
|
||||||
|
|
||||||
* `:validate_email` - Validates the uniqueness of the email, in case
|
|
||||||
you don't want to validate the uniqueness of the email (like when
|
|
||||||
using this changeset for validations on a LiveView form before
|
|
||||||
submitting the form), this option can be set to `false`.
|
|
||||||
Defaults to `true`.
|
|
||||||
"""
|
|
||||||
def registration_changeset(user, attrs, opts \\ []) do
|
|
||||||
user
|
|
||||||
|> cast(attrs, [:email, :password])
|
|
||||||
|> validate_email(opts)
|
|
||||||
|> validate_password(opts)
|
|
||||||
end
|
|
||||||
|
|
||||||
defp validate_email(changeset, opts) do
|
|
||||||
changeset
|
|
||||||
|> validate_required([:email])
|
|
||||||
|> validate_format(:email, ~r/^[^\s]+@[^\s]+$/, message: "must have the @ sign and no spaces")
|
|
||||||
|> validate_length(:email, max: 160)
|
|
||||||
|> maybe_validate_unique_email(opts)
|
|
||||||
end
|
|
||||||
|
|
||||||
defp validate_password(changeset, opts) do
|
|
||||||
changeset
|
|
||||||
|> validate_required([:password])
|
|
||||||
|> validate_length(:password, min: 12, max: 72)
|
|
||||||
# |> validate_format(:password, ~r/[a-z]/, message: "at least one lower case character")
|
|
||||||
# |> validate_format(:password, ~r/[A-Z]/, message: "at least one upper case character")
|
|
||||||
# |> validate_format(:password, ~r/[!?@#$%^&*_0-9]/, message: "at least one digit or punctuation character")
|
|
||||||
|> maybe_hash_password(opts)
|
|
||||||
end
|
|
||||||
|
|
||||||
defp maybe_hash_password(changeset, opts) do
|
|
||||||
hash_password? = Keyword.get(opts, :hash_password, true)
|
|
||||||
password = get_change(changeset, :password)
|
|
||||||
|
|
||||||
if hash_password? && password && changeset.valid? do
|
|
||||||
changeset
|
|
||||||
# If using Bcrypt, then further validate it is at most 72 bytes long
|
|
||||||
|> validate_length(:password, max: 72, count: :bytes)
|
|
||||||
|> put_change(:hashed_password, Bcrypt.hash_pwd_salt(password))
|
|
||||||
|> delete_change(:password)
|
|
||||||
else
|
|
||||||
changeset
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
defp maybe_validate_unique_email(changeset, opts) do
|
|
||||||
if Keyword.get(opts, :validate_email, true) do
|
|
||||||
changeset
|
|
||||||
|> unsafe_validate_unique(:email, Malarkey.Repo)
|
|
||||||
|> unique_constraint(:email)
|
|
||||||
else
|
|
||||||
changeset
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
@doc """
|
|
||||||
A user changeset for changing the email.
|
|
||||||
|
|
||||||
It requires the email to change otherwise an error is added.
|
|
||||||
"""
|
|
||||||
def email_changeset(user, attrs, opts \\ []) do
|
|
||||||
user
|
|
||||||
|> cast(attrs, [:email])
|
|
||||||
|> validate_email(opts)
|
|
||||||
|> case do
|
|
||||||
%{changes: %{email: _}} = changeset -> changeset
|
|
||||||
%{} = changeset -> add_error(changeset, :email, "did not change")
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
@doc """
|
|
||||||
A user changeset for changing the password.
|
|
||||||
|
|
||||||
## Options
|
|
||||||
|
|
||||||
* `:hash_password` - Hashes the password so it can be stored securely
|
|
||||||
in the database and ensures the password field is cleared to prevent
|
|
||||||
leaks in the logs. If password hashing is not needed and clearing the
|
|
||||||
password field is not desired (like when using this changeset for
|
|
||||||
validations on a LiveView form), this option can be set to `false`.
|
|
||||||
Defaults to `true`.
|
|
||||||
"""
|
|
||||||
def password_changeset(user, attrs, opts \\ []) do
|
|
||||||
user
|
|
||||||
|> cast(attrs, [:password])
|
|
||||||
|> validate_confirmation(:password, message: "does not match password")
|
|
||||||
|> validate_password(opts)
|
|
||||||
end
|
|
||||||
|
|
||||||
@doc """
|
|
||||||
Confirms the account by setting `confirmed_at`.
|
|
||||||
"""
|
|
||||||
def confirm_changeset(user) do
|
|
||||||
now = NaiveDateTime.utc_now() |> NaiveDateTime.truncate(:second)
|
|
||||||
change(user, confirmed_at: now)
|
|
||||||
end
|
|
||||||
|
|
||||||
@doc """
|
|
||||||
Verifies the password.
|
|
||||||
|
|
||||||
If there is no user or the user doesn't have a password, we call
|
|
||||||
`Bcrypt.no_user_verify/0` to avoid timing attacks.
|
|
||||||
"""
|
|
||||||
def valid_password?(%Malarkey.Accounts.User{hashed_password: hashed_password}, password)
|
|
||||||
when is_binary(hashed_password) and byte_size(password) > 0 do
|
|
||||||
Bcrypt.verify_pass(password, hashed_password)
|
|
||||||
end
|
|
||||||
|
|
||||||
def valid_password?(_, _) do
|
|
||||||
Bcrypt.no_user_verify()
|
|
||||||
false
|
|
||||||
end
|
|
||||||
|
|
||||||
@doc """
|
|
||||||
Validates the current password otherwise adds an error to the changeset.
|
|
||||||
"""
|
|
||||||
def validate_current_password(changeset, password) do
|
|
||||||
if valid_password?(changeset.data, password) do
|
|
||||||
changeset
|
|
||||||
else
|
|
||||||
add_error(changeset, :current_password, "is not valid")
|
|
||||||
end
|
|
||||||
end
|
|
||||||
end
|
|
||||||
@@ -1,79 +0,0 @@
|
|||||||
defmodule Malarkey.Accounts.UserNotifier do
|
|
||||||
import Swoosh.Email
|
|
||||||
|
|
||||||
alias Malarkey.Mailer
|
|
||||||
|
|
||||||
# Delivers the email using the application mailer.
|
|
||||||
defp deliver(recipient, subject, body) do
|
|
||||||
email =
|
|
||||||
new()
|
|
||||||
|> to(recipient)
|
|
||||||
|> from({"Malarkey", "contact@example.com"})
|
|
||||||
|> subject(subject)
|
|
||||||
|> text_body(body)
|
|
||||||
|
|
||||||
with {:ok, _metadata} <- Mailer.deliver(email) do
|
|
||||||
{:ok, email}
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
@doc """
|
|
||||||
Deliver instructions to confirm account.
|
|
||||||
"""
|
|
||||||
def deliver_confirmation_instructions(user, url) do
|
|
||||||
deliver(user.email, "Confirmation instructions", """
|
|
||||||
|
|
||||||
==============================
|
|
||||||
|
|
||||||
Hi #{user.email},
|
|
||||||
|
|
||||||
You can confirm your account by visiting the URL below:
|
|
||||||
|
|
||||||
#{url}
|
|
||||||
|
|
||||||
If you didn't create an account with us, please ignore this.
|
|
||||||
|
|
||||||
==============================
|
|
||||||
""")
|
|
||||||
end
|
|
||||||
|
|
||||||
@doc """
|
|
||||||
Deliver instructions to reset a user password.
|
|
||||||
"""
|
|
||||||
def deliver_reset_password_instructions(user, url) do
|
|
||||||
deliver(user.email, "Reset password instructions", """
|
|
||||||
|
|
||||||
==============================
|
|
||||||
|
|
||||||
Hi #{user.email},
|
|
||||||
|
|
||||||
You can reset your password by visiting the URL below:
|
|
||||||
|
|
||||||
#{url}
|
|
||||||
|
|
||||||
If you didn't request this change, please ignore this.
|
|
||||||
|
|
||||||
==============================
|
|
||||||
""")
|
|
||||||
end
|
|
||||||
|
|
||||||
@doc """
|
|
||||||
Deliver instructions to update a user email.
|
|
||||||
"""
|
|
||||||
def deliver_update_email_instructions(user, url) do
|
|
||||||
deliver(user.email, "Update email instructions", """
|
|
||||||
|
|
||||||
==============================
|
|
||||||
|
|
||||||
Hi #{user.email},
|
|
||||||
|
|
||||||
You can change your email by visiting the URL below:
|
|
||||||
|
|
||||||
#{url}
|
|
||||||
|
|
||||||
If you didn't request this change, please ignore this.
|
|
||||||
|
|
||||||
==============================
|
|
||||||
""")
|
|
||||||
end
|
|
||||||
end
|
|
||||||
@@ -1,179 +0,0 @@
|
|||||||
defmodule Malarkey.Accounts.UserToken do
|
|
||||||
use Ecto.Schema
|
|
||||||
import Ecto.Query
|
|
||||||
alias Malarkey.Accounts.UserToken
|
|
||||||
|
|
||||||
@hash_algorithm :sha256
|
|
||||||
@rand_size 32
|
|
||||||
|
|
||||||
# It is very important to keep the reset password token expiry short,
|
|
||||||
# since someone with access to the email may take over the account.
|
|
||||||
@reset_password_validity_in_days 1
|
|
||||||
@confirm_validity_in_days 7
|
|
||||||
@change_email_validity_in_days 7
|
|
||||||
@session_validity_in_days 60
|
|
||||||
|
|
||||||
schema "users_tokens" do
|
|
||||||
field :token, :binary
|
|
||||||
field :context, :string
|
|
||||||
field :sent_to, :string
|
|
||||||
belongs_to :user, Malarkey.Accounts.User
|
|
||||||
|
|
||||||
timestamps(updated_at: false)
|
|
||||||
end
|
|
||||||
|
|
||||||
@doc """
|
|
||||||
Generates a token that will be stored in a signed place,
|
|
||||||
such as session or cookie. As they are signed, those
|
|
||||||
tokens do not need to be hashed.
|
|
||||||
|
|
||||||
The reason why we store session tokens in the database, even
|
|
||||||
though Phoenix already provides a session cookie, is because
|
|
||||||
Phoenix' default session cookies are not persisted, they are
|
|
||||||
simply signed and potentially encrypted. This means they are
|
|
||||||
valid indefinitely, unless you change the signing/encryption
|
|
||||||
salt.
|
|
||||||
|
|
||||||
Therefore, storing them allows individual user
|
|
||||||
sessions to be expired. The token system can also be extended
|
|
||||||
to store additional data, such as the device used for logging in.
|
|
||||||
You could then use this information to display all valid sessions
|
|
||||||
and devices in the UI and allow users to explicitly expire any
|
|
||||||
session they deem invalid.
|
|
||||||
"""
|
|
||||||
def build_session_token(user) do
|
|
||||||
token = :crypto.strong_rand_bytes(@rand_size)
|
|
||||||
{token, %UserToken{token: token, context: "session", user_id: user.id}}
|
|
||||||
end
|
|
||||||
|
|
||||||
@doc """
|
|
||||||
Checks if the token is valid and returns its underlying lookup query.
|
|
||||||
|
|
||||||
The query returns the user found by the token, if any.
|
|
||||||
|
|
||||||
The token is valid if it matches the value in the database and it has
|
|
||||||
not expired (after @session_validity_in_days).
|
|
||||||
"""
|
|
||||||
def verify_session_token_query(token) do
|
|
||||||
query =
|
|
||||||
from token in token_and_context_query(token, "session"),
|
|
||||||
join: user in assoc(token, :user),
|
|
||||||
where: token.inserted_at > ago(@session_validity_in_days, "day"),
|
|
||||||
select: user
|
|
||||||
|
|
||||||
{:ok, query}
|
|
||||||
end
|
|
||||||
|
|
||||||
@doc """
|
|
||||||
Builds a token and its hash to be delivered to the user's email.
|
|
||||||
|
|
||||||
The non-hashed token is sent to the user email while the
|
|
||||||
hashed part is stored in the database. The original token cannot be reconstructed,
|
|
||||||
which means anyone with read-only access to the database cannot directly use
|
|
||||||
the token in the application to gain access. Furthermore, if the user changes
|
|
||||||
their email in the system, the tokens sent to the previous email are no longer
|
|
||||||
valid.
|
|
||||||
|
|
||||||
Users can easily adapt the existing code to provide other types of delivery methods,
|
|
||||||
for example, by phone numbers.
|
|
||||||
"""
|
|
||||||
def build_email_token(user, context) do
|
|
||||||
build_hashed_token(user, context, user.email)
|
|
||||||
end
|
|
||||||
|
|
||||||
defp build_hashed_token(user, context, sent_to) do
|
|
||||||
token = :crypto.strong_rand_bytes(@rand_size)
|
|
||||||
hashed_token = :crypto.hash(@hash_algorithm, token)
|
|
||||||
|
|
||||||
{Base.url_encode64(token, padding: false),
|
|
||||||
%UserToken{
|
|
||||||
token: hashed_token,
|
|
||||||
context: context,
|
|
||||||
sent_to: sent_to,
|
|
||||||
user_id: user.id
|
|
||||||
}}
|
|
||||||
end
|
|
||||||
|
|
||||||
@doc """
|
|
||||||
Checks if the token is valid and returns its underlying lookup query.
|
|
||||||
|
|
||||||
The query returns the user found by the token, if any.
|
|
||||||
|
|
||||||
The given token is valid if it matches its hashed counterpart in the
|
|
||||||
database and the user email has not changed. This function also checks
|
|
||||||
if the token is being used within a certain period, depending on the
|
|
||||||
context. The default contexts supported by this function are either
|
|
||||||
"confirm", for account confirmation emails, and "reset_password",
|
|
||||||
for resetting the password. For verifying requests to change the email,
|
|
||||||
see `verify_change_email_token_query/2`.
|
|
||||||
"""
|
|
||||||
def verify_email_token_query(token, context) do
|
|
||||||
case Base.url_decode64(token, padding: false) do
|
|
||||||
{:ok, decoded_token} ->
|
|
||||||
hashed_token = :crypto.hash(@hash_algorithm, decoded_token)
|
|
||||||
days = days_for_context(context)
|
|
||||||
|
|
||||||
query =
|
|
||||||
from token in token_and_context_query(hashed_token, context),
|
|
||||||
join: user in assoc(token, :user),
|
|
||||||
where: token.inserted_at > ago(^days, "day") and token.sent_to == user.email,
|
|
||||||
select: user
|
|
||||||
|
|
||||||
{:ok, query}
|
|
||||||
|
|
||||||
:error ->
|
|
||||||
:error
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
defp days_for_context("confirm"), do: @confirm_validity_in_days
|
|
||||||
defp days_for_context("reset_password"), do: @reset_password_validity_in_days
|
|
||||||
|
|
||||||
@doc """
|
|
||||||
Checks if the token is valid and returns its underlying lookup query.
|
|
||||||
|
|
||||||
The query returns the user found by the token, if any.
|
|
||||||
|
|
||||||
This is used to validate requests to change the user
|
|
||||||
email. It is different from `verify_email_token_query/2` precisely because
|
|
||||||
`verify_email_token_query/2` validates the email has not changed, which is
|
|
||||||
the starting point by this function.
|
|
||||||
|
|
||||||
The given token is valid if it matches its hashed counterpart in the
|
|
||||||
database and if it has not expired (after @change_email_validity_in_days).
|
|
||||||
The context must always start with "change:".
|
|
||||||
"""
|
|
||||||
def verify_change_email_token_query(token, "change:" <> _ = context) do
|
|
||||||
case Base.url_decode64(token, padding: false) do
|
|
||||||
{:ok, decoded_token} ->
|
|
||||||
hashed_token = :crypto.hash(@hash_algorithm, decoded_token)
|
|
||||||
|
|
||||||
query =
|
|
||||||
from token in token_and_context_query(hashed_token, context),
|
|
||||||
where: token.inserted_at > ago(@change_email_validity_in_days, "day")
|
|
||||||
|
|
||||||
{:ok, query}
|
|
||||||
|
|
||||||
:error ->
|
|
||||||
:error
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
@doc """
|
|
||||||
Returns the token struct for the given token value and context.
|
|
||||||
"""
|
|
||||||
def token_and_context_query(token, context) do
|
|
||||||
from UserToken, where: [token: ^token, context: ^context]
|
|
||||||
end
|
|
||||||
|
|
||||||
@doc """
|
|
||||||
Gets all tokens for the given user for the given contexts.
|
|
||||||
"""
|
|
||||||
def user_and_contexts_query(user, :all) do
|
|
||||||
from t in UserToken, where: t.user_id == ^user.id
|
|
||||||
end
|
|
||||||
|
|
||||||
def user_and_contexts_query(user, [_ | _] = contexts) do
|
|
||||||
from t in UserToken, where: t.user_id == ^user.id and t.context in ^contexts
|
|
||||||
end
|
|
||||||
end
|
|
||||||
@@ -1,38 +0,0 @@
|
|||||||
defmodule Malarkey.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
|
|
||||||
MalarkeyWeb.Telemetry,
|
|
||||||
# Start the Ecto repository
|
|
||||||
Malarkey.Repo,
|
|
||||||
# Start the PubSub system
|
|
||||||
{Phoenix.PubSub, name: Malarkey.PubSub},
|
|
||||||
# Start Finch
|
|
||||||
{Finch, name: Malarkey.Finch},
|
|
||||||
# Start the Endpoint (http/https)
|
|
||||||
MalarkeyWeb.Endpoint
|
|
||||||
# Start a worker by calling: Malarkey.Worker.start_link(arg)
|
|
||||||
# {Malarkey.Worker, arg}
|
|
||||||
]
|
|
||||||
|
|
||||||
# See https://hexdocs.pm/elixir/Supervisor.html
|
|
||||||
# for other strategies and supported options
|
|
||||||
opts = [strategy: :one_for_one, name: Malarkey.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
|
|
||||||
MalarkeyWeb.Endpoint.config_change(changed, removed)
|
|
||||||
:ok
|
|
||||||
end
|
|
||||||
end
|
|
||||||
@@ -1,3 +0,0 @@
|
|||||||
defmodule Malarkey.Mailer do
|
|
||||||
use Swoosh.Mailer, otp_app: :malarkey
|
|
||||||
end
|
|
||||||
@@ -1,28 +0,0 @@
|
|||||||
defmodule Malarkey.Release do
|
|
||||||
@moduledoc """
|
|
||||||
Used for executing DB release tasks when run in production without Mix
|
|
||||||
installed.
|
|
||||||
"""
|
|
||||||
@app :malarkey
|
|
||||||
|
|
||||||
def migrate do
|
|
||||||
load_app()
|
|
||||||
|
|
||||||
for repo <- repos() do
|
|
||||||
{:ok, _, _} = Ecto.Migrator.with_repo(repo, &Ecto.Migrator.run(&1, :up, all: true))
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
def rollback(repo, version) do
|
|
||||||
load_app()
|
|
||||||
{:ok, _, _} = Ecto.Migrator.with_repo(repo, &Ecto.Migrator.run(&1, :down, to: version))
|
|
||||||
end
|
|
||||||
|
|
||||||
defp repos do
|
|
||||||
Application.fetch_env!(@app, :ecto_repos)
|
|
||||||
end
|
|
||||||
|
|
||||||
defp load_app do
|
|
||||||
Application.load(@app)
|
|
||||||
end
|
|
||||||
end
|
|
||||||
@@ -1,5 +0,0 @@
|
|||||||
defmodule Malarkey.Repo do
|
|
||||||
use Ecto.Repo,
|
|
||||||
otp_app: :malarkey,
|
|
||||||
adapter: Ecto.Adapters.Postgres
|
|
||||||
end
|
|
||||||
@@ -1,87 +0,0 @@
|
|||||||
defmodule Malarkey.Timeline do
|
|
||||||
@moduledoc """
|
|
||||||
The Timeline context.
|
|
||||||
"""
|
|
||||||
|
|
||||||
import Ecto.Query, warn: false
|
|
||||||
|
|
||||||
alias Malarkey.Timeline.PostUserLike
|
|
||||||
alias Malarkey.Repo
|
|
||||||
alias Malarkey.Timeline.Post
|
|
||||||
|
|
||||||
def list_posts() do
|
|
||||||
# preloads = Keyword.get(opts, :preloads, [])
|
|
||||||
|
|
||||||
Post
|
|
||||||
|> order_by(desc: :inserted_at)
|
|
||||||
|> Repo.all()
|
|
||||||
|> Repo.preload(:user)
|
|
||||||
|> Repo.preload(:liked_by)
|
|
||||||
|> Repo.preload(:disliked_by)
|
|
||||||
|> Repo.preload(:reposted_by)
|
|
||||||
end
|
|
||||||
|
|
||||||
def get_post!(id) do
|
|
||||||
Repo.get!(Post, id)
|
|
||||||
|> Repo.preload(:user)
|
|
||||||
|> Repo.preload(:liked_by)
|
|
||||||
|> Repo.preload(:disliked_by)
|
|
||||||
|> Repo.preload(:reposted_by)
|
|
||||||
end
|
|
||||||
|
|
||||||
def create_post(user, attrs \\ %{}) do
|
|
||||||
%Post{user_id: user.id}
|
|
||||||
|> Post.changeset(attrs)
|
|
||||||
|> Repo.insert()
|
|
||||||
|> broadcast(:post_created)
|
|
||||||
end
|
|
||||||
|
|
||||||
def update_post(%Post{} = post, attrs \\ %{}) do
|
|
||||||
post
|
|
||||||
|> Post.changeset(attrs)
|
|
||||||
|> Repo.update()
|
|
||||||
|> broadcast(:post_updated)
|
|
||||||
end
|
|
||||||
|
|
||||||
def delete_post(%Post{} = post) do
|
|
||||||
Repo.delete(post)
|
|
||||||
end
|
|
||||||
|
|
||||||
def change_post(%Post{} = post, attrs \\ %{}) do
|
|
||||||
Post.changeset(post, attrs)
|
|
||||||
end
|
|
||||||
|
|
||||||
def add_like(user, post) do
|
|
||||||
if user in post.liked_by do
|
|
||||||
Repo.delete_all(PostUserLike.user_post_like_query(user, post))
|
|
||||||
|> broadcast(:post_updated)
|
|
||||||
else
|
|
||||||
post
|
|
||||||
|> Repo.preload(:liked_by)
|
|
||||||
|> Repo.preload(:user)
|
|
||||||
|> Post.changeset_add_like(user)
|
|
||||||
|> Repo.update()
|
|
||||||
|> broadcast(:post_updated)
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
def add_repost(user, post) do
|
|
||||||
post
|
|
||||||
|> Repo.preload(:reposted_by)
|
|
||||||
|> Repo.preload(:user)
|
|
||||||
|> Post.changeset_add_repost(user)
|
|
||||||
|> Repo.update()
|
|
||||||
|> broadcast(:post_created)
|
|
||||||
end
|
|
||||||
|
|
||||||
def subscribe do
|
|
||||||
Phoenix.PubSub.subscribe(Malarkey.PubSub, "posts")
|
|
||||||
end
|
|
||||||
|
|
||||||
defp broadcast({:error, _reason} = error, _event), do: error
|
|
||||||
|
|
||||||
defp broadcast({:ok, post}, event) do
|
|
||||||
Phoenix.PubSub.broadcast!(Malarkey.PubSub, "posts", {event, post})
|
|
||||||
{:ok, post}
|
|
||||||
end
|
|
||||||
end
|
|
||||||
@@ -1,63 +0,0 @@
|
|||||||
defmodule Malarkey.Timeline.Post do
|
|
||||||
use Ecto.Schema
|
|
||||||
import Ecto.Changeset
|
|
||||||
alias Malarkey.Accounts.User
|
|
||||||
|
|
||||||
schema "posts" do
|
|
||||||
field :body, :string
|
|
||||||
belongs_to(:user, User)
|
|
||||||
|
|
||||||
many_to_many(
|
|
||||||
:liked_by,
|
|
||||||
User,
|
|
||||||
join_through: Malarkey.Timeline.PostUserLike,
|
|
||||||
on_replace: :delete
|
|
||||||
)
|
|
||||||
|
|
||||||
many_to_many(
|
|
||||||
:disliked_by,
|
|
||||||
User,
|
|
||||||
join_through: Malarkey.Timeline.PostUserDislike,
|
|
||||||
on_replace: :delete
|
|
||||||
)
|
|
||||||
|
|
||||||
many_to_many(
|
|
||||||
:reposted_by,
|
|
||||||
User,
|
|
||||||
join_through: Malarkey.Timeline.PostUserRepost,
|
|
||||||
on_replace: :delete
|
|
||||||
)
|
|
||||||
|
|
||||||
timestamps()
|
|
||||||
end
|
|
||||||
|
|
||||||
@spec changeset(
|
|
||||||
{map, map}
|
|
||||||
| %{
|
|
||||||
:__struct__ => atom | %{:__changeset__ => map, optional(any) => any},
|
|
||||||
optional(atom) => any
|
|
||||||
},
|
|
||||||
:invalid | %{optional(:__struct__) => none, optional(atom | binary) => any}
|
|
||||||
) :: Ecto.Changeset.t()
|
|
||||||
@doc false
|
|
||||||
def changeset(post, attrs) do
|
|
||||||
post
|
|
||||||
|> cast(attrs, [:body, :user_id])
|
|
||||||
|> cast_assoc(:liked_by, required: false)
|
|
||||||
|> validate_required([:body])
|
|
||||||
|> validate_required([:user_id])
|
|
||||||
|> validate_length(:body, min: 2, max: 250)
|
|
||||||
end
|
|
||||||
|
|
||||||
def changeset_add_like(post, user, attrs \\ %{}) do
|
|
||||||
post
|
|
||||||
|> changeset(attrs)
|
|
||||||
|> put_assoc(:liked_by, [user])
|
|
||||||
end
|
|
||||||
|
|
||||||
def changeset_add_repost(post, user, attrs \\ %{}) do
|
|
||||||
post
|
|
||||||
|> changeset(attrs)
|
|
||||||
|> put_assoc(:reposted_by, [user])
|
|
||||||
end
|
|
||||||
end
|
|
||||||
@@ -1,10 +0,0 @@
|
|||||||
defmodule Malarkey.Timeline.PostUserDislike do
|
|
||||||
use Ecto.Schema
|
|
||||||
|
|
||||||
@primary_key false
|
|
||||||
schema "user_dislikes" do
|
|
||||||
belongs_to :user, Malarkey.Accounts.User, primary_key: true
|
|
||||||
belongs_to :post, Malarkey.Timeline.Post, primary_key: true
|
|
||||||
timestamps()
|
|
||||||
end
|
|
||||||
end
|
|
||||||
@@ -1,16 +0,0 @@
|
|||||||
defmodule Malarkey.Timeline.PostUserLike do
|
|
||||||
import Ecto.Query
|
|
||||||
use Ecto.Schema
|
|
||||||
alias Malarkey.Timeline.PostUserLike
|
|
||||||
|
|
||||||
@primary_key false
|
|
||||||
schema "user_likes" do
|
|
||||||
belongs_to :user, Malarkey.Accounts.User, primary_key: true
|
|
||||||
belongs_to :post, Malarkey.Timeline.Post, primary_key: true
|
|
||||||
timestamps()
|
|
||||||
end
|
|
||||||
|
|
||||||
def user_post_like_query(user, post) do
|
|
||||||
from PostUserLike, where: [user_id: ^user.id, post_id: ^post.id]
|
|
||||||
end
|
|
||||||
end
|
|
||||||
@@ -1,10 +0,0 @@
|
|||||||
defmodule Malarkey.Timeline.PostUserRepost do
|
|
||||||
use Ecto.Schema
|
|
||||||
|
|
||||||
@primary_key false
|
|
||||||
schema "user_reposts" do
|
|
||||||
belongs_to :user, Malarkey.Accounts.User, primary_key: true
|
|
||||||
belongs_to :post, Malarkey.Timeline.Post, primary_key: true
|
|
||||||
timestamps()
|
|
||||||
end
|
|
||||||
end
|
|
||||||
@@ -1,117 +0,0 @@
|
|||||||
defmodule MalarkeyWeb do
|
|
||||||
@moduledoc """
|
|
||||||
The entrypoint for defining your web interface, such
|
|
||||||
as controllers, components, channels, and so on.
|
|
||||||
|
|
||||||
This can be used in your application as:
|
|
||||||
|
|
||||||
use MalarkeyWeb, :controller
|
|
||||||
use MalarkeyWeb, :html
|
|
||||||
|
|
||||||
The definitions below will be executed for every controller,
|
|
||||||
component, etc, so keep them short and clean, focused
|
|
||||||
on imports, uses and aliases.
|
|
||||||
|
|
||||||
Do NOT define functions inside the quoted expressions
|
|
||||||
below. Instead, define additional modules and import
|
|
||||||
those modules here.
|
|
||||||
"""
|
|
||||||
|
|
||||||
def static_paths, do: ~w(assets fonts images favicon.ico robots.txt)
|
|
||||||
|
|
||||||
def router do
|
|
||||||
quote do
|
|
||||||
use Phoenix.Router, helpers: false
|
|
||||||
|
|
||||||
# Import common connection and controller functions to use in pipelines
|
|
||||||
import Plug.Conn
|
|
||||||
import Phoenix.Controller
|
|
||||||
import Phoenix.LiveView.Router
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
def channel do
|
|
||||||
quote do
|
|
||||||
use Phoenix.Channel
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
def controller do
|
|
||||||
quote do
|
|
||||||
use Phoenix.Controller,
|
|
||||||
namespace: MalarkeyWeb,
|
|
||||||
formats: [:html, :json],
|
|
||||||
layouts: [html: MalarkeyWeb.Layouts]
|
|
||||||
|
|
||||||
import Plug.Conn
|
|
||||||
import MalarkeyWeb.Gettext
|
|
||||||
|
|
||||||
unquote(verified_routes())
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
def live_view do
|
|
||||||
quote do
|
|
||||||
use Phoenix.LiveView,
|
|
||||||
layout: {MalarkeyWeb.Layouts, :app}
|
|
||||||
|
|
||||||
unquote(html_helpers())
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
def live_component do
|
|
||||||
quote do
|
|
||||||
use Phoenix.LiveComponent
|
|
||||||
|
|
||||||
unquote(html_helpers())
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
@spec html ::
|
|
||||||
{:__block__, [],
|
|
||||||
[{:__block__, [], [...]} | {:import, [...], [...]} | {:use, [...], [...]}, ...]}
|
|
||||||
def html do
|
|
||||||
quote do
|
|
||||||
use Phoenix.Component
|
|
||||||
|
|
||||||
# Import convenience functions from controllers
|
|
||||||
import Phoenix.Controller,
|
|
||||||
only: [get_csrf_token: 0, view_module: 1, view_template: 1]
|
|
||||||
|
|
||||||
# Include general helpers for rendering HTML
|
|
||||||
unquote(html_helpers())
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
defp html_helpers do
|
|
||||||
quote do
|
|
||||||
# HTML escaping functionality
|
|
||||||
import Phoenix.HTML
|
|
||||||
# Core UI components and translation
|
|
||||||
import MalarkeyWeb.CoreComponents
|
|
||||||
import MalarkeyWeb.Gettext
|
|
||||||
|
|
||||||
# Shortcut for generating JS commands
|
|
||||||
alias Phoenix.LiveView.JS
|
|
||||||
|
|
||||||
# Routes generation with the ~p sigil
|
|
||||||
unquote(verified_routes())
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
def verified_routes do
|
|
||||||
quote do
|
|
||||||
use Phoenix.VerifiedRoutes,
|
|
||||||
endpoint: MalarkeyWeb.Endpoint,
|
|
||||||
router: MalarkeyWeb.Router,
|
|
||||||
statics: MalarkeyWeb.static_paths()
|
|
||||||
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
|
|
||||||
@@ -1,636 +0,0 @@
|
|||||||
defmodule MalarkeyWeb.CoreComponents do
|
|
||||||
@moduledoc """
|
|
||||||
Provides core UI components.
|
|
||||||
|
|
||||||
The components in this module use Tailwind CSS, a utility-first CSS framework.
|
|
||||||
See the [Tailwind CSS documentation](https://tailwindcss.com) to learn how to
|
|
||||||
customize the generated components in this module.
|
|
||||||
|
|
||||||
Icons are provided by [heroicons](https://heroicons.com), using the
|
|
||||||
[heroicons_elixir](https://github.com/mveytsman/heroicons_elixir) project.
|
|
||||||
"""
|
|
||||||
use Phoenix.Component
|
|
||||||
|
|
||||||
alias Phoenix.LiveView.JS
|
|
||||||
import MalarkeyWeb.Gettext
|
|
||||||
|
|
||||||
@doc """
|
|
||||||
Renders a modal.
|
|
||||||
|
|
||||||
## Examples
|
|
||||||
|
|
||||||
<.modal id="confirm-modal">
|
|
||||||
Are you sure?
|
|
||||||
<:confirm>OK</:confirm>
|
|
||||||
<:cancel>Cancel</:cancel>
|
|
||||||
</.modal>
|
|
||||||
|
|
||||||
JS commands may be passed to the `:on_cancel` and `on_confirm` attributes
|
|
||||||
for the caller to react to each button press, for example:
|
|
||||||
|
|
||||||
<.modal id="confirm" on_confirm={JS.push("delete")} on_cancel={JS.navigate(~p"/posts")}>
|
|
||||||
Are you sure you?
|
|
||||||
<:confirm>OK</:confirm>
|
|
||||||
<:cancel>Cancel</:cancel>
|
|
||||||
</.modal>
|
|
||||||
"""
|
|
||||||
attr :id, :string, required: true
|
|
||||||
attr :show, :boolean, default: false
|
|
||||||
attr :on_cancel, JS, default: %JS{}
|
|
||||||
attr :on_confirm, JS, default: %JS{}
|
|
||||||
|
|
||||||
slot :inner_block, required: true
|
|
||||||
slot :title
|
|
||||||
slot :subtitle
|
|
||||||
slot :confirm
|
|
||||||
slot :cancel
|
|
||||||
|
|
||||||
def modal(assigns) do
|
|
||||||
~H"""
|
|
||||||
<div id={@id} phx-mounted={@show && show_modal(@id)} class="relative z-50 hidden">
|
|
||||||
<div id={"#{@id}-bg"} class="fixed inset-0 transition-opacity bg-zinc-50/90" aria-hidden="true" />
|
|
||||||
<div
|
|
||||||
class="fixed inset-0 overflow-y-auto"
|
|
||||||
aria-labelledby={"#{@id}-title"}
|
|
||||||
aria-describedby={"#{@id}-description"}
|
|
||||||
role="dialog"
|
|
||||||
aria-modal="true"
|
|
||||||
tabindex="0"
|
|
||||||
>
|
|
||||||
<div class="flex items-center justify-center min-h-full">
|
|
||||||
<div class="w-full max-w-3xl p-4 sm:p-6 lg:py-8">
|
|
||||||
<.focus_wrap
|
|
||||||
id={"#{@id}-container"}
|
|
||||||
phx-mounted={@show && show_modal(@id)}
|
|
||||||
phx-window-keydown={hide_modal(@on_cancel, @id)}
|
|
||||||
phx-key="escape"
|
|
||||||
phx-click-away={hide_modal(@on_cancel, @id)}
|
|
||||||
class="relative hidden transition bg-white shadow-lg rounded-2xl p-14 shadow-zinc-700/10 ring-1 ring-zinc-700/10"
|
|
||||||
>
|
|
||||||
<div class="absolute top-6 right-5">
|
|
||||||
<button
|
|
||||||
phx-click={hide_modal(@on_cancel, @id)}
|
|
||||||
type="button"
|
|
||||||
class="flex-none p-3 -m-3 opacity-20 hover:opacity-40"
|
|
||||||
aria-label={gettext("close")}
|
|
||||||
>
|
|
||||||
<%= Heroicons.icon("x-mark", type: "solid", class: "w-5 h-5 stroke-current") %>
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
<div id={"#{@id}-content"}>
|
|
||||||
<header :if={@title != []}>
|
|
||||||
<h1 id={"#{@id}-title"} class="text-lg font-semibold leading-8 text-zinc-800">
|
|
||||||
<%= render_slot(@title) %>
|
|
||||||
</h1>
|
|
||||||
<p :if={@subtitle != []} class="mt-2 text-sm leading-6 text-zinc-600">
|
|
||||||
<%= render_slot(@subtitle) %>
|
|
||||||
</p>
|
|
||||||
</header>
|
|
||||||
<%= render_slot(@inner_block) %>
|
|
||||||
<div :if={@confirm != [] or @cancel != []} class="flex items-center gap-5 mb-4 ml-6">
|
|
||||||
<.button
|
|
||||||
:for={confirm <- @confirm}
|
|
||||||
id={"#{@id}-confirm"}
|
|
||||||
phx-click={@on_confirm}
|
|
||||||
phx-disable-with
|
|
||||||
class="px-3 py-2"
|
|
||||||
>
|
|
||||||
<%= render_slot(confirm) %>
|
|
||||||
</.button>
|
|
||||||
<.link
|
|
||||||
:for={cancel <- @cancel}
|
|
||||||
phx-click={hide_modal(@on_cancel, @id)}
|
|
||||||
class="text-sm font-semibold leading-6 text-zinc-900 hover:text-zinc-700"
|
|
||||||
>
|
|
||||||
<%= render_slot(cancel) %>
|
|
||||||
</.link>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</.focus_wrap>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
"""
|
|
||||||
end
|
|
||||||
|
|
||||||
@doc """
|
|
||||||
Renders flash notices.
|
|
||||||
|
|
||||||
## Examples
|
|
||||||
|
|
||||||
<.flash kind={:info} flash={@flash} />
|
|
||||||
<.flash kind={:info} phx-mounted={show("#flash")}>Welcome Back!</.flash>
|
|
||||||
"""
|
|
||||||
attr :id, :string, default: "flash", doc: "the optional id of flash container"
|
|
||||||
attr :flash, :map, default: %{}, doc: "the map of flash messages to display"
|
|
||||||
attr :title, :string, default: nil
|
|
||||||
attr :kind, :atom, values: [:info, :error], doc: "used for styling and flash lookup"
|
|
||||||
attr :autoshow, :boolean, default: true, doc: "whether to auto show the flash on mount"
|
|
||||||
attr :close, :boolean, default: true, doc: "whether the flash can be closed"
|
|
||||||
attr :rest, :global, doc: "the arbitrary HTML attributes to add to the flash container"
|
|
||||||
|
|
||||||
slot :inner_block, doc: "the optional inner block that renders the flash message"
|
|
||||||
|
|
||||||
def flash(assigns) do
|
|
||||||
~H"""
|
|
||||||
<div
|
|
||||||
:if={msg = render_slot(@inner_block) || Phoenix.Flash.get(@flash, @kind)}
|
|
||||||
id={@id}
|
|
||||||
phx-mounted={@autoshow && show("##{@id}")}
|
|
||||||
phx-click={JS.push("lv:clear-flash", value: %{key: @kind}) |> hide("#flash")}
|
|
||||||
role="alert"
|
|
||||||
class={[
|
|
||||||
"fixed hidden top-2 right-2 w-80 sm:w-96 z-50 rounded-lg p-3 shadow-md shadow-zinc-900/5 ring-1",
|
|
||||||
@kind == :info && "bg-emerald-50 text-emerald-800 ring-emerald-500 fill-cyan-900",
|
|
||||||
@kind == :error && "bg-rose-50 p-3 text-rose-900 shadow-md ring-rose-500 fill-rose-900"
|
|
||||||
]}
|
|
||||||
{@rest}
|
|
||||||
>
|
|
||||||
<p :if={@title} class="flex items-center gap-1.5 text-[0.8125rem] font-semibold leading-6">
|
|
||||||
<%= if @kind == :info do %>
|
|
||||||
<%= Heroicons.icon("information-circle", type: "mini", class: "w-4 h-4") %>
|
|
||||||
<% end %>
|
|
||||||
<%= if @kind == :error do %>
|
|
||||||
<%= Heroicons.icon("exclamation-circle", type: "mini", class: "w-4 h-4") %>
|
|
||||||
<% end %>
|
|
||||||
<%= @title %>
|
|
||||||
</p>
|
|
||||||
<p class="mt-2 text-[0.8125rem] leading-5"><%= msg %></p>
|
|
||||||
<button
|
|
||||||
:if={@close}
|
|
||||||
type="button"
|
|
||||||
class="absolute p-2 group top-2 right-1"
|
|
||||||
aria-label={gettext("close")}
|
|
||||||
>
|
|
||||||
<%= Heroicons.icon("x-mark",
|
|
||||||
type: "solid",
|
|
||||||
class: "w-5 h-5 stroke-current opacity-40 group-hover:opacity-70"
|
|
||||||
) %>
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
"""
|
|
||||||
end
|
|
||||||
|
|
||||||
@doc """
|
|
||||||
Renders a simple form.
|
|
||||||
|
|
||||||
## Examples
|
|
||||||
|
|
||||||
<.simple_form :let={f} for={:user} phx-change="validate" phx-submit="save">
|
|
||||||
<.input field={{f, :email}} label="Email"/>
|
|
||||||
<.input field={{f, :username}} label="Username" />
|
|
||||||
<:actions>
|
|
||||||
<.button>Save</.button>
|
|
||||||
</:actions>
|
|
||||||
</.simple_form>
|
|
||||||
"""
|
|
||||||
attr :for, :any, default: nil, doc: "the datastructure for the form"
|
|
||||||
attr :as, :any, default: nil, doc: "the server side parameter to collect all input under"
|
|
||||||
|
|
||||||
attr :rest, :global,
|
|
||||||
include: ~w(autocomplete name rel action enctype method novalidate target),
|
|
||||||
doc: "the arbitrary HTML attributes to apply to the form tag"
|
|
||||||
|
|
||||||
slot :inner_block, required: true
|
|
||||||
slot :actions, doc: "the slot for form actions, such as a submit button"
|
|
||||||
|
|
||||||
def simple_form(assigns) do
|
|
||||||
~H"""
|
|
||||||
<.form :let={f} for={@for} as={@as} {@rest}>
|
|
||||||
<div class="mt-10 space-y-8 bg-white">
|
|
||||||
<%= render_slot(@inner_block, f) %>
|
|
||||||
<div :for={action <- @actions} class="flex items-center justify-between gap-6 mt-2">
|
|
||||||
<%= render_slot(action, f) %>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</.form>
|
|
||||||
"""
|
|
||||||
end
|
|
||||||
|
|
||||||
@doc """
|
|
||||||
Renders a button.
|
|
||||||
|
|
||||||
## Examples
|
|
||||||
|
|
||||||
<.button>Send!</.button>
|
|
||||||
<.button phx-click="go" class="ml-2">Send!</.button>
|
|
||||||
"""
|
|
||||||
attr :type, :string, default: nil
|
|
||||||
attr :class, :string, default: nil
|
|
||||||
attr :rest, :global, include: ~w(disabled form name value)
|
|
||||||
|
|
||||||
slot :inner_block, required: true
|
|
||||||
|
|
||||||
def button(assigns) do
|
|
||||||
~H"""
|
|
||||||
<button
|
|
||||||
type={@type}
|
|
||||||
class={[
|
|
||||||
"phx-submit-loading:opacity-75 rounded-lg bg-zinc-900 hover:bg-zinc-700 py-2 px-3",
|
|
||||||
"text-sm font-semibold leading-6 text-white active:text-white/80",
|
|
||||||
@class
|
|
||||||
]}
|
|
||||||
{@rest}
|
|
||||||
>
|
|
||||||
<%= render_slot(@inner_block) %>
|
|
||||||
</button>
|
|
||||||
"""
|
|
||||||
end
|
|
||||||
|
|
||||||
@doc """
|
|
||||||
Renders an input with label and error messages.
|
|
||||||
|
|
||||||
A `%Phoenix.HTML.Form{}` and field name may be passed to the input
|
|
||||||
to build input names and error messages, or all the attributes and
|
|
||||||
errors may be passed explicitly.
|
|
||||||
|
|
||||||
## Examples
|
|
||||||
|
|
||||||
<.input field={{f, :email}} type="email" />
|
|
||||||
<.input name="my-input" errors={["oh no!"]} />
|
|
||||||
"""
|
|
||||||
attr :id, :any
|
|
||||||
attr :name, :any
|
|
||||||
attr :label, :string, default: nil
|
|
||||||
|
|
||||||
attr :type, :string,
|
|
||||||
default: "text",
|
|
||||||
values: ~w(checkbox color date datetime-local email file hidden month number password
|
|
||||||
range radio search select tel text textarea time url week)
|
|
||||||
|
|
||||||
attr :value, :any
|
|
||||||
attr :field, :any, doc: "a %Phoenix.HTML.Form{}/field name tuple, for example: {f, :email}"
|
|
||||||
attr :errors, :list
|
|
||||||
attr :checked, :boolean, doc: "the checked flag for checkbox inputs"
|
|
||||||
attr :prompt, :string, default: nil, doc: "the prompt for select inputs"
|
|
||||||
attr :options, :list, doc: "the options to pass to Phoenix.HTML.Form.options_for_select/2"
|
|
||||||
attr :multiple, :boolean, default: false, doc: "the multiple flag for select inputs"
|
|
||||||
attr :rest, :global, include: ~w(autocomplete disabled form max maxlength min minlength
|
|
||||||
pattern placeholder readonly required size step)
|
|
||||||
slot :inner_block
|
|
||||||
|
|
||||||
def input(%{field: {f, field}} = assigns) do
|
|
||||||
assigns
|
|
||||||
|> assign(field: nil)
|
|
||||||
|> assign_new(:name, fn ->
|
|
||||||
name = Phoenix.HTML.Form.input_name(f, field)
|
|
||||||
if assigns.multiple, do: name <> "[]", else: name
|
|
||||||
end)
|
|
||||||
|> assign_new(:id, fn -> Phoenix.HTML.Form.input_id(f, field) end)
|
|
||||||
|> assign_new(:value, fn -> Phoenix.HTML.Form.input_value(f, field) end)
|
|
||||||
|> assign_new(:errors, fn -> translate_errors(f.errors || [], field) end)
|
|
||||||
|> input()
|
|
||||||
end
|
|
||||||
|
|
||||||
def input(%{type: "checkbox"} = assigns) do
|
|
||||||
assigns = assign_new(assigns, :checked, fn -> input_equals?(assigns.value, "true") end)
|
|
||||||
|
|
||||||
~H"""
|
|
||||||
<label phx-feedback-for={@name} class="flex items-center gap-4 text-sm leading-6 text-zinc-600">
|
|
||||||
<input type="hidden" name={@name} value="false" />
|
|
||||||
<input
|
|
||||||
type="checkbox"
|
|
||||||
id={@id || @name}
|
|
||||||
name={@name}
|
|
||||||
value="true"
|
|
||||||
checked={@checked}
|
|
||||||
class="rounded border-zinc-300 text-zinc-900 focus:ring-zinc-900"
|
|
||||||
{@rest}
|
|
||||||
/>
|
|
||||||
<%= @label %>
|
|
||||||
</label>
|
|
||||||
"""
|
|
||||||
end
|
|
||||||
|
|
||||||
def input(%{type: "select"} = assigns) do
|
|
||||||
~H"""
|
|
||||||
<div phx-feedback-for={@name}>
|
|
||||||
<.label for={@id}><%= @label %></.label>
|
|
||||||
<select
|
|
||||||
id={@id}
|
|
||||||
name={@name}
|
|
||||||
class="block w-full px-3 py-2 mt-1 bg-white border border-gray-300 rounded-md shadow-sm focus:outline-none focus:ring-zinc-500 focus:border-zinc-500 sm:text-sm"
|
|
||||||
multiple={@multiple}
|
|
||||||
{@rest}
|
|
||||||
>
|
|
||||||
<option :if={@prompt}><%= @prompt %></option>
|
|
||||||
<%= Phoenix.HTML.Form.options_for_select(@options, @value) %>
|
|
||||||
</select>
|
|
||||||
<.error :for={msg <- @errors}><%= msg %></.error>
|
|
||||||
</div>
|
|
||||||
"""
|
|
||||||
end
|
|
||||||
|
|
||||||
def input(%{type: "textarea"} = assigns) do
|
|
||||||
~H"""
|
|
||||||
<div phx-feedback-for={@name}>
|
|
||||||
<.label for={@id}><%= @label %></.label>
|
|
||||||
<textarea
|
|
||||||
id={@id || @name}
|
|
||||||
name={@name}
|
|
||||||
class={[
|
|
||||||
input_border(@errors),
|
|
||||||
"mt-2 block min-h-[6rem] w-full rounded-lg border-zinc-300 py-[7px] px-[11px]",
|
|
||||||
"text-zinc-900 focus:border-zinc-400 focus:outline-none focus:ring-4 focus:ring-zinc-800/5 sm:text-sm sm:leading-6",
|
|
||||||
"phx-no-feedback:border-zinc-300 phx-no-feedback:focus:border-zinc-400 phx-no-feedback:focus:ring-zinc-800/5"
|
|
||||||
]}
|
|
||||||
{@rest}
|
|
||||||
>
|
|
||||||
|
|
||||||
<%= @value %></textarea>
|
|
||||||
<.error :for={msg <- @errors}><%= msg %></.error>
|
|
||||||
</div>
|
|
||||||
"""
|
|
||||||
end
|
|
||||||
|
|
||||||
def input(assigns) do
|
|
||||||
~H"""
|
|
||||||
<div phx-feedback-for={@name}>
|
|
||||||
<.label for={@id}><%= @label %></.label>
|
|
||||||
<input
|
|
||||||
type={@type}
|
|
||||||
name={@name}
|
|
||||||
id={@id || @name}
|
|
||||||
value={@value}
|
|
||||||
class={[
|
|
||||||
input_border(@errors),
|
|
||||||
"mt-2 block w-full rounded-lg border-zinc-300 py-[7px] px-[11px]",
|
|
||||||
"text-zinc-900 focus:outline-none focus:ring-4 sm:text-sm sm:leading-6",
|
|
||||||
"phx-no-feedback:border-zinc-300 phx-no-feedback:focus:border-zinc-400 phx-no-feedback:focus:ring-zinc-800/5"
|
|
||||||
]}
|
|
||||||
{@rest}
|
|
||||||
/>
|
|
||||||
<.error :for={msg <- @errors}><%= msg %></.error>
|
|
||||||
</div>
|
|
||||||
"""
|
|
||||||
end
|
|
||||||
|
|
||||||
defp input_border([] = _errors),
|
|
||||||
do: "border-zinc-300 focus:border-zinc-400 focus:ring-zinc-800/5"
|
|
||||||
|
|
||||||
defp input_border([_ | _] = _errors),
|
|
||||||
do: "border-rose-400 focus:border-rose-400 focus:ring-rose-400/10"
|
|
||||||
|
|
||||||
@doc """
|
|
||||||
Renders a label.
|
|
||||||
"""
|
|
||||||
attr :for, :string, default: nil
|
|
||||||
slot :inner_block, required: true
|
|
||||||
|
|
||||||
def label(assigns) do
|
|
||||||
~H"""
|
|
||||||
<label for={@for} class="block text-sm font-semibold leading-6 text-zinc-800">
|
|
||||||
<%= render_slot(@inner_block) %>
|
|
||||||
</label>
|
|
||||||
"""
|
|
||||||
end
|
|
||||||
|
|
||||||
@doc """
|
|
||||||
Generates a generic error message.
|
|
||||||
"""
|
|
||||||
slot :inner_block, required: true
|
|
||||||
|
|
||||||
def error(assigns) do
|
|
||||||
~H"""
|
|
||||||
<p class="flex gap-3 mt-3 text-sm leading-6 phx-no-feedback:hidden text-rose-600">
|
|
||||||
<%= Heroicons.icon("exclamation-circle",
|
|
||||||
type: "mini",
|
|
||||||
class: "mt-0.5 h-5 w-5 flex-none fill-rose-500"
|
|
||||||
) %>
|
|
||||||
<%= render_slot(@inner_block) %>
|
|
||||||
</p>
|
|
||||||
"""
|
|
||||||
end
|
|
||||||
|
|
||||||
@doc """
|
|
||||||
Renders a header with title.
|
|
||||||
"""
|
|
||||||
attr :class, :string, default: nil
|
|
||||||
|
|
||||||
slot :inner_block, required: true
|
|
||||||
slot :subtitle
|
|
||||||
slot :actions
|
|
||||||
|
|
||||||
def header(assigns) do
|
|
||||||
~H"""
|
|
||||||
<header class={[@actions != [] && "flex items-center justify-between gap-6", @class]}>
|
|
||||||
<div>
|
|
||||||
<h1 class="text-lg font-semibold leading-8 text-zinc-800">
|
|
||||||
<%= render_slot(@inner_block) %>
|
|
||||||
</h1>
|
|
||||||
<p :if={@subtitle != []} class="mt-2 text-sm leading-6 text-zinc-600">
|
|
||||||
<%= render_slot(@subtitle) %>
|
|
||||||
</p>
|
|
||||||
</div>
|
|
||||||
<div class="flex-none"><%= render_slot(@actions) %></div>
|
|
||||||
</header>
|
|
||||||
"""
|
|
||||||
end
|
|
||||||
|
|
||||||
@doc ~S"""
|
|
||||||
Renders a table with generic styling.
|
|
||||||
|
|
||||||
## Examples
|
|
||||||
|
|
||||||
<.table id="users" rows={@users}>
|
|
||||||
<:col :let={user} label="id"><%= user.id %></:col>
|
|
||||||
<:col :let={user} label="username"><%= user.username %></:col>
|
|
||||||
</.table>
|
|
||||||
"""
|
|
||||||
attr :id, :string, required: true
|
|
||||||
attr :row_click, :any, default: nil
|
|
||||||
attr :rows, :list, required: true
|
|
||||||
|
|
||||||
slot :col, required: true do
|
|
||||||
attr :label, :string
|
|
||||||
end
|
|
||||||
|
|
||||||
slot :action, doc: "the slot for showing user actions in the last table column"
|
|
||||||
|
|
||||||
def table(assigns) do
|
|
||||||
~H"""
|
|
||||||
<div id={@id} class="px-4 overflow-y-auto sm:overflow-visible sm:px-0">
|
|
||||||
<table class="mt-11 w-[40rem] sm:w-full">
|
|
||||||
<thead class="text-left text-[0.8125rem] leading-6 text-zinc-500">
|
|
||||||
<tr>
|
|
||||||
<th :for={col <- @col} class="p-0 pb-4 pr-6 font-normal"><%= col[:label] %></th>
|
|
||||||
<th class="relative p-0 pb-4"><span class="sr-only"><%= gettext("Actions") %></span></th>
|
|
||||||
</tr>
|
|
||||||
</thead>
|
|
||||||
<tbody class="relative text-sm leading-6 border-t divide-y divide-zinc-100 border-zinc-200 text-zinc-700">
|
|
||||||
<tr
|
|
||||||
:for={row <- @rows}
|
|
||||||
id={"#{@id}-#{Phoenix.Param.to_param(row)}"}
|
|
||||||
class="relative group hover:bg-zinc-50"
|
|
||||||
>
|
|
||||||
<td
|
|
||||||
:for={{col, i} <- Enum.with_index(@col)}
|
|
||||||
phx-click={@row_click && @row_click.(row)}
|
|
||||||
class={["p-0", @row_click && "hover:cursor-pointer"]}
|
|
||||||
>
|
|
||||||
<div :if={i == 0}>
|
|
||||||
<span class="absolute top-0 w-4 h-full -left-4 group-hover:bg-zinc-50 sm:rounded-l-xl" />
|
|
||||||
<span class="absolute top-0 w-4 h-full -right-4 group-hover:bg-zinc-50 sm:rounded-r-xl" />
|
|
||||||
</div>
|
|
||||||
<div class="block py-4 pr-6">
|
|
||||||
<span class={["relative", i == 0 && "font-semibold text-zinc-900"]}>
|
|
||||||
<%= render_slot(col, row) %>
|
|
||||||
</span>
|
|
||||||
</div>
|
|
||||||
</td>
|
|
||||||
<td :if={@action != []} class="p-0 w-14">
|
|
||||||
<div class="relative py-4 text-sm font-medium text-right whitespace-nowrap">
|
|
||||||
<span
|
|
||||||
:for={action <- @action}
|
|
||||||
class="relative ml-4 font-semibold leading-6 text-zinc-900 hover:text-zinc-700"
|
|
||||||
>
|
|
||||||
<%= render_slot(action, row) %>
|
|
||||||
</span>
|
|
||||||
</div>
|
|
||||||
</td>
|
|
||||||
</tr>
|
|
||||||
</tbody>
|
|
||||||
</table>
|
|
||||||
</div>
|
|
||||||
"""
|
|
||||||
end
|
|
||||||
|
|
||||||
@doc """
|
|
||||||
Renders a data list.
|
|
||||||
|
|
||||||
## Examples
|
|
||||||
|
|
||||||
<.list>
|
|
||||||
<:item title="Title"><%= @post.title %></:item>
|
|
||||||
<:item title="Views"><%= @post.views %></:item>
|
|
||||||
</.list>
|
|
||||||
"""
|
|
||||||
slot :item, required: true do
|
|
||||||
attr :title, :string, required: true
|
|
||||||
end
|
|
||||||
|
|
||||||
def list(assigns) do
|
|
||||||
~H"""
|
|
||||||
<div class="mt-14">
|
|
||||||
<dl class="-my-4 divide-y divide-zinc-100">
|
|
||||||
<div :for={item <- @item} class="flex gap-4 py-4 sm:gap-8">
|
|
||||||
<dt class="w-1/4 flex-none text-[0.8125rem] leading-6 text-zinc-500"><%= item.title %></dt>
|
|
||||||
<dd class="text-sm leading-6 text-zinc-700"><%= render_slot(item) %></dd>
|
|
||||||
</div>
|
|
||||||
</dl>
|
|
||||||
</div>
|
|
||||||
"""
|
|
||||||
end
|
|
||||||
|
|
||||||
@doc """
|
|
||||||
Renders a back navigation link.
|
|
||||||
|
|
||||||
## Examples
|
|
||||||
|
|
||||||
<.back navigate={~p"/posts"}>Back to posts</.back>
|
|
||||||
"""
|
|
||||||
attr :navigate, :any, required: true
|
|
||||||
slot :inner_block, required: true
|
|
||||||
|
|
||||||
def back(assigns) do
|
|
||||||
~H"""
|
|
||||||
<div class="mt-16">
|
|
||||||
<.link
|
|
||||||
navigate={@navigate}
|
|
||||||
class="text-sm font-semibold leading-6 text-zinc-900 hover:text-zinc-700"
|
|
||||||
>
|
|
||||||
<%= Heroicons.icon("arrow-left",
|
|
||||||
type: "solid",
|
|
||||||
class: "inline w-3 h-3 stroke-current"
|
|
||||||
) %>
|
|
||||||
<%= render_slot(@inner_block) %>
|
|
||||||
</.link>
|
|
||||||
</div>
|
|
||||||
"""
|
|
||||||
end
|
|
||||||
|
|
||||||
## JS Commands
|
|
||||||
|
|
||||||
def show(js \\ %JS{}, selector) do
|
|
||||||
JS.show(js,
|
|
||||||
to: selector,
|
|
||||||
transition:
|
|
||||||
{"transition-all transform ease-out duration-300",
|
|
||||||
"opacity-0 translate-y-4 sm:translate-y-0 sm:scale-95",
|
|
||||||
"opacity-100 translate-y-0 sm:scale-100"}
|
|
||||||
)
|
|
||||||
end
|
|
||||||
|
|
||||||
def hide(js \\ %JS{}, selector) do
|
|
||||||
JS.hide(js,
|
|
||||||
to: selector,
|
|
||||||
time: 200,
|
|
||||||
transition:
|
|
||||||
{"transition-all transform ease-in duration-200",
|
|
||||||
"opacity-100 translate-y-0 sm:scale-100",
|
|
||||||
"opacity-0 translate-y-4 sm:translate-y-0 sm:scale-95"}
|
|
||||||
)
|
|
||||||
end
|
|
||||||
|
|
||||||
def show_modal(js \\ %JS{}, id) when is_binary(id) do
|
|
||||||
js
|
|
||||||
|> JS.show(to: "##{id}")
|
|
||||||
|> JS.show(
|
|
||||||
to: "##{id}-bg",
|
|
||||||
transition: {"transition-all transform ease-out duration-300", "opacity-0", "opacity-100"}
|
|
||||||
)
|
|
||||||
|> show("##{id}-container")
|
|
||||||
|> JS.focus_first(to: "##{id}-content")
|
|
||||||
end
|
|
||||||
|
|
||||||
def hide_modal(js \\ %JS{}, id) do
|
|
||||||
js
|
|
||||||
|> JS.hide(
|
|
||||||
to: "##{id}-bg",
|
|
||||||
transition: {"transition-all transform ease-in duration-200", "opacity-100", "opacity-0"}
|
|
||||||
)
|
|
||||||
|> hide("##{id}-container")
|
|
||||||
|> JS.hide(to: "##{id}", transition: {"block", "block", "hidden"})
|
|
||||||
|> JS.pop_focus()
|
|
||||||
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(MalarkeyWeb.Gettext, "errors", msg, msg, count, opts)
|
|
||||||
else
|
|
||||||
Gettext.dgettext(MalarkeyWeb.Gettext, "errors", msg, opts)
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
@doc """
|
|
||||||
Translates the errors for a field from a keyword list of errors.
|
|
||||||
"""
|
|
||||||
def translate_errors(errors, field) when is_list(errors) do
|
|
||||||
for {^field, {msg, opts}} <- errors, do: translate_error({msg, opts})
|
|
||||||
end
|
|
||||||
|
|
||||||
defp input_equals?(val1, val2) do
|
|
||||||
Phoenix.HTML.html_escape(val1) == Phoenix.HTML.html_escape(val2)
|
|
||||||
end
|
|
||||||
end
|
|
||||||
@@ -1,5 +0,0 @@
|
|||||||
defmodule MalarkeyWeb.Layouts do
|
|
||||||
use MalarkeyWeb, :html
|
|
||||||
|
|
||||||
embed_templates "layouts/*"
|
|
||||||
end
|
|
||||||
@@ -1,19 +0,0 @@
|
|||||||
|
|
||||||
<main class="px-4 py-4 sm:px-6 lg:px-8">
|
|
||||||
<div class="max-w-2xl mx-auto">
|
|
||||||
<.flash kind={:info} title="Success!" flash={@flash} />
|
|
||||||
<.flash kind={:error} title="Error!" flash={@flash} />
|
|
||||||
<.flash
|
|
||||||
id="disconnected"
|
|
||||||
kind={:error}
|
|
||||||
title="We can't find the internet"
|
|
||||||
close={false}
|
|
||||||
autoshow={false}
|
|
||||||
phx-disconnected={show("#disconnected")}
|
|
||||||
phx-connected={hide("#disconnected")}
|
|
||||||
>
|
|
||||||
Attempting to reconnect <%= Heroicons.icon("arrow-path", type: "solid", class: "inline w-3 h-3 ml-1 animate-spin") %> />
|
|
||||||
</.flash>
|
|
||||||
<%= @inner_content %>
|
|
||||||
</div>
|
|
||||||
</main>
|
|
||||||
@@ -1,53 +0,0 @@
|
|||||||
<!DOCTYPE html>
|
|
||||||
<html lang="en">
|
|
||||||
<head>
|
|
||||||
<meta charset="utf-8" />
|
|
||||||
<meta name="viewport" content="width=device-width, initial-scale=1" />
|
|
||||||
<meta name="csrf-token" content={get_csrf_token()} />
|
|
||||||
<.live_title suffix=" · Phoenix Framework">
|
|
||||||
<%= assigns[:page_title] || "Malarkey" %>
|
|
||||||
</.live_title>
|
|
||||||
<link phx-track-static rel="stylesheet" href={~p"/assets/app.css"} />
|
|
||||||
<script defer phx-track-static type="text/javascript" src={~p"/assets/app.js"}>
|
|
||||||
</script>
|
|
||||||
</head>
|
|
||||||
<body class="antialiased bg-white">
|
|
||||||
<header class="px-4 sm:px-6 lg:px-8">
|
|
||||||
<div class="flex items-center justify-between py-3 border-b border-zinc-100">
|
|
||||||
<a href="/">
|
|
||||||
<div class="flex items-center gap-4">
|
|
||||||
<img class="h-8" src="/images/logo.svg" />
|
|
||||||
<p class="px-2 font-medium leading-6 rounded-full text-md text-brand">
|
|
||||||
Malarkey
|
|
||||||
</p>
|
|
||||||
</div>
|
|
||||||
</a>
|
|
||||||
<div class="flex items-center gap-4">
|
|
||||||
<%= if @current_user do %>
|
|
||||||
<.link
|
|
||||||
href={~p"/users/log_out"}
|
|
||||||
method="delete"
|
|
||||||
class="rounded-lg bg-zinc-100 px-2 py-1 text-[0.8125rem] font-semibold leading-6 text-zinc-900 hover:bg-zinc-200/80 active:text-zinc-900/70"
|
|
||||||
>
|
|
||||||
Logout <span aria-hidden="true">→</span>
|
|
||||||
</.link>
|
|
||||||
<% else %>
|
|
||||||
<.link
|
|
||||||
href={~p"/users/register"}
|
|
||||||
class="rounded-lg bg-zinc-100 px-2 py-1 text-[0.8125rem] font-semibold leading-6 text-zinc-900 hover:bg-zinc-200/80 active:text-zinc-900/70"
|
|
||||||
>
|
|
||||||
Register <span aria-hidden="true">→</span>
|
|
||||||
</.link>
|
|
||||||
<.link
|
|
||||||
href={~p"/users/log_in"}
|
|
||||||
class="rounded-lg bg-zinc-100 px-2 py-1 text-[0.8125rem] font-semibold leading-6 text-zinc-900 hover:bg-zinc-200/80 active:text-zinc-900/70"
|
|
||||||
>
|
|
||||||
Login <span aria-hidden="true">→</span>
|
|
||||||
</.link>
|
|
||||||
<% end %>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</header>
|
|
||||||
<%= @inner_content %>
|
|
||||||
</body>
|
|
||||||
</html>
|
|
||||||
@@ -1,19 +0,0 @@
|
|||||||
defmodule MalarkeyWeb.ErrorHTML do
|
|
||||||
use MalarkeyWeb, :html
|
|
||||||
|
|
||||||
# If you want to customize your error pages,
|
|
||||||
# uncomment the embed_templates/1 call below
|
|
||||||
# and add pages to the error directory:
|
|
||||||
#
|
|
||||||
# * lib/malarkey_web/controllers/error/404.html.heex
|
|
||||||
# * lib/malarkey_web/controllers/error/500.html.heex
|
|
||||||
#
|
|
||||||
# embed_templates "error/*"
|
|
||||||
|
|
||||||
# The default is to render a plain text page based on
|
|
||||||
# the template name. For example, "404.html" becomes
|
|
||||||
# "Not Found".
|
|
||||||
def render(template, _assigns) do
|
|
||||||
Phoenix.Controller.status_message_from_template(template)
|
|
||||||
end
|
|
||||||
end
|
|
||||||
@@ -1,15 +0,0 @@
|
|||||||
defmodule MalarkeyWeb.ErrorJSON do
|
|
||||||
# If you want to customize a particular status code,
|
|
||||||
# you may add your own clauses, such as:
|
|
||||||
#
|
|
||||||
# def render("500.json", _assigns) do
|
|
||||||
# %{errors: %{detail: "Internal Server Error"}}
|
|
||||||
# end
|
|
||||||
|
|
||||||
# By default, Phoenix returns the status message from
|
|
||||||
# the template name. For example, "404.json" becomes
|
|
||||||
# "Not Found".
|
|
||||||
def render(template, _assigns) do
|
|
||||||
%{errors: %{detail: Phoenix.Controller.status_message_from_template(template)}}
|
|
||||||
end
|
|
||||||
end
|
|
||||||
@@ -1,9 +0,0 @@
|
|||||||
defmodule MalarkeyWeb.PageController do
|
|
||||||
use MalarkeyWeb, :controller
|
|
||||||
|
|
||||||
def home(conn, _params) do
|
|
||||||
# The home page is often custom made,
|
|
||||||
# so skip the default app layout.
|
|
||||||
render(conn, :home, layout: false)
|
|
||||||
end
|
|
||||||
end
|
|
||||||
@@ -1,5 +0,0 @@
|
|||||||
defmodule MalarkeyWeb.PageHTML do
|
|
||||||
use MalarkeyWeb, :html
|
|
||||||
|
|
||||||
embed_templates "page_html/*"
|
|
||||||
end
|
|
||||||
@@ -1,236 +0,0 @@
|
|||||||
<div class="fixed inset-y-0 right-0 left-[40rem] hidden lg:block xl:left-[50rem]">
|
|
||||||
<svg
|
|
||||||
viewBox="0 0 1480 957"
|
|
||||||
fill="none"
|
|
||||||
aria-hidden="true"
|
|
||||||
class="absolute inset-0 w-full h-full"
|
|
||||||
preserveAspectRatio="xMinYMid slice"
|
|
||||||
>
|
|
||||||
<path fill="#EE7868" d="M0 0h1480v957H0z" />
|
|
||||||
<path
|
|
||||||
d="M137.542 466.27c-582.851-48.41-988.806-82.127-1608.412 658.2l67.39 810 3083.15-256.51L1535.94-49.622l-98.36 8.183C1269.29 281.468 734.115 515.799 146.47 467.012l-8.928-.742Z"
|
|
||||||
fill="#FF9F92"
|
|
||||||
/>
|
|
||||||
<path
|
|
||||||
d="M371.028 528.664C-169.369 304.988-545.754 149.198-1361.45 665.565l-182.58 792.025 3014.73 694.98 389.42-1689.25-96.18-22.171C1505.28 697.438 924.153 757.586 379.305 532.09l-8.277-3.426Z"
|
|
||||||
fill="#FA8372"
|
|
||||||
/>
|
|
||||||
<path
|
|
||||||
d="M359.326 571.714C-104.765 215.795-428.003-32.102-1349.55 255.554l-282.3 1224.596 3047.04 722.01 312.24-1354.467C1411.25 1028.3 834.355 935.995 366.435 577.166l-7.109-5.452Z"
|
|
||||||
fill="#E96856"
|
|
||||||
fill-opacity=".6"
|
|
||||||
/>
|
|
||||||
<path
|
|
||||||
d="M1593.87 1236.88c-352.15 92.63-885.498-145.85-1244.602-613.557l-5.455-7.105C-12.347 152.31-260.41-170.8-1225-131.458l-368.63 1599.048 3057.19 704.76 130.31-935.47Z"
|
|
||||||
fill="#C42652"
|
|
||||||
fill-opacity=".2"
|
|
||||||
/>
|
|
||||||
<path
|
|
||||||
d="M1411.91 1526.93c-363.79 15.71-834.312-330.6-1085.883-863.909l-3.822-8.102C72.704 125.95-101.074-242.476-1052.01-408.907l-699.85 1484.267 2837.75 1338.01 326.02-886.44Z"
|
|
||||||
fill="#A41C42"
|
|
||||||
fill-opacity=".2"
|
|
||||||
/>
|
|
||||||
<path
|
|
||||||
d="M1116.26 1863.69c-355.457-78.98-720.318-535.27-825.287-1115.521l-1.594-8.816C185.286 163.833 112.786-237.016-762.678-643.898L-1822.83 608.665 571.922 2635.55l544.338-771.86Z"
|
|
||||||
fill="#A41C42"
|
|
||||||
fill-opacity=".2"
|
|
||||||
/>
|
|
||||||
</svg>
|
|
||||||
</div>
|
|
||||||
<div class="px-4 py-10 sm:py-28 sm:px-6 lg:px-8 xl:py-32 xl:px-28">
|
|
||||||
<div class="max-w-xl mx-auto lg:mx-0">
|
|
||||||
<svg viewBox="0 0 71 48" class="h-12" aria-hidden="true">
|
|
||||||
<path
|
|
||||||
d="m26.371 33.477-.552-.1c-3.92-.729-6.397-3.1-7.57-6.829-.733-2.324.597-4.035 3.035-4.148 1.995-.092 3.362 1.055 4.57 2.39 1.557 1.72 2.984 3.558 4.514 5.305 2.202 2.515 4.797 4.134 8.347 3.634 3.183-.448 5.958-1.725 8.371-3.828.363-.316.761-.592 1.144-.886l-.241-.284c-2.027.63-4.093.841-6.205.735-3.195-.16-6.24-.828-8.964-2.582-2.486-1.601-4.319-3.746-5.19-6.611-.704-2.315.736-3.934 3.135-3.6.948.133 1.746.56 2.463 1.165.583.493 1.143 1.015 1.738 1.493 2.8 2.25 6.712 2.375 10.265-.068-5.842-.026-9.817-3.24-13.308-7.313-1.366-1.594-2.7-3.216-4.095-4.785-2.698-3.036-5.692-5.71-9.79-6.623C12.8-.623 7.745.14 2.893 2.361 1.926 2.804.997 3.319 0 4.149c.494 0 .763.006 1.032 0 2.446-.064 4.28 1.023 5.602 3.024.962 1.457 1.415 3.104 1.761 4.798.513 2.515.247 5.078.544 7.605.761 6.494 4.08 11.026 10.26 13.346 2.267.852 4.591 1.135 7.172.555ZM10.751 3.852c-.976.246-1.756-.148-2.56-.962 1.377-.343 2.592-.476 3.897-.528-.107.848-.607 1.306-1.336 1.49Zm32.002 37.924c-.085-.626-.62-.901-1.04-1.228-1.857-1.446-4.03-1.958-6.333-2-1.375-.026-2.735-.128-4.031-.61-.595-.22-1.26-.505-1.244-1.272.015-.78.693-1 1.31-1.184.505-.15 1.026-.247 1.6-.382-1.46-.936-2.886-1.065-4.787-.3-2.993 1.202-5.943 1.06-8.926-.017-1.684-.608-3.179-1.563-4.735-2.408l-.043.03a2.96 2.96 0 0 0 .04-.029c-.038-.117-.107-.12-.197-.054l.122.107c1.29 2.115 3.034 3.817 5.004 5.271 3.793 2.8 7.936 4.471 12.784 3.73A66.714 66.714 0 0 1 37 40.877c1.98-.16 3.866.398 5.753.899Zm-9.14-30.345c-.105-.076-.206-.266-.42-.069 1.745 2.36 3.985 4.098 6.683 5.193 4.354 1.767 8.773 2.07 13.293.51 3.51-1.21 6.033-.028 7.343 3.38.19-3.955-2.137-6.837-5.843-7.401-2.084-.318-4.01.373-5.962.94-5.434 1.575-10.485.798-15.094-2.553Zm27.085 15.425c.708.059 1.416.123 2.124.185-1.6-1.405-3.55-1.517-5.523-1.404-3.003.17-5.167 1.903-7.14 3.972-1.739 1.824-3.31 3.87-5.903 4.604.043.078.054.117.066.117.35.005.699.021 1.047.005 3.768-.17 7.317-.965 10.14-3.7.89-.86 1.685-1.817 2.544-2.71.716-.746 1.584-1.159 2.645-1.07Zm-8.753-4.67c-2.812.246-5.254 1.409-7.548 2.943-1.766 1.18-3.654 1.738-5.776 1.37-.374-.066-.75-.114-1.124-.17l-.013.156c.135.07.265.151.405.207.354.14.702.308 1.07.395 4.083.971 7.992.474 11.516-1.803 2.221-1.435 4.521-1.707 7.013-1.336.252.038.503.083.756.107.234.022.479.255.795.003-2.179-1.574-4.526-2.096-7.094-1.872Zm-10.049-9.544c1.475.051 2.943-.142 4.486-1.059-.452.04-.643.04-.827.076-2.126.424-4.033-.04-5.733-1.383-.623-.493-1.257-.974-1.889-1.457-2.503-1.914-5.374-2.555-8.514-2.5.05.154.054.26.108.315 3.417 3.455 7.371 5.836 12.369 6.008Zm24.727 17.731c-2.114-2.097-4.952-2.367-7.578-.537 1.738.078 3.043.632 4.101 1.728.374.388.763.768 1.182 1.106 1.6 1.29 4.311 1.352 5.896.155-1.861-.726-1.861-.726-3.601-2.452Zm-21.058 16.06c-1.858-3.46-4.981-4.24-8.59-4.008a9.667 9.667 0 0 1 2.977 1.39c.84.586 1.547 1.311 2.243 2.055 1.38 1.473 3.534 2.376 4.962 2.07-.656-.412-1.238-.848-1.592-1.507Zm17.29-19.32c0-.023.001-.045.003-.068l-.006.006.006-.006-.036-.004.021.018.012.053Zm-20 14.744a7.61 7.61 0 0 0-.072-.041.127.127 0 0 0 .015.043c.005.008.038 0 .058-.002Zm-.072-.041-.008-.034-.008.01.008-.01-.022-.006.005.026.024.014Z"
|
|
||||||
fill="#FD4F00"
|
|
||||||
/>
|
|
||||||
</svg>
|
|
||||||
<h1 class="flex items-center mt-10 text-sm font-semibold leading-6 text-brand">
|
|
||||||
Phoenix Framework
|
|
||||||
<small class="ml-3 rounded-full bg-brand/5 px-2 text-[0.8125rem] font-medium leading-6">
|
|
||||||
v1.7
|
|
||||||
</small>
|
|
||||||
</h1>
|
|
||||||
<p class="mt-4 text-[2rem] font-semibold leading-10 tracking-tighter text-zinc-900">
|
|
||||||
Peace of mind from prototype to production.
|
|
||||||
</p>
|
|
||||||
<p class="mt-4 text-base leading-7 text-zinc-600">
|
|
||||||
Build rich, interactive web applications quickly, with less code and fewer moving parts. Join our growing community of developers using Phoenix to craft APIs, HTML5 apps and more, for fun or at scale.
|
|
||||||
</p>
|
|
||||||
<div class="flex">
|
|
||||||
<div class="w-full sm:w-auto">
|
|
||||||
<div class="grid grid-cols-1 mt-10 gap-x-6 gap-y-4 sm:grid-cols-3">
|
|
||||||
<a
|
|
||||||
href="https://hexdocs.pm/phoenix/overview.html"
|
|
||||||
class="relative px-6 py-4 text-sm font-semibold leading-6 group rounded-2xl text-zinc-900 sm:py-6"
|
|
||||||
>
|
|
||||||
<span class="absolute inset-0 transition rounded-2xl bg-zinc-50 group-hover:bg-zinc-100 sm:group-hover:scale-105">
|
|
||||||
</span>
|
|
||||||
<span class="relative flex items-center gap-4 sm:flex-col">
|
|
||||||
<svg viewBox="0 0 24 24" fill="none" aria-hidden="true" class="w-6 h-6">
|
|
||||||
<path d="m12 4 10-2v18l-10 2V4Z" fill="#18181B" fill-opacity=".15" />
|
|
||||||
<path
|
|
||||||
d="M12 4 2 2v18l10 2m0-18v18m0-18 10-2v18l-10 2"
|
|
||||||
stroke="#18181B"
|
|
||||||
stroke-width="2"
|
|
||||||
stroke-linecap="round"
|
|
||||||
stroke-linejoin="round"
|
|
||||||
/>
|
|
||||||
</svg>
|
|
||||||
Guides & Docs
|
|
||||||
</span>
|
|
||||||
</a>
|
|
||||||
<a
|
|
||||||
href="https://github.com/phoenixframework/phoenix"
|
|
||||||
class="relative px-6 py-4 text-sm font-semibold leading-6 group rounded-2xl text-zinc-900 sm:py-6"
|
|
||||||
>
|
|
||||||
<span class="absolute inset-0 transition rounded-2xl bg-zinc-50 group-hover:bg-zinc-100 sm:group-hover:scale-105">
|
|
||||||
</span>
|
|
||||||
<span class="relative flex items-center gap-4 sm:flex-col">
|
|
||||||
<svg viewBox="0 0 24 24" aria-hidden="true" class="w-6 h-6">
|
|
||||||
<path
|
|
||||||
fill-rule="evenodd"
|
|
||||||
clip-rule="evenodd"
|
|
||||||
d="M12 0C5.37 0 0 5.506 0 12.303c0 5.445 3.435 10.043 8.205 11.674.6.107.825-.262.825-.585 0-.292-.015-1.261-.015-2.291C6 21.67 5.22 20.346 4.98 19.654c-.135-.354-.72-1.446-1.23-1.738-.42-.23-1.02-.8-.015-.815.945-.015 1.62.892 1.845 1.261 1.08 1.86 2.805 1.338 3.495 1.015.105-.8.42-1.338.765-1.645-2.67-.308-5.46-1.37-5.46-6.075 0-1.338.465-2.446 1.23-3.307-.12-.308-.54-1.569.12-3.26 0 0 1.005-.323 3.3 1.26.96-.276 1.98-.415 3-.415s2.04.139 3 .416c2.295-1.6 3.3-1.261 3.3-1.261.66 1.691.24 2.952.12 3.26.765.861 1.23 1.953 1.23 3.307 0 4.721-2.805 5.767-5.475 6.075.435.384.81 1.122.81 2.276 0 1.645-.015 2.968-.015 3.383 0 .323.225.707.825.585a12.047 12.047 0 0 0 5.919-4.489A12.536 12.536 0 0 0 24 12.304C24 5.505 18.63 0 12 0Z"
|
|
||||||
fill="#18181B"
|
|
||||||
/>
|
|
||||||
</svg>
|
|
||||||
Source Code
|
|
||||||
</span>
|
|
||||||
</a>
|
|
||||||
<a
|
|
||||||
href="https://github.com/phoenixframework/phoenix/blob/v1.7/CHANGELOG.md"
|
|
||||||
class="relative px-6 py-4 text-sm font-semibold leading-6 group rounded-2xl text-zinc-900 sm:py-6"
|
|
||||||
>
|
|
||||||
<span class="absolute inset-0 transition rounded-2xl bg-zinc-50 group-hover:bg-zinc-100 sm:group-hover:scale-105">
|
|
||||||
</span>
|
|
||||||
<span class="relative flex items-center gap-4 sm:flex-col">
|
|
||||||
<svg viewBox="0 0 24 24" fill="none" aria-hidden="true" class="w-6 h-6">
|
|
||||||
<path
|
|
||||||
d="M12 1v6M12 17v6"
|
|
||||||
stroke="#18181B"
|
|
||||||
stroke-width="2"
|
|
||||||
stroke-linecap="round"
|
|
||||||
stroke-linejoin="round"
|
|
||||||
/>
|
|
||||||
<circle
|
|
||||||
cx="12"
|
|
||||||
cy="12"
|
|
||||||
r="4"
|
|
||||||
fill="#18181B"
|
|
||||||
fill-opacity=".15"
|
|
||||||
stroke="#18181B"
|
|
||||||
stroke-width="2"
|
|
||||||
stroke-linecap="round"
|
|
||||||
stroke-linejoin="round"
|
|
||||||
/>
|
|
||||||
</svg>
|
|
||||||
Changelog
|
|
||||||
</span>
|
|
||||||
</a>
|
|
||||||
</div>
|
|
||||||
<div class="grid grid-cols-1 mt-10 text-sm leading-6 gap-y-4 text-zinc-700 sm:grid-cols-2">
|
|
||||||
<div>
|
|
||||||
<a
|
|
||||||
href="https://twitter.com/elixirphoenix"
|
|
||||||
class="group -mx-2 -my-0.5 inline-flex items-center gap-3 rounded-lg px-2 py-0.5 hover:bg-zinc-50 hover:text-zinc-900"
|
|
||||||
>
|
|
||||||
<svg
|
|
||||||
viewBox="0 0 16 16"
|
|
||||||
aria-hidden="true"
|
|
||||||
class="w-4 h-4 fill-zinc-400 group-hover:fill-zinc-600"
|
|
||||||
>
|
|
||||||
<path d="M5.403 14c5.283 0 8.172-4.617 8.172-8.62 0-.131 0-.262-.008-.391A6.033 6.033 0 0 0 15 3.419a5.503 5.503 0 0 1-1.65.477 3.018 3.018 0 0 0 1.263-1.676 5.579 5.579 0 0 1-1.824.736 2.832 2.832 0 0 0-1.63-.916 2.746 2.746 0 0 0-1.821.319A2.973 2.973 0 0 0 8.076 3.78a3.185 3.185 0 0 0-.182 1.938 7.826 7.826 0 0 1-3.279-.918 8.253 8.253 0 0 1-2.64-2.247 3.176 3.176 0 0 0-.315 2.208 3.037 3.037 0 0 0 1.203 1.836A2.739 2.739 0 0 1 1.56 6.22v.038c0 .7.23 1.377.65 1.919.42.54 1.004.912 1.654 1.05-.423.122-.866.14-1.297.052.184.602.541 1.129 1.022 1.506a2.78 2.78 0 0 0 1.662.598 5.656 5.656 0 0 1-2.007 1.074A5.475 5.475 0 0 1 1 12.64a7.827 7.827 0 0 0 4.403 1.358" />
|
|
||||||
</svg>
|
|
||||||
Follow on Twitter
|
|
||||||
</a>
|
|
||||||
</div>
|
|
||||||
<div>
|
|
||||||
<a
|
|
||||||
href="https://elixirforum.com"
|
|
||||||
class="group -mx-2 -my-0.5 inline-flex items-center gap-3 rounded-lg px-2 py-0.5 hover:bg-zinc-50 hover:text-zinc-900"
|
|
||||||
>
|
|
||||||
<svg
|
|
||||||
viewBox="0 0 16 16"
|
|
||||||
aria-hidden="true"
|
|
||||||
class="w-4 h-4 fill-zinc-400 group-hover:fill-zinc-600"
|
|
||||||
>
|
|
||||||
<path d="M8 13.833c3.866 0 7-2.873 7-6.416C15 3.873 11.866 1 8 1S1 3.873 1 7.417c0 1.081.292 2.1.808 2.995.606 1.05.806 2.399.086 3.375l-.208.283c-.285.386-.01.905.465.85.852-.098 2.048-.318 3.137-.81a3.717 3.717 0 0 1 1.91-.318c.263.027.53.041.802.041Z" />
|
|
||||||
</svg>
|
|
||||||
Discuss on the Elixir forum
|
|
||||||
</a>
|
|
||||||
</div>
|
|
||||||
<div>
|
|
||||||
<a
|
|
||||||
href="https://elixir-slackin.herokuapp.com"
|
|
||||||
class="group -mx-2 -my-0.5 inline-flex items-center gap-3 rounded-lg px-2 py-0.5 hover:bg-zinc-50 hover:text-zinc-900"
|
|
||||||
>
|
|
||||||
<svg
|
|
||||||
viewBox="0 0 16 16"
|
|
||||||
aria-hidden="true"
|
|
||||||
class="w-4 h-4 fill-zinc-400 group-hover:fill-zinc-600"
|
|
||||||
>
|
|
||||||
<path d="M3.95 9.85a1.47 1.47 0 1 1-2.94 0 1.47 1.47 0 0 1 1.47-1.472h1.47v1.471Zm.735 0a1.47 1.47 0 1 1 2.94 0v3.678a1.47 1.47 0 1 1-2.94 0V9.85ZM6.156 3.942a1.47 1.47 0 0 1-1.47-1.472 1.47 1.47 0 1 1 2.94 0v1.472h-1.47Zm0 .747c.813 0 1.47.658 1.47 1.471a1.47 1.47 0 0 1-1.47 1.472H2.47A1.47 1.47 0 0 1 1 6.16 1.47 1.47 0 0 1 2.47 4.69h3.686ZM12.048 6.16a1.47 1.47 0 1 1 2.94 0 1.47 1.47 0 0 1-1.47 1.472h-1.47V6.16Zm-.735 0a1.47 1.47 0 1 1-2.94 0V2.47a1.47 1.47 0 1 1 2.94 0v3.69ZM9.843 12.057c.813 0 1.47.657 1.47 1.471a1.47 1.47 0 1 1-2.94 0v-1.471h1.47Zm0-.736a1.47 1.47 0 0 1-1.47-1.472 1.47 1.47 0 0 1 1.47-1.471h3.686c.813 0 1.47.658 1.47 1.471a1.47 1.47 0 0 1-1.47 1.472H9.843Z" />
|
|
||||||
</svg>
|
|
||||||
Join our Slack channel
|
|
||||||
</a>
|
|
||||||
</div>
|
|
||||||
<div>
|
|
||||||
<a
|
|
||||||
href="https://web.libera.chat/#elixir"
|
|
||||||
class="group -mx-2 -my-0.5 inline-flex items-center gap-3 rounded-lg px-2 py-0.5 hover:bg-zinc-50 hover:text-zinc-900"
|
|
||||||
>
|
|
||||||
<svg
|
|
||||||
viewBox="0 0 16 16"
|
|
||||||
aria-hidden="true"
|
|
||||||
class="w-4 h-4 fill-zinc-400 group-hover:fill-zinc-600"
|
|
||||||
>
|
|
||||||
<path
|
|
||||||
fill-rule="evenodd"
|
|
||||||
clip-rule="evenodd"
|
|
||||||
d="M6.356 2.007a.75.75 0 0 1 .637.849l-1.5 10.5a.75.75 0 1 1-1.485-.212l1.5-10.5a.75.75 0 0 1 .848-.637ZM11.356 2.008a.75.75 0 0 1 .637.848l-1.5 10.5a.75.75 0 0 1-1.485-.212l1.5-10.5a.75.75 0 0 1 .848-.636Z"
|
|
||||||
/>
|
|
||||||
<path
|
|
||||||
fill-rule="evenodd"
|
|
||||||
clip-rule="evenodd"
|
|
||||||
d="M14 5.25a.75.75 0 0 1-.75.75h-9.5a.75.75 0 0 1 0-1.5h9.5a.75.75 0 0 1 .75.75ZM13 10.75a.75.75 0 0 1-.75.75h-9.5a.75.75 0 0 1 0-1.5h9.5a.75.75 0 0 1 .75.75Z"
|
|
||||||
/>
|
|
||||||
</svg>
|
|
||||||
Chat on Libera IRC
|
|
||||||
</a>
|
|
||||||
</div>
|
|
||||||
<div>
|
|
||||||
<a
|
|
||||||
href="https://discord.gg/elixir"
|
|
||||||
class="group -mx-2 -my-0.5 inline-flex items-center gap-3 rounded-lg px-2 py-0.5 hover:bg-zinc-50 hover:text-zinc-900"
|
|
||||||
>
|
|
||||||
<svg
|
|
||||||
viewBox="0 0 16 16"
|
|
||||||
aria-hidden="true"
|
|
||||||
class="w-4 h-4 fill-zinc-400 group-hover:fill-zinc-600"
|
|
||||||
>
|
|
||||||
<path d="M13.545 2.995c-1.02-.46-2.114-.8-3.257-.994a.05.05 0 0 0-.052.024c-.141.246-.297.567-.406.82a12.377 12.377 0 0 0-3.658 0 8.238 8.238 0 0 0-.412-.82.052.052 0 0 0-.052-.024 13.315 13.315 0 0 0-3.257.994.046.046 0 0 0-.021.018C.356 6.063-.213 9.036.066 11.973c.001.015.01.029.02.038a13.353 13.353 0 0 0 3.996 1.987.052.052 0 0 0 .056-.018c.308-.414.582-.85.818-1.309a.05.05 0 0 0-.028-.069 8.808 8.808 0 0 1-1.248-.585.05.05 0 0 1-.005-.084c.084-.062.168-.126.248-.191a.05.05 0 0 1 .051-.007c2.619 1.176 5.454 1.176 8.041 0a.05.05 0 0 1 .053.006c.08.065.164.13.248.192a.05.05 0 0 1-.004.084c-.399.23-.813.423-1.249.585a.05.05 0 0 0-.027.07c.24.457.514.893.817 1.307a.051.051 0 0 0 .056.019 13.31 13.31 0 0 0 4.001-1.987.05.05 0 0 0 .021-.037c.334-3.396-.559-6.345-2.365-8.96a.04.04 0 0 0-.021-.02Zm-8.198 7.19c-.789 0-1.438-.712-1.438-1.587 0-.874.637-1.586 1.438-1.586.807 0 1.45.718 1.438 1.586 0 .875-.637 1.587-1.438 1.587Zm5.316 0c-.788 0-1.438-.712-1.438-1.587 0-.874.637-1.586 1.438-1.586.807 0 1.45.718 1.438 1.586 0 .875-.63 1.587-1.438 1.587Z" />
|
|
||||||
</svg>
|
|
||||||
Join our Discord server
|
|
||||||
</a>
|
|
||||||
</div>
|
|
||||||
<div>
|
|
||||||
<a
|
|
||||||
href="https://fly.io/docs/getting-started/elixir/"
|
|
||||||
class="group -mx-2 -my-0.5 inline-flex items-center gap-3 rounded-lg px-2 py-0.5 hover:bg-zinc-50 hover:text-zinc-900"
|
|
||||||
>
|
|
||||||
<svg
|
|
||||||
viewBox="0 0 20 20"
|
|
||||||
aria-hidden="true"
|
|
||||||
class="w-4 h-4 fill-zinc-400 group-hover:fill-zinc-600"
|
|
||||||
>
|
|
||||||
<path d="M1 12.5A4.5 4.5 0 005.5 17H15a4 4 0 001.866-7.539 3.504 3.504 0 00-4.504-4.272A4.5 4.5 0 004.06 8.235 4.502 4.502 0 001 12.5z" />
|
|
||||||
</svg>
|
|
||||||
Deploy your application
|
|
||||||
</a>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
@@ -1,42 +0,0 @@
|
|||||||
defmodule MalarkeyWeb.UserSessionController do
|
|
||||||
use MalarkeyWeb, :controller
|
|
||||||
|
|
||||||
alias Malarkey.Accounts
|
|
||||||
alias MalarkeyWeb.UserAuth
|
|
||||||
|
|
||||||
def create(conn, %{"_action" => "registered"} = params) do
|
|
||||||
create(conn, params, "Account created successfully!")
|
|
||||||
end
|
|
||||||
|
|
||||||
def create(conn, %{"_action" => "password_updated"} = params) do
|
|
||||||
conn
|
|
||||||
|> put_session(:user_return_to, ~p"/users/settings")
|
|
||||||
|> create(params, "Password updated successfully!")
|
|
||||||
end
|
|
||||||
|
|
||||||
def create(conn, params) do
|
|
||||||
create(conn, params, "Welcome back!")
|
|
||||||
end
|
|
||||||
|
|
||||||
defp create(conn, %{"user" => user_params}, info) do
|
|
||||||
%{"email" => email, "password" => password} = user_params
|
|
||||||
|
|
||||||
if user = Accounts.get_user_by_email_and_password(email, password) do
|
|
||||||
conn
|
|
||||||
|> put_flash(:info, info)
|
|
||||||
|> UserAuth.log_in_user(user, user_params)
|
|
||||||
else
|
|
||||||
# In order to prevent user enumeration attacks, don't disclose whether the email is registered.
|
|
||||||
conn
|
|
||||||
|> put_flash(:error, "Invalid email or password")
|
|
||||||
|> put_flash(:email, String.slice(email, 0, 160))
|
|
||||||
|> redirect(to: ~p"/users/log_in")
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
def delete(conn, _params) do
|
|
||||||
conn
|
|
||||||
|> put_flash(:info, "Logged out successfully.")
|
|
||||||
|> UserAuth.log_out_user()
|
|
||||||
end
|
|
||||||
end
|
|
||||||
@@ -1,51 +0,0 @@
|
|||||||
defmodule MalarkeyWeb.Endpoint do
|
|
||||||
use Phoenix.Endpoint, otp_app: :malarkey
|
|
||||||
|
|
||||||
# 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: "_malarkey_key",
|
|
||||||
signing_salt: "nOkCv+jj",
|
|
||||||
same_site: "Lax"
|
|
||||||
]
|
|
||||||
|
|
||||||
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: :malarkey,
|
|
||||||
gzip: false,
|
|
||||||
only: MalarkeyWeb.static_paths()
|
|
||||||
|
|
||||||
# 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
|
|
||||||
plug Phoenix.Ecto.CheckRepoStatus, otp_app: :malarkey
|
|
||||||
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 MalarkeyWeb.Router
|
|
||||||
end
|
|
||||||
@@ -1,24 +0,0 @@
|
|||||||
defmodule MalarkeyWeb.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 MalarkeyWeb.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: :malarkey
|
|
||||||
end
|
|
||||||
@@ -1,95 +0,0 @@
|
|||||||
defmodule MalarkeyWeb.PostLive.FormComponent do
|
|
||||||
use MalarkeyWeb, :live_component
|
|
||||||
require Logger
|
|
||||||
|
|
||||||
alias Malarkey.Timeline
|
|
||||||
|
|
||||||
@impl true
|
|
||||||
def render(assigns) do
|
|
||||||
~H"""
|
|
||||||
<div>
|
|
||||||
<.header>
|
|
||||||
<%= @title %>
|
|
||||||
<:subtitle>Dazzle us, <%= @user.fullname %>!!</:subtitle>
|
|
||||||
</.header>
|
|
||||||
|
|
||||||
<.simple_form
|
|
||||||
:let={f}
|
|
||||||
for={@changeset}
|
|
||||||
id="post-form"
|
|
||||||
phx-target={@myself}
|
|
||||||
phx-change="validate"
|
|
||||||
phx-submit="save"
|
|
||||||
>
|
|
||||||
<.input field={{f, :body}} type="textarea" label="body" />
|
|
||||||
<:actions>
|
|
||||||
<.button phx-disable-with="Saving...">Save Post</.button>
|
|
||||||
</:actions>
|
|
||||||
</.simple_form>
|
|
||||||
</div>
|
|
||||||
"""
|
|
||||||
end
|
|
||||||
|
|
||||||
@impl true
|
|
||||||
def update(%{post: post} = assigns, socket) do
|
|
||||||
changeset = Timeline.change_post(post)
|
|
||||||
|
|
||||||
{:ok,
|
|
||||||
socket
|
|
||||||
|> assign(assigns)
|
|
||||||
|> assign(:changeset, changeset)}
|
|
||||||
end
|
|
||||||
|
|
||||||
@impl true
|
|
||||||
def handle_event("validate", %{"post" => post_params}, socket) do
|
|
||||||
changeset =
|
|
||||||
socket.assigns.post
|
|
||||||
|> Timeline.change_post(post_params)
|
|
||||||
|> Map.put(:action, :validate)
|
|
||||||
|
|
||||||
{:noreply, assign(socket, :changeset, changeset)}
|
|
||||||
end
|
|
||||||
|
|
||||||
def handle_event("save", %{"post" => post_params}, socket) do
|
|
||||||
save_post(socket, socket.assigns.action, post_params)
|
|
||||||
end
|
|
||||||
|
|
||||||
defp save_post(socket, :like, _post_params) do
|
|
||||||
case Timeline.add_like(socket.assigns.current_user, socket.assigns.post) do
|
|
||||||
{:ok, _post} ->
|
|
||||||
{:noreply,
|
|
||||||
socket
|
|
||||||
|> put_flash(:info, "Post updated successfully")
|
|
||||||
|> push_navigate(to: socket.assigns.navigate)}
|
|
||||||
|
|
||||||
{:error, %Ecto.Changeset{} = changeset} ->
|
|
||||||
{:noreply, assign(socket, :changeset, changeset)}
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
defp save_post(socket, :edit, post_params) do
|
|
||||||
case Timeline.update_post(socket.assigns.post, post_params) do
|
|
||||||
{:ok, _post} ->
|
|
||||||
{:noreply,
|
|
||||||
socket
|
|
||||||
|> put_flash(:info, "Post updated successfully")
|
|
||||||
|> push_navigate(to: socket.assigns.navigate)}
|
|
||||||
|
|
||||||
{:error, %Ecto.Changeset{} = changeset} ->
|
|
||||||
{:noreply, assign(socket, :changeset, changeset)}
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
defp save_post(socket, :new, post_params) do
|
|
||||||
case Timeline.create_post(socket.assigns.user, post_params) do
|
|
||||||
{:ok, _post} ->
|
|
||||||
{:noreply,
|
|
||||||
socket
|
|
||||||
|> put_flash(:info, "Post created successfully")
|
|
||||||
|> push_navigate(to: socket.assigns.navigate)}
|
|
||||||
|
|
||||||
{:error, %Ecto.Changeset{} = changeset} ->
|
|
||||||
{:noreply, assign(socket, changeset: changeset)}
|
|
||||||
end
|
|
||||||
end
|
|
||||||
end
|
|
||||||
@@ -1,87 +0,0 @@
|
|||||||
defmodule MalarkeyWeb.PostLive.Index do
|
|
||||||
use MalarkeyWeb, :live_view
|
|
||||||
require Logger
|
|
||||||
alias Malarkey.Timeline
|
|
||||||
alias Malarkey.Timeline.Post
|
|
||||||
on_mount MalarkeyWeb.UserLiveAuth
|
|
||||||
|
|
||||||
@impl true
|
|
||||||
def mount(_params, _session, socket) do
|
|
||||||
if connected?(socket), do: Timeline.subscribe()
|
|
||||||
{:ok, assign(socket, :posts, list_posts()), temporary_assigns: [posts: []]}
|
|
||||||
end
|
|
||||||
|
|
||||||
@impl true
|
|
||||||
def handle_params(params, _url, socket) do
|
|
||||||
{:noreply, apply_action(socket, socket.assigns.live_action, params)}
|
|
||||||
end
|
|
||||||
|
|
||||||
defp apply_action(socket, :like, %{"id" => id}) do
|
|
||||||
socket
|
|
||||||
|> assign(:page_title, "Edit Post")
|
|
||||||
|> assign(:post, Timeline.get_post!(id))
|
|
||||||
end
|
|
||||||
|
|
||||||
defp apply_action(socket, :edit, %{"id" => id}) do
|
|
||||||
socket
|
|
||||||
|> assign(:page_title, "Edit Post")
|
|
||||||
|> assign(:post, Timeline.get_post!(id))
|
|
||||||
end
|
|
||||||
|
|
||||||
defp apply_action(socket, :new, _params) do
|
|
||||||
socket
|
|
||||||
|> assign(:page_title, "More Malarkey")
|
|
||||||
|> assign(:post, %Post{})
|
|
||||||
end
|
|
||||||
|
|
||||||
defp apply_action(socket, :index, _params) do
|
|
||||||
socket
|
|
||||||
|> assign(:page_title, "Listing Posts")
|
|
||||||
|> assign(:post, nil)
|
|
||||||
end
|
|
||||||
|
|
||||||
@impl true
|
|
||||||
def handle_event("login", _id, socket) do
|
|
||||||
{:noreply,
|
|
||||||
socket
|
|
||||||
|> push_redirect(to: "/users/log_in")}
|
|
||||||
end
|
|
||||||
|
|
||||||
@impl true
|
|
||||||
def handle_event("like", %{"id" => id}, socket) do
|
|
||||||
post = Timeline.get_post!(id)
|
|
||||||
{:ok, _} = Timeline.add_like(socket.assigns.current_user, post)
|
|
||||||
|
|
||||||
{:noreply, assign(socket, :posts, list_posts())}
|
|
||||||
end
|
|
||||||
|
|
||||||
@impl true
|
|
||||||
def handle_event("repost", %{"id" => id}, socket) do
|
|
||||||
post = Timeline.get_post!(id)
|
|
||||||
{:ok, _} = Timeline.add_repost(socket.assigns.current_user, post)
|
|
||||||
|
|
||||||
{:noreply, assign(socket, :posts, list_posts())}
|
|
||||||
end
|
|
||||||
|
|
||||||
@impl true
|
|
||||||
def handle_event("delete", %{"id" => id}, socket) do
|
|
||||||
post = Timeline.get_post!(id)
|
|
||||||
{:ok, _} = Timeline.delete_post(post)
|
|
||||||
|
|
||||||
{:noreply, assign(socket, :posts, list_posts())}
|
|
||||||
end
|
|
||||||
|
|
||||||
@impl true
|
|
||||||
def handle_info({:post_created, post}, socket) do
|
|
||||||
{:noreply, update(socket, :posts, fn posts -> [post | posts] end)}
|
|
||||||
end
|
|
||||||
|
|
||||||
@impl true
|
|
||||||
def handle_info({:post_updated, post}, socket) do
|
|
||||||
{:noreply, update(socket, :posts, fn posts -> [post | posts] end)}
|
|
||||||
end
|
|
||||||
|
|
||||||
defp list_posts do
|
|
||||||
Timeline.list_posts()
|
|
||||||
end
|
|
||||||
end
|
|
||||||
@@ -1,42 +0,0 @@
|
|||||||
<.header>
|
|
||||||
<%= if !!@current_user do %>
|
|
||||||
<div id="user-greeting">Hi, <%= !!@current_user.fullname || "Sailor" %></div>
|
|
||||||
<% else %>
|
|
||||||
<div id="anonymous-greeting">Hello, Sailor!</div>
|
|
||||||
<% end %>
|
|
||||||
<:actions>
|
|
||||||
<.link patch={(!!@current_user && ~p"/posts/new") || "/users/log_in"}>
|
|
||||||
<button class="inline-flex items-center px-3 py-2 text-sm font-semibold leading-6 text-white rounded-lg bg-brand phx-submit-loading:opacity-75 hover:bg-zinc-700 active:text-white/80">
|
|
||||||
<%= Heroicons.icon("plus-circle", type: "outline", class: "w-5 h-5 stroke-current mr-1") %>
|
|
||||||
<span>Add Malarkey</span>
|
|
||||||
</button>
|
|
||||||
</.link>
|
|
||||||
</:actions>
|
|
||||||
</.header>
|
|
||||||
|
|
||||||
<div id="posts" phx-update="prepend">
|
|
||||||
<%= for post <- @posts do %>
|
|
||||||
<.live_component
|
|
||||||
module={MalarkeyWeb.PostLive.PostComponent}
|
|
||||||
id={post.id}
|
|
||||||
post={post}
|
|
||||||
user={@current_user}
|
|
||||||
/>
|
|
||||||
<% end %>
|
|
||||||
</div>
|
|
||||||
<.modal
|
|
||||||
:if={@live_action in [:new, :edit]}
|
|
||||||
id="post-modal"
|
|
||||||
show
|
|
||||||
on_cancel={JS.navigate(~p"/posts")}
|
|
||||||
>
|
|
||||||
<.live_component
|
|
||||||
module={MalarkeyWeb.PostLive.FormComponent}
|
|
||||||
id={@post.id || :new}
|
|
||||||
title={@page_title}
|
|
||||||
action={@live_action}
|
|
||||||
post={@post}
|
|
||||||
navigate={~p"/posts"}
|
|
||||||
user={@current_user}
|
|
||||||
/>
|
|
||||||
</.modal>
|
|
||||||
@@ -1,106 +0,0 @@
|
|||||||
defmodule MalarkeyWeb.PostLive.PostComponent do
|
|
||||||
use MalarkeyWeb, :live_component
|
|
||||||
alias Malarkey.Accounts
|
|
||||||
|
|
||||||
@impl true
|
|
||||||
@spec render(any) :: Phoenix.LiveView.Rendered.t()
|
|
||||||
def render(assigns) do
|
|
||||||
~H"""
|
|
||||||
<div class="max-w-xl mx-auto my-6">
|
|
||||||
<article class="flex flex-wrap items-start p-2 border-t border-b border-gray-400 cursor-pointer hover:bg-gray-100">
|
|
||||||
<img src="https://i.pravatar.cc/300?u=#{@post.id}" class="w-12 h-12 mr-3 rounded-full" />
|
|
||||||
<div class="flex flex-wrap items-start justify-start flex-1">
|
|
||||||
<div class="flex items-center flex-1">
|
|
||||||
<div class="flex items-center flex-1">
|
|
||||||
<h3 class="mr-2 font-bold hover:underline">
|
|
||||||
<a href="#"><%= @post.user.fullname %></a>
|
|
||||||
</h3>
|
|
||||||
<span class="mr-2">
|
|
||||||
<svg
|
|
||||||
class="w-4 h-4"
|
|
||||||
fill="#1da1f2"
|
|
||||||
viewBox="0 0 24 24"
|
|
||||||
aria-label="Verified account"
|
|
||||||
class=""
|
|
||||||
>
|
|
||||||
<g>
|
|
||||||
<path d="M22.5 12.5c0-1.58-.875-2.95-2.148-3.6.154-.435.238-.905.238-1.4 0-2.21-1.71-3.998-3.818-3.998-.47 0-.92.084-1.336.25C14.818 2.415 13.51 1.5 12 1.5s-2.816.917-3.437 2.25c-.415-.165-.866-.25-1.336-.25-2.11 0-3.818 1.79-3.818 4 0 .494.083.964.237 1.4-1.272.65-2.147 2.018-2.147 3.6 0 1.495.782 2.798 1.942 3.486-.02.17-.032.34-.032.514 0 2.21 1.708 4 3.818 4 .47 0 .92-.086 1.335-.25.62 1.334 1.926 2.25 3.437 2.25 1.512 0 2.818-.916 3.437-2.25.415.163.865.248 1.336.248 2.11 0 3.818-1.79 3.818-4 0-.174-.012-.344-.033-.513 1.158-.687 1.943-1.99 1.943-3.484zm-6.616-3.334l-4.334 6.5c-.145.217-.382.334-.625.334-.143 0-.288-.04-.416-.126l-.115-.094-2.415-2.415c-.293-.293-.293-.768 0-1.06s.768-.294 1.06 0l1.77 1.767 3.825-5.74c.23-.345.696-.436 1.04-.207.346.23.44.696.21 1.04z">
|
|
||||||
</path>
|
|
||||||
</g>
|
|
||||||
</svg>
|
|
||||||
</span>
|
|
||||||
<span class="mr-1 text-sm text-gray-600">@<%= @post.user.username %></span>
|
|
||||||
<span class="mr-1 text-sm text-gray-600">·</span>
|
|
||||||
<span class="text-sm text-gray-600">Apr 7</span>
|
|
||||||
</div>
|
|
||||||
<div class="text-gray-600">
|
|
||||||
<a
|
|
||||||
href="#"
|
|
||||||
class="flex items-center justify-center w-6 h-6 bg-transparent rounded-full hover:bg-blue-200 hover:text-blue-600"
|
|
||||||
>
|
|
||||||
<svg viewBox="0 0 24 24" class="w-3 h-3 fill-current">
|
|
||||||
<g>
|
|
||||||
<path d="M20.207 8.147c-.39-.39-1.023-.39-1.414 0L12 14.94 5.207 8.147c-.39-.39-1.023-.39-1.414 0-.39.39-.39 1.023 0 1.414l7.5 7.5c.195.196.45.294.707.294s.512-.098.707-.293l7.5-7.5c.39-.39.39-1.022 0-1.413z">
|
|
||||||
</path>
|
|
||||||
</g>
|
|
||||||
</svg>
|
|
||||||
</a>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="w-full">
|
|
||||||
<p class="my-1"><%= @post.body %></p>
|
|
||||||
<%!-- Item has embedded media --%>
|
|
||||||
<%= if false do %>
|
|
||||||
<div class="rounded-lg">
|
|
||||||
<img
|
|
||||||
src="https://www.fillmurray.com/640/360"
|
|
||||||
class="object-cover w-full h-64 border rounded-lg"
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
<% end %>
|
|
||||||
|
|
||||||
<div class="flex items-center justify-between py-2">
|
|
||||||
<div class="flex items-center mr-8 text-gray-600 hover:text-blue-500">
|
|
||||||
<a
|
|
||||||
href="#"
|
|
||||||
class="flex items-center justify-center w-8 h-8 rounded-full hover:bg-blue-200 hover:text-blue-500"
|
|
||||||
>
|
|
||||||
<%= Heroicons.icon("chat-bubble-oval-left", type: "outline", class: "w-5 h-5") %>
|
|
||||||
</a>
|
|
||||||
<span class="ml-1">1.5K</span>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="flex items-center mr-8 text-gray-600 hover:text-green-500">
|
|
||||||
<.link phx-click={
|
|
||||||
(@user && JS.push("repost", value: %{id: @post.id})) ||
|
|
||||||
JS.push("login")
|
|
||||||
}>
|
|
||||||
<%= Heroicons.icon("arrow-path-rounded-square", type: "outline", class: "w-5 h-5") %>
|
|
||||||
</.link>
|
|
||||||
<span class="ml-1"><%= length(@post.reposted_by) %></span>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="flex items-center mr-6 text-gray-600 hover:text-red-500">
|
|
||||||
<.link phx-click={
|
|
||||||
(@user && JS.push("like", value: %{id: @post.id})) ||
|
|
||||||
JS.push("login")
|
|
||||||
}>
|
|
||||||
<%= Heroicons.icon("heart", type: "outline", class: "w-5 h-5") %>
|
|
||||||
</.link>
|
|
||||||
<span class="ml-1"><%= length(@post.liked_by) %></span>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="flex items-center mr-6 text-gray-600 hover:text-red-500 pull">
|
|
||||||
<.link patch={~p"/posts/#{@post}/edit"}>
|
|
||||||
<%= Heroicons.icon("pencil-square", type: "outline", class: "w-5 h-5") %>
|
|
||||||
</.link>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</article>
|
|
||||||
</div>
|
|
||||||
"""
|
|
||||||
end
|
|
||||||
end
|
|
||||||
@@ -1,21 +0,0 @@
|
|||||||
defmodule MalarkeyWeb.PostLive.Show do
|
|
||||||
use MalarkeyWeb, :live_view
|
|
||||||
|
|
||||||
alias Malarkey.Timeline
|
|
||||||
|
|
||||||
@impl true
|
|
||||||
def mount(_params, _session, socket) do
|
|
||||||
{:ok, socket}
|
|
||||||
end
|
|
||||||
|
|
||||||
@impl true
|
|
||||||
def handle_params(%{"id" => id}, _, socket) do
|
|
||||||
{:noreply,
|
|
||||||
socket
|
|
||||||
|> assign(:page_title, page_title(socket.assigns.live_action))
|
|
||||||
|> assign(:post, Timeline.get_post!(id))}
|
|
||||||
end
|
|
||||||
|
|
||||||
defp page_title(:show), do: "Show Post"
|
|
||||||
defp page_title(:edit), do: "Edit Post"
|
|
||||||
end
|
|
||||||
@@ -1,29 +0,0 @@
|
|||||||
<.header>
|
|
||||||
Post <%= @post.id %>
|
|
||||||
<:subtitle>This is a post record from your database.</:subtitle>
|
|
||||||
<:actions>
|
|
||||||
<.link patch={~p"/posts/#{@post}/show/edit"} phx-click={JS.push_focus()}>
|
|
||||||
<.button>Edit post</.button>
|
|
||||||
</.link>
|
|
||||||
</:actions>
|
|
||||||
</.header>
|
|
||||||
|
|
||||||
<.list>
|
|
||||||
<:item title="Username"><%= @post.user.username %></:item>
|
|
||||||
<:item title="Body"><%= @post.body %></:item>
|
|
||||||
<:item title="Likes count"><%= @post.likes_count %></:item>
|
|
||||||
<:item title="Repost count"><%= @post.repost_count %></:item>
|
|
||||||
</.list>
|
|
||||||
|
|
||||||
<.back navigate={~p"/posts"}>Back to posts</.back>
|
|
||||||
|
|
||||||
<.modal :if={@live_action == :edit} id="post-modal" show on_cancel={JS.patch(~p"/posts/#{@post}")}>
|
|
||||||
<.live_component
|
|
||||||
module={MalarkeyWeb.PostLive.FormComponent}
|
|
||||||
id={@post.id}
|
|
||||||
title={@page_title}
|
|
||||||
action={@live_action}
|
|
||||||
post={@post}
|
|
||||||
navigate={~p"/posts/#{@post}"}
|
|
||||||
/>
|
|
||||||
</.modal>
|
|
||||||
@@ -1,45 +0,0 @@
|
|||||||
defmodule MalarkeyWeb.UserConfirmationInstructionsLive do
|
|
||||||
use MalarkeyWeb, :live_view
|
|
||||||
|
|
||||||
alias Malarkey.Accounts
|
|
||||||
|
|
||||||
def render(assigns) do
|
|
||||||
~H"""
|
|
||||||
<.header>Resend confirmation instructions</.header>
|
|
||||||
|
|
||||||
<.simple_form :let={f} for={:user} id="resend_confirmation_form" phx-submit="send_instructions">
|
|
||||||
<.input field={{f, :email}} type="email" label="Email" required />
|
|
||||||
<:actions>
|
|
||||||
<.button phx-disable-with="Sending...">Resend confirmation instructions</.button>
|
|
||||||
</:actions>
|
|
||||||
</.simple_form>
|
|
||||||
|
|
||||||
<p>
|
|
||||||
<.link href={~p"/users/register"}>Register</.link>
|
|
||||||
|
|
|
||||||
<.link href={~p"/users/log_in"}>Log in</.link>
|
|
||||||
</p>
|
|
||||||
"""
|
|
||||||
end
|
|
||||||
|
|
||||||
def mount(_params, _session, socket) do
|
|
||||||
{:ok, socket}
|
|
||||||
end
|
|
||||||
|
|
||||||
def handle_event("send_instructions", %{"user" => %{"email" => email}}, socket) do
|
|
||||||
if user = Accounts.get_user_by_email(email) do
|
|
||||||
Accounts.deliver_user_confirmation_instructions(
|
|
||||||
user,
|
|
||||||
&url(~p"/users/confirm/#{&1}")
|
|
||||||
)
|
|
||||||
end
|
|
||||||
|
|
||||||
info =
|
|
||||||
"If your email is in our system and it has not been confirmed yet, you will receive an email with instructions shortly."
|
|
||||||
|
|
||||||
{:noreply,
|
|
||||||
socket
|
|
||||||
|> put_flash(:info, info)
|
|
||||||
|> redirect(to: ~p"/")}
|
|
||||||
end
|
|
||||||
end
|
|
||||||
@@ -1,56 +0,0 @@
|
|||||||
defmodule MalarkeyWeb.UserConfirmationLive do
|
|
||||||
use MalarkeyWeb, :live_view
|
|
||||||
|
|
||||||
alias Malarkey.Accounts
|
|
||||||
|
|
||||||
def render(%{live_action: :edit} = assigns) do
|
|
||||||
~H"""
|
|
||||||
<.header>Confirm Account</.header>
|
|
||||||
|
|
||||||
<.simple_form :let={f} for={:user} id="confirmation_form" phx-submit="confirm_account">
|
|
||||||
<.input field={{f, :token}} type="hidden" value={@token} />
|
|
||||||
<:actions>
|
|
||||||
<.button phx-disable-with="Confirming...">Confirm my account</.button>
|
|
||||||
</:actions>
|
|
||||||
</.simple_form>
|
|
||||||
|
|
||||||
<p>
|
|
||||||
<.link href={~p"/users/register"}>Register</.link>
|
|
||||||
|
|
|
||||||
<.link href={~p"/users/log_in"}>Log in</.link>
|
|
||||||
</p>
|
|
||||||
"""
|
|
||||||
end
|
|
||||||
|
|
||||||
def mount(params, _session, socket) do
|
|
||||||
{:ok, assign(socket, token: params["token"]), temporary_assigns: [token: nil]}
|
|
||||||
end
|
|
||||||
|
|
||||||
# Do not log in the user after confirmation to avoid a
|
|
||||||
# leaked token giving the user access to the account.
|
|
||||||
def handle_event("confirm_account", %{"user" => %{"token" => token}}, socket) do
|
|
||||||
case Accounts.confirm_user(token) do
|
|
||||||
{:ok, _} ->
|
|
||||||
{:noreply,
|
|
||||||
socket
|
|
||||||
|> put_flash(:info, "User confirmed successfully.")
|
|
||||||
|> redirect(to: ~p"/")}
|
|
||||||
|
|
||||||
:error ->
|
|
||||||
# If there is a current user and the account was already confirmed,
|
|
||||||
# then odds are that the confirmation link was already visited, either
|
|
||||||
# by some automation or by the user themselves, so we redirect without
|
|
||||||
# a warning message.
|
|
||||||
case socket.assigns do
|
|
||||||
%{current_user: %{confirmed_at: confirmed_at}} when not is_nil(confirmed_at) ->
|
|
||||||
{:noreply, redirect(socket, to: ~p"/")}
|
|
||||||
|
|
||||||
%{} ->
|
|
||||||
{:noreply,
|
|
||||||
socket
|
|
||||||
|> put_flash(:error, "User confirmation link is invalid or it has expired.")
|
|
||||||
|> redirect(to: ~p"/")}
|
|
||||||
end
|
|
||||||
end
|
|
||||||
end
|
|
||||||
end
|
|
||||||
@@ -1,46 +0,0 @@
|
|||||||
defmodule MalarkeyWeb.UserForgotPasswordLive do
|
|
||||||
use MalarkeyWeb, :live_view
|
|
||||||
|
|
||||||
alias Malarkey.Accounts
|
|
||||||
|
|
||||||
def render(assigns) do
|
|
||||||
~H"""
|
|
||||||
<div class="mx-auto max-w-sm">
|
|
||||||
<.header class="text-center">
|
|
||||||
Forgot your password?
|
|
||||||
<:subtitle>We'll send a password reset link to your inbox</:subtitle>
|
|
||||||
</.header>
|
|
||||||
|
|
||||||
<.simple_form :let={f} id="reset_password_form" for={:user} phx-submit="send_email">
|
|
||||||
<.input field={{f, :email}} type="email" placeholder="Email" required />
|
|
||||||
<:actions>
|
|
||||||
<.button phx-disable-with="Sending..." class="w-full">
|
|
||||||
Send password reset instructions
|
|
||||||
</.button>
|
|
||||||
</:actions>
|
|
||||||
</.simple_form>
|
|
||||||
</div>
|
|
||||||
"""
|
|
||||||
end
|
|
||||||
|
|
||||||
def mount(_params, _session, socket) do
|
|
||||||
{:ok, socket}
|
|
||||||
end
|
|
||||||
|
|
||||||
def handle_event("send_email", %{"user" => %{"email" => email}}, socket) do
|
|
||||||
if user = Accounts.get_user_by_email(email) do
|
|
||||||
Accounts.deliver_user_reset_password_instructions(
|
|
||||||
user,
|
|
||||||
&url(~p"/users/reset_password/#{&1}")
|
|
||||||
)
|
|
||||||
end
|
|
||||||
|
|
||||||
info =
|
|
||||||
"If your email is in our system, you will receive instructions to reset your password shortly."
|
|
||||||
|
|
||||||
{:noreply,
|
|
||||||
socket
|
|
||||||
|> put_flash(:info, info)
|
|
||||||
|> redirect(to: ~p"/")}
|
|
||||||
end
|
|
||||||
end
|
|
||||||
@@ -1,49 +0,0 @@
|
|||||||
defmodule MalarkeyWeb.UserLoginLive do
|
|
||||||
use MalarkeyWeb, :live_view
|
|
||||||
|
|
||||||
def render(assigns) do
|
|
||||||
~H"""
|
|
||||||
<div class="mx-auto max-w-sm">
|
|
||||||
<.header class="text-center">
|
|
||||||
Sign in to account
|
|
||||||
<:subtitle>
|
|
||||||
Don't have an account?
|
|
||||||
<.link navigate={~p"/users/register"} class="font-semibold text-brand hover:underline">
|
|
||||||
Sign up
|
|
||||||
</.link>
|
|
||||||
for an account now.
|
|
||||||
</:subtitle>
|
|
||||||
</.header>
|
|
||||||
|
|
||||||
<.simple_form
|
|
||||||
:let={f}
|
|
||||||
id="login_form"
|
|
||||||
for={:user}
|
|
||||||
action={~p"/users/log_in"}
|
|
||||||
as={:user}
|
|
||||||
phx-update="ignore"
|
|
||||||
>
|
|
||||||
<.input field={{f, :email}} type="email" label="Email" required />
|
|
||||||
<.input field={{f, :password}} type="password" label="Password" required />
|
|
||||||
|
|
||||||
<:actions :let={f}>
|
|
||||||
<.input field={{f, :remember_me}} type="checkbox" label="Keep me logged in" />
|
|
||||||
<.link href={~p"/users/reset_password"} class="text-sm font-semibold">
|
|
||||||
Forgot your password?
|
|
||||||
</.link>
|
|
||||||
</:actions>
|
|
||||||
<:actions>
|
|
||||||
<.button phx-disable-with="Sigining in..." class="w-full">
|
|
||||||
Sign in <span aria-hidden="true">→</span>
|
|
||||||
</.button>
|
|
||||||
</:actions>
|
|
||||||
</.simple_form>
|
|
||||||
</div>
|
|
||||||
"""
|
|
||||||
end
|
|
||||||
|
|
||||||
def mount(_params, _session, socket) do
|
|
||||||
email = live_flash(socket.assigns.flash, :email)
|
|
||||||
{:ok, assign(socket, email: email), temporary_assigns: [email: nil]}
|
|
||||||
end
|
|
||||||
end
|
|
||||||
@@ -1,74 +0,0 @@
|
|||||||
defmodule MalarkeyWeb.UserRegistrationLive do
|
|
||||||
use MalarkeyWeb, :live_view
|
|
||||||
|
|
||||||
alias Malarkey.Accounts
|
|
||||||
alias Malarkey.Accounts.User
|
|
||||||
|
|
||||||
def render(assigns) do
|
|
||||||
~H"""
|
|
||||||
<div class="max-w-sm mx-auto">
|
|
||||||
<.header class="text-center">
|
|
||||||
Register for an account
|
|
||||||
<:subtitle>
|
|
||||||
Already registered?
|
|
||||||
<.link navigate={~p"/users/log_in"} class="font-semibold text-brand hover:underline">
|
|
||||||
Sign in
|
|
||||||
</.link>
|
|
||||||
to your account now.
|
|
||||||
</:subtitle>
|
|
||||||
</.header>
|
|
||||||
|
|
||||||
<.simple_form
|
|
||||||
:let={f}
|
|
||||||
id="registration_form"
|
|
||||||
for={@changeset}
|
|
||||||
phx-submit="save"
|
|
||||||
phx-change="validate"
|
|
||||||
phx-trigger-action={@trigger_submit}
|
|
||||||
action={~p"/users/log_in?_action=registered"}
|
|
||||||
method="post"
|
|
||||||
as={:user}
|
|
||||||
>
|
|
||||||
<.error :if={@changeset.action == :insert}>
|
|
||||||
Oops, something went wrong! Please check the errors below.
|
|
||||||
</.error>
|
|
||||||
|
|
||||||
<.input field={{f, :email}} type="email" label="Email" required />
|
|
||||||
<.input field={{f, :password}} type="password" label="Password" required />
|
|
||||||
|
|
||||||
<:actions>
|
|
||||||
<.button phx-disable-with="Creating account..." class="w-full">Create an account</.button>
|
|
||||||
</:actions>
|
|
||||||
</.simple_form>
|
|
||||||
</div>
|
|
||||||
"""
|
|
||||||
end
|
|
||||||
|
|
||||||
def mount(_params, _session, socket) do
|
|
||||||
changeset = Accounts.change_user_registration(%User{})
|
|
||||||
socket = assign(socket, changeset: changeset, trigger_submit: false)
|
|
||||||
{:ok, socket, temporary_assigns: [changeset: nil]}
|
|
||||||
end
|
|
||||||
|
|
||||||
def handle_event("save", %{"user" => user_params}, socket) do
|
|
||||||
case Accounts.register_user(user_params) do
|
|
||||||
{:ok, user} ->
|
|
||||||
{:ok, _} =
|
|
||||||
Accounts.deliver_user_confirmation_instructions(
|
|
||||||
user,
|
|
||||||
&url(~p"/users/confirm/#{&1}")
|
|
||||||
)
|
|
||||||
|
|
||||||
changeset = Accounts.change_user_registration(user)
|
|
||||||
{:noreply, assign(socket, trigger_submit: true, changeset: changeset)}
|
|
||||||
|
|
||||||
{:error, %Ecto.Changeset{} = changeset} ->
|
|
||||||
{:noreply, assign(socket, :changeset, changeset)}
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
def handle_event("validate", %{"user" => user_params}, socket) do
|
|
||||||
changeset = Accounts.change_user_registration(%User{}, user_params)
|
|
||||||
{:noreply, assign(socket, changeset: Map.put(changeset, :action, :validate))}
|
|
||||||
end
|
|
||||||
end
|
|
||||||
@@ -1,85 +0,0 @@
|
|||||||
defmodule MalarkeyWeb.UserResetPasswordLive do
|
|
||||||
use MalarkeyWeb, :live_view
|
|
||||||
|
|
||||||
alias Malarkey.Accounts
|
|
||||||
|
|
||||||
def render(assigns) do
|
|
||||||
~H"""
|
|
||||||
<.header>Reset Password</.header>
|
|
||||||
|
|
||||||
<.simple_form
|
|
||||||
:let={f}
|
|
||||||
for={@changeset}
|
|
||||||
id="reset_password_form"
|
|
||||||
phx-submit="reset_password"
|
|
||||||
phx-change="validate"
|
|
||||||
>
|
|
||||||
<.error :if={@changeset.action == :insert}>
|
|
||||||
Oops, something went wrong! Please check the errors below.
|
|
||||||
</.error>
|
|
||||||
|
|
||||||
<.input field={{f, :password}} type="password" label="New password" required />
|
|
||||||
<.input
|
|
||||||
field={{f, :password_confirmation}}
|
|
||||||
type="password"
|
|
||||||
label="Confirm new password"
|
|
||||||
required
|
|
||||||
/>
|
|
||||||
<:actions>
|
|
||||||
<.button phx-disable-with="Resetting...">Reset Password</.button>
|
|
||||||
</:actions>
|
|
||||||
</.simple_form>
|
|
||||||
|
|
||||||
<p>
|
|
||||||
<.link href={~p"/users/register"}>Register</.link>
|
|
||||||
|
|
|
||||||
<.link href={~p"/users/log_in"}>Log in</.link>
|
|
||||||
</p>
|
|
||||||
"""
|
|
||||||
end
|
|
||||||
|
|
||||||
def mount(params, _session, socket) do
|
|
||||||
socket = assign_user_and_token(socket, params)
|
|
||||||
|
|
||||||
socket =
|
|
||||||
case socket.assigns do
|
|
||||||
%{user: user} ->
|
|
||||||
assign(socket, :changeset, Accounts.change_user_password(user))
|
|
||||||
|
|
||||||
_ ->
|
|
||||||
socket
|
|
||||||
end
|
|
||||||
|
|
||||||
{:ok, socket, temporary_assigns: [changeset: nil]}
|
|
||||||
end
|
|
||||||
|
|
||||||
# Do not log in the user after reset password to avoid a
|
|
||||||
# leaked token giving the user access to the account.
|
|
||||||
def handle_event("reset_password", %{"user" => user_params}, socket) do
|
|
||||||
case Accounts.reset_user_password(socket.assigns.user, user_params) do
|
|
||||||
{:ok, _} ->
|
|
||||||
{:noreply,
|
|
||||||
socket
|
|
||||||
|> put_flash(:info, "Password reset successfully.")
|
|
||||||
|> redirect(to: ~p"/users/log_in")}
|
|
||||||
|
|
||||||
{:error, changeset} ->
|
|
||||||
{:noreply, assign(socket, :changeset, Map.put(changeset, :action, :insert))}
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
def handle_event("validate", %{"user" => user_params}, socket) do
|
|
||||||
changeset = Accounts.change_user_password(socket.assigns.user, user_params)
|
|
||||||
{:noreply, assign(socket, changeset: Map.put(changeset, :action, :validate))}
|
|
||||||
end
|
|
||||||
|
|
||||||
defp assign_user_and_token(socket, %{"token" => token}) do
|
|
||||||
if user = Accounts.get_user_by_reset_password_token(token) do
|
|
||||||
assign(socket, user: user, token: token)
|
|
||||||
else
|
|
||||||
socket
|
|
||||||
|> put_flash(:error, "Reset password link is invalid or it has expired.")
|
|
||||||
|> redirect(to: ~p"/")
|
|
||||||
end
|
|
||||||
end
|
|
||||||
end
|
|
||||||
@@ -1,161 +0,0 @@
|
|||||||
defmodule MalarkeyWeb.UserSettingsLive do
|
|
||||||
use MalarkeyWeb, :live_view
|
|
||||||
|
|
||||||
alias Malarkey.Accounts
|
|
||||||
|
|
||||||
def render(assigns) do
|
|
||||||
~H"""
|
|
||||||
<.header>Change Email</.header>
|
|
||||||
|
|
||||||
<.simple_form
|
|
||||||
:let={f}
|
|
||||||
id="email_form"
|
|
||||||
for={@email_changeset}
|
|
||||||
phx-submit="update_email"
|
|
||||||
phx-change="validate_email"
|
|
||||||
>
|
|
||||||
<.error :if={@email_changeset.action == :insert}>
|
|
||||||
Oops, something went wrong! Please check the errors below.
|
|
||||||
</.error>
|
|
||||||
|
|
||||||
<.input field={{f, :email}} type="email" label="Email" required />
|
|
||||||
|
|
||||||
<.input
|
|
||||||
field={{f, :current_password}}
|
|
||||||
name="current_password"
|
|
||||||
id="current_password_for_email"
|
|
||||||
type="password"
|
|
||||||
label="Current password"
|
|
||||||
value={@email_form_current_password}
|
|
||||||
required
|
|
||||||
/>
|
|
||||||
<:actions>
|
|
||||||
<.button phx-disable-with="Changing...">Change Email</.button>
|
|
||||||
</:actions>
|
|
||||||
</.simple_form>
|
|
||||||
|
|
||||||
<.header>Change Password</.header>
|
|
||||||
|
|
||||||
<.simple_form
|
|
||||||
:let={f}
|
|
||||||
id="password_form"
|
|
||||||
for={@password_changeset}
|
|
||||||
action={~p"/users/log_in?_action=password_updated"}
|
|
||||||
method="post"
|
|
||||||
phx-change="validate_password"
|
|
||||||
phx-submit="update_password"
|
|
||||||
phx-trigger-action={@trigger_submit}
|
|
||||||
>
|
|
||||||
<.error :if={@password_changeset.action == :insert}>
|
|
||||||
Oops, something went wrong! Please check the errors below.
|
|
||||||
</.error>
|
|
||||||
|
|
||||||
<.input field={{f, :email}} type="hidden" value={@current_email} />
|
|
||||||
|
|
||||||
<.input field={{f, :password}} type="password" label="New password" required />
|
|
||||||
<.input field={{f, :password_confirmation}} type="password" label="Confirm new password" />
|
|
||||||
<.input
|
|
||||||
field={{f, :current_password}}
|
|
||||||
name="current_password"
|
|
||||||
type="password"
|
|
||||||
label="Current password"
|
|
||||||
id="current_password_for_password"
|
|
||||||
value={@current_password}
|
|
||||||
required
|
|
||||||
/>
|
|
||||||
<:actions>
|
|
||||||
<.button phx-disable-with="Changing...">Change Password</.button>
|
|
||||||
</:actions>
|
|
||||||
</.simple_form>
|
|
||||||
"""
|
|
||||||
end
|
|
||||||
|
|
||||||
def mount(%{"token" => token}, _session, socket) do
|
|
||||||
socket =
|
|
||||||
case Accounts.update_user_email(socket.assigns.current_user, token) do
|
|
||||||
:ok ->
|
|
||||||
put_flash(socket, :info, "Email changed successfully.")
|
|
||||||
|
|
||||||
:error ->
|
|
||||||
put_flash(socket, :error, "Email change link is invalid or it has expired.")
|
|
||||||
end
|
|
||||||
|
|
||||||
{:ok, push_navigate(socket, to: ~p"/users/settings")}
|
|
||||||
end
|
|
||||||
|
|
||||||
def mount(_params, _session, socket) do
|
|
||||||
user = socket.assigns.current_user
|
|
||||||
|
|
||||||
socket =
|
|
||||||
socket
|
|
||||||
|> assign(:current_password, nil)
|
|
||||||
|> assign(:email_form_current_password, nil)
|
|
||||||
|> assign(:current_email, user.email)
|
|
||||||
|> assign(:email_changeset, Accounts.change_user_email(user))
|
|
||||||
|> assign(:password_changeset, Accounts.change_user_password(user))
|
|
||||||
|> assign(:trigger_submit, false)
|
|
||||||
|
|
||||||
{:ok, socket}
|
|
||||||
end
|
|
||||||
|
|
||||||
def handle_event("validate_email", params, socket) do
|
|
||||||
%{"current_password" => password, "user" => user_params} = params
|
|
||||||
email_changeset = Accounts.change_user_email(socket.assigns.current_user, user_params)
|
|
||||||
|
|
||||||
socket =
|
|
||||||
assign(socket,
|
|
||||||
email_changeset: Map.put(email_changeset, :action, :validate),
|
|
||||||
email_form_current_password: password
|
|
||||||
)
|
|
||||||
|
|
||||||
{:noreply, socket}
|
|
||||||
end
|
|
||||||
|
|
||||||
def handle_event("update_email", params, socket) do
|
|
||||||
%{"current_password" => password, "user" => user_params} = params
|
|
||||||
user = socket.assigns.current_user
|
|
||||||
|
|
||||||
case Accounts.apply_user_email(user, password, user_params) do
|
|
||||||
{:ok, applied_user} ->
|
|
||||||
Accounts.deliver_user_update_email_instructions(
|
|
||||||
applied_user,
|
|
||||||
user.email,
|
|
||||||
&url(~p"/users/settings/confirm_email/#{&1}")
|
|
||||||
)
|
|
||||||
|
|
||||||
info = "A link to confirm your email change has been sent to the new address."
|
|
||||||
{:noreply, put_flash(socket, :info, info)}
|
|
||||||
|
|
||||||
{:error, changeset} ->
|
|
||||||
{:noreply, assign(socket, :email_changeset, Map.put(changeset, :action, :insert))}
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
def handle_event("validate_password", params, socket) do
|
|
||||||
%{"current_password" => password, "user" => user_params} = params
|
|
||||||
password_changeset = Accounts.change_user_password(socket.assigns.current_user, user_params)
|
|
||||||
|
|
||||||
{:noreply,
|
|
||||||
socket
|
|
||||||
|> assign(:password_changeset, Map.put(password_changeset, :action, :validate))
|
|
||||||
|> assign(:current_password, password)}
|
|
||||||
end
|
|
||||||
|
|
||||||
def handle_event("update_password", params, socket) do
|
|
||||||
%{"current_password" => password, "user" => user_params} = params
|
|
||||||
user = socket.assigns.current_user
|
|
||||||
|
|
||||||
case Accounts.update_user_password(user, password, user_params) do
|
|
||||||
{:ok, user} ->
|
|
||||||
socket =
|
|
||||||
socket
|
|
||||||
|> assign(:trigger_submit, true)
|
|
||||||
|> assign(:password_changeset, Accounts.change_user_password(user, user_params))
|
|
||||||
|
|
||||||
{:noreply, socket}
|
|
||||||
|
|
||||||
{:error, changeset} ->
|
|
||||||
{:noreply, assign(socket, :password_changeset, changeset)}
|
|
||||||
end
|
|
||||||
end
|
|
||||||
end
|
|
||||||
@@ -1,20 +0,0 @@
|
|||||||
defmodule MalarkeyWeb.UserLiveAuth do
|
|
||||||
require Logger
|
|
||||||
import Phoenix.Component
|
|
||||||
alias Malarkey.Accounts
|
|
||||||
alias Accounts.User
|
|
||||||
|
|
||||||
@spec on_mount(:default, any, any, map) ::
|
|
||||||
{:cont, %{:assigns => atom | map, optional(any) => any}}
|
|
||||||
def on_mount(:default, _params, session, socket) do
|
|
||||||
socket = assign_new(socket, :current_user, fn -> get_current_user(session) end)
|
|
||||||
{:cont, socket}
|
|
||||||
end
|
|
||||||
|
|
||||||
defp get_current_user(session) do
|
|
||||||
with user_token when not is_nil(user_token) <- session["user_token"],
|
|
||||||
%User{} = user <- Accounts.get_user_by_session_token(session["user_token"]) do
|
|
||||||
user
|
|
||||||
end
|
|
||||||
end
|
|
||||||
end
|
|
||||||
@@ -1,92 +0,0 @@
|
|||||||
defmodule MalarkeyWeb.Router do
|
|
||||||
use MalarkeyWeb, :router
|
|
||||||
|
|
||||||
import MalarkeyWeb.UserAuth
|
|
||||||
|
|
||||||
pipeline :browser do
|
|
||||||
plug :accepts, ["html"]
|
|
||||||
plug :fetch_session
|
|
||||||
plug :fetch_live_flash
|
|
||||||
plug :put_root_layout, {MalarkeyWeb.Layouts, :root}
|
|
||||||
plug :protect_from_forgery
|
|
||||||
plug :put_secure_browser_headers
|
|
||||||
plug :fetch_current_user
|
|
||||||
end
|
|
||||||
|
|
||||||
pipeline :api do
|
|
||||||
plug :accepts, ["json"]
|
|
||||||
end
|
|
||||||
|
|
||||||
scope "/", MalarkeyWeb do
|
|
||||||
pipe_through :browser
|
|
||||||
|
|
||||||
live "/", PostLive.Index, :index
|
|
||||||
live "/posts", PostLive.Index, :index
|
|
||||||
live "/posts/new", PostLive.Index, :new
|
|
||||||
live "/posts/:id/like", PostLive.Index, :like
|
|
||||||
live "/posts/:id/edit", PostLive.Index, :edit
|
|
||||||
|
|
||||||
live "/posts/:id", PostLive.Show, :show
|
|
||||||
live "/posts/:id/show/edit", PostLive.Show, :edit
|
|
||||||
end
|
|
||||||
|
|
||||||
# Other scopes may use custom stacks.
|
|
||||||
# scope "/api", MalarkeyWeb do
|
|
||||||
# pipe_through :api
|
|
||||||
# end
|
|
||||||
|
|
||||||
# Enable LiveDashboard and Swoosh mailbox preview in development
|
|
||||||
if Application.compile_env(:malarkey, :dev_routes) do
|
|
||||||
# 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).
|
|
||||||
import Phoenix.LiveDashboard.Router
|
|
||||||
|
|
||||||
scope "/dev" do
|
|
||||||
pipe_through :browser
|
|
||||||
|
|
||||||
live_dashboard "/dashboard", metrics: MalarkeyWeb.Telemetry
|
|
||||||
forward "/mailbox", Plug.Swoosh.MailboxPreview
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
## Authentication routes
|
|
||||||
|
|
||||||
scope "/", MalarkeyWeb do
|
|
||||||
pipe_through [:browser, :redirect_if_user_is_authenticated]
|
|
||||||
|
|
||||||
live_session :redirect_if_user_is_authenticated,
|
|
||||||
on_mount: [{MalarkeyWeb.UserAuth, :redirect_if_user_is_authenticated}] do
|
|
||||||
live "/users/register", UserRegistrationLive, :new
|
|
||||||
live "/users/log_in", UserLoginLive, :new
|
|
||||||
live "/users/reset_password", UserForgotPasswordLive, :new
|
|
||||||
live "/users/reset_password/:token", UserResetPasswordLive, :edit
|
|
||||||
end
|
|
||||||
|
|
||||||
post "/users/log_in", UserSessionController, :create
|
|
||||||
end
|
|
||||||
|
|
||||||
scope "/", MalarkeyWeb do
|
|
||||||
pipe_through [:browser, :require_authenticated_user]
|
|
||||||
|
|
||||||
live_session :require_authenticated_user,
|
|
||||||
on_mount: [{MalarkeyWeb.UserAuth, :ensure_authenticated}] do
|
|
||||||
live "/users/settings", UserSettingsLive, :edit
|
|
||||||
live "/users/settings/confirm_email/:token", UserSettingsLive, :confirm_email
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
scope "/", MalarkeyWeb do
|
|
||||||
pipe_through [:browser]
|
|
||||||
|
|
||||||
delete "/users/log_out", UserSessionController, :delete
|
|
||||||
|
|
||||||
live_session :current_user,
|
|
||||||
on_mount: [{MalarkeyWeb.UserAuth, :mount_current_user}] do
|
|
||||||
live "/users/confirm/:token", UserConfirmationLive, :edit
|
|
||||||
live "/users/confirm", UserConfirmationInstructionsLive, :new
|
|
||||||
end
|
|
||||||
end
|
|
||||||
end
|
|
||||||
@@ -1,92 +0,0 @@
|
|||||||
defmodule MalarkeyWeb.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.start.system_time",
|
|
||||||
unit: {:native, :millisecond}
|
|
||||||
),
|
|
||||||
summary("phoenix.endpoint.stop.duration",
|
|
||||||
unit: {:native, :millisecond}
|
|
||||||
),
|
|
||||||
summary("phoenix.router_dispatch.start.system_time",
|
|
||||||
tags: [:route],
|
|
||||||
unit: {:native, :millisecond}
|
|
||||||
),
|
|
||||||
summary("phoenix.router_dispatch.exception.duration",
|
|
||||||
tags: [:route],
|
|
||||||
unit: {:native, :millisecond}
|
|
||||||
),
|
|
||||||
summary("phoenix.router_dispatch.stop.duration",
|
|
||||||
tags: [:route],
|
|
||||||
unit: {:native, :millisecond}
|
|
||||||
),
|
|
||||||
summary("phoenix.socket_connected.duration",
|
|
||||||
unit: {:native, :millisecond}
|
|
||||||
),
|
|
||||||
summary("phoenix.channel_join.duration",
|
|
||||||
unit: {:native, :millisecond}
|
|
||||||
),
|
|
||||||
summary("phoenix.channel_handled_in.duration",
|
|
||||||
tags: [:event],
|
|
||||||
unit: {:native, :millisecond}
|
|
||||||
),
|
|
||||||
|
|
||||||
# Database Metrics
|
|
||||||
summary("malarkey.repo.query.total_time",
|
|
||||||
unit: {:native, :millisecond},
|
|
||||||
description: "The sum of the other measurements"
|
|
||||||
),
|
|
||||||
summary("malarkey.repo.query.decode_time",
|
|
||||||
unit: {:native, :millisecond},
|
|
||||||
description: "The time spent decoding the data received from the database"
|
|
||||||
),
|
|
||||||
summary("malarkey.repo.query.query_time",
|
|
||||||
unit: {:native, :millisecond},
|
|
||||||
description: "The time spent executing the query"
|
|
||||||
),
|
|
||||||
summary("malarkey.repo.query.queue_time",
|
|
||||||
unit: {:native, :millisecond},
|
|
||||||
description: "The time spent waiting for a database connection"
|
|
||||||
),
|
|
||||||
summary("malarkey.repo.query.idle_time",
|
|
||||||
unit: {:native, :millisecond},
|
|
||||||
description:
|
|
||||||
"The time the connection spent waiting before being checked out for the query"
|
|
||||||
),
|
|
||||||
|
|
||||||
# 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.
|
|
||||||
# {MalarkeyWeb, :count_users, []}
|
|
||||||
]
|
|
||||||
end
|
|
||||||
end
|
|
||||||
@@ -1,230 +0,0 @@
|
|||||||
defmodule MalarkeyWeb.UserAuth do
|
|
||||||
use MalarkeyWeb, :verified_routes
|
|
||||||
import Plug.Conn
|
|
||||||
import Phoenix.Controller
|
|
||||||
|
|
||||||
alias Malarkey.Accounts
|
|
||||||
|
|
||||||
# Make the remember me cookie valid for 60 days.
|
|
||||||
# If you want bump or reduce this value, also change
|
|
||||||
# the token expiry itself in UserToken.
|
|
||||||
@max_age 60 * 60 * 24 * 60
|
|
||||||
@remember_me_cookie "_malarkey_web_user_remember_me"
|
|
||||||
@remember_me_options [sign: true, max_age: @max_age, same_site: "Lax"]
|
|
||||||
|
|
||||||
@doc """
|
|
||||||
Logs the user in.
|
|
||||||
|
|
||||||
It renews the session ID and clears the whole session
|
|
||||||
to avoid fixation attacks. See the renew_session
|
|
||||||
function to customize this behaviour.
|
|
||||||
|
|
||||||
It also sets a `:live_socket_id` key in the session,
|
|
||||||
so LiveView sessions are identified and automatically
|
|
||||||
disconnected on log out. The line can be safely removed
|
|
||||||
if you are not using LiveView.
|
|
||||||
"""
|
|
||||||
def log_in_user(conn, user, params \\ %{}) do
|
|
||||||
token = Accounts.generate_user_session_token(user)
|
|
||||||
user_return_to = get_session(conn, :user_return_to)
|
|
||||||
|
|
||||||
conn
|
|
||||||
|> renew_session()
|
|
||||||
|> put_token_in_session(token)
|
|
||||||
|> maybe_write_remember_me_cookie(token, params)
|
|
||||||
|> redirect(to: user_return_to || signed_in_path(conn))
|
|
||||||
end
|
|
||||||
|
|
||||||
defp maybe_write_remember_me_cookie(conn, token, %{"remember_me" => "true"}) do
|
|
||||||
put_resp_cookie(conn, @remember_me_cookie, token, @remember_me_options)
|
|
||||||
end
|
|
||||||
|
|
||||||
defp maybe_write_remember_me_cookie(conn, _token, _params) do
|
|
||||||
conn
|
|
||||||
end
|
|
||||||
|
|
||||||
# This function renews the session ID and erases the whole
|
|
||||||
# session to avoid fixation attacks. If there is any data
|
|
||||||
# in the session you may want to preserve after log in/log out,
|
|
||||||
# you must explicitly fetch the session data before clearing
|
|
||||||
# and then immediately set it after clearing, for example:
|
|
||||||
#
|
|
||||||
# defp renew_session(conn) do
|
|
||||||
# preferred_locale = get_session(conn, :preferred_locale)
|
|
||||||
#
|
|
||||||
# conn
|
|
||||||
# |> configure_session(renew: true)
|
|
||||||
# |> clear_session()
|
|
||||||
# |> put_session(:preferred_locale, preferred_locale)
|
|
||||||
# end
|
|
||||||
#
|
|
||||||
defp renew_session(conn) do
|
|
||||||
conn
|
|
||||||
|> configure_session(renew: true)
|
|
||||||
|> clear_session()
|
|
||||||
end
|
|
||||||
|
|
||||||
@doc """
|
|
||||||
Logs the user out.
|
|
||||||
|
|
||||||
It clears all session data for safety. See renew_session.
|
|
||||||
"""
|
|
||||||
def log_out_user(conn) do
|
|
||||||
user_token = get_session(conn, :user_token)
|
|
||||||
user_token && Accounts.delete_user_session_token(user_token)
|
|
||||||
|
|
||||||
if live_socket_id = get_session(conn, :live_socket_id) do
|
|
||||||
MalarkeyWeb.Endpoint.broadcast(live_socket_id, "disconnect", %{})
|
|
||||||
end
|
|
||||||
|
|
||||||
conn
|
|
||||||
|> renew_session()
|
|
||||||
|> delete_resp_cookie(@remember_me_cookie)
|
|
||||||
|> redirect(to: "/")
|
|
||||||
end
|
|
||||||
|
|
||||||
@doc """
|
|
||||||
Authenticates the user by looking into the session
|
|
||||||
and remember me token.
|
|
||||||
"""
|
|
||||||
def fetch_current_user(conn, _opts) do
|
|
||||||
{user_token, conn} = ensure_user_token(conn)
|
|
||||||
user = user_token && Accounts.get_user_by_session_token(user_token)
|
|
||||||
assign(conn, :current_user, user || nil)
|
|
||||||
end
|
|
||||||
|
|
||||||
defp ensure_user_token(conn) do
|
|
||||||
if token = get_session(conn, :user_token) do
|
|
||||||
{token, conn}
|
|
||||||
else
|
|
||||||
conn = fetch_cookies(conn, signed: [@remember_me_cookie])
|
|
||||||
|
|
||||||
if token = conn.cookies[@remember_me_cookie] do
|
|
||||||
{token, put_token_in_session(conn, token)}
|
|
||||||
else
|
|
||||||
{nil, conn}
|
|
||||||
end
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
@doc """
|
|
||||||
Handles mounting and authenticating the current_user in LiveViews.
|
|
||||||
|
|
||||||
## `on_mount` arguments
|
|
||||||
|
|
||||||
* `:mount_current_user` - Assigns current_user
|
|
||||||
to socket assigns based on user_token, or nil if
|
|
||||||
there's no user_token or no matching user.
|
|
||||||
|
|
||||||
* `:ensure_authenticated` - Authenticates the user from the session,
|
|
||||||
and assigns the current_user to socket assigns based
|
|
||||||
on user_token.
|
|
||||||
Redirects to login page if there's no logged user.
|
|
||||||
|
|
||||||
* `:redirect_if_user_is_authenticated` - Authenticates the user from the session.
|
|
||||||
Redirects to signed_in_path if there's a logged user.
|
|
||||||
|
|
||||||
## Examples
|
|
||||||
|
|
||||||
Use the `on_mount` lifecycle macro in LiveViews to mount or authenticate
|
|
||||||
the current_user:
|
|
||||||
|
|
||||||
defmodule MalarkeyWeb.PageLive do
|
|
||||||
use MalarkeyWeb, :live_view
|
|
||||||
|
|
||||||
on_mount {MalarkeyWeb.UserAuth, :mount_current_user}
|
|
||||||
...
|
|
||||||
end
|
|
||||||
|
|
||||||
Or use the `live_session` of your router to invoke the on_mount callback:
|
|
||||||
|
|
||||||
live_session :authenticated, on_mount: [{MalarkeyWeb.UserAuth, :ensure_authenticated}] do
|
|
||||||
live "/profile", ProfileLive, :index
|
|
||||||
end
|
|
||||||
"""
|
|
||||||
def on_mount(:mount_current_user, _params, session, socket) do
|
|
||||||
{:cont, mount_current_user(session, socket)}
|
|
||||||
end
|
|
||||||
|
|
||||||
def on_mount(:ensure_authenticated, _params, session, socket) do
|
|
||||||
socket = mount_current_user(session, socket)
|
|
||||||
|
|
||||||
if socket.assigns.current_user do
|
|
||||||
{:cont, socket}
|
|
||||||
else
|
|
||||||
socket =
|
|
||||||
socket
|
|
||||||
|> Phoenix.LiveView.put_flash(:error, "You must log in to access this page.")
|
|
||||||
|> Phoenix.LiveView.redirect(to: ~p"/users/log_in")
|
|
||||||
|
|
||||||
{:halt, socket}
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
def on_mount(:redirect_if_user_is_authenticated, _params, session, socket) do
|
|
||||||
socket = mount_current_user(session, socket)
|
|
||||||
|
|
||||||
if socket.assigns.current_user do
|
|
||||||
{:halt, Phoenix.LiveView.redirect(socket, to: signed_in_path(socket))}
|
|
||||||
else
|
|
||||||
{:cont, socket}
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
defp mount_current_user(session, socket) do
|
|
||||||
case session do
|
|
||||||
%{"user_token" => user_token} ->
|
|
||||||
Phoenix.Component.assign_new(socket, :current_user, fn ->
|
|
||||||
Accounts.get_user_by_session_token(user_token)
|
|
||||||
end)
|
|
||||||
|
|
||||||
%{} ->
|
|
||||||
Phoenix.Component.assign_new(socket, :current_user, fn -> nil end)
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
@doc """
|
|
||||||
Used for routes that require the user to not be authenticated.
|
|
||||||
"""
|
|
||||||
def redirect_if_user_is_authenticated(conn, _opts) do
|
|
||||||
if conn.assigns[:current_user] do
|
|
||||||
conn
|
|
||||||
|> redirect(to: signed_in_path(conn))
|
|
||||||
|> halt()
|
|
||||||
else
|
|
||||||
conn
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
@doc """
|
|
||||||
Used for routes that require the user to be authenticated.
|
|
||||||
|
|
||||||
If you want to enforce the user email is confirmed before
|
|
||||||
they use the application at all, here would be a good place.
|
|
||||||
"""
|
|
||||||
def require_authenticated_user(conn, _opts) do
|
|
||||||
if conn.assigns[:current_user] do
|
|
||||||
conn
|
|
||||||
else
|
|
||||||
conn
|
|
||||||
|> put_flash(:error, "You must log in to access this page.")
|
|
||||||
|> maybe_store_return_to()
|
|
||||||
|> redirect(to: ~p"/users/log_in")
|
|
||||||
|> halt()
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
defp put_token_in_session(conn, token) do
|
|
||||||
conn
|
|
||||||
|> put_session(:user_token, token)
|
|
||||||
|> put_session(:live_socket_id, "users_sessions:#{Base.url_encode64(token)}")
|
|
||||||
end
|
|
||||||
|
|
||||||
defp maybe_store_return_to(%{method: "GET"} = conn) do
|
|
||||||
put_session(conn, :user_return_to, current_path(conn))
|
|
||||||
end
|
|
||||||
|
|
||||||
defp maybe_store_return_to(conn), do: conn
|
|
||||||
|
|
||||||
defp signed_in_path(_conn), do: ~p"/"
|
|
||||||
end
|
|
||||||
73
mix.exs
73
mix.exs
@@ -1,73 +0,0 @@
|
|||||||
defmodule Malarkey.MixProject do
|
|
||||||
use Mix.Project
|
|
||||||
|
|
||||||
def project do
|
|
||||||
[
|
|
||||||
app: :malarkey,
|
|
||||||
version: "0.1.0",
|
|
||||||
elixir: "~> 1.14",
|
|
||||||
elixirc_paths: elixirc_paths(Mix.env()),
|
|
||||||
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: {Malarkey.Application, []},
|
|
||||||
extra_applications: [:logger, :runtime_tools]
|
|
||||||
]
|
|
||||||
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
|
|
||||||
[
|
|
||||||
{:bcrypt_elixir, "~> 3.0"},
|
|
||||||
{:phoenix, "~> 1.7.0-rc.0", override: true},
|
|
||||||
{:phoenix_ecto, "~> 4.4"},
|
|
||||||
{:ecto_sql, "~> 3.6"},
|
|
||||||
{:postgrex, ">= 0.0.0"},
|
|
||||||
{:phoenix_html, "~> 3.0"},
|
|
||||||
{:phoenix_live_reload, "~> 1.2", only: :dev},
|
|
||||||
{:phoenix_live_view, "~> 0.18.3"},
|
|
||||||
{:floki, ">= 0.30.0", only: :test},
|
|
||||||
{:phoenix_live_dashboard, "~> 0.7.2"},
|
|
||||||
{:esbuild, "~> 0.5", runtime: Mix.env() == :dev},
|
|
||||||
{:tailwind, "~> 0.1.8", runtime: Mix.env() == :dev},
|
|
||||||
{:swoosh, "~> 1.3"},
|
|
||||||
{:finch, "~> 0.13"},
|
|
||||||
{:telemetry_metrics, "~> 0.6"},
|
|
||||||
{:telemetry_poller, "~> 1.0"},
|
|
||||||
{:gettext, "~> 0.20"},
|
|
||||||
{:jason, "~> 1.2"},
|
|
||||||
{:plug_cowboy, "~> 2.5"},
|
|
||||||
{:ex_heroicons, "~> 2.0.0"}
|
|
||||||
]
|
|
||||||
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", "ecto.setup"],
|
|
||||||
"ecto.setup": ["ecto.create", "ecto.migrate", "run priv/repo/seeds.exs"],
|
|
||||||
"ecto.reset": ["ecto.drop", "ecto.setup"],
|
|
||||||
test: ["ecto.create --quiet", "ecto.migrate --quiet", "test"],
|
|
||||||
"assets.deploy": ["tailwind default --minify", "esbuild default --minify", "phx.digest"]
|
|
||||||
]
|
|
||||||
end
|
|
||||||
end
|
|
||||||
47
mix.lock
47
mix.lock
@@ -1,47 +0,0 @@
|
|||||||
%{
|
|
||||||
"bcrypt_elixir": {:hex, :bcrypt_elixir, "3.0.1", "9be815469e6bfefec40fa74658ecbbe6897acfb57614df1416eeccd4903f602c", [:make, :mix], [{:comeonin, "~> 5.3", [hex: :comeonin, repo: "hexpm", optional: false]}, {:elixir_make, "~> 0.6", [hex: :elixir_make, repo: "hexpm", optional: false]}], "hexpm", "486bb95efb645d1efc6794c1ddd776a186a9a713abf06f45708a6ce324fb96cf"},
|
|
||||||
"castore": {:hex, :castore, "0.1.19", "a2c3e46d62b7f3aa2e6f88541c21d7400381e53704394462b9fd4f06f6d42bb6", [:mix], [], "hexpm", "e96e0161a5dc82ef441da24d5fa74aefc40d920f3a6645d15e1f9f3e66bb2109"},
|
|
||||||
"comeonin": {:hex, :comeonin, "5.3.3", "2c564dac95a35650e9b6acfe6d2952083d8a08e4a89b93a481acb552b325892e", [:mix], [], "hexpm", "3e38c9c2cb080828116597ca8807bb482618a315bfafd98c90bc22a821cc84df"},
|
|
||||||
"connection": {:hex, :connection, "1.1.0", "ff2a49c4b75b6fb3e674bfc5536451607270aac754ffd1bdfe175abe4a6d7a68", [:mix], [], "hexpm", "722c1eb0a418fbe91ba7bd59a47e28008a189d47e37e0e7bb85585a016b2869c"},
|
|
||||||
"cowboy": {:hex, :cowboy, "2.9.0", "865dd8b6607e14cf03282e10e934023a1bd8be6f6bacf921a7e2a96d800cd452", [:make, :rebar3], [{:cowlib, "2.11.0", [hex: :cowlib, repo: "hexpm", optional: false]}, {:ranch, "1.8.0", [hex: :ranch, repo: "hexpm", optional: false]}], "hexpm", "2c729f934b4e1aa149aff882f57c6372c15399a20d54f65c8d67bef583021bde"},
|
|
||||||
"cowboy_telemetry": {:hex, :cowboy_telemetry, "0.4.0", "f239f68b588efa7707abce16a84d0d2acf3a0f50571f8bb7f56a15865aae820c", [:rebar3], [{:cowboy, "~> 2.7", [hex: :cowboy, repo: "hexpm", optional: false]}, {:telemetry, "~> 1.0", [hex: :telemetry, repo: "hexpm", optional: false]}], "hexpm", "7d98bac1ee4565d31b62d59f8823dfd8356a169e7fcbb83831b8a5397404c9de"},
|
|
||||||
"cowlib": {:hex, :cowlib, "2.11.0", "0b9ff9c346629256c42ebe1eeb769a83c6cb771a6ee5960bd110ab0b9b872063", [:make, :rebar3], [], "hexpm", "2b3e9da0b21c4565751a6d4901c20d1b4cc25cbb7fd50d91d2ab6dd287bc86a9"},
|
|
||||||
"db_connection": {:hex, :db_connection, "2.4.2", "f92e79aff2375299a16bcb069a14ee8615c3414863a6fef93156aee8e86c2ff3", [:mix], [{:connection, "~> 1.0", [hex: :connection, repo: "hexpm", optional: false]}, {:telemetry, "~> 0.4 or ~> 1.0", [hex: :telemetry, repo: "hexpm", optional: false]}], "hexpm", "4fe53ca91b99f55ea249693a0229356a08f4d1a7931d8ffa79289b145fe83668"},
|
|
||||||
"decimal": {:hex, :decimal, "2.1.1", "5611dca5d4b2c3dd497dec8f68751f1f1a54755e8ed2a966c2633cf885973ad6", [:mix], [], "hexpm", "53cfe5f497ed0e7771ae1a475575603d77425099ba5faef9394932b35020ffcc"},
|
|
||||||
"ecto": {:hex, :ecto, "3.9.6", "2f420c173efcb2e22fa4f8fc41e75e02b3c5bd4cffef12085cae5418c12e530d", [:mix], [{:decimal, "~> 1.6 or ~> 2.0", [hex: :decimal, repo: "hexpm", optional: false]}, {:jason, "~> 1.0", [hex: :jason, repo: "hexpm", optional: true]}, {:telemetry, "~> 0.4 or ~> 1.0", [hex: :telemetry, repo: "hexpm", optional: false]}], "hexpm", "df17bc06ba6f78a7b764e4a14ef877fe5f4499332c5a105ace11fe7013b72c84"},
|
|
||||||
"ecto_sql": {:hex, :ecto_sql, "3.9.1", "9bd5894eecc53d5b39d0c95180d4466aff00e10679e13a5cfa725f6f85c03c22", [:mix], [{:db_connection, "~> 2.5 or ~> 2.4.1", [hex: :db_connection, repo: "hexpm", optional: false]}, {:ecto, "~> 3.9.0", [hex: :ecto, repo: "hexpm", optional: false]}, {:myxql, "~> 0.6.0", [hex: :myxql, repo: "hexpm", optional: true]}, {:postgrex, "~> 0.16.0 or ~> 1.0", [hex: :postgrex, repo: "hexpm", optional: true]}, {:tds, "~> 2.1.1 or ~> 2.2", [hex: :tds, repo: "hexpm", optional: true]}, {:telemetry, "~> 0.4.0 or ~> 1.0", [hex: :telemetry, repo: "hexpm", optional: false]}], "hexpm", "5fd470a4fff2e829bbf9dcceb7f3f9f6d1e49b4241e802f614de6b8b67c51118"},
|
|
||||||
"elixir_make": {:hex, :elixir_make, "0.6.3", "bc07d53221216838d79e03a8019d0839786703129599e9619f4ab74c8c096eac", [:mix], [], "hexpm", "f5cbd651c5678bcaabdbb7857658ee106b12509cd976c2c2fca99688e1daf716"},
|
|
||||||
"esbuild": {:hex, :esbuild, "0.5.0", "d5bb08ff049d7880ee3609ed5c4b864bd2f46445ea40b16b4acead724fb4c4a3", [:mix], [{:castore, ">= 0.0.0", [hex: :castore, repo: "hexpm", optional: false]}], "hexpm", "f183a0b332d963c4cfaf585477695ea59eef9a6f2204fdd0efa00e099694ffe5"},
|
|
||||||
"ex_heroicons": {:hex, :ex_heroicons, "2.0.0", "701ba2a314c0ff542d8e44486fbf482d29700bfccc291ee189b0154789dece54", [:mix], [{:phoenix_html, "~> 3.2", [hex: :phoenix_html, repo: "hexpm", optional: false]}, {:phoenix_live_view, "~> 0.17", [hex: :phoenix_live_view, repo: "hexpm", optional: true]}, {:surface, "~> 0.7", [hex: :surface, repo: "hexpm", optional: true]}], "hexpm", "029fb3bab5d45bf3777113733cfe944dc3607015d69a15a3ba87e321457e95ac"},
|
|
||||||
"file_system": {:hex, :file_system, "0.2.10", "fb082005a9cd1711c05b5248710f8826b02d7d1784e7c3451f9c1231d4fc162d", [:mix], [], "hexpm", "41195edbfb562a593726eda3b3e8b103a309b733ad25f3d642ba49696bf715dc"},
|
|
||||||
"finch": {:hex, :finch, "0.13.0", "c881e5460ec563bf02d4f4584079e62201db676ed4c0ef3e59189331c4eddf7b", [:mix], [{:castore, "~> 0.1", [hex: :castore, repo: "hexpm", optional: false]}, {:mime, "~> 1.0 or ~> 2.0", [hex: :mime, repo: "hexpm", optional: false]}, {:mint, "~> 1.3", [hex: :mint, repo: "hexpm", optional: false]}, {:nimble_options, "~> 0.4.0", [hex: :nimble_options, repo: "hexpm", optional: false]}, {:nimble_pool, "~> 0.2.6", [hex: :nimble_pool, repo: "hexpm", optional: false]}, {:telemetry, "~> 0.4 or ~> 1.0", [hex: :telemetry, repo: "hexpm", optional: false]}], "hexpm", "49957dcde10dcdc042a123a507a9c5ec5a803f53646d451db2f7dea696fba6cc"},
|
|
||||||
"floki": {:hex, :floki, "0.34.0", "002d0cc194b48794d74711731db004fafeb328fe676976f160685262d43706a8", [:mix], [], "hexpm", "9c3a9f43f40dde00332a589bd9d389b90c1f518aef500364d00636acc5ebc99c"},
|
|
||||||
"gettext": {:hex, :gettext, "0.20.0", "75ad71de05f2ef56991dbae224d35c68b098dd0e26918def5bb45591d5c8d429", [:mix], [], "hexpm", "1c03b177435e93a47441d7f681a7040bd2a816ece9e2666d1c9001035121eb3d"},
|
|
||||||
"heroicons": {:hex, :heroicons, "0.5.1", "cca0dcca07af5f74d8a7d111e40418d3615d65e6773c0ea10e20cef070fd30aa", [:mix], [{:castore, ">= 0.0.0", [hex: :castore, repo: "hexpm", optional: false]}, {:phoenix_live_view, "~> 0.18.2", [hex: :phoenix_live_view, repo: "hexpm", optional: false]}], "hexpm", "4b096d0a1d50e9054df9b12cc637c9f65c3972ff086791d3f2d1846f0653117e"},
|
|
||||||
"hpax": {:hex, :hpax, "0.1.2", "09a75600d9d8bbd064cdd741f21fc06fc1f4cf3d0fcc335e5aa19be1a7235c84", [:mix], [], "hexpm", "2c87843d5a23f5f16748ebe77969880e29809580efdaccd615cd3bed628a8c13"},
|
|
||||||
"jason": {:hex, :jason, "1.4.1", "af1504e35f629ddcdd6addb3513c3853991f694921b1b9368b0bd32beb9f1b63", [:mix], [{:decimal, "~> 1.0 or ~> 2.0", [hex: :decimal, repo: "hexpm", optional: true]}], "hexpm", "fbb01ecdfd565b56261302f7e1fcc27c4fb8f32d56eab74db621fc154604a7a1"},
|
|
||||||
"mime": {:hex, :mime, "2.0.3", "3676436d3d1f7b81b5a2d2bd8405f412c677558c81b1c92be58c00562bb59095", [:mix], [], "hexpm", "27a30bf0db44d25eecba73755acf4068cbfe26a4372f9eb3e4ea3a45956bff6b"},
|
|
||||||
"mint": {:hex, :mint, "1.4.2", "50330223429a6e1260b2ca5415f69b0ab086141bc76dc2fbf34d7c389a6675b2", [:mix], [{:castore, "~> 0.1.0", [hex: :castore, repo: "hexpm", optional: true]}, {:hpax, "~> 0.1.1", [hex: :hpax, repo: "hexpm", optional: false]}], "hexpm", "ce75a5bbcc59b4d7d8d70f8b2fc284b1751ffb35c7b6a6302b5192f8ab4ddd80"},
|
|
||||||
"nimble_options": {:hex, :nimble_options, "0.4.0", "c89babbab52221a24b8d1ff9e7d838be70f0d871be823165c94dd3418eea728f", [:mix], [], "hexpm", "e6701c1af326a11eea9634a3b1c62b475339ace9456c1a23ec3bc9a847bca02d"},
|
|
||||||
"nimble_pool": {:hex, :nimble_pool, "0.2.6", "91f2f4c357da4c4a0a548286c84a3a28004f68f05609b4534526871a22053cde", [:mix], [], "hexpm", "1c715055095d3f2705c4e236c18b618420a35490da94149ff8b580a2144f653f"},
|
|
||||||
"phoenix": {:hex, :phoenix, "1.7.0-rc.0", "8e328572f496b5170e879da94baa57c5f878f354d50eac052c9a7c6d57c2cf54", [:mix], [{:castore, ">= 0.0.0", [hex: :castore, repo: "hexpm", optional: false]}, {:jason, "~> 1.0", [hex: :jason, repo: "hexpm", optional: true]}, {:phoenix_pubsub, "~> 2.1", [hex: :phoenix_pubsub, repo: "hexpm", optional: false]}, {:phoenix_template, "~> 1.0", [hex: :phoenix_template, repo: "hexpm", optional: false]}, {:phoenix_view, "~> 2.0", [hex: :phoenix_view, repo: "hexpm", optional: true]}, {:plug, "~> 1.14", [hex: :plug, repo: "hexpm", optional: false]}, {:plug_cowboy, "~> 2.6", [hex: :plug_cowboy, repo: "hexpm", optional: true]}, {:plug_crypto, "~> 1.2", [hex: :plug_crypto, repo: "hexpm", optional: false]}, {:telemetry, "~> 0.4 or ~> 1.0", [hex: :telemetry, repo: "hexpm", optional: false]}, {:websock_adapter, "~> 0.4", [hex: :websock_adapter, repo: "hexpm", optional: false]}], "hexpm", "ed503f6c55184afc0a453e44e6ab2a09f014f59b7fdd682313fdc52ec2f82859"},
|
|
||||||
"phoenix_ecto": {:hex, :phoenix_ecto, "4.4.0", "0672ed4e4808b3fbed494dded89958e22fb882de47a97634c0b13e7b0b5f7720", [:mix], [{:ecto, "~> 3.3", [hex: :ecto, repo: "hexpm", optional: false]}, {:phoenix_html, "~> 2.14.2 or ~> 3.0", [hex: :phoenix_html, repo: "hexpm", optional: true]}, {:plug, "~> 1.9", [hex: :plug, repo: "hexpm", optional: false]}], "hexpm", "09864e558ed31ee00bd48fcc1d4fc58ae9678c9e81649075431e69dbabb43cc1"},
|
|
||||||
"phoenix_html": {:hex, :phoenix_html, "3.2.0", "1c1219d4b6cb22ac72f12f73dc5fad6c7563104d083f711c3fcd8551a1f4ae11", [:mix], [{:plug, "~> 1.5", [hex: :plug, repo: "hexpm", optional: true]}], "hexpm", "36ec97ba56d25c0136ef1992c37957e4246b649d620958a1f9fa86165f8bc54f"},
|
|
||||||
"phoenix_live_dashboard": {:hex, :phoenix_live_dashboard, "0.7.2", "97cc4ff2dba1ebe504db72cb45098cb8e91f11160528b980bd282cc45c73b29c", [:mix], [{:ecto, "~> 3.6.2 or ~> 3.7", [hex: :ecto, repo: "hexpm", optional: true]}, {:ecto_mysql_extras, "~> 0.5", [hex: :ecto_mysql_extras, repo: "hexpm", optional: true]}, {:ecto_psql_extras, "~> 0.7", [hex: :ecto_psql_extras, repo: "hexpm", optional: true]}, {:mime, "~> 1.6 or ~> 2.0", [hex: :mime, repo: "hexpm", optional: false]}, {:phoenix_live_view, "~> 0.18.3", [hex: :phoenix_live_view, repo: "hexpm", optional: false]}, {:telemetry_metrics, "~> 0.6 or ~> 1.0", [hex: :telemetry_metrics, repo: "hexpm", optional: false]}], "hexpm", "0e5fdf063c7a3b620c566a30fcf68b7ee02e5e46fe48ee46a6ec3ba382dc05b7"},
|
|
||||||
"phoenix_live_reload": {:hex, :phoenix_live_reload, "1.4.0", "4fe222c0be55fdc3f9c711e24955fc42a7cd9b7a2f5f406f2580a567c335a573", [: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", "bebf0fc2d2113b61cb5968f585367234b7b4c21d963d691de7b4b2dc6cdaae6f"},
|
|
||||||
"phoenix_live_view": {:hex, :phoenix_live_view, "0.18.3", "2e3d009422addf8b15c3dccc65ce53baccbe26f7cfd21d264680b5867789a9c1", [:mix], [{:jason, "~> 1.0", [hex: :jason, repo: "hexpm", optional: true]}, {:phoenix, "~> 1.6.15 or ~> 1.7.0", [hex: :phoenix, repo: "hexpm", optional: false]}, {:phoenix_html, "~> 3.1", [hex: :phoenix_html, repo: "hexpm", optional: false]}, {:phoenix_template, "~> 1.0", [hex: :phoenix_template, repo: "hexpm", optional: false]}, {:phoenix_view, "~> 2.0", [hex: :phoenix_view, repo: "hexpm", optional: true]}, {:telemetry, "~> 0.4.2 or ~> 1.0", [hex: :telemetry, repo: "hexpm", optional: false]}], "hexpm", "c8845177a866e017dcb7083365393c8f00ab061b8b6b2bda575891079dce81b2"},
|
|
||||||
"phoenix_pubsub": {:hex, :phoenix_pubsub, "2.1.1", "ba04e489ef03763bf28a17eb2eaddc2c20c6d217e2150a61e3298b0f4c2012b5", [:mix], [], "hexpm", "81367c6d1eea5878ad726be80808eb5a787a23dee699f96e72b1109c57cdd8d9"},
|
|
||||||
"phoenix_template": {:hex, :phoenix_template, "1.0.0", "c57bc5044f25f007dc86ab21895688c098a9f846a8dda6bc40e2d0ddc146e38f", [:mix], [{:phoenix_html, "~> 2.14.2 or ~> 3.0", [hex: :phoenix_html, repo: "hexpm", optional: true]}], "hexpm", "1b066f99a26fd22064c12b2600a9a6e56700f591bf7b20b418054ea38b4d4357"},
|
|
||||||
"plug": {:hex, :plug, "1.14.0", "ba4f558468f69cbd9f6b356d25443d0b796fbdc887e03fa89001384a9cac638f", [:mix], [{:mime, "~> 1.0 or ~> 2.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.3 or ~> 1.0", [hex: :telemetry, repo: "hexpm", optional: false]}], "hexpm", "bf020432c7d4feb7b3af16a0c2701455cbbbb95e5b6866132cb09eb0c29adc14"},
|
|
||||||
"plug_cowboy": {:hex, :plug_cowboy, "2.6.0", "d1cf12ff96a1ca4f52207c5271a6c351a4733f413803488d75b70ccf44aebec2", [:mix], [{:cowboy, "~> 2.7", [hex: :cowboy, repo: "hexpm", optional: false]}, {:cowboy_telemetry, "~> 0.3", [hex: :cowboy_telemetry, repo: "hexpm", optional: false]}, {:plug, "~> 1.14", [hex: :plug, repo: "hexpm", optional: false]}], "hexpm", "073cf20b753ce6682ed72905cd62a2d4bd9bad1bf9f7feb02a1b8e525bd94fa6"},
|
|
||||||
"plug_crypto": {:hex, :plug_crypto, "1.2.3", "8f77d13aeb32bfd9e654cb68f0af517b371fb34c56c9f2b58fe3df1235c1251a", [:mix], [], "hexpm", "b5672099c6ad5c202c45f5a403f21a3411247f164e4a8fab056e5cd8a290f4a2"},
|
|
||||||
"postgrex": {:hex, :postgrex, "0.16.5", "fcc4035cc90e23933c5d69a9cd686e329469446ef7abba2cf70f08e2c4b69810", [:mix], [{:connection, "~> 1.1", [hex: :connection, repo: "hexpm", optional: false]}, {:db_connection, "~> 2.1", [hex: :db_connection, repo: "hexpm", optional: false]}, {:decimal, "~> 1.5 or ~> 2.0", [hex: :decimal, repo: "hexpm", optional: false]}, {:jason, "~> 1.0", [hex: :jason, repo: "hexpm", optional: true]}, {:table, "~> 0.1.0", [hex: :table, repo: "hexpm", optional: true]}], "hexpm", "edead639dc6e882618c01d8fc891214c481ab9a3788dfe38dd5e37fd1d5fb2e8"},
|
|
||||||
"ranch": {:hex, :ranch, "1.8.0", "8c7a100a139fd57f17327b6413e4167ac559fbc04ca7448e9be9057311597a1d", [:make, :rebar3], [], "hexpm", "49fbcfd3682fab1f5d109351b61257676da1a2fdbe295904176d5e521a2ddfe5"},
|
|
||||||
"swoosh": {:hex, :swoosh, "1.8.2", "af9a22ab2c0d20b266f61acca737fa11a121902de9466a39e91bacdce012101c", [:mix], [{:cowboy, "~> 1.1 or ~> 2.4", [hex: :cowboy, repo: "hexpm", optional: true]}, {:ex_aws, "~> 2.1", [hex: :ex_aws, repo: "hexpm", optional: true]}, {:finch, "~> 0.6", [hex: :finch, repo: "hexpm", optional: true]}, {:gen_smtp, "~> 0.13 or ~> 1.0", [hex: :gen_smtp, repo: "hexpm", optional: true]}, {:hackney, "~> 1.9", [hex: :hackney, repo: "hexpm", optional: true]}, {:jason, "~> 1.0", [hex: :jason, repo: "hexpm", optional: false]}, {:mail, "~> 0.2", [hex: :mail, repo: "hexpm", optional: true]}, {:mime, "~> 1.1 or ~> 2.0", [hex: :mime, repo: "hexpm", optional: false]}, {:plug_cowboy, ">= 1.0.0", [hex: :plug_cowboy, repo: "hexpm", optional: true]}, {:telemetry, "~> 0.4.2 or ~> 1.0", [hex: :telemetry, repo: "hexpm", optional: false]}], "hexpm", "d058ba750eafadb6c09a84a352c14c5d1eeeda6e84945fcc95785b7f3067b7db"},
|
|
||||||
"tailwind": {:hex, :tailwind, "0.1.9", "25ba09d42f7bfabe170eb67683a76d6ec2061952dc9bd263a52a99ba3d24bd4d", [:mix], [{:castore, ">= 0.0.0", [hex: :castore, repo: "hexpm", optional: false]}], "hexpm", "9213f87709c458aaec313bb5f2df2b4d2cedc2b630e4ae821bf3c54c47a56d0b"},
|
|
||||||
"telemetry": {:hex, :telemetry, "1.2.1", "68fdfe8d8f05a8428483a97d7aab2f268aaff24b49e0f599faa091f1d4e7f61c", [:rebar3], [], "hexpm", "dad9ce9d8effc621708f99eac538ef1cbe05d6a874dd741de2e689c47feafed5"},
|
|
||||||
"telemetry_metrics": {:hex, :telemetry_metrics, "0.6.1", "315d9163a1d4660aedc3fee73f33f1d355dcc76c5c3ab3d59e76e3edf80eef1f", [:mix], [{:telemetry, "~> 0.4 or ~> 1.0", [hex: :telemetry, repo: "hexpm", optional: false]}], "hexpm", "7be9e0871c41732c233be71e4be11b96e56177bf15dde64a8ac9ce72ac9834c6"},
|
|
||||||
"telemetry_poller": {:hex, :telemetry_poller, "1.0.0", "db91bb424e07f2bb6e73926fcafbfcbcb295f0193e0a00e825e589a0a47e8453", [:rebar3], [{:telemetry, "~> 1.0", [hex: :telemetry, repo: "hexpm", optional: false]}], "hexpm", "b3a24eafd66c3f42da30fc3ca7dda1e9d546c12250a2d60d7b81d264fbec4f6e"},
|
|
||||||
"websock": {:hex, :websock, "0.4.3", "184ac396bdcd3dfceb5b74c17d221af659dd559a95b1b92041ecb51c9b728093", [:mix], [], "hexpm", "5e4dd85f305f43fd3d3e25d70bec4a45228dfed60f0f3b072d8eddff335539cf"},
|
|
||||||
"websock_adapter": {:hex, :websock_adapter, "0.4.5", "30038a3715067f51a9580562c05a3a8d501126030336ffc6edb53bf57d6d2d26", [:mix], [{:bandit, "~> 0.6", [hex: :bandit, repo: "hexpm", optional: true]}, {:plug, "~> 1.14", [hex: :plug, repo: "hexpm", optional: false]}, {:plug_cowboy, "~> 2.6", [hex: :plug_cowboy, repo: "hexpm", optional: true]}, {:websock, "~> 0.4", [hex: :websock, repo: "hexpm", optional: false]}], "hexpm", "1d9812dc7e703c205049426fd4fe0852a247a825f91b099e53dc96f68bafe4c8"},
|
|
||||||
}
|
|
||||||
@@ -1,112 +0,0 @@
|
|||||||
## `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"
|
|
||||||
|
|
||||||
## From Ecto.Changeset.cast/4
|
|
||||||
msgid "can't be blank"
|
|
||||||
msgstr ""
|
|
||||||
|
|
||||||
## From Ecto.Changeset.unique_constraint/3
|
|
||||||
msgid "has already been taken"
|
|
||||||
msgstr ""
|
|
||||||
|
|
||||||
## From Ecto.Changeset.put_change/3
|
|
||||||
msgid "is invalid"
|
|
||||||
msgstr ""
|
|
||||||
|
|
||||||
## From Ecto.Changeset.validate_acceptance/3
|
|
||||||
msgid "must be accepted"
|
|
||||||
msgstr ""
|
|
||||||
|
|
||||||
## From Ecto.Changeset.validate_format/3
|
|
||||||
msgid "has invalid format"
|
|
||||||
msgstr ""
|
|
||||||
|
|
||||||
## From Ecto.Changeset.validate_subset/3
|
|
||||||
msgid "has an invalid entry"
|
|
||||||
msgstr ""
|
|
||||||
|
|
||||||
## From Ecto.Changeset.validate_exclusion/3
|
|
||||||
msgid "is reserved"
|
|
||||||
msgstr ""
|
|
||||||
|
|
||||||
## From Ecto.Changeset.validate_confirmation/3
|
|
||||||
msgid "does not match confirmation"
|
|
||||||
msgstr ""
|
|
||||||
|
|
||||||
## From Ecto.Changeset.no_assoc_constraint/3
|
|
||||||
msgid "is still associated with this entry"
|
|
||||||
msgstr ""
|
|
||||||
|
|
||||||
msgid "are still associated with this entry"
|
|
||||||
msgstr ""
|
|
||||||
|
|
||||||
## From Ecto.Changeset.validate_length/3
|
|
||||||
msgid "should have %{count} item(s)"
|
|
||||||
msgid_plural "should have %{count} item(s)"
|
|
||||||
msgstr[0] ""
|
|
||||||
msgstr[1] ""
|
|
||||||
|
|
||||||
msgid "should be %{count} character(s)"
|
|
||||||
msgid_plural "should be %{count} character(s)"
|
|
||||||
msgstr[0] ""
|
|
||||||
msgstr[1] ""
|
|
||||||
|
|
||||||
msgid "should be %{count} byte(s)"
|
|
||||||
msgid_plural "should be %{count} byte(s)"
|
|
||||||
msgstr[0] ""
|
|
||||||
msgstr[1] ""
|
|
||||||
|
|
||||||
msgid "should have at least %{count} item(s)"
|
|
||||||
msgid_plural "should have at least %{count} item(s)"
|
|
||||||
msgstr[0] ""
|
|
||||||
msgstr[1] ""
|
|
||||||
|
|
||||||
msgid "should be at least %{count} character(s)"
|
|
||||||
msgid_plural "should be at least %{count} character(s)"
|
|
||||||
msgstr[0] ""
|
|
||||||
msgstr[1] ""
|
|
||||||
|
|
||||||
msgid "should be at least %{count} byte(s)"
|
|
||||||
msgid_plural "should be at least %{count} byte(s)"
|
|
||||||
msgstr[0] ""
|
|
||||||
msgstr[1] ""
|
|
||||||
|
|
||||||
msgid "should have at most %{count} item(s)"
|
|
||||||
msgid_plural "should have at most %{count} item(s)"
|
|
||||||
msgstr[0] ""
|
|
||||||
msgstr[1] ""
|
|
||||||
|
|
||||||
msgid "should be at most %{count} character(s)"
|
|
||||||
msgid_plural "should be at most %{count} character(s)"
|
|
||||||
msgstr[0] ""
|
|
||||||
msgstr[1] ""
|
|
||||||
|
|
||||||
msgid "should be at most %{count} byte(s)"
|
|
||||||
msgid_plural "should be at most %{count} byte(s)"
|
|
||||||
msgstr[0] ""
|
|
||||||
msgstr[1] ""
|
|
||||||
|
|
||||||
## From Ecto.Changeset.validate_number/3
|
|
||||||
msgid "must be less than %{number}"
|
|
||||||
msgstr ""
|
|
||||||
|
|
||||||
msgid "must be greater than %{number}"
|
|
||||||
msgstr ""
|
|
||||||
|
|
||||||
msgid "must be less than or equal to %{number}"
|
|
||||||
msgstr ""
|
|
||||||
|
|
||||||
msgid "must be greater than or equal to %{number}"
|
|
||||||
msgstr ""
|
|
||||||
|
|
||||||
msgid "must be equal to %{number}"
|
|
||||||
msgstr ""
|
|
||||||
@@ -1,110 +0,0 @@
|
|||||||
## 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.
|
|
||||||
|
|
||||||
## From Ecto.Changeset.cast/4
|
|
||||||
msgid "can't be blank"
|
|
||||||
msgstr ""
|
|
||||||
|
|
||||||
## From Ecto.Changeset.unique_constraint/3
|
|
||||||
msgid "has already been taken"
|
|
||||||
msgstr ""
|
|
||||||
|
|
||||||
## From Ecto.Changeset.put_change/3
|
|
||||||
msgid "is invalid"
|
|
||||||
msgstr ""
|
|
||||||
|
|
||||||
## From Ecto.Changeset.validate_acceptance/3
|
|
||||||
msgid "must be accepted"
|
|
||||||
msgstr ""
|
|
||||||
|
|
||||||
## From Ecto.Changeset.validate_format/3
|
|
||||||
msgid "has invalid format"
|
|
||||||
msgstr ""
|
|
||||||
|
|
||||||
## From Ecto.Changeset.validate_subset/3
|
|
||||||
msgid "has an invalid entry"
|
|
||||||
msgstr ""
|
|
||||||
|
|
||||||
## From Ecto.Changeset.validate_exclusion/3
|
|
||||||
msgid "is reserved"
|
|
||||||
msgstr ""
|
|
||||||
|
|
||||||
## From Ecto.Changeset.validate_confirmation/3
|
|
||||||
msgid "does not match confirmation"
|
|
||||||
msgstr ""
|
|
||||||
|
|
||||||
## From Ecto.Changeset.no_assoc_constraint/3
|
|
||||||
msgid "is still associated with this entry"
|
|
||||||
msgstr ""
|
|
||||||
|
|
||||||
msgid "are still associated with this entry"
|
|
||||||
msgstr ""
|
|
||||||
|
|
||||||
## From Ecto.Changeset.validate_length/3
|
|
||||||
msgid "should have %{count} item(s)"
|
|
||||||
msgid_plural "should have %{count} item(s)"
|
|
||||||
msgstr[0] ""
|
|
||||||
msgstr[1] ""
|
|
||||||
|
|
||||||
msgid "should be %{count} character(s)"
|
|
||||||
msgid_plural "should be %{count} character(s)"
|
|
||||||
msgstr[0] ""
|
|
||||||
msgstr[1] ""
|
|
||||||
|
|
||||||
msgid "should be %{count} byte(s)"
|
|
||||||
msgid_plural "should be %{count} byte(s)"
|
|
||||||
msgstr[0] ""
|
|
||||||
msgstr[1] ""
|
|
||||||
|
|
||||||
msgid "should have at least %{count} item(s)"
|
|
||||||
msgid_plural "should have at least %{count} item(s)"
|
|
||||||
msgstr[0] ""
|
|
||||||
msgstr[1] ""
|
|
||||||
|
|
||||||
msgid "should be at least %{count} character(s)"
|
|
||||||
msgid_plural "should be at least %{count} character(s)"
|
|
||||||
msgstr[0] ""
|
|
||||||
msgstr[1] ""
|
|
||||||
|
|
||||||
msgid "should be at least %{count} byte(s)"
|
|
||||||
msgid_plural "should be at least %{count} byte(s)"
|
|
||||||
msgstr[0] ""
|
|
||||||
msgstr[1] ""
|
|
||||||
|
|
||||||
msgid "should have at most %{count} item(s)"
|
|
||||||
msgid_plural "should have at most %{count} item(s)"
|
|
||||||
msgstr[0] ""
|
|
||||||
msgstr[1] ""
|
|
||||||
|
|
||||||
msgid "should be at most %{count} character(s)"
|
|
||||||
msgid_plural "should be at most %{count} character(s)"
|
|
||||||
msgstr[0] ""
|
|
||||||
msgstr[1] ""
|
|
||||||
|
|
||||||
msgid "should be at most %{count} byte(s)"
|
|
||||||
msgid_plural "should be at most %{count} byte(s)"
|
|
||||||
msgstr[0] ""
|
|
||||||
msgstr[1] ""
|
|
||||||
|
|
||||||
## From Ecto.Changeset.validate_number/3
|
|
||||||
msgid "must be less than %{number}"
|
|
||||||
msgstr ""
|
|
||||||
|
|
||||||
msgid "must be greater than %{number}"
|
|
||||||
msgstr ""
|
|
||||||
|
|
||||||
msgid "must be less than or equal to %{number}"
|
|
||||||
msgstr ""
|
|
||||||
|
|
||||||
msgid "must be greater than or equal to %{number}"
|
|
||||||
msgstr ""
|
|
||||||
|
|
||||||
msgid "must be equal to %{number}"
|
|
||||||
msgstr ""
|
|
||||||
@@ -1,4 +0,0 @@
|
|||||||
[
|
|
||||||
import_deps: [:ecto_sql],
|
|
||||||
inputs: ["*.exs"]
|
|
||||||
]
|
|
||||||
@@ -1,29 +0,0 @@
|
|||||||
defmodule Malarkey.Repo.Migrations.CreateUsersAuthTables do
|
|
||||||
use Ecto.Migration
|
|
||||||
|
|
||||||
def change do
|
|
||||||
execute "CREATE EXTENSION IF NOT EXISTS citext", ""
|
|
||||||
|
|
||||||
create table(:users) do
|
|
||||||
add :email, :citext, null: false
|
|
||||||
add :hashed_password, :string, null: false
|
|
||||||
add :confirmed_at, :naive_datetime
|
|
||||||
add :username, :string
|
|
||||||
add :fullname, :string
|
|
||||||
timestamps()
|
|
||||||
end
|
|
||||||
|
|
||||||
create unique_index(:users, [:email])
|
|
||||||
|
|
||||||
create table(:users_tokens) do
|
|
||||||
add :user_id, references(:users, on_delete: :delete_all), null: false
|
|
||||||
add :token, :binary, null: false
|
|
||||||
add :context, :string, null: false
|
|
||||||
add :sent_to, :string
|
|
||||||
timestamps(updated_at: false)
|
|
||||||
end
|
|
||||||
|
|
||||||
create index(:users_tokens, [:user_id])
|
|
||||||
create unique_index(:users_tokens, [:context, :token])
|
|
||||||
end
|
|
||||||
end
|
|
||||||
@@ -1,12 +0,0 @@
|
|||||||
defmodule Malarkey.Repo.Migrations.CreatePosts do
|
|
||||||
use Ecto.Migration
|
|
||||||
|
|
||||||
def change do
|
|
||||||
create table(:posts) do
|
|
||||||
add :body, :string
|
|
||||||
add :user_id, references(:users, on_delete: :delete_all), null: false
|
|
||||||
|
|
||||||
timestamps()
|
|
||||||
end
|
|
||||||
end
|
|
||||||
end
|
|
||||||
@@ -1,16 +0,0 @@
|
|||||||
defmodule Malarkey.Repo.Migrations.AddUserLikes do
|
|
||||||
use Ecto.Migration
|
|
||||||
|
|
||||||
def change do
|
|
||||||
create table(:user_likes, primary_key: false) do
|
|
||||||
add(:post_id, references(:posts, on_delete: :delete_all), primary_key: true)
|
|
||||||
add(:user_id, references(:users, on_delete: :delete_all), primary_key: true)
|
|
||||||
timestamps()
|
|
||||||
end
|
|
||||||
|
|
||||||
create(index(:user_likes, [:post_id]))
|
|
||||||
create(index(:user_likes, [:user_id]))
|
|
||||||
|
|
||||||
create(unique_index(:user_likes, [:user_id, :post_id], name: :user_likes_unique_index))
|
|
||||||
end
|
|
||||||
end
|
|
||||||
@@ -1,16 +0,0 @@
|
|||||||
defmodule Malarkey.Repo.Migrations.AddUserDislikes do
|
|
||||||
use Ecto.Migration
|
|
||||||
|
|
||||||
def change do
|
|
||||||
create table(:user_dislikes, primary_key: false) do
|
|
||||||
add(:post_id, references(:posts, on_delete: :delete_all), primary_key: true)
|
|
||||||
add(:user_id, references(:users, on_delete: :delete_all), primary_key: true)
|
|
||||||
timestamps()
|
|
||||||
end
|
|
||||||
|
|
||||||
create(index(:user_dislikes, [:post_id]))
|
|
||||||
create(index(:user_dislikes, [:user_id]))
|
|
||||||
|
|
||||||
create(unique_index(:user_dislikes, [:user_id, :post_id], name: :user_dislikes_unique_index))
|
|
||||||
end
|
|
||||||
end
|
|
||||||
@@ -1,16 +0,0 @@
|
|||||||
defmodule Malarkey.Repo.Migrations.AddUserReposts do
|
|
||||||
use Ecto.Migration
|
|
||||||
|
|
||||||
def change do
|
|
||||||
create table(:user_reposts, primary_key: false) do
|
|
||||||
add(:post_id, references(:posts, on_delete: :delete_all), primary_key: true)
|
|
||||||
add(:user_id, references(:users, on_delete: :delete_all), primary_key: true)
|
|
||||||
timestamps()
|
|
||||||
end
|
|
||||||
|
|
||||||
create(index(:user_reposts, [:post_id]))
|
|
||||||
create(index(:user_reposts, [:user_id]))
|
|
||||||
|
|
||||||
create(unique_index(:user_reposts, [:user_id, :post_id], name: :user_reposts_unique_index))
|
|
||||||
end
|
|
||||||
end
|
|
||||||
@@ -1,11 +0,0 @@
|
|||||||
# Script for populating the database. You can run it as:
|
|
||||||
#
|
|
||||||
# mix run priv/repo/seeds.exs
|
|
||||||
#
|
|
||||||
# Inside the script, you can read and write to any of your
|
|
||||||
# repositories directly:
|
|
||||||
#
|
|
||||||
# Malarkey.Repo.insert!(%Malarkey.SomeSchema{})
|
|
||||||
#
|
|
||||||
# We recommend using the bang functions (`insert!`, `update!`
|
|
||||||
# and so on) as they will fail if something goes wrong.
|
|
||||||
Binary file not shown.
|
Before Width: | Height: | Size: 1.2 KiB |
@@ -1,23 +0,0 @@
|
|||||||
<svg xmlns="http://www.w3.org/2000/svg"
|
|
||||||
width="32"
|
|
||||||
height="32"
|
|
||||||
version="1">
|
|
||||||
<g fill-rule="evenodd"
|
|
||||||
stroke="#373748"
|
|
||||||
stroke-linecap="round"
|
|
||||||
stroke-linejoin="round"
|
|
||||||
stroke-width="1.026">
|
|
||||||
<path fill="#f55"
|
|
||||||
d="M1130 1842.362c5 2 15 2 20 0 0 13-3 17-10 20-7-3-10-7-10-20z"
|
|
||||||
transform="matrix(.97483 0 0 .97458 -1101.06 -1787.312)"/>
|
|
||||||
<path fill="#f2f2f2"
|
|
||||||
d="M1136.5 1854.362c2.5 1 4.5 1 7 0-.536 2-1.5 3-3.5 4-2-1-2.964-2-3.5-4zm-3.25-6.5c2 1.5 3 1.5 5 0-2-1.5-3.007-1.5-5 0zm8.5 0c2 1.5 3 1.5 5 0-2-1.5-3.007-1.5-5 0z"
|
|
||||||
transform="matrix(.97483 0 0 .97458 -1101.06 -1787.312)"/>
|
|
||||||
<path fill="#5fd38d"
|
|
||||||
d="M1142.482 1838.32c4.312 3.225 13.971 5.813 19.318 5.176-3.364 12.557-7.297 15.644-14.835 16.73-5.985-4.71-7.848-9.35-4.483-21.907z"
|
|
||||||
transform="matrix(.97483 0 0 .97458 -1101.06 -1787.312)"/>
|
|
||||||
<path fill="#f2f2f2"
|
|
||||||
d="M1145.655 1851.593c2.156 1.613 4.087 2.13 6.761 1.811-1.035 1.793-2.225 2.51-4.416 2.958-1.673-1.483-2.345-2.699-2.345-4.77zm-1.458-7.12c1.544 1.966 2.51 2.225 4.83 1.294-1.544-1.967-2.516-2.227-4.83-1.294zm8.21 2.2c1.544 1.966 2.51 2.225 4.83 1.294-1.544-1.967-2.516-2.227-4.83-1.294z"
|
|
||||||
transform="matrix(.97483 0 0 .97458 -1101.06 -1787.312)"/>
|
|
||||||
</g>
|
|
||||||
</svg>
|
|
||||||
|
Before Width: | Height: | Size: 1.3 KiB |
Binary file not shown.
|
Before Width: | Height: | Size: 14 KiB |
@@ -1,5 +0,0 @@
|
|||||||
# See https://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: /
|
|
||||||
@@ -1,7 +0,0 @@
|
|||||||
#!/usr/bin/env bash
|
|
||||||
echo Dropping exiting db
|
|
||||||
dropdb -f --if-exists -h localhost -U postgres malarkey_dev
|
|
||||||
echo Creating new db
|
|
||||||
createdb -h localhost -U postgres malarkey_dev
|
|
||||||
|
|
||||||
mix ecto.migrate
|
|
||||||
@@ -1,508 +0,0 @@
|
|||||||
defmodule Malarkey.AccountsTest do
|
|
||||||
use Malarkey.DataCase
|
|
||||||
|
|
||||||
alias Malarkey.Accounts
|
|
||||||
|
|
||||||
import Malarkey.AccountsFixtures
|
|
||||||
alias Malarkey.Accounts.{User, UserToken}
|
|
||||||
|
|
||||||
describe "get_user_by_email/1" do
|
|
||||||
test "does not return the user if the email does not exist" do
|
|
||||||
refute Accounts.get_user_by_email("unknown@example.com")
|
|
||||||
end
|
|
||||||
|
|
||||||
test "returns the user if the email exists" do
|
|
||||||
%{id: id} = user = user_fixture()
|
|
||||||
assert %User{id: ^id} = Accounts.get_user_by_email(user.email)
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
describe "get_user_by_email_and_password/2" do
|
|
||||||
test "does not return the user if the email does not exist" do
|
|
||||||
refute Accounts.get_user_by_email_and_password("unknown@example.com", "hello world!")
|
|
||||||
end
|
|
||||||
|
|
||||||
test "does not return the user if the password is not valid" do
|
|
||||||
user = user_fixture()
|
|
||||||
refute Accounts.get_user_by_email_and_password(user.email, "invalid")
|
|
||||||
end
|
|
||||||
|
|
||||||
test "returns the user if the email and password are valid" do
|
|
||||||
%{id: id} = user = user_fixture()
|
|
||||||
|
|
||||||
assert %User{id: ^id} =
|
|
||||||
Accounts.get_user_by_email_and_password(user.email, valid_user_password())
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
describe "get_user!/1" do
|
|
||||||
test "raises if id is invalid" do
|
|
||||||
assert_raise Ecto.NoResultsError, fn ->
|
|
||||||
Accounts.get_user!(-1)
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
test "returns the user with the given id" do
|
|
||||||
%{id: id} = user = user_fixture()
|
|
||||||
assert %User{id: ^id} = Accounts.get_user!(user.id)
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
describe "register_user/1" do
|
|
||||||
test "requires email and password to be set" do
|
|
||||||
{:error, changeset} = Accounts.register_user(%{})
|
|
||||||
|
|
||||||
assert %{
|
|
||||||
password: ["can't be blank"],
|
|
||||||
email: ["can't be blank"]
|
|
||||||
} = errors_on(changeset)
|
|
||||||
end
|
|
||||||
|
|
||||||
test "validates email and password when given" do
|
|
||||||
{:error, changeset} = Accounts.register_user(%{email: "not valid", password: "not valid"})
|
|
||||||
|
|
||||||
assert %{
|
|
||||||
email: ["must have the @ sign and no spaces"],
|
|
||||||
password: ["should be at least 12 character(s)"]
|
|
||||||
} = errors_on(changeset)
|
|
||||||
end
|
|
||||||
|
|
||||||
test "validates maximum values for email and password for security" do
|
|
||||||
too_long = String.duplicate("db", 100)
|
|
||||||
{:error, changeset} = Accounts.register_user(%{email: too_long, password: too_long})
|
|
||||||
assert "should be at most 160 character(s)" in errors_on(changeset).email
|
|
||||||
assert "should be at most 72 character(s)" in errors_on(changeset).password
|
|
||||||
end
|
|
||||||
|
|
||||||
test "validates email uniqueness" do
|
|
||||||
%{email: email} = user_fixture()
|
|
||||||
{:error, changeset} = Accounts.register_user(%{email: email})
|
|
||||||
assert "has already been taken" in errors_on(changeset).email
|
|
||||||
|
|
||||||
# Now try with the upper cased email too, to check that email case is ignored.
|
|
||||||
{:error, changeset} = Accounts.register_user(%{email: String.upcase(email)})
|
|
||||||
assert "has already been taken" in errors_on(changeset).email
|
|
||||||
end
|
|
||||||
|
|
||||||
test "registers users with a hashed password" do
|
|
||||||
email = unique_user_email()
|
|
||||||
{:ok, user} = Accounts.register_user(valid_user_attributes(email: email))
|
|
||||||
assert user.email == email
|
|
||||||
assert is_binary(user.hashed_password)
|
|
||||||
assert is_nil(user.confirmed_at)
|
|
||||||
assert is_nil(user.password)
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
describe "change_user_registration/2" do
|
|
||||||
test "returns a changeset" do
|
|
||||||
assert %Ecto.Changeset{} = changeset = Accounts.change_user_registration(%User{})
|
|
||||||
assert changeset.required == [:password, :email]
|
|
||||||
end
|
|
||||||
|
|
||||||
test "allows fields to be set" do
|
|
||||||
email = unique_user_email()
|
|
||||||
password = valid_user_password()
|
|
||||||
|
|
||||||
changeset =
|
|
||||||
Accounts.change_user_registration(
|
|
||||||
%User{},
|
|
||||||
valid_user_attributes(email: email, password: password)
|
|
||||||
)
|
|
||||||
|
|
||||||
assert changeset.valid?
|
|
||||||
assert get_change(changeset, :email) == email
|
|
||||||
assert get_change(changeset, :password) == password
|
|
||||||
assert is_nil(get_change(changeset, :hashed_password))
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
describe "change_user_email/2" do
|
|
||||||
test "returns a user changeset" do
|
|
||||||
assert %Ecto.Changeset{} = changeset = Accounts.change_user_email(%User{})
|
|
||||||
assert changeset.required == [:email]
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
describe "apply_user_email/3" do
|
|
||||||
setup do
|
|
||||||
%{user: user_fixture()}
|
|
||||||
end
|
|
||||||
|
|
||||||
test "requires email to change", %{user: user} do
|
|
||||||
{:error, changeset} = Accounts.apply_user_email(user, valid_user_password(), %{})
|
|
||||||
assert %{email: ["did not change"]} = errors_on(changeset)
|
|
||||||
end
|
|
||||||
|
|
||||||
test "validates email", %{user: user} do
|
|
||||||
{:error, changeset} =
|
|
||||||
Accounts.apply_user_email(user, valid_user_password(), %{email: "not valid"})
|
|
||||||
|
|
||||||
assert %{email: ["must have the @ sign and no spaces"]} = errors_on(changeset)
|
|
||||||
end
|
|
||||||
|
|
||||||
test "validates maximum value for email for security", %{user: user} do
|
|
||||||
too_long = String.duplicate("db", 100)
|
|
||||||
|
|
||||||
{:error, changeset} =
|
|
||||||
Accounts.apply_user_email(user, valid_user_password(), %{email: too_long})
|
|
||||||
|
|
||||||
assert "should be at most 160 character(s)" in errors_on(changeset).email
|
|
||||||
end
|
|
||||||
|
|
||||||
test "validates email uniqueness", %{user: user} do
|
|
||||||
%{email: email} = user_fixture()
|
|
||||||
password = valid_user_password()
|
|
||||||
|
|
||||||
{:error, changeset} = Accounts.apply_user_email(user, password, %{email: email})
|
|
||||||
|
|
||||||
assert "has already been taken" in errors_on(changeset).email
|
|
||||||
end
|
|
||||||
|
|
||||||
test "validates current password", %{user: user} do
|
|
||||||
{:error, changeset} =
|
|
||||||
Accounts.apply_user_email(user, "invalid", %{email: unique_user_email()})
|
|
||||||
|
|
||||||
assert %{current_password: ["is not valid"]} = errors_on(changeset)
|
|
||||||
end
|
|
||||||
|
|
||||||
test "applies the email without persisting it", %{user: user} do
|
|
||||||
email = unique_user_email()
|
|
||||||
{:ok, user} = Accounts.apply_user_email(user, valid_user_password(), %{email: email})
|
|
||||||
assert user.email == email
|
|
||||||
assert Accounts.get_user!(user.id).email != email
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
describe "deliver_user_update_email_instructions/3" do
|
|
||||||
setup do
|
|
||||||
%{user: user_fixture()}
|
|
||||||
end
|
|
||||||
|
|
||||||
test "sends token through notification", %{user: user} do
|
|
||||||
token =
|
|
||||||
extract_user_token(fn url ->
|
|
||||||
Accounts.deliver_user_update_email_instructions(user, "current@example.com", url)
|
|
||||||
end)
|
|
||||||
|
|
||||||
{:ok, token} = Base.url_decode64(token, padding: false)
|
|
||||||
assert user_token = Repo.get_by(UserToken, token: :crypto.hash(:sha256, token))
|
|
||||||
assert user_token.user_id == user.id
|
|
||||||
assert user_token.sent_to == user.email
|
|
||||||
assert user_token.context == "change:current@example.com"
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
describe "update_user_email/2" do
|
|
||||||
setup do
|
|
||||||
user = user_fixture()
|
|
||||||
email = unique_user_email()
|
|
||||||
|
|
||||||
token =
|
|
||||||
extract_user_token(fn url ->
|
|
||||||
Accounts.deliver_user_update_email_instructions(%{user | email: email}, user.email, url)
|
|
||||||
end)
|
|
||||||
|
|
||||||
%{user: user, token: token, email: email}
|
|
||||||
end
|
|
||||||
|
|
||||||
test "updates the email with a valid token", %{user: user, token: token, email: email} do
|
|
||||||
assert Accounts.update_user_email(user, token) == :ok
|
|
||||||
changed_user = Repo.get!(User, user.id)
|
|
||||||
assert changed_user.email != user.email
|
|
||||||
assert changed_user.email == email
|
|
||||||
assert changed_user.confirmed_at
|
|
||||||
assert changed_user.confirmed_at != user.confirmed_at
|
|
||||||
refute Repo.get_by(UserToken, user_id: user.id)
|
|
||||||
end
|
|
||||||
|
|
||||||
test "does not update email with invalid token", %{user: user} do
|
|
||||||
assert Accounts.update_user_email(user, "oops") == :error
|
|
||||||
assert Repo.get!(User, user.id).email == user.email
|
|
||||||
assert Repo.get_by(UserToken, user_id: user.id)
|
|
||||||
end
|
|
||||||
|
|
||||||
test "does not update email if user email changed", %{user: user, token: token} do
|
|
||||||
assert Accounts.update_user_email(%{user | email: "current@example.com"}, token) == :error
|
|
||||||
assert Repo.get!(User, user.id).email == user.email
|
|
||||||
assert Repo.get_by(UserToken, user_id: user.id)
|
|
||||||
end
|
|
||||||
|
|
||||||
test "does not update email if token expired", %{user: user, token: token} do
|
|
||||||
{1, nil} = Repo.update_all(UserToken, set: [inserted_at: ~N[2020-01-01 00:00:00]])
|
|
||||||
assert Accounts.update_user_email(user, token) == :error
|
|
||||||
assert Repo.get!(User, user.id).email == user.email
|
|
||||||
assert Repo.get_by(UserToken, user_id: user.id)
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
describe "change_user_password/2" do
|
|
||||||
test "returns a user changeset" do
|
|
||||||
assert %Ecto.Changeset{} = changeset = Accounts.change_user_password(%User{})
|
|
||||||
assert changeset.required == [:password]
|
|
||||||
end
|
|
||||||
|
|
||||||
test "allows fields to be set" do
|
|
||||||
changeset =
|
|
||||||
Accounts.change_user_password(%User{}, %{
|
|
||||||
"password" => "new valid password"
|
|
||||||
})
|
|
||||||
|
|
||||||
assert changeset.valid?
|
|
||||||
assert get_change(changeset, :password) == "new valid password"
|
|
||||||
assert is_nil(get_change(changeset, :hashed_password))
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
describe "update_user_password/3" do
|
|
||||||
setup do
|
|
||||||
%{user: user_fixture()}
|
|
||||||
end
|
|
||||||
|
|
||||||
test "validates password", %{user: user} do
|
|
||||||
{:error, changeset} =
|
|
||||||
Accounts.update_user_password(user, valid_user_password(), %{
|
|
||||||
password: "not valid",
|
|
||||||
password_confirmation: "another"
|
|
||||||
})
|
|
||||||
|
|
||||||
assert %{
|
|
||||||
password: ["should be at least 12 character(s)"],
|
|
||||||
password_confirmation: ["does not match password"]
|
|
||||||
} = errors_on(changeset)
|
|
||||||
end
|
|
||||||
|
|
||||||
test "validates maximum values for password for security", %{user: user} do
|
|
||||||
too_long = String.duplicate("db", 100)
|
|
||||||
|
|
||||||
{:error, changeset} =
|
|
||||||
Accounts.update_user_password(user, valid_user_password(), %{password: too_long})
|
|
||||||
|
|
||||||
assert "should be at most 72 character(s)" in errors_on(changeset).password
|
|
||||||
end
|
|
||||||
|
|
||||||
test "validates current password", %{user: user} do
|
|
||||||
{:error, changeset} =
|
|
||||||
Accounts.update_user_password(user, "invalid", %{password: valid_user_password()})
|
|
||||||
|
|
||||||
assert %{current_password: ["is not valid"]} = errors_on(changeset)
|
|
||||||
end
|
|
||||||
|
|
||||||
test "updates the password", %{user: user} do
|
|
||||||
{:ok, user} =
|
|
||||||
Accounts.update_user_password(user, valid_user_password(), %{
|
|
||||||
password: "new valid password"
|
|
||||||
})
|
|
||||||
|
|
||||||
assert is_nil(user.password)
|
|
||||||
assert Accounts.get_user_by_email_and_password(user.email, "new valid password")
|
|
||||||
end
|
|
||||||
|
|
||||||
test "deletes all tokens for the given user", %{user: user} do
|
|
||||||
_ = Accounts.generate_user_session_token(user)
|
|
||||||
|
|
||||||
{:ok, _} =
|
|
||||||
Accounts.update_user_password(user, valid_user_password(), %{
|
|
||||||
password: "new valid password"
|
|
||||||
})
|
|
||||||
|
|
||||||
refute Repo.get_by(UserToken, user_id: user.id)
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
describe "generate_user_session_token/1" do
|
|
||||||
setup do
|
|
||||||
%{user: user_fixture()}
|
|
||||||
end
|
|
||||||
|
|
||||||
test "generates a token", %{user: user} do
|
|
||||||
token = Accounts.generate_user_session_token(user)
|
|
||||||
assert user_token = Repo.get_by(UserToken, token: token)
|
|
||||||
assert user_token.context == "session"
|
|
||||||
|
|
||||||
# Creating the same token for another user should fail
|
|
||||||
assert_raise Ecto.ConstraintError, fn ->
|
|
||||||
Repo.insert!(%UserToken{
|
|
||||||
token: user_token.token,
|
|
||||||
user_id: user_fixture().id,
|
|
||||||
context: "session"
|
|
||||||
})
|
|
||||||
end
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
describe "get_user_by_session_token/1" do
|
|
||||||
setup do
|
|
||||||
user = user_fixture()
|
|
||||||
token = Accounts.generate_user_session_token(user)
|
|
||||||
%{user: user, token: token}
|
|
||||||
end
|
|
||||||
|
|
||||||
test "returns user by token", %{user: user, token: token} do
|
|
||||||
assert session_user = Accounts.get_user_by_session_token(token)
|
|
||||||
assert session_user.id == user.id
|
|
||||||
end
|
|
||||||
|
|
||||||
test "does not return user for invalid token" do
|
|
||||||
refute Accounts.get_user_by_session_token("oops")
|
|
||||||
end
|
|
||||||
|
|
||||||
test "does not return user for expired token", %{token: token} do
|
|
||||||
{1, nil} = Repo.update_all(UserToken, set: [inserted_at: ~N[2020-01-01 00:00:00]])
|
|
||||||
refute Accounts.get_user_by_session_token(token)
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
describe "delete_user_session_token/1" do
|
|
||||||
test "deletes the token" do
|
|
||||||
user = user_fixture()
|
|
||||||
token = Accounts.generate_user_session_token(user)
|
|
||||||
assert Accounts.delete_user_session_token(token) == :ok
|
|
||||||
refute Accounts.get_user_by_session_token(token)
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
describe "deliver_user_confirmation_instructions/2" do
|
|
||||||
setup do
|
|
||||||
%{user: user_fixture()}
|
|
||||||
end
|
|
||||||
|
|
||||||
test "sends token through notification", %{user: user} do
|
|
||||||
token =
|
|
||||||
extract_user_token(fn url ->
|
|
||||||
Accounts.deliver_user_confirmation_instructions(user, url)
|
|
||||||
end)
|
|
||||||
|
|
||||||
{:ok, token} = Base.url_decode64(token, padding: false)
|
|
||||||
assert user_token = Repo.get_by(UserToken, token: :crypto.hash(:sha256, token))
|
|
||||||
assert user_token.user_id == user.id
|
|
||||||
assert user_token.sent_to == user.email
|
|
||||||
assert user_token.context == "confirm"
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
describe "confirm_user/1" do
|
|
||||||
setup do
|
|
||||||
user = user_fixture()
|
|
||||||
|
|
||||||
token =
|
|
||||||
extract_user_token(fn url ->
|
|
||||||
Accounts.deliver_user_confirmation_instructions(user, url)
|
|
||||||
end)
|
|
||||||
|
|
||||||
%{user: user, token: token}
|
|
||||||
end
|
|
||||||
|
|
||||||
test "confirms the email with a valid token", %{user: user, token: token} do
|
|
||||||
assert {:ok, confirmed_user} = Accounts.confirm_user(token)
|
|
||||||
assert confirmed_user.confirmed_at
|
|
||||||
assert confirmed_user.confirmed_at != user.confirmed_at
|
|
||||||
assert Repo.get!(User, user.id).confirmed_at
|
|
||||||
refute Repo.get_by(UserToken, user_id: user.id)
|
|
||||||
end
|
|
||||||
|
|
||||||
test "does not confirm with invalid token", %{user: user} do
|
|
||||||
assert Accounts.confirm_user("oops") == :error
|
|
||||||
refute Repo.get!(User, user.id).confirmed_at
|
|
||||||
assert Repo.get_by(UserToken, user_id: user.id)
|
|
||||||
end
|
|
||||||
|
|
||||||
test "does not confirm email if token expired", %{user: user, token: token} do
|
|
||||||
{1, nil} = Repo.update_all(UserToken, set: [inserted_at: ~N[2020-01-01 00:00:00]])
|
|
||||||
assert Accounts.confirm_user(token) == :error
|
|
||||||
refute Repo.get!(User, user.id).confirmed_at
|
|
||||||
assert Repo.get_by(UserToken, user_id: user.id)
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
describe "deliver_user_reset_password_instructions/2" do
|
|
||||||
setup do
|
|
||||||
%{user: user_fixture()}
|
|
||||||
end
|
|
||||||
|
|
||||||
test "sends token through notification", %{user: user} do
|
|
||||||
token =
|
|
||||||
extract_user_token(fn url ->
|
|
||||||
Accounts.deliver_user_reset_password_instructions(user, url)
|
|
||||||
end)
|
|
||||||
|
|
||||||
{:ok, token} = Base.url_decode64(token, padding: false)
|
|
||||||
assert user_token = Repo.get_by(UserToken, token: :crypto.hash(:sha256, token))
|
|
||||||
assert user_token.user_id == user.id
|
|
||||||
assert user_token.sent_to == user.email
|
|
||||||
assert user_token.context == "reset_password"
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
describe "get_user_by_reset_password_token/1" do
|
|
||||||
setup do
|
|
||||||
user = user_fixture()
|
|
||||||
|
|
||||||
token =
|
|
||||||
extract_user_token(fn url ->
|
|
||||||
Accounts.deliver_user_reset_password_instructions(user, url)
|
|
||||||
end)
|
|
||||||
|
|
||||||
%{user: user, token: token}
|
|
||||||
end
|
|
||||||
|
|
||||||
test "returns the user with valid token", %{user: %{id: id}, token: token} do
|
|
||||||
assert %User{id: ^id} = Accounts.get_user_by_reset_password_token(token)
|
|
||||||
assert Repo.get_by(UserToken, user_id: id)
|
|
||||||
end
|
|
||||||
|
|
||||||
test "does not return the user with invalid token", %{user: user} do
|
|
||||||
refute Accounts.get_user_by_reset_password_token("oops")
|
|
||||||
assert Repo.get_by(UserToken, user_id: user.id)
|
|
||||||
end
|
|
||||||
|
|
||||||
test "does not return the user if token expired", %{user: user, token: token} do
|
|
||||||
{1, nil} = Repo.update_all(UserToken, set: [inserted_at: ~N[2020-01-01 00:00:00]])
|
|
||||||
refute Accounts.get_user_by_reset_password_token(token)
|
|
||||||
assert Repo.get_by(UserToken, user_id: user.id)
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
describe "reset_user_password/2" do
|
|
||||||
setup do
|
|
||||||
%{user: user_fixture()}
|
|
||||||
end
|
|
||||||
|
|
||||||
test "validates password", %{user: user} do
|
|
||||||
{:error, changeset} =
|
|
||||||
Accounts.reset_user_password(user, %{
|
|
||||||
password: "not valid",
|
|
||||||
password_confirmation: "another"
|
|
||||||
})
|
|
||||||
|
|
||||||
assert %{
|
|
||||||
password: ["should be at least 12 character(s)"],
|
|
||||||
password_confirmation: ["does not match password"]
|
|
||||||
} = errors_on(changeset)
|
|
||||||
end
|
|
||||||
|
|
||||||
test "validates maximum values for password for security", %{user: user} do
|
|
||||||
too_long = String.duplicate("db", 100)
|
|
||||||
{:error, changeset} = Accounts.reset_user_password(user, %{password: too_long})
|
|
||||||
assert "should be at most 72 character(s)" in errors_on(changeset).password
|
|
||||||
end
|
|
||||||
|
|
||||||
test "updates the password", %{user: user} do
|
|
||||||
{:ok, updated_user} = Accounts.reset_user_password(user, %{password: "new valid password"})
|
|
||||||
assert is_nil(updated_user.password)
|
|
||||||
assert Accounts.get_user_by_email_and_password(user.email, "new valid password")
|
|
||||||
end
|
|
||||||
|
|
||||||
test "deletes all tokens for the given user", %{user: user} do
|
|
||||||
_ = Accounts.generate_user_session_token(user)
|
|
||||||
{:ok, _} = Accounts.reset_user_password(user, %{password: "new valid password"})
|
|
||||||
refute Repo.get_by(UserToken, user_id: user.id)
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
describe "inspect/2 for the User module" do
|
|
||||||
test "does not include password" do
|
|
||||||
refute inspect(%User{password: "123456"}) =~ "password: \"123456\""
|
|
||||||
end
|
|
||||||
end
|
|
||||||
end
|
|
||||||
@@ -1,65 +0,0 @@
|
|||||||
defmodule Malarkey.TimelineTest do
|
|
||||||
use Malarkey.DataCase
|
|
||||||
|
|
||||||
alias Malarkey.Timeline
|
|
||||||
|
|
||||||
describe "posts" do
|
|
||||||
alias Malarkey.Timeline.Post
|
|
||||||
|
|
||||||
import Malarkey.TimelineFixtures
|
|
||||||
|
|
||||||
@invalid_attrs %{body: nil, likes_count: nil, repost_count: nil, username: nil}
|
|
||||||
|
|
||||||
test "list_posts/0 returns all posts" do
|
|
||||||
post = post_fixture()
|
|
||||||
assert Timeline.list_posts() == [post]
|
|
||||||
end
|
|
||||||
|
|
||||||
test "get_post!/1 returns the post with given id" do
|
|
||||||
post = post_fixture()
|
|
||||||
assert Timeline.get_post!(post.id) == post
|
|
||||||
end
|
|
||||||
|
|
||||||
test "create_post/1 with valid data creates a post" do
|
|
||||||
valid_attrs = %{body: "some body", likes_count: 42, repost_count: 42, username: "some username"}
|
|
||||||
|
|
||||||
assert {:ok, %Post{} = post} = Timeline.create_post(valid_attrs)
|
|
||||||
assert post.body == "some body"
|
|
||||||
assert post.likes_count == 42
|
|
||||||
assert post.repost_count == 42
|
|
||||||
assert post.username == "some username"
|
|
||||||
end
|
|
||||||
|
|
||||||
test "create_post/1 with invalid data returns error changeset" do
|
|
||||||
assert {:error, %Ecto.Changeset{}} = Timeline.create_post(@invalid_attrs)
|
|
||||||
end
|
|
||||||
|
|
||||||
test "update_post/2 with valid data updates the post" do
|
|
||||||
post = post_fixture()
|
|
||||||
update_attrs = %{body: "some updated body", likes_count: 43, repost_count: 43, username: "some updated username"}
|
|
||||||
|
|
||||||
assert {:ok, %Post{} = post} = Timeline.update_post(post, update_attrs)
|
|
||||||
assert post.body == "some updated body"
|
|
||||||
assert post.likes_count == 43
|
|
||||||
assert post.repost_count == 43
|
|
||||||
assert post.username == "some updated username"
|
|
||||||
end
|
|
||||||
|
|
||||||
test "update_post/2 with invalid data returns error changeset" do
|
|
||||||
post = post_fixture()
|
|
||||||
assert {:error, %Ecto.Changeset{}} = Timeline.update_post(post, @invalid_attrs)
|
|
||||||
assert post == Timeline.get_post!(post.id)
|
|
||||||
end
|
|
||||||
|
|
||||||
test "delete_post/1 deletes the post" do
|
|
||||||
post = post_fixture()
|
|
||||||
assert {:ok, %Post{}} = Timeline.delete_post(post)
|
|
||||||
assert_raise Ecto.NoResultsError, fn -> Timeline.get_post!(post.id) end
|
|
||||||
end
|
|
||||||
|
|
||||||
test "change_post/1 returns a post changeset" do
|
|
||||||
post = post_fixture()
|
|
||||||
assert %Ecto.Changeset{} = Timeline.change_post(post)
|
|
||||||
end
|
|
||||||
end
|
|
||||||
end
|
|
||||||
@@ -1,14 +0,0 @@
|
|||||||
defmodule MalarkeyWeb.ErrorHTMLTest do
|
|
||||||
use MalarkeyWeb.ConnCase, async: true
|
|
||||||
|
|
||||||
# Bring render_to_string/3 for testing custom views
|
|
||||||
import Phoenix.Template
|
|
||||||
|
|
||||||
test "renders 404.html" do
|
|
||||||
assert render_to_string(MalarkeyWeb.ErrorHTML, "404", "html", []) == "Not Found"
|
|
||||||
end
|
|
||||||
|
|
||||||
test "renders 500.html" do
|
|
||||||
assert render_to_string(MalarkeyWeb.ErrorHTML, "500", "html", []) == "Internal Server Error"
|
|
||||||
end
|
|
||||||
end
|
|
||||||
@@ -1,12 +0,0 @@
|
|||||||
defmodule MalarkeyWeb.ErrorJSONTest do
|
|
||||||
use MalarkeyWeb.ConnCase, async: true
|
|
||||||
|
|
||||||
test "renders 404" do
|
|
||||||
assert MalarkeyWeb.ErrorJSON.render("404.json", %{}) == %{errors: %{detail: "Not Found"}}
|
|
||||||
end
|
|
||||||
|
|
||||||
test "renders 500" do
|
|
||||||
assert MalarkeyWeb.ErrorJSON.render("500.json", %{}) ==
|
|
||||||
%{errors: %{detail: "Internal Server Error"}}
|
|
||||||
end
|
|
||||||
end
|
|
||||||
@@ -1,8 +0,0 @@
|
|||||||
defmodule MalarkeyWeb.PageControllerTest do
|
|
||||||
use MalarkeyWeb.ConnCase
|
|
||||||
|
|
||||||
test "GET /", %{conn: conn} do
|
|
||||||
conn = get(conn, ~p"/")
|
|
||||||
assert html_response(conn, 200) =~ "Peace of mind from prototype to production"
|
|
||||||
end
|
|
||||||
end
|
|
||||||
@@ -1,113 +0,0 @@
|
|||||||
defmodule MalarkeyWeb.UserSessionControllerTest do
|
|
||||||
use MalarkeyWeb.ConnCase, async: true
|
|
||||||
|
|
||||||
import Malarkey.AccountsFixtures
|
|
||||||
|
|
||||||
setup do
|
|
||||||
%{user: user_fixture()}
|
|
||||||
end
|
|
||||||
|
|
||||||
describe "POST /users/log_in" do
|
|
||||||
test "logs the user in", %{conn: conn, user: user} do
|
|
||||||
conn =
|
|
||||||
post(conn, ~p"/users/log_in", %{
|
|
||||||
"user" => %{"email" => user.email, "password" => valid_user_password()}
|
|
||||||
})
|
|
||||||
|
|
||||||
assert get_session(conn, :user_token)
|
|
||||||
assert redirected_to(conn) == ~p"/"
|
|
||||||
|
|
||||||
# Now do a logged in request and assert on the menu
|
|
||||||
conn = get(conn, ~p"/")
|
|
||||||
response = html_response(conn, 200)
|
|
||||||
assert response =~ user.email
|
|
||||||
assert response =~ "Settings</a>"
|
|
||||||
assert response =~ "Log out</a>"
|
|
||||||
end
|
|
||||||
|
|
||||||
test "logs the user in with remember me", %{conn: conn, user: user} do
|
|
||||||
conn =
|
|
||||||
post(conn, ~p"/users/log_in", %{
|
|
||||||
"user" => %{
|
|
||||||
"email" => user.email,
|
|
||||||
"password" => valid_user_password(),
|
|
||||||
"remember_me" => "true"
|
|
||||||
}
|
|
||||||
})
|
|
||||||
|
|
||||||
assert conn.resp_cookies["_malarkey_web_user_remember_me"]
|
|
||||||
assert redirected_to(conn) == ~p"/"
|
|
||||||
end
|
|
||||||
|
|
||||||
test "logs the user in with return to", %{conn: conn, user: user} do
|
|
||||||
conn =
|
|
||||||
conn
|
|
||||||
|> init_test_session(user_return_to: "/foo/bar")
|
|
||||||
|> post(~p"/users/log_in", %{
|
|
||||||
"user" => %{
|
|
||||||
"email" => user.email,
|
|
||||||
"password" => valid_user_password()
|
|
||||||
}
|
|
||||||
})
|
|
||||||
|
|
||||||
assert redirected_to(conn) == "/foo/bar"
|
|
||||||
assert Phoenix.Flash.get(conn.assigns.flash, :info) =~ "Welcome back!"
|
|
||||||
end
|
|
||||||
|
|
||||||
test "login following registration", %{conn: conn, user: user} do
|
|
||||||
conn =
|
|
||||||
conn
|
|
||||||
|> post(~p"/users/log_in", %{
|
|
||||||
"_action" => "registered",
|
|
||||||
"user" => %{
|
|
||||||
"email" => user.email,
|
|
||||||
"password" => valid_user_password()
|
|
||||||
}
|
|
||||||
})
|
|
||||||
|
|
||||||
assert redirected_to(conn) == ~p"/"
|
|
||||||
assert Phoenix.Flash.get(conn.assigns.flash, :info) =~ "Account created successfully"
|
|
||||||
end
|
|
||||||
|
|
||||||
test "login following password update", %{conn: conn, user: user} do
|
|
||||||
conn =
|
|
||||||
conn
|
|
||||||
|> post(~p"/users/log_in", %{
|
|
||||||
"_action" => "password_updated",
|
|
||||||
"user" => %{
|
|
||||||
"email" => user.email,
|
|
||||||
"password" => valid_user_password()
|
|
||||||
}
|
|
||||||
})
|
|
||||||
|
|
||||||
assert redirected_to(conn) == ~p"/users/settings"
|
|
||||||
assert Phoenix.Flash.get(conn.assigns.flash, :info) =~ "Password updated successfully"
|
|
||||||
end
|
|
||||||
|
|
||||||
test "redirects to login page with invalid credentials", %{conn: conn} do
|
|
||||||
conn =
|
|
||||||
post(conn, ~p"/users/log_in", %{
|
|
||||||
"user" => %{"email" => "invalid@email.com", "password" => "invalid_password"}
|
|
||||||
})
|
|
||||||
|
|
||||||
assert Phoenix.Flash.get(conn.assigns.flash, :error) == "Invalid email or password"
|
|
||||||
assert redirected_to(conn) == ~p"/users/log_in"
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
describe "DELETE /users/log_out" do
|
|
||||||
test "logs the user out", %{conn: conn, user: user} do
|
|
||||||
conn = conn |> log_in_user(user) |> delete(~p"/users/log_out")
|
|
||||||
assert redirected_to(conn) == ~p"/"
|
|
||||||
refute get_session(conn, :user_token)
|
|
||||||
assert Phoenix.Flash.get(conn.assigns.flash, :info) =~ "Logged out successfully"
|
|
||||||
end
|
|
||||||
|
|
||||||
test "succeeds even if the user is not logged in", %{conn: conn} do
|
|
||||||
conn = delete(conn, ~p"/users/log_out")
|
|
||||||
assert redirected_to(conn) == ~p"/"
|
|
||||||
refute get_session(conn, :user_token)
|
|
||||||
assert Phoenix.Flash.get(conn.assigns.flash, :info) =~ "Logged out successfully"
|
|
||||||
end
|
|
||||||
end
|
|
||||||
end
|
|
||||||
@@ -1,110 +0,0 @@
|
|||||||
defmodule MalarkeyWeb.PostLiveTest do
|
|
||||||
use MalarkeyWeb.ConnCase
|
|
||||||
|
|
||||||
import Phoenix.LiveViewTest
|
|
||||||
import Malarkey.TimelineFixtures
|
|
||||||
|
|
||||||
@create_attrs %{body: "some body", likes_count: 42, repost_count: 42, username: "some username"}
|
|
||||||
@update_attrs %{body: "some updated body", likes_count: 43, repost_count: 43, username: "some updated username"}
|
|
||||||
@invalid_attrs %{body: nil, likes_count: nil, repost_count: nil, username: nil}
|
|
||||||
|
|
||||||
defp create_post(_) do
|
|
||||||
post = post_fixture()
|
|
||||||
%{post: post}
|
|
||||||
end
|
|
||||||
|
|
||||||
describe "Index" do
|
|
||||||
setup [:create_post]
|
|
||||||
|
|
||||||
test "lists all posts", %{conn: conn, post: post} do
|
|
||||||
{:ok, _index_live, html} = live(conn, ~p"/posts")
|
|
||||||
|
|
||||||
assert html =~ "Listing Posts"
|
|
||||||
assert html =~ post.body
|
|
||||||
end
|
|
||||||
|
|
||||||
test "saves new post", %{conn: conn} do
|
|
||||||
{:ok, index_live, _html} = live(conn, ~p"/posts")
|
|
||||||
|
|
||||||
assert index_live |> element("a", "New Post") |> render_click() =~
|
|
||||||
"New Post"
|
|
||||||
|
|
||||||
assert_patch(index_live, ~p"/posts/new")
|
|
||||||
|
|
||||||
assert index_live
|
|
||||||
|> form("#post-form", post: @invalid_attrs)
|
|
||||||
|> render_change() =~ "can't be blank"
|
|
||||||
|
|
||||||
{:ok, _, html} =
|
|
||||||
index_live
|
|
||||||
|> form("#post-form", post: @create_attrs)
|
|
||||||
|> render_submit()
|
|
||||||
|> follow_redirect(conn, ~p"/posts")
|
|
||||||
|
|
||||||
assert html =~ "Post created successfully"
|
|
||||||
assert html =~ "some body"
|
|
||||||
end
|
|
||||||
|
|
||||||
test "updates post in listing", %{conn: conn, post: post} do
|
|
||||||
{:ok, index_live, _html} = live(conn, ~p"/posts")
|
|
||||||
|
|
||||||
assert index_live |> element("#posts-#{post.id} a", "Edit") |> render_click() =~
|
|
||||||
"Edit Post"
|
|
||||||
|
|
||||||
assert_patch(index_live, ~p"/posts/#{post}/edit")
|
|
||||||
|
|
||||||
assert index_live
|
|
||||||
|> form("#post-form", post: @invalid_attrs)
|
|
||||||
|> render_change() =~ "can't be blank"
|
|
||||||
|
|
||||||
{:ok, _, html} =
|
|
||||||
index_live
|
|
||||||
|> form("#post-form", post: @update_attrs)
|
|
||||||
|> render_submit()
|
|
||||||
|> follow_redirect(conn, ~p"/posts")
|
|
||||||
|
|
||||||
assert html =~ "Post updated successfully"
|
|
||||||
assert html =~ "some updated body"
|
|
||||||
end
|
|
||||||
|
|
||||||
test "deletes post in listing", %{conn: conn, post: post} do
|
|
||||||
{:ok, index_live, _html} = live(conn, ~p"/posts")
|
|
||||||
|
|
||||||
assert index_live |> element("#posts-#{post.id} a", "Delete") |> render_click()
|
|
||||||
refute has_element?(index_live, "#post-#{post.id}")
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
describe "Show" do
|
|
||||||
setup [:create_post]
|
|
||||||
|
|
||||||
test "displays post", %{conn: conn, post: post} do
|
|
||||||
{:ok, _show_live, html} = live(conn, ~p"/posts/#{post}")
|
|
||||||
|
|
||||||
assert html =~ "Show Post"
|
|
||||||
assert html =~ post.body
|
|
||||||
end
|
|
||||||
|
|
||||||
test "updates post within modal", %{conn: conn, post: post} do
|
|
||||||
{:ok, show_live, _html} = live(conn, ~p"/posts/#{post}")
|
|
||||||
|
|
||||||
assert show_live |> element("a", "Edit") |> render_click() =~
|
|
||||||
"Edit Post"
|
|
||||||
|
|
||||||
assert_patch(show_live, ~p"/posts/#{post}/show/edit")
|
|
||||||
|
|
||||||
assert show_live
|
|
||||||
|> form("#post-form", post: @invalid_attrs)
|
|
||||||
|> render_change() =~ "can't be blank"
|
|
||||||
|
|
||||||
{:ok, _, html} =
|
|
||||||
show_live
|
|
||||||
|> form("#post-form", post: @update_attrs)
|
|
||||||
|> render_submit()
|
|
||||||
|> follow_redirect(conn, ~p"/posts/#{post}")
|
|
||||||
|
|
||||||
assert html =~ "Post updated successfully"
|
|
||||||
assert html =~ "some updated body"
|
|
||||||
end
|
|
||||||
end
|
|
||||||
end
|
|
||||||
@@ -1,67 +0,0 @@
|
|||||||
defmodule MalarkeyWeb.UserConfirmationInstructionsLiveTest do
|
|
||||||
use MalarkeyWeb.ConnCase
|
|
||||||
|
|
||||||
import Phoenix.LiveViewTest
|
|
||||||
import Malarkey.AccountsFixtures
|
|
||||||
|
|
||||||
alias Malarkey.Accounts
|
|
||||||
alias Malarkey.Repo
|
|
||||||
|
|
||||||
setup do
|
|
||||||
%{user: user_fixture()}
|
|
||||||
end
|
|
||||||
|
|
||||||
describe "Resend confirmation" do
|
|
||||||
test "renders the resend confirmation page", %{conn: conn} do
|
|
||||||
{:ok, _lv, html} = live(conn, ~p"/users/confirm")
|
|
||||||
assert html =~ "Resend confirmation instructions"
|
|
||||||
end
|
|
||||||
|
|
||||||
test "sends a new confirmation token", %{conn: conn, user: user} do
|
|
||||||
{:ok, lv, _html} = live(conn, ~p"/users/confirm")
|
|
||||||
|
|
||||||
{:ok, conn} =
|
|
||||||
lv
|
|
||||||
|> form("#resend_confirmation_form", user: %{email: user.email})
|
|
||||||
|> render_submit()
|
|
||||||
|> follow_redirect(conn, ~p"/")
|
|
||||||
|
|
||||||
assert Phoenix.Flash.get(conn.assigns.flash, :info) =~
|
|
||||||
"If your email is in our system"
|
|
||||||
|
|
||||||
assert Repo.get_by!(Accounts.UserToken, user_id: user.id).context == "confirm"
|
|
||||||
end
|
|
||||||
|
|
||||||
test "does not send confirmation token if user is confirmed", %{conn: conn, user: user} do
|
|
||||||
Repo.update!(Accounts.User.confirm_changeset(user))
|
|
||||||
|
|
||||||
{:ok, lv, _html} = live(conn, ~p"/users/confirm")
|
|
||||||
|
|
||||||
{:ok, conn} =
|
|
||||||
lv
|
|
||||||
|> form("#resend_confirmation_form", user: %{email: user.email})
|
|
||||||
|> render_submit()
|
|
||||||
|> follow_redirect(conn, ~p"/")
|
|
||||||
|
|
||||||
assert Phoenix.Flash.get(conn.assigns.flash, :info) =~
|
|
||||||
"If your email is in our system"
|
|
||||||
|
|
||||||
refute Repo.get_by(Accounts.UserToken, user_id: user.id)
|
|
||||||
end
|
|
||||||
|
|
||||||
test "does not send confirmation token if email is invalid", %{conn: conn} do
|
|
||||||
{:ok, lv, _html} = live(conn, ~p"/users/confirm")
|
|
||||||
|
|
||||||
{:ok, conn} =
|
|
||||||
lv
|
|
||||||
|> form("#resend_confirmation_form", user: %{email: "unknown@example.com"})
|
|
||||||
|> render_submit()
|
|
||||||
|> follow_redirect(conn, ~p"/")
|
|
||||||
|
|
||||||
assert Phoenix.Flash.get(conn.assigns.flash, :info) =~
|
|
||||||
"If your email is in our system"
|
|
||||||
|
|
||||||
assert Repo.all(Accounts.UserToken) == []
|
|
||||||
end
|
|
||||||
end
|
|
||||||
end
|
|
||||||
@@ -1,88 +0,0 @@
|
|||||||
defmodule MalarkeyWeb.UserConfirmationLiveTest do
|
|
||||||
use MalarkeyWeb.ConnCase
|
|
||||||
|
|
||||||
import Phoenix.LiveViewTest
|
|
||||||
import Malarkey.AccountsFixtures
|
|
||||||
|
|
||||||
alias Malarkey.Accounts
|
|
||||||
alias Malarkey.Repo
|
|
||||||
|
|
||||||
setup do
|
|
||||||
%{user: user_fixture()}
|
|
||||||
end
|
|
||||||
|
|
||||||
describe "Confirm user" do
|
|
||||||
test "renders confirmation page", %{conn: conn} do
|
|
||||||
{:ok, _lv, html} = live(conn, ~p"/users/confirm/some-token")
|
|
||||||
assert html =~ "Confirm Account"
|
|
||||||
end
|
|
||||||
|
|
||||||
test "confirms the given token once", %{conn: conn, user: user} do
|
|
||||||
token =
|
|
||||||
extract_user_token(fn url ->
|
|
||||||
Accounts.deliver_user_confirmation_instructions(user, url)
|
|
||||||
end)
|
|
||||||
|
|
||||||
{:ok, lv, _html} = live(conn, ~p"/users/confirm/#{token}")
|
|
||||||
|
|
||||||
result =
|
|
||||||
lv
|
|
||||||
|> form("#confirmation_form")
|
|
||||||
|> render_submit()
|
|
||||||
|> follow_redirect(conn, "/")
|
|
||||||
|
|
||||||
assert {:ok, conn} = result
|
|
||||||
|
|
||||||
assert Phoenix.Flash.get(conn.assigns.flash, :info) =~
|
|
||||||
"User confirmed successfully"
|
|
||||||
|
|
||||||
assert Accounts.get_user!(user.id).confirmed_at
|
|
||||||
refute get_session(conn, :user_token)
|
|
||||||
assert Repo.all(Accounts.UserToken) == []
|
|
||||||
|
|
||||||
# when not logged in
|
|
||||||
{:ok, lv, _html} = live(conn, ~p"/users/confirm/#{token}")
|
|
||||||
|
|
||||||
result =
|
|
||||||
lv
|
|
||||||
|> form("#confirmation_form")
|
|
||||||
|> render_submit()
|
|
||||||
|> follow_redirect(conn, "/")
|
|
||||||
|
|
||||||
assert {:ok, conn} = result
|
|
||||||
|
|
||||||
assert Phoenix.Flash.get(conn.assigns.flash, :error) =~
|
|
||||||
"User confirmation link is invalid or it has expired"
|
|
||||||
|
|
||||||
# when logged in
|
|
||||||
{:ok, lv, _html} =
|
|
||||||
build_conn()
|
|
||||||
|> log_in_user(user)
|
|
||||||
|> live(~p"/users/confirm/#{token}")
|
|
||||||
|
|
||||||
result =
|
|
||||||
lv
|
|
||||||
|> form("#confirmation_form")
|
|
||||||
|> render_submit()
|
|
||||||
|> follow_redirect(conn, "/")
|
|
||||||
|
|
||||||
assert {:ok, conn} = result
|
|
||||||
refute Phoenix.Flash.get(conn.assigns.flash, :error)
|
|
||||||
end
|
|
||||||
|
|
||||||
test "does not confirm email with invalid token", %{conn: conn, user: user} do
|
|
||||||
{:ok, lv, _html} = live(conn, ~p"/users/confirm/invalid-token")
|
|
||||||
|
|
||||||
{:ok, conn} =
|
|
||||||
lv
|
|
||||||
|> form("#confirmation_form")
|
|
||||||
|> render_submit()
|
|
||||||
|> follow_redirect(conn, ~p"/")
|
|
||||||
|
|
||||||
assert Phoenix.Flash.get(conn.assigns.flash, :error) =~
|
|
||||||
"User confirmation link is invalid or it has expired"
|
|
||||||
|
|
||||||
refute Accounts.get_user!(user.id).confirmed_at
|
|
||||||
end
|
|
||||||
end
|
|
||||||
end
|
|
||||||
@@ -1,63 +0,0 @@
|
|||||||
defmodule MalarkeyWeb.UserForgotPasswordLiveTest do
|
|
||||||
use MalarkeyWeb.ConnCase
|
|
||||||
|
|
||||||
import Phoenix.LiveViewTest
|
|
||||||
import Malarkey.AccountsFixtures
|
|
||||||
|
|
||||||
alias Malarkey.Accounts
|
|
||||||
alias Malarkey.Repo
|
|
||||||
|
|
||||||
describe "Forgot password page" do
|
|
||||||
test "renders email page", %{conn: conn} do
|
|
||||||
{:ok, _lv, html} = live(conn, ~p"/users/reset_password")
|
|
||||||
|
|
||||||
assert html =~ "Forgot your password?"
|
|
||||||
assert html =~ "Register</a>"
|
|
||||||
assert html =~ "Log in</a>"
|
|
||||||
end
|
|
||||||
|
|
||||||
test "redirects if already logged in", %{conn: conn} do
|
|
||||||
result =
|
|
||||||
conn
|
|
||||||
|> log_in_user(user_fixture())
|
|
||||||
|> live(~p"/users/reset_password")
|
|
||||||
|> follow_redirect(conn, ~p"/")
|
|
||||||
|
|
||||||
assert {:ok, _conn} = result
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
describe "Reset link" do
|
|
||||||
setup do
|
|
||||||
%{user: user_fixture()}
|
|
||||||
end
|
|
||||||
|
|
||||||
test "sends a new reset password token", %{conn: conn, user: user} do
|
|
||||||
{:ok, lv, _html} = live(conn, ~p"/users/reset_password")
|
|
||||||
|
|
||||||
{:ok, conn} =
|
|
||||||
lv
|
|
||||||
|> form("#reset_password_form", user: %{"email" => user.email})
|
|
||||||
|> render_submit()
|
|
||||||
|> follow_redirect(conn, "/")
|
|
||||||
|
|
||||||
assert Phoenix.Flash.get(conn.assigns.flash, :info) =~ "If your email is in our system"
|
|
||||||
|
|
||||||
assert Repo.get_by!(Accounts.UserToken, user_id: user.id).context ==
|
|
||||||
"reset_password"
|
|
||||||
end
|
|
||||||
|
|
||||||
test "does not send reset password token if email is invalid", %{conn: conn} do
|
|
||||||
{:ok, lv, _html} = live(conn, ~p"/users/reset_password")
|
|
||||||
|
|
||||||
{:ok, conn} =
|
|
||||||
lv
|
|
||||||
|> form("#reset_password_form", user: %{"email" => "unknown@example.com"})
|
|
||||||
|> render_submit()
|
|
||||||
|> follow_redirect(conn, "/")
|
|
||||||
|
|
||||||
assert Phoenix.Flash.get(conn.assigns.flash, :info) =~ "If your email is in our system"
|
|
||||||
assert Repo.all(Accounts.UserToken) == []
|
|
||||||
end
|
|
||||||
end
|
|
||||||
end
|
|
||||||
@@ -1,87 +0,0 @@
|
|||||||
defmodule MalarkeyWeb.UserLoginLiveTest do
|
|
||||||
use MalarkeyWeb.ConnCase
|
|
||||||
|
|
||||||
import Phoenix.LiveViewTest
|
|
||||||
import Malarkey.AccountsFixtures
|
|
||||||
|
|
||||||
describe "Log in page" do
|
|
||||||
test "renders log in page", %{conn: conn} do
|
|
||||||
{:ok, _lv, html} = live(conn, ~p"/users/log_in")
|
|
||||||
|
|
||||||
assert html =~ "Log in"
|
|
||||||
assert html =~ "Register"
|
|
||||||
assert html =~ "Forgot your password?"
|
|
||||||
end
|
|
||||||
|
|
||||||
test "redirects if already logged in", %{conn: conn} do
|
|
||||||
result =
|
|
||||||
conn
|
|
||||||
|> log_in_user(user_fixture())
|
|
||||||
|> live(~p"/users/log_in")
|
|
||||||
|> follow_redirect(conn, "/")
|
|
||||||
|
|
||||||
assert {:ok, _conn} = result
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
describe "user login" do
|
|
||||||
test "redirects if user login with valid credentials", %{conn: conn} do
|
|
||||||
password = "123456789abcd"
|
|
||||||
user = user_fixture(%{password: password})
|
|
||||||
|
|
||||||
{:ok, lv, _html} = live(conn, ~p"/users/log_in")
|
|
||||||
|
|
||||||
form =
|
|
||||||
form(lv, "#login_form", user: %{email: user.email, password: password, remember_me: true})
|
|
||||||
|
|
||||||
conn = submit_form(form, conn)
|
|
||||||
|
|
||||||
assert redirected_to(conn) == ~p"/"
|
|
||||||
end
|
|
||||||
|
|
||||||
test "redirects to login page with a flash error if there are no valid credentials", %{
|
|
||||||
conn: conn
|
|
||||||
} do
|
|
||||||
{:ok, lv, _html} = live(conn, ~p"/users/log_in")
|
|
||||||
|
|
||||||
form =
|
|
||||||
form(lv, "#login_form",
|
|
||||||
user: %{email: "test@email.com", password: "123456", remember_me: true}
|
|
||||||
)
|
|
||||||
|
|
||||||
conn = submit_form(form, conn)
|
|
||||||
|
|
||||||
assert Phoenix.Flash.get(conn.assigns.flash, :error) == "Invalid email or password"
|
|
||||||
|
|
||||||
assert redirected_to(conn) == "/users/log_in"
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
describe "login navigation" do
|
|
||||||
test "redirects to registration page when the Register button is clicked", %{conn: conn} do
|
|
||||||
{:ok, lv, _html} = live(conn, ~p"/users/log_in")
|
|
||||||
|
|
||||||
{:ok, _login_live, login_html} =
|
|
||||||
lv
|
|
||||||
|> element(~s|a:fl-contains("Sign up")|)
|
|
||||||
|> render_click()
|
|
||||||
|> follow_redirect(conn, ~p"/users/register")
|
|
||||||
|
|
||||||
assert login_html =~ "Register"
|
|
||||||
end
|
|
||||||
|
|
||||||
test "redirects to forgot password page when the Forgot Password button is clicked", %{
|
|
||||||
conn: conn
|
|
||||||
} do
|
|
||||||
{:ok, lv, _html} = live(conn, ~p"/users/log_in")
|
|
||||||
|
|
||||||
{:ok, conn} =
|
|
||||||
lv
|
|
||||||
|> element(~s{a:fl-contains('Forgot your password?')})
|
|
||||||
|> render_click()
|
|
||||||
|> follow_redirect(conn, ~p"/users/reset_password")
|
|
||||||
|
|
||||||
assert conn.resp_body =~ "Forgot your password?"
|
|
||||||
end
|
|
||||||
end
|
|
||||||
end
|
|
||||||
@@ -1,84 +0,0 @@
|
|||||||
defmodule MalarkeyWeb.UserRegistrationLiveTest do
|
|
||||||
use MalarkeyWeb.ConnCase
|
|
||||||
|
|
||||||
import Phoenix.LiveViewTest
|
|
||||||
import Malarkey.AccountsFixtures
|
|
||||||
|
|
||||||
describe "Registration page" do
|
|
||||||
test "renders registration page", %{conn: conn} do
|
|
||||||
{:ok, _lv, html} = live(conn, ~p"/users/register")
|
|
||||||
|
|
||||||
assert html =~ "Register"
|
|
||||||
assert html =~ "Log in"
|
|
||||||
end
|
|
||||||
|
|
||||||
test "redirects if already logged in", %{conn: conn} do
|
|
||||||
result =
|
|
||||||
conn
|
|
||||||
|> log_in_user(user_fixture())
|
|
||||||
|> live(~p"/users/register")
|
|
||||||
|> follow_redirect(conn, "/")
|
|
||||||
|
|
||||||
assert {:ok, _conn} = result
|
|
||||||
end
|
|
||||||
|
|
||||||
test "renders errors for invalid data", %{conn: conn} do
|
|
||||||
{:ok, lv, _html} = live(conn, ~p"/users/register")
|
|
||||||
|
|
||||||
result =
|
|
||||||
lv
|
|
||||||
|> element("#registration_form")
|
|
||||||
|> render_change(user: %{"email" => "with spaces", "password" => "too short"})
|
|
||||||
|
|
||||||
assert result =~ "Register"
|
|
||||||
assert result =~ "must have the @ sign and no spaces"
|
|
||||||
assert result =~ "should be at least 12 character"
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
describe "register user" do
|
|
||||||
test "creates account and logs the user in", %{conn: conn} do
|
|
||||||
{:ok, lv, _html} = live(conn, ~p"/users/register")
|
|
||||||
|
|
||||||
email = unique_user_email()
|
|
||||||
form = form(lv, "#registration_form", user: valid_user_attributes(email: email))
|
|
||||||
render_submit(form)
|
|
||||||
conn = follow_trigger_action(form, conn)
|
|
||||||
|
|
||||||
assert redirected_to(conn) == ~p"/"
|
|
||||||
|
|
||||||
# Now do a logged in request and assert on the menu
|
|
||||||
conn = get(conn, "/")
|
|
||||||
response = html_response(conn, 200)
|
|
||||||
assert response =~ email
|
|
||||||
assert response =~ "Settings"
|
|
||||||
assert response =~ "Log out"
|
|
||||||
end
|
|
||||||
|
|
||||||
test "renders errors for duplicated email", %{conn: conn} do
|
|
||||||
{:ok, lv, _html} = live(conn, ~p"/users/register")
|
|
||||||
|
|
||||||
user = user_fixture(%{email: "test@email.com"})
|
|
||||||
|
|
||||||
lv
|
|
||||||
|> form("#registration_form",
|
|
||||||
user: %{"email" => user.email, "password" => "valid_password"}
|
|
||||||
)
|
|
||||||
|> render_submit() =~ "has already been taken"
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
describe "registration navigation" do
|
|
||||||
test "redirects to login page when the Log in button is clicked", %{conn: conn} do
|
|
||||||
{:ok, lv, _html} = live(conn, ~p"/users/register")
|
|
||||||
|
|
||||||
{:ok, _login_live, login_html} =
|
|
||||||
lv
|
|
||||||
|> element(~s|main a:fl-contains("Sign in")|)
|
|
||||||
|> render_click()
|
|
||||||
|> follow_redirect(conn, ~p"/users/log_in")
|
|
||||||
|
|
||||||
assert login_html =~ "Log in"
|
|
||||||
end
|
|
||||||
end
|
|
||||||
end
|
|
||||||
@@ -1,118 +0,0 @@
|
|||||||
defmodule MalarkeyWeb.UserResetPasswordLiveTest do
|
|
||||||
use MalarkeyWeb.ConnCase
|
|
||||||
|
|
||||||
import Phoenix.LiveViewTest
|
|
||||||
import Malarkey.AccountsFixtures
|
|
||||||
|
|
||||||
alias Malarkey.Accounts
|
|
||||||
|
|
||||||
setup do
|
|
||||||
user = user_fixture()
|
|
||||||
|
|
||||||
token =
|
|
||||||
extract_user_token(fn url ->
|
|
||||||
Accounts.deliver_user_reset_password_instructions(user, url)
|
|
||||||
end)
|
|
||||||
|
|
||||||
%{token: token, user: user}
|
|
||||||
end
|
|
||||||
|
|
||||||
describe "Reset password page" do
|
|
||||||
test "renders reset password with valid token", %{conn: conn, token: token} do
|
|
||||||
{:ok, _lv, html} = live(conn, ~p"/users/reset_password/#{token}")
|
|
||||||
|
|
||||||
assert html =~ "Reset Password"
|
|
||||||
end
|
|
||||||
|
|
||||||
test "does not render reset password with invalid token", %{conn: conn} do
|
|
||||||
{:error, {:redirect, to}} = live(conn, ~p"/users/reset_password/invalid")
|
|
||||||
|
|
||||||
assert to == %{
|
|
||||||
flash: %{"error" => "Reset password link is invalid or it has expired."},
|
|
||||||
to: ~p"/"
|
|
||||||
}
|
|
||||||
end
|
|
||||||
|
|
||||||
test "renders errors for invalid data", %{conn: conn, token: token} do
|
|
||||||
{:ok, lv, _html} = live(conn, ~p"/users/reset_password/#{token}")
|
|
||||||
|
|
||||||
result =
|
|
||||||
lv
|
|
||||||
|> element("#reset_password_form")
|
|
||||||
|> render_change(
|
|
||||||
user: %{"password" => "secret12", "confirmation_password" => "secret123456"}
|
|
||||||
)
|
|
||||||
|
|
||||||
assert result =~ "should be at least 12 character"
|
|
||||||
assert result =~ "does not match password"
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
describe "Reset Password" do
|
|
||||||
test "resets password once", %{conn: conn, token: token, user: user} do
|
|
||||||
{:ok, lv, _html} = live(conn, ~p"/users/reset_password/#{token}")
|
|
||||||
|
|
||||||
{:ok, conn} =
|
|
||||||
lv
|
|
||||||
|> form("#reset_password_form",
|
|
||||||
user: %{
|
|
||||||
"password" => "new valid password",
|
|
||||||
"password_confirmation" => "new valid password"
|
|
||||||
}
|
|
||||||
)
|
|
||||||
|> render_submit()
|
|
||||||
|> follow_redirect(conn, ~p"/users/log_in")
|
|
||||||
|
|
||||||
refute get_session(conn, :user_token)
|
|
||||||
assert Phoenix.Flash.get(conn.assigns.flash, :info) =~ "Password reset successfully"
|
|
||||||
assert Accounts.get_user_by_email_and_password(user.email, "new valid password")
|
|
||||||
end
|
|
||||||
|
|
||||||
test "does not reset password on invalid data", %{conn: conn, token: token} do
|
|
||||||
{:ok, lv, _html} = live(conn, ~p"/users/reset_password/#{token}")
|
|
||||||
|
|
||||||
result =
|
|
||||||
lv
|
|
||||||
|> form("#reset_password_form",
|
|
||||||
user: %{
|
|
||||||
"password" => "too short",
|
|
||||||
"password_confirmation" => "does not match"
|
|
||||||
}
|
|
||||||
)
|
|
||||||
|> render_submit()
|
|
||||||
|
|
||||||
assert result =~ "Reset Password"
|
|
||||||
assert result =~ "should be at least 12 character(s)"
|
|
||||||
assert result =~ "does not match password"
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
describe "Reset password navigation" do
|
|
||||||
test "redirects to login page when the Log in button is clicked", %{conn: conn, token: token} do
|
|
||||||
{:ok, lv, _html} = live(conn, ~p"/users/reset_password/#{token}")
|
|
||||||
|
|
||||||
{:ok, conn} =
|
|
||||||
lv
|
|
||||||
|> element(~s|main a:fl-contains("Log in")|)
|
|
||||||
|> render_click()
|
|
||||||
|> follow_redirect(conn, ~p"/users/log_in")
|
|
||||||
|
|
||||||
assert conn.resp_body =~ "Log in"
|
|
||||||
end
|
|
||||||
|
|
||||||
test "redirects to password reset page when the Register button is clicked", %{
|
|
||||||
conn: conn,
|
|
||||||
token: token
|
|
||||||
} do
|
|
||||||
{:ok, lv, _html} = live(conn, ~p"/users/reset_password/#{token}")
|
|
||||||
|
|
||||||
{:ok, conn} =
|
|
||||||
lv
|
|
||||||
|> element(~s|main a:fl-contains("Register")|)
|
|
||||||
|> render_click()
|
|
||||||
|> follow_redirect(conn, ~p"/users/register")
|
|
||||||
|
|
||||||
assert conn.resp_body =~ "Register"
|
|
||||||
end
|
|
||||||
end
|
|
||||||
end
|
|
||||||
@@ -1,210 +0,0 @@
|
|||||||
defmodule MalarkeyWeb.UserSettingsLiveTest do
|
|
||||||
use MalarkeyWeb.ConnCase
|
|
||||||
|
|
||||||
alias Malarkey.Accounts
|
|
||||||
import Phoenix.LiveViewTest
|
|
||||||
import Malarkey.AccountsFixtures
|
|
||||||
|
|
||||||
describe "Settings page" do
|
|
||||||
test "renders settings page", %{conn: conn} do
|
|
||||||
{:ok, _lv, html} =
|
|
||||||
conn
|
|
||||||
|> log_in_user(user_fixture())
|
|
||||||
|> live(~p"/users/settings")
|
|
||||||
|
|
||||||
assert html =~ "Change Email"
|
|
||||||
assert html =~ "Change Password"
|
|
||||||
end
|
|
||||||
|
|
||||||
test "redirects if user is not logged in", %{conn: conn} do
|
|
||||||
assert {:error, redirect} = live(conn, ~p"/users/settings")
|
|
||||||
|
|
||||||
assert {:redirect, %{to: path, flash: flash}} = redirect
|
|
||||||
assert path == ~p"/users/log_in"
|
|
||||||
assert %{"error" => "You must log in to access this page."} = flash
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
describe "update email form" do
|
|
||||||
setup %{conn: conn} do
|
|
||||||
password = valid_user_password()
|
|
||||||
user = user_fixture(%{password: password})
|
|
||||||
%{conn: log_in_user(conn, user), user: user, password: password}
|
|
||||||
end
|
|
||||||
|
|
||||||
test "updates the user email", %{conn: conn, password: password, user: user} do
|
|
||||||
new_email = unique_user_email()
|
|
||||||
|
|
||||||
{:ok, lv, _html} = live(conn, ~p"/users/settings")
|
|
||||||
|
|
||||||
result =
|
|
||||||
lv
|
|
||||||
|> form("#email_form", %{
|
|
||||||
"current_password" => password,
|
|
||||||
"user" => %{"email" => new_email}
|
|
||||||
})
|
|
||||||
|> render_submit()
|
|
||||||
|
|
||||||
assert result =~ "A link to confirm your email"
|
|
||||||
assert Accounts.get_user_by_email(user.email)
|
|
||||||
end
|
|
||||||
|
|
||||||
test "renders errors with invalid data (phx-change)", %{conn: conn} do
|
|
||||||
{:ok, lv, _html} = live(conn, ~p"/users/settings")
|
|
||||||
|
|
||||||
result =
|
|
||||||
lv
|
|
||||||
|> element("#email_form")
|
|
||||||
|> render_change(%{
|
|
||||||
"action" => "update_email",
|
|
||||||
"current_password" => "invalid",
|
|
||||||
"user" => %{"email" => "with spaces"}
|
|
||||||
})
|
|
||||||
|
|
||||||
assert result =~ "Change Email"
|
|
||||||
assert result =~ "must have the @ sign and no spaces"
|
|
||||||
end
|
|
||||||
|
|
||||||
test "renders errors with invalid data (phx-submit)", %{conn: conn, user: user} do
|
|
||||||
{:ok, lv, _html} = live(conn, ~p"/users/settings")
|
|
||||||
|
|
||||||
result =
|
|
||||||
lv
|
|
||||||
|> form("#email_form", %{
|
|
||||||
"current_password" => "invalid",
|
|
||||||
"user" => %{"email" => user.email}
|
|
||||||
})
|
|
||||||
|> render_submit()
|
|
||||||
|
|
||||||
assert result =~ "Change Email"
|
|
||||||
assert result =~ "did not change"
|
|
||||||
assert result =~ "is not valid"
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
describe "update password form" do
|
|
||||||
setup %{conn: conn} do
|
|
||||||
password = valid_user_password()
|
|
||||||
user = user_fixture(%{password: password})
|
|
||||||
%{conn: log_in_user(conn, user), user: user, password: password}
|
|
||||||
end
|
|
||||||
|
|
||||||
test "updates the user password", %{conn: conn, user: user, password: password} do
|
|
||||||
new_password = valid_user_password()
|
|
||||||
|
|
||||||
{:ok, lv, _html} = live(conn, ~p"/users/settings")
|
|
||||||
|
|
||||||
form =
|
|
||||||
form(lv, "#password_form", %{
|
|
||||||
"current_password" => password,
|
|
||||||
"user" => %{
|
|
||||||
"email" => user.email,
|
|
||||||
"password" => new_password,
|
|
||||||
"password_confirmation" => new_password
|
|
||||||
}
|
|
||||||
})
|
|
||||||
|
|
||||||
render_submit(form)
|
|
||||||
|
|
||||||
new_password_conn = follow_trigger_action(form, conn)
|
|
||||||
|
|
||||||
assert redirected_to(new_password_conn) == ~p"/users/settings"
|
|
||||||
|
|
||||||
assert get_session(new_password_conn, :user_token) != get_session(conn, :user_token)
|
|
||||||
|
|
||||||
assert Phoenix.Flash.get(new_password_conn.assigns.flash, :info) =~
|
|
||||||
"Password updated successfully"
|
|
||||||
|
|
||||||
assert Accounts.get_user_by_email_and_password(user.email, new_password)
|
|
||||||
end
|
|
||||||
|
|
||||||
test "renders errors with invalid data (phx-change)", %{conn: conn} do
|
|
||||||
{:ok, lv, _html} = live(conn, ~p"/users/settings")
|
|
||||||
|
|
||||||
result =
|
|
||||||
lv
|
|
||||||
|> element("#password_form")
|
|
||||||
|> render_change(%{
|
|
||||||
"current_password" => "invalid",
|
|
||||||
"user" => %{
|
|
||||||
"password" => "too short",
|
|
||||||
"password_confirmation" => "does not match"
|
|
||||||
}
|
|
||||||
})
|
|
||||||
|
|
||||||
assert result =~ "Change Password"
|
|
||||||
assert result =~ "should be at least 12 character(s)"
|
|
||||||
assert result =~ "does not match password"
|
|
||||||
end
|
|
||||||
|
|
||||||
test "renders errors with invalid data (phx-submit)", %{conn: conn} do
|
|
||||||
{:ok, lv, _html} = live(conn, ~p"/users/settings")
|
|
||||||
|
|
||||||
result =
|
|
||||||
lv
|
|
||||||
|> form("#password_form", %{
|
|
||||||
"current_password" => "invalid",
|
|
||||||
"user" => %{
|
|
||||||
"password" => "too short",
|
|
||||||
"password_confirmation" => "does not match"
|
|
||||||
}
|
|
||||||
})
|
|
||||||
|> render_submit()
|
|
||||||
|
|
||||||
assert result =~ "Change Password"
|
|
||||||
assert result =~ "should be at least 12 character(s)"
|
|
||||||
assert result =~ "does not match password"
|
|
||||||
assert result =~ "is not valid"
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
describe "confirm email" do
|
|
||||||
setup %{conn: conn} do
|
|
||||||
user = user_fixture()
|
|
||||||
email = unique_user_email()
|
|
||||||
|
|
||||||
token =
|
|
||||||
extract_user_token(fn url ->
|
|
||||||
Accounts.deliver_user_update_email_instructions(%{user | email: email}, user.email, url)
|
|
||||||
end)
|
|
||||||
|
|
||||||
%{conn: log_in_user(conn, user), token: token, email: email, user: user}
|
|
||||||
end
|
|
||||||
|
|
||||||
test "updates the user email once", %{conn: conn, user: user, token: token, email: email} do
|
|
||||||
{:error, redirect} = live(conn, ~p"/users/settings/confirm_email/#{token}")
|
|
||||||
|
|
||||||
assert {:live_redirect, %{to: path, flash: flash}} = redirect
|
|
||||||
assert path == ~p"/users/settings"
|
|
||||||
assert %{"info" => message} = flash
|
|
||||||
assert message == "Email changed successfully."
|
|
||||||
refute Accounts.get_user_by_email(user.email)
|
|
||||||
assert Accounts.get_user_by_email(email)
|
|
||||||
|
|
||||||
# use confirm token again
|
|
||||||
{:error, redirect} = live(conn, ~p"/users/settings/confirm_email/#{token}")
|
|
||||||
assert {:live_redirect, %{to: path, flash: flash}} = redirect
|
|
||||||
assert path == ~p"/users/settings"
|
|
||||||
assert %{"error" => message} = flash
|
|
||||||
assert message == "Email change link is invalid or it has expired."
|
|
||||||
end
|
|
||||||
|
|
||||||
test "does not update email with invalid token", %{conn: conn, user: user} do
|
|
||||||
{:error, redirect} = live(conn, ~p"/users/settings/confirm_email/oops")
|
|
||||||
assert {:live_redirect, %{to: path, flash: flash}} = redirect
|
|
||||||
assert path == ~p"/users/settings"
|
|
||||||
assert %{"error" => message} = flash
|
|
||||||
assert message == "Email change link is invalid or it has expired."
|
|
||||||
assert Accounts.get_user_by_email(user.email)
|
|
||||||
end
|
|
||||||
|
|
||||||
test "redirects if user is not logged in", %{token: token} do
|
|
||||||
conn = build_conn()
|
|
||||||
{:error, redirect} = live(conn, ~p"/users/settings/confirm_email/#{token}")
|
|
||||||
assert {:redirect, %{to: path, flash: flash}} = redirect
|
|
||||||
assert path == ~p"/users/log_in"
|
|
||||||
assert %{"error" => message} = flash
|
|
||||||
assert message == "You must log in to access this page."
|
|
||||||
end
|
|
||||||
end
|
|
||||||
end
|
|
||||||
@@ -1,272 +0,0 @@
|
|||||||
defmodule MalarkeyWeb.UserAuthTest do
|
|
||||||
use MalarkeyWeb.ConnCase, async: true
|
|
||||||
|
|
||||||
alias Phoenix.LiveView
|
|
||||||
alias Malarkey.Accounts
|
|
||||||
alias MalarkeyWeb.UserAuth
|
|
||||||
import Malarkey.AccountsFixtures
|
|
||||||
|
|
||||||
@remember_me_cookie "_malarkey_web_user_remember_me"
|
|
||||||
|
|
||||||
setup %{conn: conn} do
|
|
||||||
conn =
|
|
||||||
conn
|
|
||||||
|> Map.replace!(:secret_key_base, MalarkeyWeb.Endpoint.config(:secret_key_base))
|
|
||||||
|> init_test_session(%{})
|
|
||||||
|
|
||||||
%{user: user_fixture(), conn: conn}
|
|
||||||
end
|
|
||||||
|
|
||||||
describe "log_in_user/3" do
|
|
||||||
test "stores the user token in the session", %{conn: conn, user: user} do
|
|
||||||
conn = UserAuth.log_in_user(conn, user)
|
|
||||||
assert token = get_session(conn, :user_token)
|
|
||||||
assert get_session(conn, :live_socket_id) == "users_sessions:#{Base.url_encode64(token)}"
|
|
||||||
assert redirected_to(conn) == ~p"/"
|
|
||||||
assert Accounts.get_user_by_session_token(token)
|
|
||||||
end
|
|
||||||
|
|
||||||
test "clears everything previously stored in the session", %{conn: conn, user: user} do
|
|
||||||
conn = conn |> put_session(:to_be_removed, "value") |> UserAuth.log_in_user(user)
|
|
||||||
refute get_session(conn, :to_be_removed)
|
|
||||||
end
|
|
||||||
|
|
||||||
test "redirects to the configured path", %{conn: conn, user: user} do
|
|
||||||
conn = conn |> put_session(:user_return_to, "/hello") |> UserAuth.log_in_user(user)
|
|
||||||
assert redirected_to(conn) == "/hello"
|
|
||||||
end
|
|
||||||
|
|
||||||
test "writes a cookie if remember_me is configured", %{conn: conn, user: user} do
|
|
||||||
conn = conn |> fetch_cookies() |> UserAuth.log_in_user(user, %{"remember_me" => "true"})
|
|
||||||
assert get_session(conn, :user_token) == conn.cookies[@remember_me_cookie]
|
|
||||||
|
|
||||||
assert %{value: signed_token, max_age: max_age} = conn.resp_cookies[@remember_me_cookie]
|
|
||||||
assert signed_token != get_session(conn, :user_token)
|
|
||||||
assert max_age == 5_184_000
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
describe "logout_user/1" do
|
|
||||||
test "erases session and cookies", %{conn: conn, user: user} do
|
|
||||||
user_token = Accounts.generate_user_session_token(user)
|
|
||||||
|
|
||||||
conn =
|
|
||||||
conn
|
|
||||||
|> put_session(:user_token, user_token)
|
|
||||||
|> put_req_cookie(@remember_me_cookie, user_token)
|
|
||||||
|> fetch_cookies()
|
|
||||||
|> UserAuth.log_out_user()
|
|
||||||
|
|
||||||
refute get_session(conn, :user_token)
|
|
||||||
refute conn.cookies[@remember_me_cookie]
|
|
||||||
assert %{max_age: 0} = conn.resp_cookies[@remember_me_cookie]
|
|
||||||
assert redirected_to(conn) == ~p"/"
|
|
||||||
refute Accounts.get_user_by_session_token(user_token)
|
|
||||||
end
|
|
||||||
|
|
||||||
test "broadcasts to the given live_socket_id", %{conn: conn} do
|
|
||||||
live_socket_id = "users_sessions:abcdef-token"
|
|
||||||
MalarkeyWeb.Endpoint.subscribe(live_socket_id)
|
|
||||||
|
|
||||||
conn
|
|
||||||
|> put_session(:live_socket_id, live_socket_id)
|
|
||||||
|> UserAuth.log_out_user()
|
|
||||||
|
|
||||||
assert_receive %Phoenix.Socket.Broadcast{event: "disconnect", topic: ^live_socket_id}
|
|
||||||
end
|
|
||||||
|
|
||||||
test "works even if user is already logged out", %{conn: conn} do
|
|
||||||
conn = conn |> fetch_cookies() |> UserAuth.log_out_user()
|
|
||||||
refute get_session(conn, :user_token)
|
|
||||||
assert %{max_age: 0} = conn.resp_cookies[@remember_me_cookie]
|
|
||||||
assert redirected_to(conn) == ~p"/"
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
describe "fetch_current_user/2" do
|
|
||||||
test "authenticates user from session", %{conn: conn, user: user} do
|
|
||||||
user_token = Accounts.generate_user_session_token(user)
|
|
||||||
conn = conn |> put_session(:user_token, user_token) |> UserAuth.fetch_current_user([])
|
|
||||||
assert conn.assigns.current_user.id == user.id
|
|
||||||
end
|
|
||||||
|
|
||||||
test "authenticates user from cookies", %{conn: conn, user: user} do
|
|
||||||
logged_in_conn =
|
|
||||||
conn |> fetch_cookies() |> UserAuth.log_in_user(user, %{"remember_me" => "true"})
|
|
||||||
|
|
||||||
user_token = logged_in_conn.cookies[@remember_me_cookie]
|
|
||||||
%{value: signed_token} = logged_in_conn.resp_cookies[@remember_me_cookie]
|
|
||||||
|
|
||||||
conn =
|
|
||||||
conn
|
|
||||||
|> put_req_cookie(@remember_me_cookie, signed_token)
|
|
||||||
|> UserAuth.fetch_current_user([])
|
|
||||||
|
|
||||||
assert conn.assigns.current_user.id == user.id
|
|
||||||
assert get_session(conn, :user_token) == user_token
|
|
||||||
|
|
||||||
assert get_session(conn, :live_socket_id) ==
|
|
||||||
"users_sessions:#{Base.url_encode64(user_token)}"
|
|
||||||
end
|
|
||||||
|
|
||||||
test "does not authenticate if data is missing", %{conn: conn, user: user} do
|
|
||||||
_ = Accounts.generate_user_session_token(user)
|
|
||||||
conn = UserAuth.fetch_current_user(conn, [])
|
|
||||||
refute get_session(conn, :user_token)
|
|
||||||
refute conn.assigns.current_user
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
describe "on_mount: mount_current_user" do
|
|
||||||
test "assigns current_user based on a valid user_token ", %{conn: conn, user: user} do
|
|
||||||
user_token = Accounts.generate_user_session_token(user)
|
|
||||||
session = conn |> put_session(:user_token, user_token) |> get_session()
|
|
||||||
|
|
||||||
{:cont, updated_socket} =
|
|
||||||
UserAuth.on_mount(:mount_current_user, %{}, session, %LiveView.Socket{})
|
|
||||||
|
|
||||||
assert updated_socket.assigns.current_user.id == user.id
|
|
||||||
end
|
|
||||||
|
|
||||||
test "assigns nil to current_ user assign if there isn't a valid user_token ", %{conn: conn} do
|
|
||||||
user_token = "invalid_token"
|
|
||||||
session = conn |> put_session(:user_token, user_token) |> get_session()
|
|
||||||
|
|
||||||
{:cont, updated_socket} =
|
|
||||||
UserAuth.on_mount(:mount_current_user, %{}, session, %LiveView.Socket{})
|
|
||||||
|
|
||||||
assert updated_socket.assigns.current_user == nil
|
|
||||||
end
|
|
||||||
|
|
||||||
test "assigns nil to current_ user assign if there isn't a user_token", %{conn: conn} do
|
|
||||||
session = conn |> get_session()
|
|
||||||
|
|
||||||
{:cont, updated_socket} =
|
|
||||||
UserAuth.on_mount(:mount_current_user, %{}, session, %LiveView.Socket{})
|
|
||||||
|
|
||||||
assert updated_socket.assigns.current_user == nil
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
describe "on_mount: ensure_authenticated" do
|
|
||||||
test "authenticates current_user based on a valid user_token ", %{conn: conn, user: user} do
|
|
||||||
user_token = Accounts.generate_user_session_token(user)
|
|
||||||
session = conn |> put_session(:user_token, user_token) |> get_session()
|
|
||||||
|
|
||||||
{:cont, updated_socket} =
|
|
||||||
UserAuth.on_mount(:ensure_authenticated, %{}, session, %LiveView.Socket{})
|
|
||||||
|
|
||||||
assert updated_socket.assigns.current_user.id == user.id
|
|
||||||
end
|
|
||||||
|
|
||||||
test "redirects to login page if there isn't a valid user_token ", %{conn: conn} do
|
|
||||||
user_token = "invalid_token"
|
|
||||||
session = conn |> put_session(:user_token, user_token) |> get_session()
|
|
||||||
|
|
||||||
socket = %LiveView.Socket{
|
|
||||||
endpoint: MalarkeyWeb.Endpoint,
|
|
||||||
assigns: %{__changed__: %{}, flash: %{}}
|
|
||||||
}
|
|
||||||
|
|
||||||
{:halt, updated_socket} = UserAuth.on_mount(:ensure_authenticated, %{}, session, socket)
|
|
||||||
assert updated_socket.assigns.current_user == nil
|
|
||||||
end
|
|
||||||
|
|
||||||
test "redirects to login page if there isn't a user_token ", %{conn: conn} do
|
|
||||||
session = conn |> get_session()
|
|
||||||
|
|
||||||
socket = %LiveView.Socket{
|
|
||||||
endpoint: MalarkeyWeb.Endpoint,
|
|
||||||
assigns: %{__changed__: %{}, flash: %{}}
|
|
||||||
}
|
|
||||||
|
|
||||||
{:halt, updated_socket} = UserAuth.on_mount(:ensure_authenticated, %{}, session, socket)
|
|
||||||
assert updated_socket.assigns.current_user == nil
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
describe "on_mount: :redirect_if_user_is_authenticated" do
|
|
||||||
test "redirects if there is an authenticated user ", %{conn: conn, user: user} do
|
|
||||||
user_token = Accounts.generate_user_session_token(user)
|
|
||||||
session = conn |> put_session(:user_token, user_token) |> get_session()
|
|
||||||
|
|
||||||
assert {:halt, _updated_socket} =
|
|
||||||
UserAuth.on_mount(
|
|
||||||
:redirect_if_user_is_authenticated,
|
|
||||||
%{},
|
|
||||||
session,
|
|
||||||
%LiveView.Socket{}
|
|
||||||
)
|
|
||||||
end
|
|
||||||
|
|
||||||
test "Don't redirect is there is no authenticated user", %{conn: conn} do
|
|
||||||
session = conn |> get_session()
|
|
||||||
|
|
||||||
assert {:cont, _updated_socket} =
|
|
||||||
UserAuth.on_mount(
|
|
||||||
:redirect_if_user_is_authenticated,
|
|
||||||
%{},
|
|
||||||
session,
|
|
||||||
%LiveView.Socket{}
|
|
||||||
)
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
describe "redirect_if_user_is_authenticated/2" do
|
|
||||||
test "redirects if user is authenticated", %{conn: conn, user: user} do
|
|
||||||
conn = conn |> assign(:current_user, user) |> UserAuth.redirect_if_user_is_authenticated([])
|
|
||||||
assert conn.halted
|
|
||||||
assert redirected_to(conn) == ~p"/"
|
|
||||||
end
|
|
||||||
|
|
||||||
test "does not redirect if user is not authenticated", %{conn: conn} do
|
|
||||||
conn = UserAuth.redirect_if_user_is_authenticated(conn, [])
|
|
||||||
refute conn.halted
|
|
||||||
refute conn.status
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
describe "require_authenticated_user/2" do
|
|
||||||
test "redirects if user is not authenticated", %{conn: conn} do
|
|
||||||
conn = conn |> fetch_flash() |> UserAuth.require_authenticated_user([])
|
|
||||||
assert conn.halted
|
|
||||||
|
|
||||||
assert redirected_to(conn) == ~p"/users/log_in"
|
|
||||||
|
|
||||||
assert Phoenix.Flash.get(conn.assigns.flash, :error) ==
|
|
||||||
"You must log in to access this page."
|
|
||||||
end
|
|
||||||
|
|
||||||
test "stores the path to redirect to on GET", %{conn: conn} do
|
|
||||||
halted_conn =
|
|
||||||
%{conn | path_info: ["foo"], query_string: ""}
|
|
||||||
|> fetch_flash()
|
|
||||||
|> UserAuth.require_authenticated_user([])
|
|
||||||
|
|
||||||
assert halted_conn.halted
|
|
||||||
assert get_session(halted_conn, :user_return_to) == "/foo"
|
|
||||||
|
|
||||||
halted_conn =
|
|
||||||
%{conn | path_info: ["foo"], query_string: "bar=baz"}
|
|
||||||
|> fetch_flash()
|
|
||||||
|> UserAuth.require_authenticated_user([])
|
|
||||||
|
|
||||||
assert halted_conn.halted
|
|
||||||
assert get_session(halted_conn, :user_return_to) == "/foo?bar=baz"
|
|
||||||
|
|
||||||
halted_conn =
|
|
||||||
%{conn | path_info: ["foo"], query_string: "bar", method: "POST"}
|
|
||||||
|> fetch_flash()
|
|
||||||
|> UserAuth.require_authenticated_user([])
|
|
||||||
|
|
||||||
assert halted_conn.halted
|
|
||||||
refute get_session(halted_conn, :user_return_to)
|
|
||||||
end
|
|
||||||
|
|
||||||
test "does not redirect if user is authenticated", %{conn: conn, user: user} do
|
|
||||||
conn = conn |> assign(:current_user, user) |> UserAuth.require_authenticated_user([])
|
|
||||||
refute conn.halted
|
|
||||||
refute conn.status
|
|
||||||
end
|
|
||||||
end
|
|
||||||
end
|
|
||||||
@@ -1,64 +0,0 @@
|
|||||||
defmodule MalarkeyWeb.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 MalarkeyWeb.ConnCase, async: true`, although
|
|
||||||
this option is not recommended for other databases.
|
|
||||||
"""
|
|
||||||
|
|
||||||
use ExUnit.CaseTemplate
|
|
||||||
|
|
||||||
using do
|
|
||||||
quote do
|
|
||||||
# The default endpoint for testing
|
|
||||||
@endpoint MalarkeyWeb.Endpoint
|
|
||||||
|
|
||||||
use MalarkeyWeb, :verified_routes
|
|
||||||
|
|
||||||
# Import conveniences for testing with connections
|
|
||||||
import Plug.Conn
|
|
||||||
import Phoenix.ConnTest
|
|
||||||
import MalarkeyWeb.ConnCase
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
setup tags do
|
|
||||||
Malarkey.DataCase.setup_sandbox(tags)
|
|
||||||
{:ok, conn: Phoenix.ConnTest.build_conn()}
|
|
||||||
end
|
|
||||||
|
|
||||||
@doc """
|
|
||||||
Setup helper that registers and logs in users.
|
|
||||||
|
|
||||||
setup :register_and_log_in_user
|
|
||||||
|
|
||||||
It stores an updated connection and a registered user in the
|
|
||||||
test context.
|
|
||||||
"""
|
|
||||||
def register_and_log_in_user(%{conn: conn}) do
|
|
||||||
user = Malarkey.AccountsFixtures.user_fixture()
|
|
||||||
%{conn: log_in_user(conn, user), user: user}
|
|
||||||
end
|
|
||||||
|
|
||||||
@doc """
|
|
||||||
Logs the given `user` into the `conn`.
|
|
||||||
|
|
||||||
It returns an updated `conn`.
|
|
||||||
"""
|
|
||||||
def log_in_user(conn, user) do
|
|
||||||
token = Malarkey.Accounts.generate_user_session_token(user)
|
|
||||||
|
|
||||||
conn
|
|
||||||
|> Phoenix.ConnTest.init_test_session(%{})
|
|
||||||
|> Plug.Conn.put_session(:user_token, token)
|
|
||||||
end
|
|
||||||
end
|
|
||||||
@@ -1,58 +0,0 @@
|
|||||||
defmodule Malarkey.DataCase do
|
|
||||||
@moduledoc """
|
|
||||||
This module defines the setup for tests requiring
|
|
||||||
access to the application's data layer.
|
|
||||||
|
|
||||||
You may define functions here to be used as helpers in
|
|
||||||
your tests.
|
|
||||||
|
|
||||||
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 Malarkey.DataCase, async: true`, although
|
|
||||||
this option is not recommended for other databases.
|
|
||||||
"""
|
|
||||||
|
|
||||||
use ExUnit.CaseTemplate
|
|
||||||
|
|
||||||
using do
|
|
||||||
quote do
|
|
||||||
alias Malarkey.Repo
|
|
||||||
|
|
||||||
import Ecto
|
|
||||||
import Ecto.Changeset
|
|
||||||
import Ecto.Query
|
|
||||||
import Malarkey.DataCase
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
setup tags do
|
|
||||||
Malarkey.DataCase.setup_sandbox(tags)
|
|
||||||
:ok
|
|
||||||
end
|
|
||||||
|
|
||||||
@doc """
|
|
||||||
Sets up the sandbox based on the test tags.
|
|
||||||
"""
|
|
||||||
def setup_sandbox(tags) do
|
|
||||||
pid = Ecto.Adapters.SQL.Sandbox.start_owner!(Malarkey.Repo, shared: not tags[:async])
|
|
||||||
on_exit(fn -> Ecto.Adapters.SQL.Sandbox.stop_owner(pid) end)
|
|
||||||
end
|
|
||||||
|
|
||||||
@doc """
|
|
||||||
A helper that transforms changeset errors into a map of messages.
|
|
||||||
|
|
||||||
assert {:error, changeset} = Accounts.create_user(%{password: "short"})
|
|
||||||
assert "password is too short" in errors_on(changeset).password
|
|
||||||
assert %{password: ["password is too short"]} = errors_on(changeset)
|
|
||||||
|
|
||||||
"""
|
|
||||||
def errors_on(changeset) do
|
|
||||||
Ecto.Changeset.traverse_errors(changeset, fn {message, opts} ->
|
|
||||||
Regex.replace(~r"%{(\w+)}", message, fn _, key ->
|
|
||||||
opts |> Keyword.get(String.to_existing_atom(key), key) |> to_string()
|
|
||||||
end)
|
|
||||||
end)
|
|
||||||
end
|
|
||||||
end
|
|
||||||
@@ -1,31 +0,0 @@
|
|||||||
defmodule Malarkey.AccountsFixtures do
|
|
||||||
@moduledoc """
|
|
||||||
This module defines test helpers for creating
|
|
||||||
entities via the `Malarkey.Accounts` context.
|
|
||||||
"""
|
|
||||||
|
|
||||||
def unique_user_email, do: "user#{System.unique_integer()}@example.com"
|
|
||||||
def valid_user_password, do: "hello world!"
|
|
||||||
|
|
||||||
def valid_user_attributes(attrs \\ %{}) do
|
|
||||||
Enum.into(attrs, %{
|
|
||||||
email: unique_user_email(),
|
|
||||||
password: valid_user_password()
|
|
||||||
})
|
|
||||||
end
|
|
||||||
|
|
||||||
def user_fixture(attrs \\ %{}) do
|
|
||||||
{:ok, user} =
|
|
||||||
attrs
|
|
||||||
|> valid_user_attributes()
|
|
||||||
|> Malarkey.Accounts.register_user()
|
|
||||||
|
|
||||||
user
|
|
||||||
end
|
|
||||||
|
|
||||||
def extract_user_token(fun) do
|
|
||||||
{:ok, captured_email} = fun.(&"[TOKEN]#{&1}[TOKEN]")
|
|
||||||
[_, token | _] = String.split(captured_email.text_body, "[TOKEN]")
|
|
||||||
token
|
|
||||||
end
|
|
||||||
end
|
|
||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user