mirror of
https://github.com/we-promise/sure.git
synced 2026-04-07 14:31:25 +00:00
* feat: add protection indicator to entries and unlock functionality - Introduced protection indicator component rendering on hover and in detail views. - Added support to unlock entries, clearing protection flags (`user_modified`, `import_locked`, and locked attributes). - Updated routes, controllers, and models to enable unlock functionality for trades and transactions. - Refactored views and localized content to support the new feature. - Added relevant tests for unlocking functionality and attribute handling. * feat: improve sync protection and turbo stream updates for entries - Added tests for turbo stream updates reflecting protection indicators. - Ensured user-modified entries lock specific attributes to prevent overwrites. - Updated controllers to mark entries as user-modified and reload for accurate rendering. - Enhanced protection indicator rendering using turbo frames. - Applied consistent lock state handling across trades and transactions. * Address PR review comments for protection indicator --------- Co-authored-by: luckyPipewrench <luckypipewrench@proton.me>
267 lines
7.6 KiB
Ruby
267 lines
7.6 KiB
Ruby
require "test_helper"
|
|
|
|
class TradesControllerTest < ActionDispatch::IntegrationTest
|
|
include EntryableResourceInterfaceTest
|
|
|
|
setup do
|
|
sign_in @user = users(:family_admin)
|
|
@entry = entries(:trade)
|
|
end
|
|
|
|
test "updates trade entry" do
|
|
assert_no_difference [ "Entry.count", "Trade.count" ] do
|
|
patch trade_url(@entry), params: {
|
|
entry: {
|
|
currency: "USD",
|
|
entryable_attributes: {
|
|
id: @entry.entryable_id,
|
|
qty: 20,
|
|
price: 20
|
|
}
|
|
}
|
|
}
|
|
end
|
|
|
|
@entry.reload
|
|
|
|
assert_enqueued_with job: SyncJob
|
|
|
|
assert_equal 20, @entry.trade.qty
|
|
assert_equal 20, @entry.trade.price
|
|
assert_equal "USD", @entry.currency
|
|
|
|
assert_redirected_to account_url(@entry.account)
|
|
end
|
|
|
|
test "creates deposit entry" do
|
|
from_account = accounts(:depository) # Account the deposit is coming from
|
|
|
|
assert_difference -> { Entry.count } => 2,
|
|
-> { Transaction.count } => 2,
|
|
-> { Transfer.count } => 1 do
|
|
post trades_url(account_id: @entry.account_id), params: {
|
|
model: {
|
|
type: "deposit",
|
|
date: Date.current,
|
|
amount: 10,
|
|
currency: "USD",
|
|
transfer_account_id: from_account.id
|
|
}
|
|
}
|
|
end
|
|
|
|
assert_redirected_to @entry.account
|
|
end
|
|
|
|
test "creates withdrawal entry" do
|
|
to_account = accounts(:depository) # Account the withdrawal is going to
|
|
|
|
assert_difference -> { Entry.count } => 2,
|
|
-> { Transaction.count } => 2,
|
|
-> { Transfer.count } => 1 do
|
|
post trades_url(account_id: @entry.account_id), params: {
|
|
model: {
|
|
type: "withdrawal",
|
|
date: Date.current,
|
|
amount: 10,
|
|
currency: "USD",
|
|
transfer_account_id: to_account.id
|
|
}
|
|
}
|
|
end
|
|
|
|
assert_redirected_to @entry.account
|
|
end
|
|
|
|
test "deposit and withdrawal has optional transfer account" do
|
|
assert_difference -> { Entry.count } => 1,
|
|
-> { Transaction.count } => 1,
|
|
-> { Transfer.count } => 0 do
|
|
post trades_url(account_id: @entry.account_id), params: {
|
|
model: {
|
|
type: "withdrawal",
|
|
date: Date.current,
|
|
amount: 10,
|
|
currency: "USD"
|
|
}
|
|
}
|
|
end
|
|
|
|
created_entry = Entry.order(created_at: :desc).first
|
|
|
|
assert created_entry.amount.positive?
|
|
assert_redirected_to @entry.account
|
|
end
|
|
|
|
test "creates interest entry" do
|
|
assert_difference [ "Entry.count", "Transaction.count" ], 1 do
|
|
post trades_url(account_id: @entry.account_id), params: {
|
|
model: {
|
|
type: "interest",
|
|
date: Date.current,
|
|
amount: 10,
|
|
currency: "USD"
|
|
}
|
|
}
|
|
end
|
|
|
|
created_entry = Entry.order(created_at: :desc).first
|
|
|
|
assert created_entry.amount.negative?
|
|
assert_redirected_to @entry.account
|
|
end
|
|
|
|
test "creates trade buy entry" do
|
|
assert_difference [ "Entry.count", "Trade.count", "Security.count" ], 1 do
|
|
post trades_url(account_id: @entry.account_id), params: {
|
|
model: {
|
|
type: "buy",
|
|
date: Date.current,
|
|
ticker: "NVDA (NASDAQ)",
|
|
qty: 10,
|
|
price: 10,
|
|
currency: "USD"
|
|
}
|
|
}
|
|
end
|
|
|
|
created_entry = Entry.order(created_at: :desc).first
|
|
|
|
assert created_entry.amount.positive?
|
|
assert created_entry.trade.qty.positive?
|
|
assert_equal "Entry created", flash[:notice]
|
|
assert_enqueued_with job: SyncJob
|
|
assert_redirected_to account_url(created_entry.account)
|
|
end
|
|
|
|
test "creates trade sell entry" do
|
|
assert_difference [ "Entry.count", "Trade.count" ], 1 do
|
|
post trades_url(account_id: @entry.account_id), params: {
|
|
model: {
|
|
type: "sell",
|
|
ticker: "AAPL (NYSE)",
|
|
date: Date.current,
|
|
currency: "USD",
|
|
qty: 10,
|
|
price: 10
|
|
}
|
|
}
|
|
end
|
|
|
|
created_entry = Entry.order(created_at: :desc).first
|
|
|
|
assert created_entry.amount.negative?
|
|
assert created_entry.trade.qty.negative?
|
|
assert_equal "Entry created", flash[:notice]
|
|
assert_enqueued_with job: SyncJob
|
|
assert_redirected_to account_url(created_entry.account)
|
|
end
|
|
|
|
test "unlock clears protection flags on user-modified entry" do
|
|
# Mark as protected with locked_attributes on both entry and entryable
|
|
@entry.update!(user_modified: true, locked_attributes: { "name" => Time.current.iso8601 })
|
|
@entry.trade.update!(locked_attributes: { "qty" => Time.current.iso8601 })
|
|
|
|
assert @entry.reload.protected_from_sync?
|
|
|
|
post unlock_trade_path(@entry.trade)
|
|
|
|
assert_redirected_to account_path(@entry.account)
|
|
assert_equal "Entry unlocked. It may be updated on next sync.", flash[:notice]
|
|
|
|
@entry.reload
|
|
assert_not @entry.user_modified?
|
|
assert_empty @entry.locked_attributes, "Entry locked_attributes should be cleared"
|
|
assert_empty @entry.trade.locked_attributes, "Trade locked_attributes should be cleared"
|
|
assert_not @entry.protected_from_sync?
|
|
end
|
|
|
|
test "unlock clears import_locked flag" do
|
|
@entry.update!(import_locked: true)
|
|
|
|
assert @entry.reload.protected_from_sync?
|
|
|
|
post unlock_trade_path(@entry.trade)
|
|
|
|
assert_redirected_to account_path(@entry.account)
|
|
@entry.reload
|
|
assert_not @entry.import_locked?
|
|
assert_not @entry.protected_from_sync?
|
|
end
|
|
|
|
test "update locks saved attributes" do
|
|
assert_not @entry.user_modified?
|
|
assert_empty @entry.trade.locked_attributes
|
|
|
|
patch trade_url(@entry), params: {
|
|
entry: {
|
|
currency: "USD",
|
|
entryable_attributes: {
|
|
id: @entry.entryable_id,
|
|
qty: 50,
|
|
price: 25
|
|
}
|
|
}
|
|
}
|
|
|
|
@entry.reload
|
|
assert @entry.user_modified?
|
|
assert @entry.trade.locked_attributes.key?("qty")
|
|
assert @entry.trade.locked_attributes.key?("price")
|
|
end
|
|
|
|
test "turbo stream update includes lock icon for protected entry" do
|
|
assert_not @entry.user_modified?
|
|
|
|
patch trade_url(@entry), params: {
|
|
entry: {
|
|
currency: "USD",
|
|
nature: "outflow",
|
|
entryable_attributes: {
|
|
id: @entry.entryable_id,
|
|
qty: 50,
|
|
price: 25
|
|
}
|
|
}
|
|
}, as: :turbo_stream
|
|
|
|
assert_response :success
|
|
assert_match(/turbo-stream/, response.content_type)
|
|
# The turbo stream should contain the lock icon link with protection tooltip
|
|
assert_match(/title="Protected from sync"/, response.body)
|
|
# And should contain the lock SVG (the path for lock icon)
|
|
assert_match(/M7 11V7a5 5 0 0 1 10 0v4/, response.body)
|
|
end
|
|
|
|
test "quick edit badge update locks activity label" do
|
|
assert_not @entry.user_modified?
|
|
assert_empty @entry.trade.locked_attributes
|
|
original_label = @entry.trade.investment_activity_label
|
|
|
|
# Mimic the quick edit badge JSON request
|
|
patch trade_url(@entry),
|
|
params: {
|
|
entry: {
|
|
entryable_attributes: {
|
|
id: @entry.entryable_id,
|
|
investment_activity_label: original_label == "Buy" ? "Sell" : "Buy"
|
|
}
|
|
}
|
|
}.to_json,
|
|
headers: {
|
|
"Content-Type" => "application/json",
|
|
"Accept" => "text/vnd.turbo-stream.html"
|
|
}
|
|
|
|
assert_response :success
|
|
assert_match(/turbo-stream/, response.content_type)
|
|
# The turbo stream should contain the lock icon
|
|
assert_match(/title="Protected from sync"/, response.body)
|
|
|
|
@entry.reload
|
|
assert @entry.user_modified?, "Entry should be marked as user_modified"
|
|
assert @entry.trade.locked_attributes.key?("investment_activity_label"), "investment_activity_label should be locked"
|
|
assert @entry.protected_from_sync?, "Entry should be protected from sync"
|
|
end
|
|
end
|