summaryrefslogtreecommitdiff
path: root/LedgerSMB.pm
blob: 6a929b47aaee93af59cd91b6fda8e49c5070ea83 (plain)
  1. #=====================================================================
  2. # LedgerSMB
  3. # Small Medium Business Accounting software
  4. # http://www.ledgersmb.org/
  5. #
  6. # Copyright (C) 2006
  7. # This work contains copyrighted information from a number of sources all used
  8. # with permission.
  9. #
  10. # This file contains source code included with or based on SQL-Ledger which
  11. # is Copyright Dieter Simader and DWS Systems Inc. 2000-2005 and licensed
  12. # under the GNU General Public License version 2 or, at your option, any later
  13. # version. For a full list including contact information of contributors,
  14. # maintainers, and copyright holders, see the CONTRIBUTORS file.
  15. #
  16. # Original Copyright Notice from SQL-Ledger 2.6.17 (before the fork):
  17. # Copyright (C) 2000
  18. #
  19. # Author: DWS Systems Inc.
  20. # Web: http://www.sql-ledger.org
  21. #
  22. # Contributors: Thomas Bayen <bayen@gmx.de>
  23. # Antti Kaihola <akaihola@siba.fi>
  24. # Moritz Bunkus (tex)
  25. # Jim Rawlings <jim@your-dba.com> (DB2)
  26. #======================================================================
  27. #
  28. # This file has undergone whitespace cleanup.
  29. #
  30. #======================================================================
  31. #
  32. # main package
  33. #
  34. #======================================================================
  35. use Math::BigFloat lib=>'GMP';
  36. use LedgerSMB::Sysconfig;
  37. use strict;
  38. package LedgerSMB;
  39. sub new {
  40. my $type = shift;
  41. my $argstr = shift;
  42. read(STDIN, $_, $ENV{CONTENT_LENGTH});
  43. if ($argstr){
  44. $_ = $argstr;
  45. }
  46. elsif ($ENV{QUERY_STRING}) {
  47. $_ = $ENV{QUERY_STRING};
  48. }
  49. elsif ($ARGV[0]) {
  50. $_ = $ARGV[0];
  51. }
  52. my $self = {};
  53. %$self = split /[&=]/;
  54. for (keys %$self) { $self->{$_} = unescape("", $self->{$_}) }
  55. if (substr($self->{action}, 0, 1) !~ /( |\.)/) {
  56. $self->{action} = lc $self->{action};
  57. $self->{action} =~ s/( |-|,|\#|\/|\.$)/_/g;
  58. }
  59. $self->{menubar} = 1 if $self->{path} =~ /lynx/i;
  60. #menubar will be deprecated, replaced with below
  61. $self->{lynx} = 1 if $self->{path} =~ /lynx/i;
  62. $self->{version} = "1.2.0 Beta 2";
  63. $self->{dbversion} = "1.2.0";
  64. bless $self, $type;
  65. }
  66. sub debug {
  67. my ($self, $file) = @_;
  68. if ($file) {
  69. open(FH, '>', "$file") or die $!;
  70. for (sort keys %$self) { print FH "$_ = $self->{$_}\n" }
  71. close(FH);
  72. } else {
  73. print "\n";
  74. for (sort keys %$self) { print "$_ = $self->{$_}\n" }
  75. }
  76. }
  77. sub escape {
  78. my ($self, $str, $beenthere) = @_;
  79. # for Apache 2 we escape strings twice
  80. if (($ENV{SERVER_SIGNATURE} =~ /Apache\/2\.(\d+)\.(\d+)/) && !$beenthere) {
  81. $str = $self->escape($str, 1) if $1 == 0 && $2 < 44;
  82. }
  83. $str =~ s/([^a-zA-Z0-9_.-])/sprintf("%%%02x", ord($1))/ge;
  84. $str;
  85. }
  86. sub unescape {
  87. my ($self, $str) = @_;
  88. $str =~ tr/+/ /;
  89. $str =~ s/\\$//;
  90. $str =~ s/%([0-9a-fA-Z]{2})/pack("c",hex($1))/eg;
  91. $str =~ s/\r?\n/\n/g;
  92. $str;
  93. }
  94. sub numtextrows {
  95. my ($self, $str, $cols, $maxrows) = @_;
  96. my $rows = 0;
  97. for (split /\n/, $str) {
  98. $rows += int (((length) - 2)/$cols) + 1
  99. }
  100. $maxrows = $rows unless defined $maxrows;
  101. return ($rows > $maxrows) ? $maxrows : $rows;
  102. }
  103. sub isblank {
  104. my ($self, $name, $msg) = @_;
  105. $self->error($msg) if $self->{$name} =~ /^\s*$/;
  106. }
  107. sub redirect {
  108. my ($self, $msg) = @_;
  109. use List::Util qw(first);
  110. if ($self->{callback} || !$msg) {
  111. main::redirect();
  112. } else {
  113. $self->info($msg);
  114. }
  115. }
  116. sub format_string {
  117. my ($self, @fields) = @_;
  118. my $format = $self->{format};
  119. if ($self->{format} =~ /(postscript|pdf)/) {
  120. $format = 'tex';
  121. }
  122. my %replace = (
  123. 'order' => {
  124. html => [ '<', '>', '\n', '\r' ],
  125. txt => [ '\n', '\r' ],
  126. tex => [ quotemeta('\\'), '&', '\n','\r',
  127. '\$', '%', '_', '#',
  128. quotemeta('^'), '{', '}', '<', '>', '£'
  129. ] },
  130. html => { '<' => '&lt;', '>' => '&gt;','\n' => '<br />',
  131. '\r' => '<br />' },
  132. txt => { '\n' => "\n", '\r' => "\r" },
  133. tex => {'&' => '\&', '$' => '\$', '%' => '\%', '_' => '\_',
  134. '#' => '\#', quotemeta('^') => '\^\\', '{' => '\{',
  135. '}' => '\}', '<' => '$<$', '>' => '$>$',
  136. '\n' => '\newline ', '\r' => '\newline ',
  137. '£' => '\pounds ', quotemeta('\\') => '/'}
  138. );
  139. my $key;
  140. foreach $key (@{ $replace{order}{$format} }) {
  141. for (@fields) { $self->{$_} =~ s/$key/$replace{$format}{$key}/g }
  142. }
  143. }
  144. sub format_amount {
  145. my ($self, $myconfig, $amount, $places, $dash) = @_;
  146. my $negative ;
  147. if ($amount){
  148. $amount = $self->parse_amount($myconfig, $amount);
  149. $negative = ($amount < 0);
  150. $amount =~ s/-//;
  151. }
  152. if ($places =~ /\d+/) {
  153. #$places = 4 if $places == 2;
  154. $amount = $self->round_amount($amount, $places);
  155. }
  156. # is the amount negative
  157. # Parse $myconfig->{numberformat}
  158. my ($ts, $ds) = ($1, $2);
  159. if ($amount) {
  160. if ($myconfig->{numberformat}) {
  161. my ($whole, $dec) = split /\./, "$amount";
  162. $amount = join '', reverse split //, $whole;
  163. if ($places) {
  164. $dec .= "0" x $places;
  165. $dec = substr($dec, 0, $places);
  166. }
  167. if ($myconfig->{numberformat} eq '1,000.00') {
  168. $amount =~ s/\d{3,}?/$&,/g;
  169. $amount =~ s/,$//;
  170. $amount = join '', reverse split //, $amount;
  171. $amount .= "\.$dec" if ($dec ne "");
  172. }
  173. if ($myconfig->{numberformat} eq '1 000.00') {
  174. $amount =~ s/\d{3,}?/$& /g;
  175. $amount =~ s/\s$//;
  176. $amount = join '', reverse split //, $amount;
  177. $amount .= "\.$dec" if ($dec ne "");
  178. }
  179. if ($myconfig->{numberformat} eq "1'000.00") {
  180. $amount =~ s/\d{3,}?/$&'/g;
  181. $amount =~ s/'$//;
  182. $amount = join '', reverse split //, $amount;
  183. $amount .= "\.$dec" if ($dec ne "");
  184. }
  185. if ($myconfig->{numberformat} eq '1.000,00') {
  186. $amount =~ s/\d{3,}?/$&./g;
  187. $amount =~ s/\.$//;
  188. $amount = join '', reverse split //, $amount;
  189. $amount .= ",$dec" if ($dec ne "");
  190. }
  191. if ($myconfig->{numberformat} eq '1000,00') {
  192. $amount = "$whole";
  193. $amount .= ",$dec" if ($dec ne "");
  194. }
  195. if ($myconfig->{numberformat} eq '1000.00') {
  196. $amount = "$whole";
  197. $amount .= ".$dec" if ($dec ne "");
  198. }
  199. if ($dash =~ /-/) {
  200. $amount = ($negative) ? "($amount)" : "$amount";
  201. } elsif ($dash =~ /DRCR/) {
  202. $amount = ($negative) ? "$amount DR" : "$amount CR";
  203. } else {
  204. $amount = ($negative) ? "-$amount" : "$amount";
  205. }
  206. }
  207. } else {
  208. if ($dash eq "0" && $places) {
  209. if ($myconfig->{numberformat} eq '1.000,00') {
  210. $amount = "0".","."0" x $places;
  211. } else {
  212. $amount = "0"."."."0" x $places;
  213. }
  214. } else {
  215. $amount = ($dash ne "") ? "$dash" : "";
  216. }
  217. }
  218. $amount;
  219. }
  220. sub parse_amount {
  221. my ($self, $myconfig, $amount) = @_;
  222. if ($amount eq '' or $amount == undef){
  223. return 0;
  224. }
  225. if (UNIVERSAL::isa($amount, 'Math::BigFloat')){ # Amount may not be an object
  226. return $amount;
  227. }
  228. my $numberformat = $myconfig->{numberformat};
  229. if (($numberformat eq '1.000,00') ||
  230. ($numberformat eq '1000,00')) {
  231. $amount =~ s/\.//g;
  232. $amount =~ s/,/./;
  233. }
  234. if ($numberformat eq '1 000.00'){
  235. $amount =~ s/\s//g;
  236. }
  237. if ($numberformat eq "1'000.00") {
  238. $amount =~ s/'//g;
  239. }
  240. $amount =~ s/,//g;
  241. if ($amount =~ s/\((\d*\.?\d*)\)/$1/){
  242. $amount = $1 * -1;
  243. }
  244. if ($amount =~ s/(\d*\.?\d*)\s?DR/$1/){
  245. $amount = $1 * -1;
  246. }
  247. $amount =~ s/\s?CR//;
  248. $amount = new Math::BigFloat($amount);
  249. return ($amount * 1);
  250. }
  251. sub round_amount {
  252. my ($self, $amount, $places) = @_;
  253. # These rounding rules follow from the previous implementation.
  254. # They should be changed to allow different rules for different accounts.
  255. Math::BigFloat->round_mode('+inf') if $amount >= 0;
  256. Math::BigFloat->round_mode('-inf') if $amount < 0;
  257. $amount = Math::BigFloat->new($amount)->ffround(-$places) if $places >= 0;
  258. $amount = Math::BigFloat->new($amount)->ffround(-($places-1)) if $places < 0;
  259. return $amount;
  260. }
  261. sub callproc {
  262. my $self = shift @_;
  263. my $procname = shift @_;
  264. my $argstr = "";
  265. my @results;
  266. for (1 .. scalar @_){
  267. $argstr .= "?, ";
  268. }
  269. $argstr =~ s/\, $//;
  270. my $query = "SELECT * FROM $procname()";
  271. $query =~ s/\(\)/($argstr)/;
  272. my $sth = $self->{dbh}->prepare($query);
  273. $sth->execute(@_);
  274. while (my $ref = $sth->fetchrow_hashref('NAME_lc')){
  275. push @results, $ref;
  276. }
  277. @results;
  278. }
  279. sub datetonum {
  280. my ($self, $myconfig, $date, $picture) = @_;
  281. my ($yy, $mm, $dd);
  282. if ($date && $date =~ /\D/) {
  283. if ($myconfig->{dateformat} =~ /^yy/) {
  284. ($yy, $mm, $dd) = split /\D/, $date;
  285. }
  286. if ($myconfig->{dateformat} =~ /^mm/) {
  287. ($mm, $dd, $yy) = split /\D/, $date;
  288. }
  289. if ($myconfig->{dateformat} =~ /^dd/) {
  290. ($dd, $mm, $yy) = split /\D/, $date;
  291. }
  292. $dd *= 1;
  293. $mm *= 1;
  294. $yy += 2000 if length $yy == 2;
  295. $dd = substr("0$dd", -2);
  296. $mm = substr("0$mm", -2);
  297. $date = "$yy$mm$dd";
  298. }
  299. $date;
  300. }
  301. # Database routines used throughout
  302. sub db_init {
  303. my ($self, $myconfig) = @_;
  304. $self->{dbh} = $self->dbconnect_noauto($myconfig) || $self->dberror();
  305. my $query =
  306. "SELECT t.extends,
  307. coalesce (t.table_name, 'custom_' || extends)
  308. || ':' || f.field_name as field_def
  309. FROM custom_table_catalog t
  310. JOIN custom_field_catalog f USING (table_id)";
  311. my $sth = $self->{dbh}->prepare($query);
  312. $sth->execute;
  313. my $ref;
  314. while ($ref = $sth->fetchrow_hashref('NAME_lc')){
  315. push @{$self->{custom_db_fields}{$ref->{extends}}},
  316. $ref->{field_def};
  317. }
  318. }
  319. sub dbconnect_noauto {
  320. my ($self, $myconfig) = @_;
  321. # connect to database
  322. my $dbh = DBI->connect($myconfig->{dbconnect}, $myconfig->{dbuser}, $myconfig->{dbpasswd}, {AutoCommit => 0}) or $self->dberror;
  323. # set db options
  324. if ($myconfig->{dboptions}) {
  325. $dbh->do($myconfig->{dboptions});
  326. }
  327. $dbh;
  328. }
  329. sub redo_rows {
  330. my ($self, $flds, $new, $count, $numrows) = @_;
  331. my @ndx = ();
  332. for (1 .. $count) {
  333. push @ndx, { num => $new->[$_-1]->{runningnumber}, ndx => $_ }
  334. }
  335. my $i = 0;
  336. my $j;
  337. # fill rows
  338. foreach my $item (sort { $a->{num} <=> $b->{num} } @ndx) {
  339. $i++;
  340. $j = $item->{ndx} - 1;
  341. for (@{$flds}) { $self->{"${_}_$i"} = $new->[$j]->{$_} }
  342. }
  343. # delete empty rows
  344. for $i ($count + 1 .. $numrows) {
  345. for (@{$flds}) { delete $self->{"${_}_$i"} }
  346. }
  347. }
  348. sub merge {
  349. my ($self, $src) = @_;
  350. for my $arg ($self, $src){
  351. shift;
  352. }
  353. my @keys;
  354. if (scalar @keys){
  355. @keys = @_;
  356. print "Keys: ". scalar @keys . "\n";
  357. }
  358. else {
  359. @keys = keys %{$src};
  360. print "Keys: ". scalar @keys . "\n";
  361. }
  362. for my $arg (keys %$src){
  363. $self->{$arg} = $src->{$arg};
  364. }
  365. }
  366. 1;