191 |
191 |
return @trans_ids;
|
192 |
192 |
}
|
193 |
193 |
|
194 |
|
sub transfer_assembly {
|
195 |
|
$main::lxdebug->enter_sub();
|
196 |
|
|
197 |
|
my $self = shift;
|
198 |
|
my %params = @_;
|
199 |
|
Common::check_params(\%params, qw(assembly_id dst_warehouse_id login qty unit dst_bin_id chargenumber bestbefore comment));
|
200 |
|
|
201 |
|
my $myconfig = \%main::myconfig;
|
202 |
|
my $form = $main::form;
|
203 |
|
my $kannNichtFertigen =""; # Falls leer dann erfolgreich
|
204 |
|
|
205 |
|
SL::DB->client->with_transaction(sub {
|
206 |
|
my $dbh = $params{dbh} || SL::DB->client->dbh;
|
207 |
|
|
208 |
|
# Ablauferklärung
|
209 |
|
#
|
210 |
|
# ... Standard-Check oben Ende. Hier die eigentliche SQL-Abfrage
|
211 |
|
# select parts_id,qty from assembly where id=1064;
|
212 |
|
# Erweiterung für bug 935 am 23.4.09 -
|
213 |
|
# Erzeugnisse können Dienstleistungen enthalten, die ja nicht 'lagerbar' sind.
|
214 |
|
# select parts_id,qty from assembly inner join parts on assembly.parts_id = parts.id
|
215 |
|
# where assembly.id=1066 and inventory_accno_id IS NOT NULL;
|
216 |
|
#
|
217 |
|
# Erweiterung für bug 23.4.09 -2 Erzeugnisse in Erzeugnissen können nicht ausgelagert werden,
|
218 |
|
# wenn assembly nicht überprüft wird ...
|
219 |
|
# patch von joachim eingespielt 24.4.2009:
|
220 |
|
# my $query = qq|select parts_id,qty from assembly inner join parts
|
221 |
|
# on assembly.parts_id = parts.id where assembly.id = ? and
|
222 |
|
# (inventory_accno_id IS NOT NULL or parts.assembly = TRUE)|;
|
223 |
|
|
224 |
|
# Lager in dem die Bestandteile gesucht werden kann entweder das Ziellager sein oder ist per Mandantenkonfig
|
225 |
|
# auf das Standardlager des Bestandteiles schaltbar
|
226 |
|
|
227 |
|
my $use_default_warehouse = $::instance_conf->get_transfer_default_warehouse_for_assembly;
|
228 |
|
|
229 |
|
my $query = qq|SELECT assembly.parts_id, assembly.qty, parts.warehouse_id
|
230 |
|
FROM assembly INNER JOIN parts ON assembly.parts_id = parts.id
|
231 |
|
WHERE assembly.id = ? AND parts.part_type != 'service'|;
|
232 |
|
|
233 |
|
my $sth_part_qty_assembly = prepare_execute_query($form, $dbh, $query, $params{assembly_id});
|
234 |
|
|
235 |
|
my @trans_ids;
|
236 |
|
|
237 |
|
# Hier wird das prepared Statement für die Schleife über alle Lagerplätze vorbereitet
|
238 |
|
my $transferPartSQL = qq|INSERT INTO inventory (parts_id, warehouse_id, bin_id, chargenumber, bestbefore, comment, employee_id, qty,
|
239 |
|
trans_id, trans_type_id, shippingdate)
|
240 |
|
VALUES (?, ?, ?, ?, ?, ?, (SELECT id FROM employee WHERE login = ?), ?, ?,
|
241 |
|
(SELECT id FROM transfer_type WHERE direction = 'out' AND description = 'used'),
|
242 |
|
(SELECT current_date))|;
|
243 |
|
my $sthTransferPartSQL = prepare_query($form, $dbh, $transferPartSQL);
|
244 |
|
my $trans_id;
|
245 |
|
|
246 |
|
# der return-string für die fehlermeldung inkl. welche waren zum fertigen noch fehlen
|
247 |
|
|
248 |
|
my $schleife_durchlaufen=0; # Falls die Schleife nicht ausgeführt wird -> Keine Einzelteile definiert. Bessere Idee? jan
|
249 |
|
while (my $hash_ref = $sth_part_qty_assembly->fetchrow_hashref()) { #Schleife für select parts_id,(...) from assembly
|
250 |
|
$schleife_durchlaufen=1; # Erzeugnis definiert
|
251 |
|
|
252 |
|
my $partsQTY = $hash_ref->{qty} * $params{qty}; # benötigte teile * anzahl erzeugnisse
|
253 |
|
my $currentPart_ID = $hash_ref->{parts_id};
|
254 |
|
|
255 |
|
my $currentPart_WH_ID = $use_default_warehouse && $hash_ref->{warehouse_id} ? $hash_ref->{warehouse_id} : $params{dst_warehouse_id};
|
256 |
|
my $no_check = 0;
|
257 |
|
|
258 |
|
# Prüfen ob Erzeugnis-Teile Standardlager haben.
|
259 |
|
if ($use_default_warehouse && ! $hash_ref->{warehouse_id}) {
|
260 |
|
# Prüfen ob in Mandantenkonfiguration ein Standardlager aktiviert isti.
|
261 |
|
if ($::instance_conf->get_transfer_default_ignore_onhand) {
|
262 |
|
$currentPart_WH_ID = $::instance_conf->get_warehouse_id_ignore_onhand;
|
263 |
|
$no_check = 1;
|
264 |
|
} else {
|
265 |
|
$kannNichtFertigen .= "Kein Standardlager: " .
|
266 |
|
" Die Ware " . $self->get_part_description(parts_id => $currentPart_ID) .
|
267 |
|
" hat kein Standardlager definiert " .
|
268 |
|
", um das Erzeugnis herzustellen. <br>";
|
269 |
|
next;
|
270 |
|
}
|
271 |
|
}
|
272 |
|
my $warehouse_info = $self->get_basic_warehouse_info('id'=> $currentPart_WH_ID);
|
273 |
|
my $warehouse_desc = $warehouse_info->{"warehouse_description"};
|
274 |
|
|
275 |
|
# Fertigen ohne Prüfung nach Bestand
|
276 |
|
if ($no_check) {
|
277 |
|
my $temppart_bin_id = $::instance_conf->get_bin_id_ignore_onhand;
|
278 |
|
my $temppart_chargenumber = "";
|
279 |
|
my $temppart_bestbefore = localtime();
|
280 |
|
my $temppart_qty = $partsQTY * -1;
|
281 |
|
($trans_id) = selectrow_query($form, $dbh, qq|SELECT nextval('id')| ) unless $trans_id;
|
282 |
|
|
283 |
|
do_statement($form, $sthTransferPartSQL, $transferPartSQL, $currentPart_ID, $currentPart_WH_ID,
|
284 |
|
$temppart_bin_id, $temppart_chargenumber, $temppart_bestbefore, 'Verbraucht für ' .
|
285 |
|
$self->get_part_description(parts_id => $params{assembly_id}), $params{login}, $temppart_qty, $trans_id);
|
286 |
|
next;
|
287 |
|
}
|
288 |
|
# Überprüfen, ob diese Anzahl gefertigt werden kann
|
289 |
|
my $max_parts = $self->get_max_qty_parts(parts_id => $currentPart_ID, # $self->method() == this.method()
|
290 |
|
warehouse_id => $currentPart_WH_ID);
|
291 |
|
|
292 |
|
if ($partsQTY > $max_parts){
|
293 |
|
# Gibt es hier ein Problem mit nicht "escapten" Zeichen?
|
294 |
|
# 25.4.09 Antwort: Ja. Aber erst wenn im Frontend die locales-Funktion aufgerufen wird
|
295 |
|
|
296 |
|
$kannNichtFertigen .= "Zum Fertigen fehlen: " . abs($partsQTY - $max_parts) .
|
297 |
|
" Einheiten der Ware: " . $self->get_part_description(parts_id => $currentPart_ID) .
|
298 |
|
" im Lager: " . $warehouse_desc .
|
299 |
|
", um das Erzeugnis herzustellen. <br>"; # Konnte die Menge nicht mit der aktuellen Anzahl der Waren fertigen
|
300 |
|
next; # die weiteren Überprüfungen sind unnötig, daher das nächste elemente prüfen (genaue Ausgabe, was noch fehlt)
|
301 |
|
}
|
302 |
|
|
303 |
|
# Eine kurze Vorabfrage, um den Lagerplatz, Chargennummer und die Mindesthaltbarkeit zu bestimmen
|
304 |
|
# Offen: Die Summe über alle Lagerplätze wird noch nicht gebildet
|
305 |
|
# Gelöst: Wir haben vorher schon die Abfrage durchgeführt, ob wir fertigen können.
|
306 |
|
# Noch besser gelöst: Wir laufen durch alle benötigten Waren zum Fertigen und geben eine Rückmeldung an den Benutzer was noch fehlt
|
307 |
|
# und lösen den Rest dann so wie bei xplace im Barcode-Programm
|
308 |
|
# S.a. Kommentar im bin/mozilla-Code mb übernimmt und macht das in ordentlich
|
309 |
|
|
310 |
|
my $tempquery = qq|SELECT SUM(qty), bin_id, chargenumber, bestbefore FROM inventory
|
311 |
|
WHERE warehouse_id = ? AND parts_id = ? GROUP BY bin_id, chargenumber, bestbefore having SUM(qty)>0|;
|
312 |
|
my $tempsth = prepare_execute_query($form, $dbh, $tempquery, $currentPart_WH_ID, $currentPart_ID);
|
313 |
|
|
314 |
|
# Alle Werte zu dem einzelnen Artikel, die wir später auslagern
|
315 |
|
my $tmpPartsQTY = $partsQTY;
|
316 |
|
|
317 |
|
while (my $temphash_ref = $tempsth->fetchrow_hashref()) {
|
318 |
|
my $temppart_bin_id = $temphash_ref->{bin_id}; # kann man hier den quelllagerplatz beim verbauen angeben?
|
319 |
|
my $temppart_chargenumber = $temphash_ref->{chargenumber};
|
320 |
|
my $temppart_bestbefore = conv_date($temphash_ref->{bestbefore});
|
321 |
|
my $temppart_qty = $temphash_ref->{sum};
|
322 |
|
|
323 |
|
($trans_id) = selectrow_query($form, $dbh, qq|SELECT nextval('id')| ) unless $trans_id;
|
324 |
|
if ($tmpPartsQTY > $temppart_qty) { # wir haben noch mehr waren zum wegbuchen.
|
325 |
|
# Wir buchen den kompletten Lagerplatzbestand und zählen die Hilfsvariable runter
|
326 |
|
$tmpPartsQTY = $tmpPartsQTY - $temppart_qty;
|
327 |
|
$temppart_qty = $temppart_qty * -1; # TODO beim analyiseren des sql-trace, war dieser wert positiv,
|
328 |
|
# wenn * -1 als berechnung in der parameter-übergabe angegeben wird.
|
329 |
|
# Dieser Wert IST und BLEIBT positiv!! Hilfe.
|
330 |
|
# Liegt das daran, dass dieser Wert aus einem SQL-Statement stammt?
|
331 |
|
push @trans_ids, $trans_id;
|
332 |
|
do_statement($form, $sthTransferPartSQL, $transferPartSQL, $currentPart_ID, $currentPart_WH_ID,
|
333 |
|
$temppart_bin_id, $temppart_chargenumber, $temppart_bestbefore, 'Verbraucht für ' .
|
334 |
|
$self->get_part_description(parts_id => $params{assembly_id}), $params{login}, $temppart_qty, $trans_id);
|
335 |
|
|
336 |
|
# hier ist noch ein fehler am besten mit definierten erzeugnissen debuggen 02/2009 jb
|
337 |
|
# idee: ausbuch algorithmus mit rekursion lösen und an- und abschaltbar machen
|
338 |
|
# das problem könnte sein, dass strict nicht an war und sth global eine andere zuweisung bekam
|
339 |
|
# auf jeden fall war der internal-server-error nach aktivierung von strict und warnings plus ein paar my-definitionen weg
|
340 |
|
} else { # okay, wir haben weniger oder gleich Waren die wir wegbuchen müssen, wir können also aufhören
|
341 |
|
$tmpPartsQTY *=-1;
|
342 |
|
do_statement($form, $sthTransferPartSQL, $transferPartSQL, $currentPart_ID, $currentPart_WH_ID,
|
343 |
|
$temppart_bin_id, $temppart_chargenumber, $temppart_bestbefore, 'Verbraucht für ' .
|
344 |
|
$self->get_part_description(parts_id => $params{assembly_id}), $params{login}, $tmpPartsQTY, $trans_id);
|
345 |
|
last; # beendet die schleife (springt zum letzten element)
|
346 |
|
}
|
347 |
|
} # ende while SELECT SUM(qty), bin_id, chargenumber, bestbefore FROM inventory WHERE warehouse_id
|
348 |
|
} #ende while select parts_id,qty from assembly where id = ?
|
349 |
|
|
350 |
|
if ($schleife_durchlaufen==0){ # falls die schleife nicht durchlaufen wurde, wurden auch
|
351 |
|
# keine einzelteile definiert
|
352 |
|
$kannNichtFertigen ="Für dieses Erzeugnis sind keine Einzelteile definiert.
|
353 |
|
Dementsprechend kann auch nichts hergestellt werden";
|
354 |
|
}
|
355 |
|
# gibt die Fehlermeldung zurück. A.) Keine Teile definiert
|
356 |
|
# B.) Artikel und Anzahl der fehlenden Teile/Dienstleistungen
|
357 |
|
die "<br><br>" . $kannNichtFertigen if ($kannNichtFertigen);
|
358 |
|
|
359 |
|
# soweit alles gut. Jetzt noch die wirkliche Lagerbewegung für das Erzeugnis ausführen ...
|
360 |
|
($trans_id) = selectrow_query($form, $dbh, qq|SELECT nextval('id')| ) unless $trans_id;
|
361 |
|
my $transferAssemblySQL = qq|INSERT INTO inventory (parts_id, warehouse_id, bin_id, chargenumber, bestbefore,
|
362 |
|
comment, employee_id, qty, trans_id, trans_type_id, shippingdate)
|
363 |
|
VALUES (?, ?, ?, ?, ?, ?, (SELECT id FROM employee WHERE login = ?), ?, ?,
|
364 |
|
(SELECT id FROM transfer_type WHERE direction = 'in' AND description = 'assembled'),
|
365 |
|
(select current_date))|;
|
366 |
|
my $sthTransferAssemblySQL = prepare_query($form, $dbh, $transferAssemblySQL);
|
367 |
|
do_statement($form, $sthTransferAssemblySQL, $transferAssemblySQL, $params{assembly_id}, $params{dst_warehouse_id},
|
368 |
|
$params{dst_bin_id}, $params{chargenumber}, conv_date($params{bestbefore}), $params{comment}, $params{login}, $params{qty}, $trans_id);
|
369 |
|
|
370 |
|
|
371 |
|
1;
|
372 |
|
}) or do { return $kannNichtFertigen };
|
373 |
|
|
374 |
|
$main::lxdebug->leave_sub();
|
375 |
|
return 1; # Alles erfolgreich
|
376 |
|
}
|
377 |
|
|
378 |
194 |
sub get_warehouse_journal {
|
379 |
195 |
$main::lxdebug->enter_sub();
|
380 |
196 |
|
WH::transfer_assembly gegen S/H/Inventory::produce_assembly getauscht
Testfälle i.O., weiteres Feature für das Erzeugnis fertigen möglich
und im Changelog kommentiert. S.a. #429