-- payment_get_open_accounts and the option to get all accounts need to be -- refactored and redesigned. -- CT CREATE OR REPLACE FUNCTION payment_get_open_accounts(in_account_class int) returns SETOF entity AS $$ DECLARE out_entity entity%ROWTYPE; BEGIN FOR out_entity IN SELECT ec.id, e.name, e.entity_class, e.created FROM entity e JOIN entity_credit_account ec ON (ec.entity_id = e.id) WHERE ec.entity_class = in_account_class AND CASE WHEN in_account_class = 1 THEN id IN (SELECT entity_id FROM ap WHERE amount <> paid GROUP BY entity_id) WHEN in_account_class = 2 THEN id IN (SELECT entity_id FROM ar WHERE amount <> paid GROUP BY entity_id) END LOOP RETURN NEXT out_entity; END LOOP; END; $$ LANGUAGE PLPGSQL; COMMENT ON FUNCTION payment_get_open_accounts(int) IS $$ This function takes a single argument (1 for vendor, 2 for customer as always) and returns all entities with open accounts of the appropriate type. $$; CREATE OR REPLACE FUNCTION payment_get_all_accounts(in_account_class int) RETURNS SETOF entity AS $$ DECLARE out_entity entity%ROWTYPE; BEGIN FOR out_entity IN SELECT ec.id, e.name, e.entity_class, e.created FROM entity e JOIN entity_credit_account ec ON (ec.entity_id = e.id) WHERE e.entity_class = in_account_class LOOP RETURN NEXT out_entity; END LOOP; END; $$ LANGUAGE PLPGSQL; COMMENT ON FUNCTION payment_get_open_accounts(int) IS $$ This function takes a single argument (1 for vendor, 2 for customer as always) and returns all entities with accounts of the appropriate type. $$; CREATE TYPE payment_invoice AS ( invoice_id int, invnumber text, invoice_date date, amount numeric, discount numeric, due numeric ); CREATE OR REPLACE FUNCTION payment_get_open_invoices (in_account_class int, in_entity_credit_id int, in_curr char(3)) RETURNS SETOF payment_invoice AS $$ DECLARE payment_inv payment_invoice; BEGIN FOR payment_inv IN SELECT a.id AS invoice_id, a.invnumber, a.transdate AS invoice_date, a.amount, CASE WHEN discount_terms > extract('days' FROM age(a.transdate)) THEN 0 ELSE (a.amount - a.paid) * c.discount / 100 END AS discount, a.amount - a.paid - CASE WHEN discount_terms > extract('days' FROM age(a.transdate)) THEN 0 ELSE (a.amount - a.paid) * c.discount / 100 END AS due FROM (SELECT id, invnumber, transdate, amount, entity_id, 1 as invoice_class, paid, curr FROM ap UNION SELECT id, invnumber, transdate, amount, entity_id, 2 AS invoice_class, paid, curr FROM ar ) a JOIN entity_credit_account c USING (entity_id) WHERE a.invoice_class = in_account_class AND c.entity_class = in_account_class AND a.amount - a.paid <> 0 AND a.curr = in_curr AND a.credit_account = coalesce(in_entity_credit_id, a.credit_account) LOOP RETURN NEXT payment_inv; END LOOP; END; $$ LANGUAGE PLPGSQL; COMMENT ON FUNCTION payment_get_open_invoices(int, int, char(3)) IS $$ This function takes three arguments: Type: 1 for vendor, 2 for customer Entity_id: The entity_id of the customer or vendor Currency: 3 characters for currency ('USD' for example). Returns all open invoices for the entity in question. $$; CREATE TYPE payment_contact_invoice AS ( contact_id int, contact_name text, account_number text, total_due numeric, invoices text[] ); CREATE OR REPLACE FUNCTION payment_get_all_contact_invoices (in_account_class int, in_business_id int, in_currency char(3), in_date_from date, in_date_to date, in_batch_id int, in_ar_ap_accno text) RETURNS SETOF payment_contact_invoice AS $$ DECLARE payment_item payment_contact_invoice; BEGIN FOR payment_item IN SELECT c.id AS contact_id, e.name AS contact_name, c.meta_number AS account_number, sum(a.amount - a.paid) AS total_due, compound_array(ARRAY[[ a.id::text, a.invnumber, a.transdate::text, a.amount::text, a.paid::text, (CASE WHEN c.discount_terms > extract('days' FROM age(a.transdate)) THEN 0 ELSE (a.amount - coalesce(a.paid, 0)) * coalesce(c.discount, 0) / 100 END)::text, (a.amount - coalesce(a.paid, 0) - (CASE WHEN c.discount_terms > extract('days' FROM age(a.transdate)) THEN 0 ELSE (a.amount - coalesce(a.paid, 0)) * coalesce(c.discount, 0) / 100 END))::text]]), bool_and(lock_record(a.id, (select max(session_id) FROM "session" where users_id = ( select id from users WHERE username = SESSION_USER)))) FROM entity e JOIN entity_credit_account c ON (e.id = c.entity_id) JOIN (SELECT id, invnumber, transdate, amount, entity_id, paid, curr, 1 as invoice_class, entity_credit_account FROM ap UNION SELECT id, invnumber, transdate, amount, entity_id, paid, curr, 2 as invoice_class, entity_credit_account FROM ar ORDER BY transdate ) a USING (entity_id) JOIN transactions t ON (a.id = t.id) WHERE a.id IN (select voucher.trans_id FROM voucher WHERE batch_id = in_batch_id) OR (a.invoice_class = in_account_class AND c.business_id = coalesce(in_business_id, c.business_id) AND ((a.transdate >= COALESCE(in_date_from, a.transdate) AND a.transdate <= COALESCE(in_date_to, a.transdate))) AND c.entity_class = in_account_class AND a.curr = in_currency AND a.entity_credit_account = c.id AND a.amount - a.paid <> 0 AND NOT (t.locked_by IS NOT NULL AND t.locked_by IN (select "session_id" FROM "session" WHERE users_id IN (select id from users where username <> SESSION_USER))) AND EXISTS (select trans_id FROM acc_trans WHERE trans_id = a.id AND chart_id = (SELECT id frOM chart WHERE accno = in_ar_ap_accno) )) GROUP BY c.id, e.name, c.meta_number, c.threshold HAVING sum(a.amount - a.paid) > c.threshold LOOP RETURN NEXT payment_item; END LOOP; END; $$ LANGUAGE plpgsql; COMMENT ON FUNCTION payment_get_all_contact_invoices (in_account_class int, in_business_type int, in_currency char(3), in_date_from date, in_date_to date, in_batch_id int, in_ar_ap_accno text) IS $$ This function takes the following arguments (all prefaced with in_ in the db): account_class: 1 for vendor, 2 for customer business_type: integer of business.id. currency: char(3) of currency (for example 'USD') date_from, date_to: These dates are inclusive. 1;3B batch_id: For payment batches, where fees are concerned. ar_ap_accno: The AR/AP account number. This then returns a set of contact information with a 2 dimensional array cnsisting of outstanding invoices. $$; CREATE OR REPLACE FUNCTION payment_create_queue_entry() RETURNS int AS $$ $$ LANGUAGE PLPGSQL; CREATE OR REPLACE FUNCTION payment_bulk_queue (in_transactions numeric[], in_batch_id int, in_source text, in_total numeric, in_ar_ap_accno text, in_cash_accno text, in_payment_date date, in_account_class int) returns int as $$ BEGIN INSERT INTO payments_queue (transactions, batch_id, source, total, ar_ap_accno, cash_accno, payment_date, account_class) VALUES (in_transactions, in_batch_id, in_source, in_total, in_ar_ap_accno, in_cash_accno, in_payment_date, in_account_class); RETURN array_upper(in_transactions, 1) - array_lower(in_transactions, 1); END; $$ LANGUAGE PLPGSQL; CREATE OR REPLACE FUNCTION job__process_payments(in_job_id int) RETURNS bool AS $$ DECLARE queue_record RECORD t_auth_name text, BEGIN -- TODO: Move the set session authorization into a utility function SELECT created_by INTO t_auth_name FROM pending_jobs WHERE id = in_job_id; EXECUTE 'SET SESSION AUTHORIZATION ' quote_ident(t_auth_name); FOR queue_record IN SELECT * from payments_queue WHERE job_id = in_job_id LOOP PERFORM payment_bulk_post (transactions, batch_id, source, total, ar_ap_accno, cash_accno, payment_date, account_class); END LOOP; UPDATE pending_job SET completed_at = timeofday()::timestamp, success = true WHERE id = in_job_id; RETURN TRUE; END; $$ language plpgsql; CREATE OR REPLACE FUNCTION job__create(in_batch_class int, in_batch_id int) RETURNS int AS $$ BEGIN INSERT INTO pending_job (batch_class, batch_id) VALUES (in_batch_class, in_batch_id); RETURN currval('pending_job_id_seq'); END; $$ LANGUAGE PLPGSQL; CREATE TYPE job__status AS ( completed int, -- 1 for completed, 0 for no success int, -- 1 for success, 0 for no completed_at timestamp, error_condition text -- error if not successful ); CREATE OR REPLACE FUNCTION job__status(in_job_id int) RETURNS job__status AS $$ DECLARE out_row job__status; BEGIN SELECT (completed_at IS NULL)::INT, success::int, completed_at, error_condition INTO out_row FROM pending_job WHERE id = in_job_id; RETURN out_row; END; $$ language plpgsql; CREATE OR REPLACE FUNCTION payment_bulk_post (in_transactions numeric[], in_batch_id int, in_source text, in_total numeric, in_ar_ap_accno text, in_cash_accno text, in_payment_date date, in_account_class int) RETURNS int AS $$ DECLARE payment_trans numeric[]; out_count int; t_voucher_id int; t_trans_id int; t_amount numeric; BEGIN IF in_batch_id IS NULL THEN t_voucher_id := NULL; ELSE INSERT INTO voucher (batch_id, batch_class, trans_id) values (in_batch_id, 3, in_transactions[1][1]); t_voucher_id := currval('voucher_id_seq'); END IF; FOR out_count IN array_lower(in_transactions, 1) .. array_upper(in_transactions, 1) LOOP INSERT INTO acc_trans (trans_id, chart_id, amount, approved, voucher_id, transdate) VALUES (in_transactions[out_count][1], case when in_account_class = 1 THEN (SELECT id FROM chart WHERE accno = in_cash_accno) WHEN in_account_class = 2 THEN (SELECT id FROM chart WHERE accno = in_ar_ap_accno) ELSE -1 END, in_transactions[out_count][2], CASE WHEN t_voucher_id IS NULL THEN true ELSE false END, t_voucher_id, in_payment_date), (in_transactions[out_count][1], case when in_account_class = 1 THEN (SELECT id FROM chart WHERE accno = in_ar_ap_accno) WHEN in_account_class = 2 THEN (SELECT id FROM chart WHERE accno = in_cash_accno) ELSE -1 END, in_transactions[out_count][2]* -1, CASE WHEN t_voucher_id IS NULL THEN true ELSE false END, t_voucher_id, in_payment_date); UPDATE ap set paid = paid +in_transactions[out_count][2] where id =in_transactions[out_count][1]; END LOOP; return out_count; END; $$ language plpgsql; COMMENT ON FUNCTION payment_bulk_post (in_transactions numeric[], in_batch_id int, in_source text, in_total numeric, in_ar_ap_accno text, in_cash_accno text, in_payment_date date, in_account_class int) IS $$ Note that in_transactions is a two-dimensional numeric array. Of each sub-array, the first element is the (integer) transaction id, and the second is the amount for that transaction. If the total of the amounts do not add up to in_total, then an error is generated. $$; CREATE OR REPLACE FUNCTION payment_post (in_trans_id int, in_batch_id int, in_source text, in_amount numeric, in_ar_ap_accno text, in_cash_accno text, in_approved bool, in_payment_date date, in_account_class int) RETURNS INT AS $$ DECLARE out_entry_id int; BEGIN INSERT INTO acc_trans (chart_id, amount, trans_id, transdate, approved, source) VALUES ((SELECT id FROM chart WHERE accno = in_ar_ap_accno), CASE WHEN in_account_class = 1 THEN in_amount * -1 ELSE amount END, in_trans_id, in_payment_date, in_approved, in_source); INSERT INTO acc_trans (chart_id, amount, trans_id, transdate, approved, source) VALUES ((SELECT id FROM chart WHERE accno = in_cash_accno), CASE WHEN in_account_class = 2 THEN in_amount * -1 ELSE amount END, in_trans_id, in_payment_date, coalesce(in_approved, true), in_source); SELECT currval('acc_trans_entry_id_seq') INTO out_entry_id; RETURN out_entry_id; END; $$ LANGUAGE PLPGSQL; COMMENT ON FUNCTION payment_post (in_trans_id int, in_source text, in_amount numeric, in_ar_ap_accno text, in_cash_accno text, in_approved bool, in_payment_date date, in_account_class int) IS $$ This function takes the following arguments (prefaced with in_ in the db): trans_id: Id for ar/ap transaction. source: text for source documnet identifier (for example, check number) amount: numeric for the amount of the transaction ar_ap_accno: AR/AP account number cash_accno: Cash Account number, i.e. the account where the payment will be held approved: False, for a voucher. This function posts the payment or saves the payment voucher. $$; -- Move this to the projects module when we start on that. CT CREATE OR REPLACE FUNCTION project_list_open(in_date date) RETURNS SETOF project AS $$ DECLARE out_project project%ROWTYPE; BEGIN FOR out_project IN SELECT * from project WHERE startdate <= in_date AND enddate >= in_date AND completed = 0 LOOP return next out_project; END LOOP; END; $$ language plpgsql; comment on function project_list_open(in_date date) is $$ This function returns all projects that were open as on the date provided as the argument.$$; -- Move this to the projects module when we start on that. CT CREATE OR REPLACE FUNCTION department_list(in_role char) RETURNS SETOF department AS $$ DECLARE out_department department%ROWTYPE; BEGIN FOR out_department IN SELECT * from department WHERE role = coalesce(in_role, role) LOOP return next out_department; END LOOP; END; $$ language plpgsql; -- Move this into another module. comment on function department_list(in_role char) is $$ This function returns all department that match the role provided as the argument.$$; CREATE OR REPLACE FUNCTION payments_get_open_currencies(in_account_class int) RETURNS SETOF char(3) AS $$ DECLARE resultrow record; BEGIN FOR resultrow IN SELECT curr AS curr FROM ar WHERE amount <> paid OR paid IS NULL AND in_account_class=2 UNION SELECT curr FROM ap WHERE amount <> paid OR paid IS NULL AND in_account_class=1 ORDER BY curr LOOP return next resultrow.curr; END LOOP; END; $$ language plpgsql; CREATE OR REPLACE FUNCTION currency_get_exchangerate(in_currency char(3), in_date date, in_account_class int) RETURNS NUMERIC AS $$ DECLARE out_exrate exchangerate.buy%TYPE; BEGIN IF in_account_class = 1 THEN SELECT INTO out_exrate buy FROM exchangerate WHERE transdate = in_date AND curr = in_currency; ELSE SELECT INTO out_exrate sell FROM exchangerate WHERE transdate = in_date AND curr = in_currency; END IF; RETURN out_exrate; END; $$ language plpgsql; COMMENT ON FUNCTION currency_get_exchangerate(in_currency char(3), in_date date, in_account_class int) IS $$ This function return the exchange rate of a given currency, date and exchange rate class (buy or sell). $$; CREATE OR REPLACE FUNCTION payment_get_vc_info(in_entity_id int) RETURNS SETOF entity AS $$ DECLARE out_info entity%ROWTYPE; BEGIN FOR out_info IN SELECT e.id, e.name FROM entity e JOIN company c ON (e.id = c.entity_id) WHERE e.id = in_entity_id --SELECT e.id, c.legal_name, l.line_one, l.city_province, cy.name FROM entity e --JOIN company c ON (e.id = c.entity_id) --JOIN company_to_location cl ON (c.id = cl.company_id) --JOIN location l ON (l.id = cl.location_id) --JOIN country cy ON (cy.id = l.country_id) LOOP return next out_info; END LOOP; IF NOT FOUND THEN RAISE EXCEPTION 'ID % not found', in_entity_id; END IF; END; $$ language plpgsql; COMMENT ON FUNCTION payment_get_vc_info(in_entity_id int) IS $$ This function return vendor or customer info, its under construction $$; CREATE TYPE payment_record AS ( amount numeric, meta_number text, company_paid text, cash_account_id int, cash_accno text, cash_account_description text, ar_ap_account_id int, ar_ap_accno text, ar_ap_description text ); CREATE OR REPLACE FUNCTION payment__retrieve (in_source text, in_meta_number text, in_account_class int, in_cash_accno text) RETURNS SETOF payment_record AS $$ DECLARE out_row payment_record; BEGIN FOR out_row IN SELECT sum(case when at.amount > 0 then at.amount else 0 end) AS amount, ec.meta_number, c.legal_name, max(cc.id), max(cc.accno), max(cc.description), max(ac.id), max(ac.accno), max(ac.description) FROM acc_trans at JOIN entity_credit_account ec ON (at.trans_id IN (select id FROM ar WHERE in_account_class = 2 AND entity_credit_account = (SELECT id FROM entity_credit_account WHERE meta_number = in_meta_number AND entity_class = in_account_class) UNION SELECT id FROM ap WHERE in_account_class = 1 AND entity_credit_account = (select id FROM entity_credit_account WHERE meta_number = in_meta_number AND entity_class = in_account_class))) JOIN company c ON (ec.entity_id = c.entity_id) LEFT JOIN chart cc ON (at.chart_id = cc.id AND cc.link LIKE '%paid%') JOIN chart ac ON (at.chart_id = ac.id AND ((in_account_class = 1 AND ac.link = 'AP') OR (in_account_class = 2 AND ac.link = 'AR'))) WHERE source = in_source GROUP BY ec.meta_number, c.legal_name HAVING max(cc.accno) = in_cash_accno LOOP return next out_row; END LOOP; END; $$ LANGUAGE plpgsql; CREATE OR REPLACE FUNCTION payment__reverse (in_source text, in_date_paid date, in_credit_id int, in_cash_accno text) RETURNS INT AS $$ DECLARE count int; BEGIN count := 0; FOR END; $$ LANGUAGE PLPGSQL;