summaryrefslogtreecommitdiff
path: root/LedgerSMB/DBObject/Payment.pm
blob: b35bb2d8a18c33aeee99b443803d14ffe1786f27 (plain)
  1. =head1 NAME
  2. LedgerSMB::Payment: Payment Handling Back-end Routines for LedgerSMB
  3. =head1 SYNOPSIS
  4. Provides the functions for generating the data structures payments made in
  5. LedgerSMB. This module currently handles only basic payment logic, and does
  6. handle overpayment logic, though these features will be moved into this module
  7. in the near future.
  8. =head1 COPYRIGHT
  9. Copyright (c) 2007 The LedgerSMB Core Team. Licensed under the GNU General
  10. Public License version 2 or at your option any later version. Please see the
  11. included COPYRIGHT and LICENSE files for more information.
  12. =cut
  13. package LedgerSMB::DBObject::Payment;
  14. use base qw(LedgerSMB::DBObject);
  15. use strict;
  16. use Math::BigFloat lib => 'GMP';
  17. our $VERSION = '0.1.0';
  18. =head1 METHODS
  19. =over
  20. =item LedgerSMB::DBObject::Payment->new()
  21. Inherited from LedgerSMB::DBObject. Please see that documnetation for details.
  22. =item $payment->get_open_accounts()
  23. This function returns a list of open accounts depending on the
  24. $payment->{account_class} property. If this property is 1, it returns a list
  25. of vendor accounts, for 2, a list of customer accounts are returned.
  26. The returned list of hashrefs is stored in the $payment->{accounts} property.
  27. Each hashref has the following keys: id (entity id), name, and entity_class.
  28. An account is considered open if there are outstanding, unpaid invoices
  29. attached to it. Customer/vendor payment threshold is not considered for this
  30. calculation.
  31. =back
  32. =cut
  33. sub __validate__ {
  34. my ($self) = shift @_;
  35. # If the account class is not set, we don't know if it is a payment or a
  36. # receipt. --CT
  37. if (!$self->{account_class}) {
  38. $self->error("account_class must be set")
  39. };
  40. # We should try to re-engineer this so that we don't have to include SQL in
  41. # this file. --CT
  42. ($self->{current_date}) = $self->{dbh}->selectrow_array('select current_date');
  43. }
  44. sub get_metadata {
  45. my ($self) = @_;
  46. $self->list_open_projects();
  47. @{$self->{departments}} = $self->exec_method(funcname => 'department_list');
  48. $self->get_open_currencies();
  49. $self->{currencies} = [];
  50. for my $c (@{$self->{openCurrencies}}){
  51. push @{$self->{currencies}}, $c->{payments_get_open_currencies};
  52. }
  53. @{$self->{businesses}} = $self->exec_method(
  54. funcname => 'business_type__list'
  55. );
  56. @{$self->{debt_accounts}} = $self->exec_method(
  57. funcname => 'chart_get_ar_ap');
  58. @{$self->{cash_accounts}} = $self->exec_method(
  59. funcname => 'chart_list_cash');
  60. for my $ref(@{$self->{cash_accounts}}){
  61. $ref->{text} = "$ref->{accno}--$ref->{description}";
  62. }
  63. }
  64. sub get_open_accounts {
  65. my ($self) = @_;
  66. @{$self->{accounts}} =
  67. $self->exec_method(funcname => 'payment_get_open_accounts');
  68. return @{$self->{accounts}};
  69. }
  70. =over
  71. =item $payment->get_all_accounts()
  72. This function returns a list of open or closed accounts depending on the
  73. $payment->{account_class} property. If this property is 1, it returns a list
  74. of vendor accounts, for 2, a list of customer accounts are returned.
  75. The returned list of hashrefs is stored in the $payment->{accounts} property.
  76. Each hashref has the following keys: id (entity id), name, and entity_class.
  77. =back
  78. =cut
  79. sub get_all_accounts {
  80. my ($self) = @_;
  81. @{$self->{accounts}} =
  82. $self->exec_method(funcname => 'payment_get_all_accounts');
  83. return @{$self->{accounts}};
  84. }
  85. =over
  86. =item $payment->get_open_invoices()
  87. This function returns a list of open invoices depending on the
  88. $payment->{account_class}, $payment->{entity_id}, and $payment->{curr}
  89. properties. Account classes follow the conventions above. This list is hence
  90. specific to a customer or vendor and currency as well.
  91. The returned list of hashrefs is stored in the $payment->{open_invoices}
  92. property. Each hashref has the following keys: id (entity id), name, and
  93. entity_class.
  94. =back
  95. =cut
  96. sub get_open_invoices {
  97. my ($self) = @_;
  98. @{$self->{open_invoices}} =
  99. $self->exec_method(funcname => 'payment_get_open_invoices');
  100. return @{$self->{open_invoices}};
  101. }
  102. =over
  103. =item $oayment->get_all_contact_invoices()
  104. This function returns a list of open accounts depending on the
  105. $payment->{account_class} property. If this property is 1, it returns a list
  106. of vendor accounts, for 2, a list of customer accounts are returned. Attached
  107. to each account is a list of open invoices. The data structure is somewhat
  108. complex.
  109. Each item in the list has the following keys: contact_id, contact_name, \
  110. account_number, total_due, and invoices.
  111. The invoices entry is a reference to an array of hashrefs. Each of these
  112. hashrefs has the following keys: invoice_id, invnumber, invoice_date, amount,
  113. discount, and due.
  114. These are filtered based on the (required) properties:
  115. $payment->{account_class}, $payment->{business_type}, $payment->{date_from},
  116. $payment->{date_to}, and $payment->{ar_ap_accno}.
  117. The $payment->{ar_ap_accno} property is used to filter out by AR or AP account.
  118. The following can also be optionally passed: $payment->{batch_id}. If this is
  119. patched, vouchers in the current batch will be picked up as well.
  120. The returned list of hashrefs is stored in the $payment->{contact} property.
  121. Each hashref has the following keys: id (entity id), name, and entity_class.
  122. =back
  123. =cut
  124. sub get_all_contact_invoices {
  125. my ($self) = @_;
  126. @{$self->{contacts}} =
  127. $self->exec_method(funcname => 'payment_get_all_contact_invoices');
  128. # When arrays of complex types are supported by all versions of Postgres
  129. # that this application supports, we should look at doing type conversions
  130. # in DBObject so this sort of logic is unncessary. -- CT
  131. for my $contact (@{$self->{contacts}}){
  132. my @invoices = $self->parse_array($contact->{invoices});
  133. my $processed_invoices = [];
  134. for my $invoice (@invoices){
  135. my $new_invoice = {};
  136. for (qw(invoice_id invnumber invoice_date amount discount due)){
  137. $new_invoice->{$_} = shift @$invoice;
  138. if ($_ =~ /^(amount|discount|due)$/){
  139. $new_invoice->{$_} =
  140. Math::BigFloat->new($new_invoice->{$_});
  141. }
  142. }
  143. push(@$processed_invoices, $new_invoice);
  144. }
  145. $contact->{invoice} = $processed_invoices;
  146. }
  147. return @{$self->{contacts}};
  148. }
  149. =over
  150. =item list_open_projects
  151. This method gets the current date attribute, and provides a list of open
  152. projects. The list is attached to $self->{projects} and returned.
  153. =back
  154. =cut
  155. sub list_open_projects {
  156. my ($self) = @_;
  157. @{$self->{projects}} = $self->call_procedure(
  158. procname => 'project_list_open', args => [$self->{current_date}]
  159. );
  160. return @{$self->{projects}};
  161. }
  162. =over
  163. =item list_departments
  164. This method gets the type of document as a parameter, and provides a list of departments
  165. of the required type.
  166. The list is attached to $self->{departments} and returned.
  167. =back
  168. =cut
  169. sub list_departments {
  170. my ($self) = shift @_;
  171. my @args = @_;
  172. @{$self->{departments}} = $self->call_procedure(
  173. procname => 'department_list',
  174. args => \@args
  175. );
  176. return @{$self->{departments}};
  177. }
  178. =item list_open_vc
  179. This method gets the type of vc (vendor or customer) as a parameter, and provides a list of departments
  180. of the required type.
  181. The list is attached to $self->{departments} and returned.
  182. =back
  183. =cut
  184. sub list_departments {
  185. my ($self) = shift @_;
  186. my @args = @_;
  187. @{$self->{departments}} = $self->call_procedure(
  188. procname => 'department_list',
  189. args => \@args
  190. );
  191. return @{$self->{departments}};
  192. }
  193. =item get_open_currencies
  194. This method gets a list of the open currencies inside the database, it requires that
  195. $self->{account_class} (must be 1 or 2) exist to work.
  196. =back
  197. =cut
  198. sub get_open_currencies {
  199. my ($self) = shift @_;
  200. @{$self->{openCurrencies}} = $self->exec_method( funcname => 'payments_get_open_currencies');
  201. return @{$self->{openCurrencies}};
  202. }
  203. =item list_accounting
  204. This method lists all accounts that match the role specified in account_class property and
  205. are availible to store the payment or receipts.
  206. =back
  207. =cut
  208. sub list_accounting {
  209. my ($self) = @_;
  210. @{$self->{pay_accounts}} = $self->exec_method( funcname => 'chart_list_cash');
  211. return @{$self->{pay_accounts}};
  212. }
  213. =item get_sources
  214. This method builds all the possible sources of money,
  215. in the future it will look inside the DB.
  216. =back
  217. =cut
  218. sub get_sources {
  219. my ($self, $locale) = @_;
  220. @{$self->{cash_sources}} = ($locale->text('cash'),
  221. $locale->text('check'),
  222. $locale->text('deposit'),
  223. $locale->text('other'));
  224. return @{$self->{cash_sources}};
  225. }
  226. =item get_exchange_rate(currency, date)
  227. This method gets the exchange rate for the specified currency and date
  228. =cut
  229. sub get_exchange_rate {
  230. my ($self) = shift @_;
  231. ($self->{currency}, $self->{date}) = @_;
  232. ($self->{exchangerate}) = $self->exec_method(funcname => 'currency_get_exchangerate');
  233. return $self->{exchangerate}->{currency_get_exchangerate};
  234. }
  235. =item get_default_currency
  236. This method gets the default currency
  237. =back
  238. =cut
  239. sub get_default_currency {
  240. my ($self) = shift @_;
  241. ($self->{default_currency}) = $self->call_procedure(procname => 'defaults_get_defaultcurrency');
  242. return $self->{default_currency}->{defaults_get_defaultcurrency};
  243. }
  244. =item get_current_date
  245. This method returns the system's current date
  246. =cut
  247. sub get_current_date {
  248. my ($self) = shift @_;
  249. return $self->{current_date};
  250. }
  251. =item get_vc_info
  252. This method returns the contact informatino for a customer or vendor according to
  253. $self->{account_class}
  254. =cut
  255. sub get_vc_info {
  256. my ($self) = @_;
  257. #@{$self->{vendor_customer_info}} = $self->call_procedure(procname => 'vendor_customer_info');
  258. #return @{$self->{vendor_customer_info}};
  259. }
  260. =item get_payment_detail_data
  261. This method sets appropriate project, department, etc. fields.
  262. =cut
  263. sub get_payment_detail_data {
  264. my ($self) = @_;
  265. $self->get_metadata();
  266. my $source_inc;
  267. my $source_src;
  268. if (defined ($self->{source_start})){
  269. $self->{source_start} =~ /(\d*)\D*$/;
  270. $source_src = $1;
  271. if ($source_src) {
  272. $source_inc = $source_src;
  273. } else {
  274. $source_inc = 0;
  275. }
  276. }
  277. my $source_length = length($source_inc);
  278. @{$self->{contact_invoices}} = $self->exec_method(
  279. funcname => 'payment_get_all_contact_invoices');
  280. for my $inv (@{$self->{contact_invoices}}){
  281. if (defined $self->{source_start}){
  282. my $source = $self->{source_start};
  283. if (length($source_inc) < $source_length){
  284. $source_inc = sprintf('%0*s', $source_length, $source_inc);
  285. }
  286. $source =~ s/$source_src(\D*)$/$source_inc$1/;
  287. ++ $source_inc;
  288. $inv->{source} = $source;
  289. }
  290. my $tmp_invoices = $inv->{invoices};
  291. $inv->{invoices} = [];
  292. @{$inv->{invoices}} = $self->_parse_array($tmp_invoices);
  293. }
  294. $self->{dbh}->commit; # Commit locks
  295. }
  296. sub post_bulk {
  297. my ($self) = @_;
  298. my $total_count = 0;
  299. my ($ref) = $self->call_procedure(
  300. procname => 'setting_get',
  301. args => ['queue_payments'],
  302. );
  303. my $queue_payments = $ref->{setting_get};
  304. if ($queue_payments){
  305. my ($job_ref) = $self->exec_method(
  306. funcname => 'job__create'
  307. );
  308. $self->{job_id} = $job_ref->{job__create};
  309. ($self->{job}) = $self->exec_method(
  310. funcname => 'job__status'
  311. );
  312. }
  313. $self->{payment_date} = $self->{datepaid};
  314. for my $contact_row (1 .. $self->{contact_count}){
  315. my $contact_id = $self->{"contact_$contact_row"};
  316. next if (!$self->{"id_$contact_id"});
  317. my $invoice_array = "{}"; # Pg Array
  318. for my $invoice_row (1 .. $self->{"invoice_count_$contact_id"}){
  319. my $invoice_id = $self->{"invoice_${contact_id}_${invoice_row}"};
  320. my $pay_amount = ($self->{"paid_$contact_id"} eq 'all' )
  321. ? $self->{"net_$invoice_id"}
  322. : $self->{"payment_$invoice_id"};
  323. next if ! $pay_amount;
  324. $pay_amount = $pay_amount * 1;
  325. my $invoice_subarray = "{$invoice_id,$pay_amount}";
  326. if ($invoice_subarray !~ /^\{\d+\,\-?\d*\.?\d+\}$/){
  327. $self->error("Invalid subarray: $invoice_subarray");
  328. }
  329. $invoice_subarray =~ s/[^0123456789{},.]//;
  330. if ($invoice_array eq '{}'){ # Omit comma
  331. $invoice_array = "{$invoice_subarray}";
  332. } else {
  333. $invoice_array =~ s/\}$/,$invoice_subarray\}/;
  334. }
  335. }
  336. $self->{transactions} = $invoice_array;
  337. $self->{source} = $self->{"source_$contact_id"};
  338. if ($queue_payments){
  339. $self->{batch_class} = 3;
  340. $self->exec_method(
  341. funcname => 'payment_bulk_queue'
  342. );
  343. } else {
  344. $self->exec_method(funcname => 'payment_bulk_post');
  345. }
  346. }
  347. $self->{queue_payments} = $queue_payments;
  348. return $self->{dbh}->commit;
  349. }
  350. sub check_job {
  351. my ($self) = @_;
  352. ($self->{job}) = $self->exec_method(funcname => 'job__status');
  353. }
  354. 1;