1 package SL::Dev::Record;
5 our @EXPORT_OK = qw(create_invoice_item
11 create_delivery_order_item
12 create_sales_delivery_order
13 create_purchase_delivery_order
14 create_project create_department
19 our %EXPORT_TAGS = (ALL => \@EXPORT_OK);
22 use SL::DB::InvoiceItem;
24 use SL::Dev::Part qw(new_part);
25 use SL::Dev::CustomerVendor qw(new_vendor new_customer);
27 use SL::DB::ProjectStatus;
28 use SL::DB::ProjectType;
31 use List::Util qw(sum);
33 use SL::Locale::String qw(t8);
36 my %record_type_to_item_type = ( sales_invoice => 'SL::DB::InvoiceItem',
37 credit_note => 'SL::DB::InvoiceItem',
38 sales_order => 'SL::DB::OrderItem',
39 purchase_order => 'SL::DB::OrderItem',
40 sales_delivery_order => 'SL::DB::DeliveryOrderItem',
43 sub create_sales_invoice {
46 my $record_type = 'sales_invoice';
47 my $invoiceitems = delete $params{invoiceitems} // _create_two_items($record_type);
48 _check_items($invoiceitems, $record_type);
50 my $customer = delete $params{customer} // new_customer(name => 'Testcustomer')->save;
51 die "illegal customer" unless defined $customer && ref($customer) eq 'SL::DB::Customer';
53 my $invoice = SL::DB::Invoice->new(
56 customer_id => $customer->id,
57 taxzone_id => $customer->taxzone->id,
58 invnumber => delete $params{invnumber} // undef,
59 currency_id => $params{currency_id} // $::instance_conf->get_currency_id,
60 taxincluded => $params{taxincluded} // 0,
61 employee_id => $params{employee_id} // SL::DB::Manager::Employee->current->id,
62 salesman_id => $params{employee_id} // SL::DB::Manager::Employee->current->id,
63 transdate => $params{transdate} // DateTime->today_local->to_kivitendo,
64 payment_id => $params{payment_id} // undef,
65 gldate => DateTime->today,
66 invoiceitems => $invoiceitems,
68 $invoice->assign_attributes(%params) if %params;
74 sub create_credit_note {
77 my $record_type = 'credit_note';
78 my $invoiceitems = delete $params{invoiceitems} // _create_two_items($record_type);
79 _check_items($invoiceitems, $record_type);
81 my $customer = delete $params{customer} // new_customer(name => 'Testcustomer')->save;
82 die "illegal customer" unless defined $customer && ref($customer) eq 'SL::DB::Customer';
84 # adjust qty for credit note items
85 $_->qty( $_->qty * -1) foreach @{$invoiceitems};
87 my $invoice = SL::DB::Invoice->new(
89 type => 'credit_note',
90 customer_id => $customer->id,
91 taxzone_id => $customer->taxzone->id,
92 invnumber => delete $params{invnumber} // undef,
93 currency_id => $params{currency_id} // $::instance_conf->get_currency_id,
94 taxincluded => $params{taxincluded} // 0,
95 employee_id => $params{employee_id} // SL::DB::Manager::Employee->current->id,
96 salesman_id => $params{employee_id} // SL::DB::Manager::Employee->current->id,
97 transdate => $params{transdate} // DateTime->today_local->to_kivitendo,
98 payment_id => $params{payment_id} // undef,
99 gldate => DateTime->today,
100 invoiceitems => $invoiceitems,
102 $invoice->assign_attributes(%params) if %params;
108 sub create_sales_delivery_order {
111 my $record_type = 'sales_delivery_order';
112 my $orderitems = delete $params{orderitems} // _create_two_items($record_type);
113 _check_items($orderitems, $record_type);
115 my $customer = $params{customer} // new_customer(name => 'Testcustomer')->save;
116 die "illegal customer" unless ref($customer) eq 'SL::DB::Customer';
118 my $delivery_order = SL::DB::DeliveryOrder->new(
119 'is_sales' => 'true',
121 customer_id => $customer->id,
122 taxzone_id => $customer->taxzone_id,
123 donumber => $params{donumber} // undef,
124 currency_id => $params{currency_id} // $::instance_conf->get_currency_id,
125 taxincluded => $params{taxincluded} // 0,
126 employee_id => $params{employee_id} // SL::DB::Manager::Employee->current->id,
127 salesman_id => $params{employee_id} // SL::DB::Manager::Employee->current->id,
128 transdate => $params{transdate} // DateTime->today,
129 orderitems => $orderitems,
131 $delivery_order->assign_attributes(%params) if %params;
132 $delivery_order->save;
133 return $delivery_order;
136 sub create_purchase_delivery_order {
139 my $record_type = 'purchase_delivery_order';
140 my $orderitems = delete $params{orderitems} // _create_two_items($record_type);
141 _check_items($orderitems, $record_type);
143 my $vendor = $params{vendor} // new_vendor(name => 'Testvendor')->save;
144 die "illegal customer" unless ref($vendor) eq 'SL::DB::Vendor';
146 my $delivery_order = SL::DB::DeliveryOrder->new(
147 'is_sales' => 'false',
149 vendor_id => $vendor->id,
150 taxzone_id => $vendor->taxzone_id,
151 donumber => $params{donumber} // undef,
152 currency_id => $params{currency_id} // $::instance_conf->get_currency_id,
153 taxincluded => $params{taxincluded} // 0,
154 employee_id => $params{employee_id} // SL::DB::Manager::Employee->current->id,
155 salesman_id => $params{employee_id} // SL::DB::Manager::Employee->current->id,
156 transdate => $params{transdate} // DateTime->today,
157 orderitems => $orderitems,
159 $delivery_order->assign_attributes(%params) if %params;
160 $delivery_order->save;
161 return $delivery_order;
164 sub create_sales_order {
167 my $record_type = 'sales_order';
168 my $orderitems = delete $params{orderitems} // _create_two_items($record_type);
169 _check_items($orderitems, $record_type);
171 my $save = delete $params{save} // 0;
173 my $customer = $params{customer} // new_customer(name => 'Testcustomer')->save;
174 die "illegal customer" unless ref($customer) eq 'SL::DB::Customer';
176 my $order = SL::DB::Order->new(
177 customer_id => delete $params{customer_id} // $customer->id,
178 taxzone_id => delete $params{taxzone_id} // $customer->taxzone->id,
179 currency_id => delete $params{currency_id} // $::instance_conf->get_currency_id,
180 taxincluded => delete $params{taxincluded} // 0,
181 employee_id => delete $params{employee_id} // SL::DB::Manager::Employee->current->id,
182 salesman_id => delete $params{employee_id} // SL::DB::Manager::Employee->current->id,
183 transdate => delete $params{transdate} // DateTime->today,
184 orderitems => $orderitems,
186 $order->assign_attributes(%params) if %params;
189 $order->calculate_prices_and_taxes;
195 sub create_purchase_order {
198 my $record_type = 'purchase_order';
199 my $orderitems = delete $params{orderitems} // _create_two_items($record_type);
200 _check_items($orderitems, $record_type);
202 my $save = delete $params{save} // 0;
204 my $vendor = $params{vendor} // new_vendor(name => 'Testvendor')->save;
205 die "illegal vendor" unless ref($vendor) eq 'SL::DB::Vendor';
207 my $order = SL::DB::Order->new(
208 vendor_id => delete $params{vendor_id} // $vendor->id,
209 taxzone_id => delete $params{taxzone_id} // $vendor->taxzone->id,
210 currency_id => delete $params{currency_id} // $::instance_conf->get_currency_id,
211 taxincluded => delete $params{taxincluded} // 0,
212 transdate => delete $params{transdate} // DateTime->today,
214 orderitems => $orderitems,
216 $order->assign_attributes(%params) if %params;
219 $order->calculate_prices_and_taxes; # not tested for purchase orders
226 my ($items, $record_type) = @_;
228 if ( scalar @{$items} == 0 or grep { ref($_) ne $record_type_to_item_type{"$record_type"} } @{$items} ) {
229 die "Error: items must be an arrayref of " . $record_type_to_item_type{"$record_type"} . "objects.";
233 sub create_invoice_item {
236 return _create_item(record_type => 'sales_invoice', %params);
239 sub create_order_item {
242 return _create_item(record_type => 'sales_order', %params);
245 sub create_delivery_order_item {
248 return _create_item(record_type => 'sales_delivery_order', %params);
254 my $record_type = delete($params{record_type});
255 my $part = delete($params{part});
257 die "illegal record type: $record_type, must be one of: " . join(' ', keys %record_type_to_item_type) unless $record_type_to_item_type{ $record_type };
258 die "part missing as param" unless $part && ref($part) eq 'SL::DB::Part';
260 my ($sellprice, $lastcost);
262 if ( $record_type =~ /^sales/ ) {
263 $sellprice = delete $params{sellprice} // $part->sellprice;
264 $lastcost = delete $params{lastcost} // $part->lastcost;
266 $sellprice = delete $params{sellprice} // $part->lastcost;
267 $lastcost = delete $params{lastcost} // 0; # $part->lastcost;
270 my $item = "$record_type_to_item_type{$record_type}"->new(
271 parts_id => $part->id,
272 sellprice => $sellprice,
273 lastcost => $lastcost,
274 description => $part->description,
276 qty => $params{qty} || 5,
278 $item->assign_attributes(%params) if %params;
282 sub _create_two_items {
283 my ($record_type) = @_;
285 my $part1 = new_part(description => 'Testpart 1',
288 my $part2 = new_part(description => 'Testpart 2',
291 my $item1 = _create_item(record_type => $record_type, part => $part1, qty => 5);
292 my $item2 = _create_item(record_type => $record_type, part => $part2, qty => 8);
293 return [ $item1, $item2 ];
298 my $project = SL::DB::Project->new(
299 projectnumber => delete $params{projectnumber} // 1,
300 description => delete $params{description} // "Test project",
303 project_status_id => SL::DB::Manager::ProjectStatus->find_by(name => "running")->id,
304 project_type_id => SL::DB::Manager::ProjectType->find_by(description => "Standard")->id,
306 $project->assign_attributes(%params) if %params;
310 sub create_department {
313 my $department = SL::DB::Department->new(
314 'description' => delete $params{description} // 'Test Department',
317 $department->assign_attributes(%params) if %params;
322 sub create_ap_transaction {
325 my $vendor = delete $params{vendor};
327 die "vendor missing or not a SL::DB::Vendor object" unless ref($vendor) eq 'SL::DB::Vendor';
329 # use default SL/Dev vendor if it exists, or create a new one
330 $vendor = SL::DB::Manager::Vendor->find_by(name => 'Testlieferant') // new_vendor->save;
333 my $taxincluded = $params{taxincluded} // 1;
334 delete $params{taxincluded};
336 my $bookings = delete $params{bookings};
338 unless ( $bookings ) {
339 my $chart_postage = SL::DB::Manager::Chart->find_by(description => 'Porto');
340 my $chart_telephone = SL::DB::Manager::Chart->find_by(description => 'Telefon');
343 chart => $chart_postage,
347 chart => $chart_telephone,
348 amount => $taxincluded ? 1190 : 1000,
354 my $project_id = delete $params{globalproject_id};
356 # if amount or netamount are given, then it compares them to the final values, and dies if they don't match
357 my $expected_amount = delete $params{amount};
358 my $expected_netamount = delete $params{netamount};
360 my $dec = delete $params{dec} // 2;
362 my $today = DateTime->today_local;
363 my $transdate = delete $params{transdate} // $today;
364 die "transdate hat to be DateTime object" unless ref($transdate) eq 'DateTime';
366 my $gldate = delete $params{gldate} // $today;
367 die "gldate hat to be DateTime object" unless ref($gldate) eq 'DateTime';
369 my $ap_chart = delete $params{ap_chart} // SL::DB::Manager::Chart->find_by( accno => '1600' );
370 die "no ap_chart found or not an AP chart" unless $ap_chart and $ap_chart->link eq 'AP';
372 my $ap_transaction = SL::DB::PurchaseInvoice->new(
373 vendor_id => $vendor->id,
376 globalproject_id => $project_id,
377 invnumber => delete $params{invnumber} // 'test ap_transaction',
378 notes => delete $params{notes} // 'test ap_transaction',
379 transdate => $transdate,
381 taxincluded => $taxincluded,
382 taxzone_id => $vendor->taxzone_id, # taxzone_id shouldn't have any effect on ap transactions
383 currency_id => $::instance_conf->get_currency_id,
384 type => undef, # isn't set for ap
385 employee_id => SL::DB::Manager::Employee->current->id,
387 # assign any parameters that weren't explicitly handled above, e.g. itime
388 $ap_transaction->assign_attributes(%params) if %params;
390 foreach my $booking ( @{$bookings} ) {
391 my $chart = delete $booking->{chart};
392 die "illegal chart" unless ref($chart) eq 'SL::DB::Chart';
394 my $tax = _transaction_tax_helper($booking, $chart, $transdate); # will die if tax can't be found
396 $ap_transaction->add_ap_amount_row(
397 amount => $booking->{amount}, # add_ap_amount_row expects the user input amount, does its own calculate_tax
400 project_id => $booking->{project_id},
404 my $acc_trans_sum = sum map { $_->amount } grep { $_->chart_link =~ 'AP_amount' } @{$ap_transaction->transactions};
405 # $main::lxdebug->message(0, sprintf("accno: %s amount: %s chart_link: %s\n",
409 # )) foreach @{$ap_transaction->transactions};
411 # determine netamount and amount from the transactions that were added via bookings
412 $ap_transaction->netamount( -1 * sum map { $_->amount } grep { $_->chart_link =~ 'AP_amount' } @{$ap_transaction->transactions} );
413 # $main::lxdebug->message(0, sprintf('found netamount %s', $ap_transaction->netamount));
415 my $taxamount = -1 * sum map { $_->amount } grep { $_->chart_link =~ /tax/ } @{$ap_transaction->transactions};
416 $ap_transaction->amount( $ap_transaction->netamount + $taxamount );
417 # additional check, add up all transactions before AP-transaction is added
418 my $refamount = -1 * sum map { $_->amount } @{$ap_transaction->transactions};
419 die "refamount = $refamount, ap_transaction->amount = " . $ap_transaction->amount unless $refamount == $ap_transaction->amount;
421 # if amount or netamount were passed as params, check if the values are still
422 # the same after recalculating them from the acc_trans entries
423 if (defined $expected_amount) {
424 die "amount doesn't match acc_trans amounts: $expected_amount != " . $ap_transaction->amount unless $expected_amount == $ap_transaction->amount;
426 if (defined $expected_netamount) {
427 die "netamount doesn't match acc_trans netamounts: $expected_netamount != " . $ap_transaction->netamount unless $expected_netamount == $ap_transaction->netamount;
430 $ap_transaction->create_ap_row(chart => $ap_chart);
431 $ap_transaction->save;
432 # $main::lxdebug->message(0, sprintf("created ap_transaction with invnumber %s and trans_id %s",
433 # $ap_transaction->invnumber,
434 # $ap_transaction->id));
435 return $ap_transaction;
438 sub create_ar_transaction {
441 my $customer = delete $params{customer};
443 die "customer missing or not a SL::DB::Customer object" unless ref($customer) eq 'SL::DB::Customer';
445 # use default SL/Dev vendor if it exists, or create a new one
446 $customer = SL::DB::Manager::Customer->find_by(name => 'Testkunde') // new_customer->save;
449 my $taxincluded = $params{taxincluded} // 1;
450 delete $params{taxincluded};
452 my $bookings = delete $params{bookings};
454 unless ( $bookings ) {
455 my $chart_19 = SL::DB::Manager::Chart->find_by(accno => '8400');
456 my $chart_7 = SL::DB::Manager::Chart->find_by(accno => '8300');
457 my $chart_0 = SL::DB::Manager::Chart->find_by(accno => '8200');
461 amount => $taxincluded ? 119 : 100,
465 amount => $taxincluded ? 107 : 100,
475 my $project_id = delete $params{globalproject_id};
477 # if amount or netamount are given, then it compares them to the final values, and dies if they don't match
478 my $expected_amount = delete $params{amount};
479 my $expected_netamount = delete $params{netamount};
481 my $dec = delete $params{dec} // 2;
483 my $today = DateTime->today_local;
484 my $transdate = delete $params{transdate} // $today;
485 die "transdate hat to be DateTime object" unless ref($transdate) eq 'DateTime';
487 my $gldate = delete $params{gldate} // $today;
488 die "gldate hat to be DateTime object" unless ref($gldate) eq 'DateTime';
490 my $ar_chart = delete $params{ar_chart} // SL::DB::Manager::Chart->find_by( accno => '1400' );
491 die "no ar_chart found or not an AR chart" unless $ar_chart and $ar_chart->link eq 'AR';
493 my $ar_transaction = SL::DB::Invoice->new(
494 customer_id => $customer->id,
497 globalproject_id => $project_id,
498 invnumber => delete $params{invnumber} // 'test ar_transaction',
499 notes => delete $params{notes} // 'test ar_transaction',
500 transdate => $transdate,
502 taxincluded => $taxincluded,
503 taxzone_id => $customer->taxzone_id, # taxzone_id shouldn't have any effect on ar transactions
504 currency_id => $::instance_conf->get_currency_id,
505 type => undef, # isn't set for ar
506 employee_id => SL::DB::Manager::Employee->current->id,
508 # assign any parameters that weren't explicitly handled above, e.g. itime
509 $ar_transaction->assign_attributes(%params) if %params;
511 foreach my $booking ( @{$bookings} ) {
512 my $chart = delete $booking->{chart};
513 die "illegal chart" unless ref($chart) eq 'SL::DB::Chart';
515 my $tax = _transaction_tax_helper($booking, $chart, $transdate); # will die if tax can't be found
517 $ar_transaction->add_ar_amount_row(
518 amount => $booking->{amount}, # add_ar_amount_row expects the user input amount, does its own calculate_tax
521 project_id => $booking->{project_id},
525 my $acc_trans_sum = sum map { $_->amount } grep { $_->chart_link =~ 'AR_amount' } @{$ar_transaction->transactions};
526 # $main::lxdebug->message(0, sprintf("accno: %s amount: %s chart_link: %s\n",
530 # )) foreach @{$ar_transaction->transactions};
532 # determine netamount and amount from the transactions that were added via bookings
533 $ar_transaction->netamount( 1 * sum map { $_->amount } grep { $_->chart_link =~ 'AR_amount' } @{$ar_transaction->transactions} );
534 # $main::lxdebug->message(0, sprintf('found netamount %s', $ar_transaction->netamount));
536 my $taxamount = 1 * sum map { $_->amount } grep { $_->chart_link =~ /tax/ } @{$ar_transaction->transactions};
537 $ar_transaction->amount( $ar_transaction->netamount + $taxamount );
538 # additional check, add up all transactions before AP-transaction is added
539 my $refamount = 1 * sum map { $_->amount } @{$ar_transaction->transactions};
540 die "refamount = $refamount, ar_transaction->amount = " . $ar_transaction->amount unless $refamount == $ar_transaction->amount;
542 # if amount or netamount were passed as params, check if the values are still
543 # the same after recalculating them from the acc_trans entries
544 if (defined $expected_amount) {
545 die "amount doesn't match acc_trans amounts: $expected_amount != " . $ar_transaction->amount unless $expected_amount == $ar_transaction->amount;
547 if (defined $expected_netamount) {
548 die "netamount doesn't match acc_trans netamounts: $expected_netamount != " . $ar_transaction->netamount unless $expected_netamount == $ar_transaction->netamount;
551 $ar_transaction->create_ar_row(chart => $ar_chart);
552 $ar_transaction->save;
553 # $main::lxdebug->message(0, sprintf("created ar_transaction with invnumber %s and trans_id %s",
554 # $ar_transaction->invnumber,
555 # $ar_transaction->id));
556 return $ar_transaction;
559 sub create_gl_transaction {
562 my $ob_transaction = delete $params{ob_transaction} // 0;
563 my $cb_transaction = delete $params{cb_transaction} // 0;
564 my $dec = delete $params{rec} // 2;
566 my $taxincluded = defined $params{taxincluded} ? $params{taxincluded} : 1;
568 my $today = DateTime->today_local;
569 my $transdate = delete $params{transdate} // $today;
570 my $gldate = delete $params{gldate} // $today;
572 my $reference = delete $params{reference} // 'reference';
573 my $description = delete $params{description} // 'description';
575 my $department_id = delete $params{department_id};
577 my $bookings = delete $params{bookings};
578 unless ( $bookings && scalar @{$bookings} ) {
579 # default bookings if left empty
580 my $expense_chart = SL::DB::Manager::Chart->find_by(accno => '4660') or die "Can't find expense chart 4660\n"; # Reisekosten
581 my $cash_chart = SL::DB::Manager::Chart->find_by(accno => '1000') or die "Can't find cash chart 1000\n"; # Kasse
585 $reference = 'Reise';
586 $description = 'Reise';
590 chart => $expense_chart, # has default tax of 19%
595 chart => $cash_chart,
602 my $gl_transaction = SL::DB::GLTransaction->new(
603 reference => $reference,
604 description => $description,
605 transdate => $transdate,
607 taxincluded => $taxincluded,
609 ob_transaction => $ob_transaction,
610 cb_transaction => $cb_transaction,
615 # assign any parameters that weren't explicitly handled above, e.g. itime
616 $gl_transaction->assign_attributes(%params) if %params;
619 if ( scalar @{$bookings} ) {
620 # there are several ways of determining the tax:
621 # * tax_id : fetches SL::DB::Tax object via id (as used in dropdown in interface)
622 # * tax : SL::DB::Tax object (where $tax->id = tax_id)
623 # * taxkey : tax is determined from startdate
624 # * none of the above defined: use the default tax for that chart
626 foreach my $booking ( @{$bookings} ) {
627 my $chart = delete $booking->{chart};
628 die "illegal chart" unless ref($chart) eq 'SL::DB::Chart';
630 die t8('Empty transaction!')
631 unless $booking->{debit} or $booking->{credit}; # must exist and not be 0
632 die t8('Cannot post transaction with a debit and credit entry for the same account!')
633 if defined($booking->{debit}) and defined($booking->{credit});
635 my $tax = _transaction_tax_helper($booking, $chart, $transdate); # will die if tax can't be found
637 $gl_transaction->add_chart_booking(
639 debit => $booking->{debit},
640 credit => $booking->{credit},
642 source => $booking->{source} // '',
643 memo => $booking->{memo} // '',
644 project_id => $booking->{project_id}
649 $gl_transaction->post;
651 return $gl_transaction;
654 sub _transaction_tax_helper {
655 # checks for hash-entries with key tax, tax_id or taxkey
656 # returns an SL::DB::Tax object
657 # can be used for booking hashref in ar_transaction, ap_transaction and gl_transaction
658 # will modify hashref, e.g. removing taxkey if tax_id was also supplied
660 my ($booking, $chart, $transdate) = @_;
662 die "_transaction_tax_helper: chart missing" unless $chart && ref($chart) eq 'SL::DB::Chart';
663 die "_transaction_tax_helper: transdate missing" unless $transdate && ref($transdate) eq 'DateTime';
667 if ( defined $booking->{tax_id} ) { # tax_id may be 0
668 delete $booking->{taxkey}; # ignore any taxkeys that may have been added, tax_id has precedence
669 $tax = SL::DB::Tax->new(id => $booking->{tax_id})->load( with => [ 'chart' ] );
670 } elsif ( $booking->{tax} ) {
671 die "illegal tax entry" unless ref($booking->{tax}) eq 'SL::DB::Tax';
672 $tax = $booking->{tax};
673 } elsif ( defined $booking->{taxkey} ) {
674 # If a taxkey is given, find the taxkey entry for that chart that
675 # matches the stored taxkey and with the correct transdate. This will only work
676 # if kivitendo has that taxkey configured for that chart, i.e. it should barf if
677 # e.g. the bank chart is called with taxkey 3.
682 # where taxkey_id = 3
683 # and chart_id = (select id from chart where accno = '8400')
684 # and startdate <= '2018-01-01'
685 # order by startdate desc
688 my $taxkey = SL::DB::Manager::TaxKey->get_first(
689 query => [ and => [ chart_id => $chart->id,
690 startdate => { le => $transdate },
691 taxkey => $booking->{taxkey}
694 sort_by => "startdate DESC",
696 with_objects => [ qw(tax) ],
698 die sprintf("Chart %s doesn't have a taxkey chart configured for taxkey %s", $chart->accno, $booking->{taxkey})
703 # use default tax for that chart if neither tax_id, tax or taxkey were defined
704 my $active_taxkey = $chart->get_active_taxkey($transdate);
705 $tax = $active_taxkey->tax;
706 # $main::lxdebug->message(0, sprintf("found default taxrate %s for chart %s", $tax->rate, $chart->displayable_name));
709 die "no tax" unless $tax && ref($tax) eq 'SL::DB::Tax';
719 SL::Dev::Record - create record objects for testing, with minimal defaults
723 =head2 C<create_sales_invoice %PARAMS>
725 Creates a new sales invoice (table ar, invoice = 1).
727 If neither customer nor invoiceitems are passed as params a customer and two
728 parts are created and used for building the invoice.
730 Minimal usage example:
732 my $invoice = SL::Dev::Record::create_sales_invoice();
736 my $invoice2 = SL::Dev::Record::create_sales_invoice(
738 transdate => DateTime->today->subtract(days => 7),
742 =head2 C<create_credit_note %PARAMS>
744 Create a credit note (sales). Use positive quantities when adding items.
746 Example including creation of parts and of credit_note:
748 my $part1 = SL::Dev::Part::new_part( partnumber => 'T4254')->save;
749 my $part2 = SL::Dev::Part::new_service(partnumber => 'Serv1')->save;
750 my $credit_note = SL::Dev::Record::create_credit_note(
753 invoiceitems => [ SL::Dev::Record::create_invoice_item(part => $part1, qty => 3, sellprice => 70),
754 SL::Dev::Record::create_invoice_item(part => $part2, qty => 10, sellprice => 50),
758 =head2 C<create_sales_order %PARAMS>
762 Create a sales order and save it directly via rose, without running
763 calculate_prices_and_taxes:
765 my $order = SL::Dev::Record::create_sales_order()->save;
767 Let create_sales_order run calculate_prices_and_taxes and save:
769 my $order = SL::Dev::Record::create_sales_order(save => 1);
772 Example including creation of part and of sales order:
774 my $part1 = SL::Dev::Part::new_part( partnumber => 'T4254')->save;
775 my $part2 = SL::Dev::Part::new_service(partnumber => 'Serv1')->save;
776 my $order = SL::Dev::Record::create_sales_order(
779 orderitems => [ SL::Dev::Record::create_order_item(part => $part1, qty => 3, sellprice => 70),
780 SL::Dev::Record::create_order_item(part => $part2, qty => 10, sellprice => 50),
784 Example: create 100 orders with the same part for 100 new customers:
786 my $part1 = SL::Dev::Part::new_part(partnumber => 'T6256')->save;
787 SL::Dev::Record::create_sales_order(
790 orderitems => [ SL::Dev::Record::create_order_item(part => $part1, qty => 1, sellprice => 9) ]
793 =head2 C<create_purchase_order %PARAMS>
795 See comments for C<create_sales_order>.
799 my $purchase_order = SL::Dev::Record::create_purchase_order(save => 1);
802 =head2 C<create_item %PARAMS>
804 Creates an item from a part object that can be added to a record.
808 record_type (sales_invoice, sales_order, sales_delivery_order)
809 part (an SL::DB::Part object)
811 Example including creation of part and of invoice:
813 my $part = SL::Dev::Part::new_part( partnumber => 'T4254')->save;
814 my $item = SL::Dev::Record::create_invoice_item(part => $part, qty => 2.5);
815 my $invoice = SL::Dev::Record::create_sales_invoice(
817 invoiceitems => [ $item ],
820 =head2 C<create_project %PARAMS>
822 Creates a default project.
824 Minimal example, creating a project with status "running" and type "Standard":
826 my $project = SL::Dev::Record::create_project();
828 $project = SL::Dev::Record::create_project(
829 projectnumber => 'p1',
830 description => 'Test project',
833 If C<$params{description}> or C<$params{projectnumber}> exists, this will override the
834 default value 'Test project'.
836 C<%params> should only contain alterable keys from the object Project.
838 =head2 C<create_department %PARAMS>
840 Creates a default department.
844 my $department = SL::Dev::Record::create_department();
846 my $department = SL::Dev::Record::create_department(
847 description => 'Hawaii',
850 If C<$params{description}> exists, this will override the
851 default value 'Test Department'.
853 C<%params> should only contain alterable keys from the object Department.
855 =head2 C<create_ap_transaction %PARAMS>
857 Creates a new AP transaction (table ap, invoice = 0), and will try to add as
858 many defaults as possible.
861 * vendor (SL::DB::Vendor object, defaults to SL::Dev default vendor)
862 * taxincluded (0 or 1, defaults to 1)
863 * transdate (DateTime object, defaults to current date)
864 * bookings (arrayref for the charts to be booked, see examples below)
865 * amount (to check if final amount matches this amount)
866 * netamount (to check if final amount matches this amount)
867 * dec (number of decimals to round to, defaults to 2)
868 * ap_chart (SL::DB::Chart object, default to accno 1600)
869 * invnumber (defaults to 'test ap_transaction')
870 * notes (defaults to 'test ap_transaction')
873 Currently doesn't support exchange rates.
875 Minimal usage example, creating an AP transaction with a default vendor and
876 default bookings (telephone, postage):
878 use SL::Dev::Record qw(create_ap_transaction);
879 my $invoice = create_ap_transaction();
881 Create an AP transaction with a specific vendor and specific charts:
883 my $vendor = SL::Dev::CustomerVendor::new_vendor(name => 'My Vendor')->save;
884 my $chart_postage = SL::DB::Manager::Chart->find_by(description => 'Porto');
885 my $chart_telephone = SL::DB::Manager::Chart->find_by(description => 'Telefon');
887 my $ap_transaction = create_ap_transaction(
889 invnumber => 'test invoice taxincluded',
891 amount => 2190, # optional param for checking whether final amount matches
892 netamount => 2000, # optional param for checking whether final netamount matches
895 chart => $chart_postage,
899 chart => $chart_telephone,
905 Or the same example with tax not included, but an old transdate and old taxrate (16%):
907 my $ap_transaction = create_ap_transaction(
909 invnumber => 'test invoice tax not included',
910 transdate => DateTime->new(year => 2000, month => 10, day => 1),
912 amount => 2160, # optional param for checking whether final amount matches
913 netamount => 2000, # optional param for checking whether final netamount matches
916 chart => $chart_postage,
920 chart => $chart_telephone,
926 Don't use the default tax, e.g. postage with 19%:
928 my $tax_9 = SL::DB::Manager::Tax->find_by(taxkey => 9, rate => 0.19);
929 my $chart_postage = SL::DB::Manager::Chart->find_by(description => 'Porto');
930 my $ap_transaction = create_ap_transaction(
931 invnumber => 'postage with tax',
935 chart => $chart_postage,
942 =head2 C<create_ar_transaction %PARAMS>
944 See C<create_ap_transaction>, except use customer instead of vendor.
946 =head2 C<create_gl_transaction %PARAMS>
948 Creates a new GL transaction (table gl), which is basically a wrapper around
949 SL::DB::GLTransaction->new(...) and add_chart_booking and post, while setting
950 as many defaults as possible.
954 * taxincluded (0 or 1, defaults to 1)
955 * transdate (DateTime object, defaults to current date)
956 * dec (number of decimals to round to, defaults to 2)
957 * bookings (arrayref for the charts and taxes to be booked, see examples below)
959 bookings must include a least:
961 * chart as an SL::DB::Chart object
962 * credit or debit, as positive numbers
963 * tax_id, tax (an SL::DB::Tax object) or taxkey (e.g. 9)
965 Can't be used to create storno transactions.
967 Minimal usage example, using all the defaults, creating a GL transaction with
970 use SL::Dev::Record qw(create_gl_transaction);
971 $gl_transaction = create_gl_transaction();
973 Create a GL transaction with a specific charts and taxes (the default taxes for
974 those charts are used if none are explicitly given in bookings):
976 my $cash = SL::DB::Manager::Chart->find_by( description => 'Kasse' );
977 my $betriebsbedarf = SL::DB::Manager::Chart->find_by( description => 'Betriebsbedarf' );
978 $gl_transaction = create_gl_transaction(
979 reference => 'betriebsbedarf',
983 chart => $betriebsbedarf,
989 chart => $betriebsbedarf,
1010 G. Richardson E<lt>grichardson@kivitec.deE<gt>