berrypod/lib/berrypod_web/controllers/order_lookup_controller.ex
jamey 34822254e3
All checks were successful
deploy / deploy (push) Successful in 1m13s
add persistent email session for order lookup and reviews
Replaces the short-lived (1 hour) session-based order lookup with a
persistent cookie-based email session lasting 30 days. This foundation
enables customers to leave reviews and view orders without re-verifying
their email each time.

- Add EmailSession module for signed cookie management
- Add EmailSession plug to load verified email into session
- Set email session on order lookup verification
- Set email session on checkout completion (via /checkout/complete)
- Update orders and order detail pages to use email session
- Add reviews system plan document

Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
2026-04-01 09:44:53 +01:00

70 lines
1.8 KiB
Elixir

defmodule BerrypodWeb.OrderLookupController do
use BerrypodWeb, :controller
alias Berrypod.EmailSession
alias Berrypod.Orders
alias Berrypod.Orders.OrderNotifier
@salt "order_lookup"
@max_age 3_600
@doc """
Looks up orders by email and sends a verification link (no-JS fallback).
"""
def lookup(conn, %{"email" => email}) when is_binary(email) and email != "" do
orders = Orders.list_orders_by_email(email)
if orders == [] do
conn
|> put_flash(
:error,
"No orders found for that address. Make sure you use the same email you checked out with."
)
|> redirect(to: R.contact())
else
token = generate_token(email)
link = BerrypodWeb.Endpoint.url() <> ~p"/orders/verify/#{token}"
OrderNotifier.deliver_order_lookup(email, link)
conn
|> put_flash(
:info,
"We've sent a link to your email address. It'll expire after an hour."
)
|> redirect(to: R.contact())
end
end
def lookup(conn, _params) do
conn
|> put_flash(:error, "Please enter your email address.")
|> redirect(to: R.contact())
end
def verify(conn, %{"token" => token}) do
case Phoenix.Token.verify(BerrypodWeb.Endpoint, @salt, token, max_age: @max_age) do
{:ok, email} ->
conn
|> EmailSession.put_session(email)
|> redirect(to: R.orders())
{:error, :expired} ->
conn
|> put_flash(:error, "That link has expired. Please request a new one.")
|> redirect(to: R.contact())
{:error, _} ->
conn
|> put_flash(:error, "That link is invalid.")
|> redirect(to: R.contact())
end
end
@doc """
Generates a signed, time-limited token for the given email address.
"""
def generate_token(email) do
Phoenix.Token.sign(BerrypodWeb.Endpoint, @salt, email)
end
end