mirror of
https://github.com/maybe-finance/maybe.git
synced 2025-07-19 21:29:38 +02:00
Move merchants to top-level namespace (#895)
This commit is contained in:
parent
2681dd96b1
commit
dc3147c101
29 changed files with 153 additions and 145 deletions
41
app/controllers/merchants_controller.rb
Normal file
41
app/controllers/merchants_controller.rb
Normal file
|
@ -0,0 +1,41 @@
|
||||||
|
class MerchantsController < ApplicationController
|
||||||
|
layout "with_sidebar"
|
||||||
|
|
||||||
|
before_action :set_merchant, only: %i[ edit update destroy ]
|
||||||
|
|
||||||
|
def index
|
||||||
|
@merchants = Current.family.merchants.alphabetically
|
||||||
|
end
|
||||||
|
|
||||||
|
def new
|
||||||
|
@merchant = Merchant.new
|
||||||
|
end
|
||||||
|
|
||||||
|
def create
|
||||||
|
Current.family.merchants.create!(merchant_params)
|
||||||
|
redirect_to merchants_path, notice: t(".success")
|
||||||
|
end
|
||||||
|
|
||||||
|
def edit
|
||||||
|
end
|
||||||
|
|
||||||
|
def update
|
||||||
|
@merchant.update!(merchant_params)
|
||||||
|
redirect_to merchants_path, notice: t(".success")
|
||||||
|
end
|
||||||
|
|
||||||
|
def destroy
|
||||||
|
@merchant.destroy!
|
||||||
|
redirect_to merchants_path, notice: t(".success")
|
||||||
|
end
|
||||||
|
|
||||||
|
private
|
||||||
|
|
||||||
|
def set_merchant
|
||||||
|
@merchant = Current.family.merchants.find(params[:id])
|
||||||
|
end
|
||||||
|
|
||||||
|
def merchant_params
|
||||||
|
params.require(:merchant).permit(:name, :color)
|
||||||
|
end
|
||||||
|
end
|
|
@ -1,41 +0,0 @@
|
||||||
class Transactions::MerchantsController < ApplicationController
|
|
||||||
layout "with_sidebar"
|
|
||||||
|
|
||||||
before_action :set_merchant, only: %i[ edit update destroy ]
|
|
||||||
|
|
||||||
def index
|
|
||||||
@merchants = Current.family.transaction_merchants.alphabetically
|
|
||||||
end
|
|
||||||
|
|
||||||
def new
|
|
||||||
@merchant = Transaction::Merchant.new
|
|
||||||
end
|
|
||||||
|
|
||||||
def create
|
|
||||||
Current.family.transaction_merchants.create!(merchant_params)
|
|
||||||
redirect_to transaction_merchants_path, notice: t(".success")
|
|
||||||
end
|
|
||||||
|
|
||||||
def edit
|
|
||||||
end
|
|
||||||
|
|
||||||
def update
|
|
||||||
@merchant.update!(merchant_params)
|
|
||||||
redirect_to transaction_merchants_path, notice: t(".success")
|
|
||||||
end
|
|
||||||
|
|
||||||
def destroy
|
|
||||||
@merchant.destroy!
|
|
||||||
redirect_to transaction_merchants_path, notice: t(".success")
|
|
||||||
end
|
|
||||||
|
|
||||||
private
|
|
||||||
|
|
||||||
def set_merchant
|
|
||||||
@merchant = Current.family.transaction_merchants.find(params[:id])
|
|
||||||
end
|
|
||||||
|
|
||||||
def merchant_params
|
|
||||||
params.require(:transaction_merchant).permit(:name, :color)
|
|
||||||
end
|
|
||||||
end
|
|
|
@ -6,7 +6,7 @@ class Family < ApplicationRecord
|
||||||
has_many :transactions, through: :accounts
|
has_many :transactions, through: :accounts
|
||||||
has_many :imports, through: :accounts
|
has_many :imports, through: :accounts
|
||||||
has_many :categories, dependent: :destroy
|
has_many :categories, dependent: :destroy
|
||||||
has_many :transaction_merchants, dependent: :destroy, class_name: "Transaction::Merchant"
|
has_many :merchants, dependent: :destroy
|
||||||
|
|
||||||
def snapshot(period = Period.all)
|
def snapshot(period = Period.all)
|
||||||
query = accounts.active.joins(:balances)
|
query = accounts.active.joins(:balances)
|
||||||
|
|
|
@ -1,4 +1,4 @@
|
||||||
class Transaction::Merchant < ApplicationRecord
|
class Merchant < ApplicationRecord
|
||||||
has_many :transactions, dependent: :nullify
|
has_many :transactions, dependent: :nullify
|
||||||
belongs_to :family
|
belongs_to :family
|
||||||
|
|
|
@ -21,7 +21,7 @@ class Transaction < ApplicationRecord
|
||||||
scope :with_categories, ->(categories) { joins(:category).where(categories: { name: categories }) }
|
scope :with_categories, ->(categories) { joins(:category).where(categories: { name: categories }) }
|
||||||
scope :with_accounts, ->(accounts) { joins(:account).where(accounts: { name: accounts }) }
|
scope :with_accounts, ->(accounts) { joins(:account).where(accounts: { name: accounts }) }
|
||||||
scope :with_account_ids, ->(account_ids) { joins(:account).where(accounts: { id: account_ids }) }
|
scope :with_account_ids, ->(account_ids) { joins(:account).where(accounts: { id: account_ids }) }
|
||||||
scope :with_merchants, ->(merchants) { joins(:merchant).where(transaction_merchants: { name: merchants }) }
|
scope :with_merchants, ->(merchants) { joins(:merchant).where(merchants: { name: merchants }) }
|
||||||
scope :on_or_after_date, ->(date) { where("transactions.date >= ?", date) }
|
scope :on_or_after_date, ->(date) { where("transactions.date >= ?", date) }
|
||||||
scope :on_or_before_date, ->(date) { where("transactions.date <= ?", date) }
|
scope :on_or_before_date, ->(date) { where("transactions.date <= ?", date) }
|
||||||
scope :with_converted_amount, ->(currency = Current.family.currency) {
|
scope :with_converted_amount, ->(currency = Current.family.currency) {
|
||||||
|
|
|
@ -23,6 +23,6 @@
|
||||||
|
|
||||||
<footer class="flex justify-between gap-4">
|
<footer class="flex justify-between gap-4">
|
||||||
<%= previous_setting("Tags", tags_path) %>
|
<%= previous_setting("Tags", tags_path) %>
|
||||||
<%= next_setting("Merchants", transaction_merchants_path) %>
|
<%= next_setting("Merchants", merchants_path) %>
|
||||||
</footer>
|
</footer>
|
||||||
</section>
|
</section>
|
||||||
|
|
|
@ -1,14 +1,14 @@
|
||||||
<% is_editing = @merchant.id.present? %>
|
<% is_editing = @merchant.id.present? %>
|
||||||
<div data-controller="merchant-avatar">
|
<div data-controller="merchant-avatar">
|
||||||
<%= form_with model: @merchant, url: is_editing ? transaction_merchant_path(@merchant) : transaction_merchants_path, method: is_editing ? :patch : :post, scope: :transaction_merchant, data: { turbo: false } do |f| %>
|
<%= form_with model: @merchant, url: is_editing ? merchant_path(@merchant) : merchants_path, method: is_editing ? :patch : :post, scope: :merchant, data: { turbo: false } do |f| %>
|
||||||
<section class="space-y-4">
|
<section class="space-y-4">
|
||||||
<div class="w-fit m-auto">
|
<div class="w-fit m-auto">
|
||||||
<%= render partial: "transactions/merchants/avatar", locals: { merchant: } %>
|
<%= render partial: "merchants/avatar", locals: { merchant: } %>
|
||||||
</div>
|
</div>
|
||||||
<div data-controller="select" data-select-active-class="bg-gray-200" data-select-selected-value="<%= @merchant&.color || Transaction::Merchant::COLORS[0] %>">
|
<div data-controller="select" data-select-active-class="bg-gray-200" data-select-selected-value="<%= @merchant&.color || Merchant::COLORS[0] %>">
|
||||||
<%= f.hidden_field :color, data: { select_target: "input", merchant_avatar_target: "color" } %>
|
<%= f.hidden_field :color, data: { select_target: "input", merchant_avatar_target: "color" } %>
|
||||||
<ul data-select-target="list" class="flex gap-2 items-center">
|
<ul data-select-target="list" class="flex gap-2 items-center">
|
||||||
<% Transaction::Merchant::COLORS.each do |color| %>
|
<% Merchant::COLORS.each do |color| %>
|
||||||
<li tabindex="0" data-select-target="option" data-action="click->select#selectOption" data-value="<%= color %>" class="flex shrink-0 justify-center items-center w-6 h-6 cursor-pointer hover:bg-gray-200 rounded-full">
|
<li tabindex="0" data-select-target="option" data-action="click->select#selectOption" data-value="<%= color %>" class="flex shrink-0 justify-center items-center w-6 h-6 cursor-pointer hover:bg-gray-200 rounded-full">
|
||||||
<div style="background-color: <%= color %>" class="shrink-0 w-4 h-4 rounded-full"></div>
|
<div style="background-color: <%= color %>" class="shrink-0 w-4 h-4 rounded-full"></div>
|
||||||
</li>
|
</li>
|
|
@ -2,7 +2,7 @@
|
||||||
<% merchants.each.with_index do |merchant, index| %>
|
<% merchants.each.with_index do |merchant, index| %>
|
||||||
<div class="flex justify-between items-center p-4 bg-white">
|
<div class="flex justify-between items-center p-4 bg-white">
|
||||||
<div class="flex w-full items-center gap-2.5">
|
<div class="flex w-full items-center gap-2.5">
|
||||||
<%= render partial: "transactions/merchants/avatar", locals: { merchant: } %>
|
<%= render partial: "merchants/avatar", locals: { merchant: } %>
|
||||||
<p class="text-gray-900 text-sm truncate">
|
<p class="text-gray-900 text-sm truncate">
|
||||||
<%= merchant.name %>
|
<%= merchant.name %>
|
||||||
</p>
|
</p>
|
||||||
|
@ -13,13 +13,13 @@
|
||||||
</button>
|
</button>
|
||||||
<div data-menu-target="content" class="absolute z-10 top-10 right-0 border border-alpha-black-25 bg-white rounded-lg shadow-xs w-48 hidden">
|
<div data-menu-target="content" class="absolute z-10 top-10 right-0 border border-alpha-black-25 bg-white rounded-lg shadow-xs w-48 hidden">
|
||||||
<div class="border-t border-b border-alpha-black-100 p-1">
|
<div class="border-t border-b border-alpha-black-100 p-1">
|
||||||
<%= button_to edit_transaction_merchant_path(merchant),
|
<%= button_to edit_merchant_path(merchant),
|
||||||
method: :get,
|
method: :get,
|
||||||
class: "flex w-full gap-1 items-center text-sm hover:bg-gray-50 rounded-lg px-3 py-2",
|
class: "flex w-full gap-1 items-center text-sm hover:bg-gray-50 rounded-lg px-3 py-2",
|
||||||
data: { turbo_frame: "modal" } do %>
|
data: { turbo_frame: "modal" } do %>
|
||||||
<%= lucide_icon("pencil-line", class: "w-5 h-5 mr-2") %> <%= t(".edit") %>
|
<%= lucide_icon("pencil-line", class: "w-5 h-5 mr-2") %> <%= t(".edit") %>
|
||||||
<% end %>
|
<% end %>
|
||||||
<%= button_to transaction_merchant_path(merchant),
|
<%= button_to merchant_path(merchant),
|
||||||
method: :delete,
|
method: :delete,
|
||||||
class: "flex w-full gap-1 items-center text-sm text-red-600 hover:text-red-800 hover:bg-gray-50 rounded-lg px-3 py-2",
|
class: "flex w-full gap-1 items-center text-sm text-red-600 hover:text-red-800 hover:bg-gray-50 rounded-lg px-3 py-2",
|
||||||
data: {
|
data: {
|
|
@ -4,7 +4,7 @@
|
||||||
<div class="space-y-4">
|
<div class="space-y-4">
|
||||||
<div class="flex items-center justify-between">
|
<div class="flex items-center justify-between">
|
||||||
<h1 class="text-xl font-medium text-gray-900"><%= t(".title") %></h1>
|
<h1 class="text-xl font-medium text-gray-900"><%= t(".title") %></h1>
|
||||||
<%= link_to new_transaction_merchant_path, class: "flex text-white text-sm font-medium items-center gap-1 bg-gray-900 rounded-lg p-2 pr-3", data: { turbo_frame: "modal" } do %>
|
<%= link_to new_merchant_path, class: "flex text-white text-sm font-medium items-center gap-1 bg-gray-900 rounded-lg p-2 pr-3", data: { turbo_frame: "modal" } do %>
|
||||||
<%= lucide_icon("plus", class: "w-5 h-5") %>
|
<%= lucide_icon("plus", class: "w-5 h-5") %>
|
||||||
<span><%= t(".new_short") %></span>
|
<span><%= t(".new_short") %></span>
|
||||||
<% end %>
|
<% end %>
|
||||||
|
@ -14,7 +14,7 @@
|
||||||
<div class="flex justify-center items-center py-20">
|
<div class="flex justify-center items-center py-20">
|
||||||
<div class="text-center flex flex-col items-center max-w-[300px]">
|
<div class="text-center flex flex-col items-center max-w-[300px]">
|
||||||
<p class="text-gray-900 mb-1 font-medium text-sm"><%= t(".empty") %></p>
|
<p class="text-gray-900 mb-1 font-medium text-sm"><%= t(".empty") %></p>
|
||||||
<%= link_to new_transaction_merchant_path, class: "w-fit flex text-white text-sm font-medium items-center gap-1 bg-gray-900 rounded-lg p-2 pr-3", data: { turbo_frame: "modal" } do %>
|
<%= link_to new_merchant_path, class: "w-fit flex text-white text-sm font-medium items-center gap-1 bg-gray-900 rounded-lg p-2 pr-3", data: { turbo_frame: "modal" } do %>
|
||||||
<%= lucide_icon("plus", class: "w-5 h-5") %>
|
<%= lucide_icon("plus", class: "w-5 h-5") %>
|
||||||
<span><%= t(".new_long") %></span>
|
<span><%= t(".new_long") %></span>
|
||||||
<% end %>
|
<% end %>
|
||||||
|
@ -27,7 +27,7 @@
|
||||||
<span class="text-gray-400 mx-2">·</span>
|
<span class="text-gray-400 mx-2">·</span>
|
||||||
<p><%= @merchants.count %></p>
|
<p><%= @merchants.count %></p>
|
||||||
</div>
|
</div>
|
||||||
<%= render partial: "transactions/merchants/list", locals: { merchants: @merchants } %>
|
<%= render partial: "merchants/list", locals: { merchants: @merchants } %>
|
||||||
</div>
|
</div>
|
||||||
<% end %>
|
<% end %>
|
||||||
</div>
|
</div>
|
|
@ -53,7 +53,7 @@
|
||||||
<%= sidebar_link_to t(".categories_label"), categories_path, icon: "tags" %>
|
<%= sidebar_link_to t(".categories_label"), categories_path, icon: "tags" %>
|
||||||
</li>
|
</li>
|
||||||
<li>
|
<li>
|
||||||
<%= sidebar_link_to t(".merchants_label"), transaction_merchants_path, icon: "store" %>
|
<%= sidebar_link_to t(".merchants_label"), merchants_path, icon: "store" %>
|
||||||
</li>
|
</li>
|
||||||
<li>
|
<li>
|
||||||
<%= sidebar_link_to t(".rules_label"), transaction_rules_path, icon: "list-checks" %>
|
<%= sidebar_link_to t(".rules_label"), transaction_rules_path, icon: "list-checks" %>
|
||||||
|
|
|
@ -29,7 +29,7 @@
|
||||||
<div class="pb-6 space-y-2">
|
<div class="pb-6 space-y-2">
|
||||||
<%= form.date_field :date, label: t(".date_label"), max: Date.current %>
|
<%= form.date_field :date, label: t(".date_label"), max: Date.current %>
|
||||||
<%= form.collection_select :category_id, Current.family.categories.alphabetically, :id, :name, { prompt: t(".category_placeholder"), label: t(".category_label"), class: "text-gray-400" } %>
|
<%= form.collection_select :category_id, Current.family.categories.alphabetically, :id, :name, { prompt: t(".category_placeholder"), label: t(".category_label"), class: "text-gray-400" } %>
|
||||||
<%= form.collection_select :merchant_id, Current.family.transaction_merchants.alphabetically, :id, :name, { prompt: t(".merchant_placeholder"), label: t(".merchant_label"), class: "text-gray-400" } %>
|
<%= form.collection_select :merchant_id, Current.family.merchants.alphabetically, :id, :name, { prompt: t(".merchant_placeholder"), label: t(".merchant_label"), class: "text-gray-400" } %>
|
||||||
</div>
|
</div>
|
||||||
</details>
|
</details>
|
||||||
|
|
||||||
|
|
|
@ -9,7 +9,7 @@
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div class="flex justify-between gap-4">
|
<div class="flex justify-between gap-4">
|
||||||
<%= previous_setting("Merchants", transaction_merchants_path) %>
|
<%= previous_setting("Merchants", merchants_path) %>
|
||||||
<%= next_setting("Imports", imports_path) %>
|
<%= next_setting("Imports", imports_path) %>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
|
@ -5,7 +5,7 @@
|
||||||
<%= lucide_icon("search", class: "w-5 h-5 text-gray-500 absolute inset-y-0 left-2 top-1/2 transform -translate-y-1/2") %>
|
<%= lucide_icon("search", class: "w-5 h-5 text-gray-500 absolute inset-y-0 left-2 top-1/2 transform -translate-y-1/2") %>
|
||||||
</div>
|
</div>
|
||||||
<div class="my-2" id="list" data-list-filter-target="list">
|
<div class="my-2" id="list" data-list-filter-target="list">
|
||||||
<% Current.family.transaction_merchants.alphabetically.each do |merchant| %>
|
<% Current.family.merchants.alphabetically.each do |merchant| %>
|
||||||
<div class="filterable-item flex items-center gap-2 p-2" data-filter-name="<%= merchant.name %>">
|
<div class="filterable-item flex items-center gap-2 p-2" data-filter-name="<%= merchant.name %>">
|
||||||
<%= form.check_box :merchants,
|
<%= form.check_box :merchants,
|
||||||
{
|
{
|
||||||
|
|
|
@ -30,7 +30,7 @@
|
||||||
|
|
||||||
<% unless @transaction.marked_as_transfer %>
|
<% unless @transaction.marked_as_transfer %>
|
||||||
<%= f.collection_select :category_id, Current.family.categories.alphabetically, :id, :name, { prompt: t(".category_placeholder"), label: t(".category_label"), class: "text-gray-400" }, "data-auto-submit-form-target": "auto" %>
|
<%= f.collection_select :category_id, Current.family.categories.alphabetically, :id, :name, { prompt: t(".category_placeholder"), label: t(".category_label"), class: "text-gray-400" }, "data-auto-submit-form-target": "auto" %>
|
||||||
<%= f.collection_select :merchant_id, Current.family.transaction_merchants.alphabetically, :id, :name, { prompt: t(".merchant_placeholder"), label: t(".merchant_label"), class: "text-gray-400" }, "data-auto-submit-form-target": "auto" %>
|
<%= f.collection_select :merchant_id, Current.family.merchants.alphabetically, :id, :name, { prompt: t(".merchant_placeholder"), label: t(".merchant_label"), class: "text-gray-400" }, "data-auto-submit-form-target": "auto" %>
|
||||||
<% end %>
|
<% end %>
|
||||||
|
|
||||||
<%= f.collection_select :account_id, Current.family.accounts.alphabetically, :id, :name, { prompt: t(".account_placeholder"), label: t(".account_label"), class: "text-gray-500" }, { class: "form-field__input cursor-not-allowed text-gray-400", disabled: "disabled" } %>
|
<%= f.collection_select :account_id, Current.family.accounts.alphabetically, :id, :name, { prompt: t(".account_placeholder"), label: t(".account_label"), class: "text-gray-500" }, { class: "form-field__input cursor-not-allowed text-gray-400", disabled: "disabled" } %>
|
||||||
|
|
29
config/locales/views/merchants/en.yml
Normal file
29
config/locales/views/merchants/en.yml
Normal file
|
@ -0,0 +1,29 @@
|
||||||
|
---
|
||||||
|
en:
|
||||||
|
merchants:
|
||||||
|
create:
|
||||||
|
success: New merchant created successfully
|
||||||
|
destroy:
|
||||||
|
success: Merchant deleted successfully
|
||||||
|
edit:
|
||||||
|
title: Edit merchant
|
||||||
|
form:
|
||||||
|
name_placeholder: Merchant name
|
||||||
|
submit_create: Add merchant
|
||||||
|
submit_edit: Update
|
||||||
|
index:
|
||||||
|
empty: No merchants yet
|
||||||
|
new_long: New merchant
|
||||||
|
new_short: New
|
||||||
|
title: Merchants
|
||||||
|
list:
|
||||||
|
confirm_accept: Delete merchant
|
||||||
|
confirm_body: Are you sure you want to delete this merchant? Removing this merchant
|
||||||
|
will unlink all associated transactions and may effect your reporting.
|
||||||
|
confirm_title: Delete merchant?
|
||||||
|
delete: Delete merchant
|
||||||
|
edit: Edit merchant
|
||||||
|
new:
|
||||||
|
title: New merchant
|
||||||
|
update:
|
||||||
|
success: Merchant updated successfully
|
|
@ -50,33 +50,6 @@ en:
|
||||||
transaction: transaction
|
transaction: transaction
|
||||||
mark_transfers:
|
mark_transfers:
|
||||||
success: Marked as transfer
|
success: Marked as transfer
|
||||||
merchants:
|
|
||||||
create:
|
|
||||||
success: New merchant created successfully
|
|
||||||
destroy:
|
|
||||||
success: Merchant deleted successfully
|
|
||||||
edit:
|
|
||||||
title: Edit merchant
|
|
||||||
form:
|
|
||||||
name_placeholder: Merchant name
|
|
||||||
submit_create: Add merchant
|
|
||||||
submit_edit: Update
|
|
||||||
index:
|
|
||||||
empty: No merchants yet
|
|
||||||
new_long: New merchant
|
|
||||||
new_short: New
|
|
||||||
title: Merchants
|
|
||||||
list:
|
|
||||||
confirm_accept: Delete merchant
|
|
||||||
confirm_body: Are you sure you want to delete this merchant? Removing this
|
|
||||||
merchant will unlink all associated transactions and may effect your reporting.
|
|
||||||
confirm_title: Delete merchant?
|
|
||||||
delete: Delete merchant
|
|
||||||
edit: Edit merchant
|
|
||||||
new:
|
|
||||||
title: New merchant
|
|
||||||
update:
|
|
||||||
success: Merchant updated successfully
|
|
||||||
selection_bar:
|
selection_bar:
|
||||||
mark_transfers: Mark as transfers?
|
mark_transfers: Mark as transfers?
|
||||||
mark_transfers_confirm: Mark as transfers
|
mark_transfers_confirm: Mark as transfers
|
||||||
|
|
|
@ -48,6 +48,8 @@ Rails.application.routes.draw do
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
resources :merchants, only: %i[ index new create edit update destroy ]
|
||||||
|
|
||||||
resources :transactions do
|
resources :transactions do
|
||||||
collection do
|
collection do
|
||||||
post "bulk_delete"
|
post "bulk_delete"
|
||||||
|
@ -59,7 +61,6 @@ Rails.application.routes.draw do
|
||||||
scope module: :transactions, as: :transaction do
|
scope module: :transactions, as: :transaction do
|
||||||
resources :rows, only: %i[ show update ]
|
resources :rows, only: %i[ show update ]
|
||||||
resources :rules, only: %i[ index ]
|
resources :rules, only: %i[ index ]
|
||||||
resources :merchants, only: %i[ index new create edit update destroy ]
|
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
5
db/migrate/20240620122201_rename_merchants_table.rb
Normal file
5
db/migrate/20240620122201_rename_merchants_table.rb
Normal file
|
@ -0,0 +1,5 @@
|
||||||
|
class RenameMerchantsTable < ActiveRecord::Migration[7.2]
|
||||||
|
def change
|
||||||
|
rename_table :transaction_merchants, :merchants
|
||||||
|
end
|
||||||
|
end
|
24
db/schema.rb
generated
24
db/schema.rb
generated
|
@ -10,7 +10,7 @@
|
||||||
#
|
#
|
||||||
# It's strongly recommended that you check this file into your version control system.
|
# It's strongly recommended that you check this file into your version control system.
|
||||||
|
|
||||||
ActiveRecord::Schema[7.2].define(version: 2024_06_20_114307) do
|
ActiveRecord::Schema[7.2].define(version: 2024_06_20_122201) do
|
||||||
# These are extensions that must be enabled in order to support this database
|
# These are extensions that must be enabled in order to support this database
|
||||||
enable_extension "pgcrypto"
|
enable_extension "pgcrypto"
|
||||||
enable_extension "plpgsql"
|
enable_extension "plpgsql"
|
||||||
|
@ -242,6 +242,15 @@ ActiveRecord::Schema[7.2].define(version: 2024_06_20_114307) do
|
||||||
t.datetime "updated_at", null: false
|
t.datetime "updated_at", null: false
|
||||||
end
|
end
|
||||||
|
|
||||||
|
create_table "merchants", id: :uuid, default: -> { "gen_random_uuid()" }, force: :cascade do |t|
|
||||||
|
t.string "name", null: false
|
||||||
|
t.string "color", default: "#e99537", null: false
|
||||||
|
t.uuid "family_id", null: false
|
||||||
|
t.datetime "created_at", null: false
|
||||||
|
t.datetime "updated_at", null: false
|
||||||
|
t.index ["family_id"], name: "index_merchants_on_family_id"
|
||||||
|
end
|
||||||
|
|
||||||
create_table "other_assets", id: :uuid, default: -> { "gen_random_uuid()" }, force: :cascade do |t|
|
create_table "other_assets", id: :uuid, default: -> { "gen_random_uuid()" }, force: :cascade do |t|
|
||||||
t.datetime "created_at", null: false
|
t.datetime "created_at", null: false
|
||||||
t.datetime "updated_at", null: false
|
t.datetime "updated_at", null: false
|
||||||
|
@ -284,15 +293,6 @@ ActiveRecord::Schema[7.2].define(version: 2024_06_20_114307) do
|
||||||
t.index ["family_id"], name: "index_tags_on_family_id"
|
t.index ["family_id"], name: "index_tags_on_family_id"
|
||||||
end
|
end
|
||||||
|
|
||||||
create_table "transaction_merchants", id: :uuid, default: -> { "gen_random_uuid()" }, force: :cascade do |t|
|
|
||||||
t.string "name", null: false
|
|
||||||
t.string "color", default: "#e99537", null: false
|
|
||||||
t.uuid "family_id", null: false
|
|
||||||
t.datetime "created_at", null: false
|
|
||||||
t.datetime "updated_at", null: false
|
|
||||||
t.index ["family_id"], name: "index_transaction_merchants_on_family_id"
|
|
||||||
end
|
|
||||||
|
|
||||||
create_table "transactions", id: :uuid, default: -> { "gen_random_uuid()" }, force: :cascade do |t|
|
create_table "transactions", id: :uuid, default: -> { "gen_random_uuid()" }, force: :cascade do |t|
|
||||||
t.string "name"
|
t.string "name"
|
||||||
t.date "date", null: false
|
t.date "date", null: false
|
||||||
|
@ -359,12 +359,12 @@ ActiveRecord::Schema[7.2].define(version: 2024_06_20_114307) do
|
||||||
add_foreign_key "categories", "families"
|
add_foreign_key "categories", "families"
|
||||||
add_foreign_key "imports", "accounts"
|
add_foreign_key "imports", "accounts"
|
||||||
add_foreign_key "institutions", "families"
|
add_foreign_key "institutions", "families"
|
||||||
|
add_foreign_key "merchants", "families"
|
||||||
add_foreign_key "taggings", "tags"
|
add_foreign_key "taggings", "tags"
|
||||||
add_foreign_key "tags", "families"
|
add_foreign_key "tags", "families"
|
||||||
add_foreign_key "transaction_merchants", "families"
|
|
||||||
add_foreign_key "transactions", "accounts", on_delete: :cascade
|
add_foreign_key "transactions", "accounts", on_delete: :cascade
|
||||||
add_foreign_key "transactions", "categories", on_delete: :nullify
|
add_foreign_key "transactions", "categories", on_delete: :nullify
|
||||||
add_foreign_key "transactions", "transaction_merchants", column: "merchant_id"
|
add_foreign_key "transactions", "merchants"
|
||||||
add_foreign_key "transactions", "transfers"
|
add_foreign_key "transactions", "transfers"
|
||||||
add_foreign_key "users", "families"
|
add_foreign_key "users", "families"
|
||||||
add_foreign_key "valuations", "accounts", on_delete: :cascade
|
add_foreign_key "valuations", "accounts", on_delete: :cascade
|
||||||
|
|
39
test/controllers/merchants_controller_test.rb
Normal file
39
test/controllers/merchants_controller_test.rb
Normal file
|
@ -0,0 +1,39 @@
|
||||||
|
require "test_helper"
|
||||||
|
|
||||||
|
class MerchantsControllerTest < ActionDispatch::IntegrationTest
|
||||||
|
setup do
|
||||||
|
sign_in @user = users(:family_admin)
|
||||||
|
@merchant = merchants(:netflix)
|
||||||
|
end
|
||||||
|
|
||||||
|
test "index" do
|
||||||
|
get merchants_path
|
||||||
|
assert_response :success
|
||||||
|
end
|
||||||
|
|
||||||
|
test "new" do
|
||||||
|
get new_merchant_path
|
||||||
|
assert_response :success
|
||||||
|
end
|
||||||
|
|
||||||
|
test "should create merchant" do
|
||||||
|
assert_difference("Merchant.count") do
|
||||||
|
post merchants_url, params: { merchant: { name: "new merchant", color: "#000000" } }
|
||||||
|
end
|
||||||
|
|
||||||
|
assert_redirected_to merchants_path
|
||||||
|
end
|
||||||
|
|
||||||
|
test "should update merchant" do
|
||||||
|
patch merchant_url(@merchant), params: { merchant: { name: "new name", color: "#000000" } }
|
||||||
|
assert_redirected_to merchants_path
|
||||||
|
end
|
||||||
|
|
||||||
|
test "should destroy merchant" do
|
||||||
|
assert_difference("Merchant.count", -1) do
|
||||||
|
delete merchant_url(@merchant)
|
||||||
|
end
|
||||||
|
|
||||||
|
assert_redirected_to merchants_path
|
||||||
|
end
|
||||||
|
end
|
|
@ -1,39 +0,0 @@
|
||||||
require "test_helper"
|
|
||||||
|
|
||||||
class Transactions::MerchantsControllerTest < ActionDispatch::IntegrationTest
|
|
||||||
setup do
|
|
||||||
sign_in @user = users(:family_admin)
|
|
||||||
@merchant = transaction_merchants(:netflix)
|
|
||||||
end
|
|
||||||
|
|
||||||
test "index" do
|
|
||||||
get transaction_merchants_path
|
|
||||||
assert_response :success
|
|
||||||
end
|
|
||||||
|
|
||||||
test "new" do
|
|
||||||
get new_transaction_merchant_path
|
|
||||||
assert_response :success
|
|
||||||
end
|
|
||||||
|
|
||||||
test "should create merchant" do
|
|
||||||
assert_difference("Transaction::Merchant.count") do
|
|
||||||
post transaction_merchants_url, params: { transaction_merchant: { name: "new merchant", color: "#000000" } }
|
|
||||||
end
|
|
||||||
|
|
||||||
assert_redirected_to transaction_merchants_path
|
|
||||||
end
|
|
||||||
|
|
||||||
test "should update merchant" do
|
|
||||||
patch transaction_merchant_url(@merchant), params: { transaction_merchant: { name: "new name", color: "#000000" } }
|
|
||||||
assert_redirected_to transaction_merchants_path
|
|
||||||
end
|
|
||||||
|
|
||||||
test "should destroy merchant" do
|
|
||||||
assert_difference("Transaction::Merchant.count", -1) do
|
|
||||||
delete transaction_merchant_url(@merchant)
|
|
||||||
end
|
|
||||||
|
|
||||||
assert_redirected_to transaction_merchants_path
|
|
||||||
end
|
|
||||||
end
|
|
|
@ -160,7 +160,7 @@ class TransactionsControllerTest < ActionDispatch::IntegrationTest
|
||||||
transaction.update! \
|
transaction.update! \
|
||||||
excluded: false,
|
excluded: false,
|
||||||
category_id: Category.first.id,
|
category_id: Category.first.id,
|
||||||
merchant_id: Transaction::Merchant.first.id,
|
merchant_id: Merchant.first.id,
|
||||||
notes: "Starting note"
|
notes: "Starting note"
|
||||||
end
|
end
|
||||||
|
|
||||||
|
@ -170,7 +170,7 @@ class TransactionsControllerTest < ActionDispatch::IntegrationTest
|
||||||
transaction_ids: transactions.map(&:id),
|
transaction_ids: transactions.map(&:id),
|
||||||
excluded: true,
|
excluded: true,
|
||||||
category_id: Category.second.id,
|
category_id: Category.second.id,
|
||||||
merchant_id: Transaction::Merchant.second.id,
|
merchant_id: Merchant.second.id,
|
||||||
notes: "Updated note"
|
notes: "Updated note"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -182,7 +182,7 @@ class TransactionsControllerTest < ActionDispatch::IntegrationTest
|
||||||
assert_equal Date.current, transaction.date
|
assert_equal Date.current, transaction.date
|
||||||
assert transaction.excluded
|
assert transaction.excluded
|
||||||
assert_equal Category.second, transaction.category
|
assert_equal Category.second, transaction.category
|
||||||
assert_equal Transaction::Merchant.second, transaction.merchant
|
assert_equal Merchant.second, transaction.merchant
|
||||||
assert_equal "Updated note", transaction.notes
|
assert_equal "Updated note", transaction.notes
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
|
@ -39,7 +39,7 @@ class FamilyTest < ActiveSupport::TestCase
|
||||||
end
|
end
|
||||||
|
|
||||||
test "should destroy dependent merchants" do
|
test "should destroy dependent merchants" do
|
||||||
assert_difference("Transaction::Merchant.count", -@family.transaction_merchants.count) do
|
assert_difference("Merchant.count", -@family.merchants.count) do
|
||||||
@family.destroy
|
@family.destroy
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
|
@ -13,7 +13,7 @@ class SettingsTest < ApplicationSystemTestCase
|
||||||
[ "Accounts", "Accounts", accounts_path ],
|
[ "Accounts", "Accounts", accounts_path ],
|
||||||
[ "Tags", "Tags", tags_path ],
|
[ "Tags", "Tags", tags_path ],
|
||||||
[ "Categories", "Categories", categories_path ],
|
[ "Categories", "Categories", categories_path ],
|
||||||
[ "Merchants", "Merchants", transaction_merchants_path ],
|
[ "Merchants", "Merchants", merchants_path ],
|
||||||
[ "Rules", "Rules", transaction_rules_path ],
|
[ "Rules", "Rules", transaction_rules_path ],
|
||||||
[ "Imports", "Imports", imports_path ],
|
[ "Imports", "Imports", imports_path ],
|
||||||
[ "What's New", "What's New", changelog_path ],
|
[ "What's New", "What's New", changelog_path ],
|
||||||
|
|
|
@ -6,7 +6,7 @@ class TransactionsTest < ApplicationSystemTestCase
|
||||||
|
|
||||||
@latest_transactions = @user.family.transactions.ordered.limit(20).to_a
|
@latest_transactions = @user.family.transactions.ordered.limit(20).to_a
|
||||||
@test_category = @user.family.categories.create! name: "System Test Category"
|
@test_category = @user.family.categories.create! name: "System Test Category"
|
||||||
@test_merchant = @user.family.transaction_merchants.create! name: "System Test Merchant"
|
@test_merchant = @user.family.merchants.create! name: "System Test Merchant"
|
||||||
@target_txn = @user.family.accounts.first.transactions.create! \
|
@target_txn = @user.family.accounts.first.transactions.create! \
|
||||||
name: "Oldest transaction",
|
name: "Oldest transaction",
|
||||||
date: 10.years.ago.to_date,
|
date: 10.years.ago.to_date,
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue