Projekt

Allgemein

Profil

« Zurück | Weiter » 

Revision 471af251

Von Sven Schöling vor etwa 8 Jahren hinzugefügt

  • ID 471af251c02dc9b72fc44d4de661dba56586a4d5
  • Vorgänger 865ea676
  • Nachfolger 20599de5

WH::transfer_assembly: single-dbh

Unterschiede anzeigen:

SL/WH.pm
163 163

  
164 164
  my $myconfig = \%main::myconfig;
165 165
  my $form     = $main::form;
166
  my $dbh      = $params{dbh} || $form->get_standard_dbh($myconfig);
167

  
168

  
169
  # Ablauferklärung
170
  #
171
  # ... Standard-Check oben Ende. Hier die eigentliche SQL-Abfrage
172
  # select parts_id,qty from assembly where id=1064;
173
  # Erweiterung für bug 935 am 23.4.09 -
174
  # Erzeugnisse können Dienstleistungen enthalten, die ja nicht 'lagerbar' sind.
175
  # select parts_id,qty from assembly inner join parts on assembly.parts_id = parts.id
176
  # where assembly.id=1066 and inventory_accno_id IS NOT NULL;
177
  #
178
  # Erweiterung für bug 23.4.09 -2 Erzeugnisse in Erzeugnissen können nicht ausgelagert werden,
179
  # wenn assembly nicht überprüft wird ...
180
  # patch von joachim eingespielt 24.4.2009:
181
  # my $query    = qq|select parts_id,qty from assembly inner join parts
182
  # on assembly.parts_id = parts.id  where assembly.id = ? and
183
  # (inventory_accno_id IS NOT NULL or parts.assembly = TRUE)|;
184

  
185
  # Lager in dem die Bestandteile gesucht werden kann entweder das Ziellager sein oder ist per Mandantenkonfig
186
  # auf das Standardlager des Bestandteiles schaltbar
187

  
188
  my $use_default_warehouse = $::instance_conf->get_transfer_default_warehouse_for_assembly;
189

  
190
  my $query = qq|SELECT assembly.parts_id, assembly.qty, parts.warehouse_id
191
                 FROM assembly INNER JOIN parts ON assembly.parts_id = parts.id
192
                 WHERE assembly.id = ? AND (inventory_accno_id IS NOT NULL OR parts.assembly = TRUE)|;
193

  
194
  my $sth_part_qty_assembly = prepare_execute_query($form, $dbh, $query, $params{assembly_id});
195

  
196
  # Hier wird das prepared Statement für die Schleife über alle Lagerplätze vorbereitet
197
  my $transferPartSQL = qq|INSERT INTO inventory (parts_id, warehouse_id, bin_id, chargenumber, bestbefore, comment, employee_id, qty,
198
                           trans_id, trans_type_id, shippingdate)
199
                           VALUES (?, ?, ?, ?, ?, ?, (SELECT id FROM employee WHERE login = ?), ?, nextval('id'),
200
                           (SELECT id FROM transfer_type WHERE direction = 'out' AND description = 'used'),
201
                           (SELECT current_date))|;
202
  my $sthTransferPartSQL   = prepare_query($form, $dbh, $transferPartSQL);
203

  
204
  # der return-string für die fehlermeldung inkl. welche waren zum fertigen noch fehlen
205

  
206 166
  my $kannNichtFertigen ="";  # Falls leer dann erfolgreich
207
  my $schleife_durchlaufen=0; # Falls die Schleife nicht ausgeführt wird -> Keine Einzelteile definiert. Bessere Idee? jan
208
  while (my $hash_ref = $sth_part_qty_assembly->fetchrow_hashref()) { #Schleife für select parts_id,(...) from assembly
209
    $schleife_durchlaufen=1;  # Erzeugnis definiert
210

  
211
    my $partsQTY          = $hash_ref->{qty} * $params{qty}; # benötigte teile * anzahl erzeugnisse
212
    my $currentPart_ID    = $hash_ref->{parts_id};
213

  
214
    my $currentPart_WH_ID = $use_default_warehouse && $hash_ref->{warehouse_id} ? $hash_ref->{warehouse_id} : $params{dst_warehouse_id};
215
    my $no_check = 0;
216

  
217
    # Prüfen ob Erzeugnis-Teile Standardlager haben.
218
    if ($use_default_warehouse && ! $hash_ref->{warehouse_id}) {
219
      # Prüfen ob in Mandantenkonfiguration ein Standardlager aktiviert isti.
220
      if ($::instance_conf->get_transfer_default_ignore_onhand) {
221
        $currentPart_WH_ID = $::instance_conf->get_warehouse_id_ignore_onhand;
222
        $no_check = 1;
223
      } else {
224
        $kannNichtFertigen .= "Kein Standardlager: " .
225
                            " Die Ware " . $self->get_part_description(parts_id => $currentPart_ID) .
226
                            " hat kein Standardlager definiert " .
227
                            ", um das Erzeugnis herzustellen. <br>";
228
        next;
167

  
168
  SL::DB->client->with_transaction(sub {
169
    my $dbh      = $params{dbh} || SL::DB->client->dbh;
170

  
171
    # Ablauferklärung
172
    #
173
    # ... Standard-Check oben Ende. Hier die eigentliche SQL-Abfrage
174
    # select parts_id,qty from assembly where id=1064;
175
    # Erweiterung für bug 935 am 23.4.09 -
176
    # Erzeugnisse können Dienstleistungen enthalten, die ja nicht 'lagerbar' sind.
177
    # select parts_id,qty from assembly inner join parts on assembly.parts_id = parts.id
178
    # where assembly.id=1066 and inventory_accno_id IS NOT NULL;
179
    #
180
    # Erweiterung für bug 23.4.09 -2 Erzeugnisse in Erzeugnissen können nicht ausgelagert werden,
181
    # wenn assembly nicht überprüft wird ...
182
    # patch von joachim eingespielt 24.4.2009:
183
    # my $query    = qq|select parts_id,qty from assembly inner join parts
184
    # on assembly.parts_id = parts.id  where assembly.id = ? and
185
    # (inventory_accno_id IS NOT NULL or parts.assembly = TRUE)|;
186

  
187
    # Lager in dem die Bestandteile gesucht werden kann entweder das Ziellager sein oder ist per Mandantenkonfig
188
    # auf das Standardlager des Bestandteiles schaltbar
189

  
190
    my $use_default_warehouse = $::instance_conf->get_transfer_default_warehouse_for_assembly;
191

  
192
    my $query = qq|SELECT assembly.parts_id, assembly.qty, parts.warehouse_id
193
                   FROM assembly INNER JOIN parts ON assembly.parts_id = parts.id
194
                   WHERE assembly.id = ? AND (inventory_accno_id IS NOT NULL OR parts.assembly = TRUE)|;
195

  
196
    my $sth_part_qty_assembly = prepare_execute_query($form, $dbh, $query, $params{assembly_id});
197

  
198
    # Hier wird das prepared Statement für die Schleife über alle Lagerplätze vorbereitet
199
    my $transferPartSQL = qq|INSERT INTO inventory (parts_id, warehouse_id, bin_id, chargenumber, bestbefore, comment, employee_id, qty,
200
                             trans_id, trans_type_id, shippingdate)
201
                             VALUES (?, ?, ?, ?, ?, ?, (SELECT id FROM employee WHERE login = ?), ?, nextval('id'),
202
                             (SELECT id FROM transfer_type WHERE direction = 'out' AND description = 'used'),
203
                             (SELECT current_date))|;
204
    my $sthTransferPartSQL   = prepare_query($form, $dbh, $transferPartSQL);
205

  
206
    # der return-string für die fehlermeldung inkl. welche waren zum fertigen noch fehlen
207

  
208
    my $schleife_durchlaufen=0; # Falls die Schleife nicht ausgeführt wird -> Keine Einzelteile definiert. Bessere Idee? jan
209
    while (my $hash_ref = $sth_part_qty_assembly->fetchrow_hashref()) { #Schleife für select parts_id,(...) from assembly
210
      $schleife_durchlaufen=1;  # Erzeugnis definiert
211

  
212
      my $partsQTY          = $hash_ref->{qty} * $params{qty}; # benötigte teile * anzahl erzeugnisse
213
      my $currentPart_ID    = $hash_ref->{parts_id};
214

  
215
      my $currentPart_WH_ID = $use_default_warehouse && $hash_ref->{warehouse_id} ? $hash_ref->{warehouse_id} : $params{dst_warehouse_id};
216
      my $no_check = 0;
217

  
218
      # Prüfen ob Erzeugnis-Teile Standardlager haben.
219
      if ($use_default_warehouse && ! $hash_ref->{warehouse_id}) {
220
        # Prüfen ob in Mandantenkonfiguration ein Standardlager aktiviert isti.
221
        if ($::instance_conf->get_transfer_default_ignore_onhand) {
222
          $currentPart_WH_ID = $::instance_conf->get_warehouse_id_ignore_onhand;
223
          $no_check = 1;
224
        } else {
225
          $kannNichtFertigen .= "Kein Standardlager: " .
226
                              " Die Ware " . $self->get_part_description(parts_id => $currentPart_ID) .
227
                              " hat kein Standardlager definiert " .
228
                              ", um das Erzeugnis herzustellen. <br>";
229
          next;
230
        }
229 231
      }
230
    }
231
    my $warehouse_info    = $self->get_basic_warehouse_info('id'=> $currentPart_WH_ID);
232
    my $warehouse_desc    = $warehouse_info->{"warehouse_description"};
233

  
234
    # Fertigen ohne Prüfung nach Bestand
235
    if ($no_check) {
236
      my $temppart_bin_id       = $::instance_conf->get_bin_id_ignore_onhand;
237
      my $temppart_chargenumber = "";
238
      my $temppart_bestbefore   = localtime();
239
      my $temppart_qty          = $partsQTY * -1;
240

  
241
      do_statement($form, $sthTransferPartSQL, $transferPartSQL, $currentPart_ID, $currentPart_WH_ID,
242
                     $temppart_bin_id, $temppart_chargenumber, $temppart_bestbefore, 'Verbraucht für ' .
243
                     $self->get_part_description(parts_id => $params{assembly_id}), $params{login}, $temppart_qty);
244
      next;
245
    }
246
    # Überprüfen, ob diese Anzahl gefertigt werden kann
247
    my $max_parts = $self->get_max_qty_parts(parts_id     => $currentPart_ID, # $self->method() == this.method()
248
                                             warehouse_id => $currentPart_WH_ID);
249

  
250
    if ($partsQTY  > $max_parts){
251
      # Gibt es hier ein Problem mit nicht "escapten" Zeichen?
252
      # 25.4.09 Antwort: Ja.  Aber erst wenn im Frontend die locales-Funktion aufgerufen wird
253

  
254
      $kannNichtFertigen .= "Zum Fertigen fehlen: " . abs($partsQTY - $max_parts) .
255
                            " Einheiten der Ware: " . $self->get_part_description(parts_id => $currentPart_ID) .
256
                            " im Lager: " . $warehouse_desc .
257
                            ", um das Erzeugnis herzustellen. <br>"; # Konnte die Menge nicht mit der aktuellen Anzahl der Waren fertigen
258
      next; # die weiteren Überprüfungen sind unnötig, daher das nächste elemente prüfen (genaue Ausgabe, was noch fehlt)
259
    }
232
      my $warehouse_info    = $self->get_basic_warehouse_info('id'=> $currentPart_WH_ID);
233
      my $warehouse_desc    = $warehouse_info->{"warehouse_description"};
234

  
235
      # Fertigen ohne Prüfung nach Bestand
236
      if ($no_check) {
237
        my $temppart_bin_id       = $::instance_conf->get_bin_id_ignore_onhand;
238
        my $temppart_chargenumber = "";
239
        my $temppart_bestbefore   = localtime();
240
        my $temppart_qty          = $partsQTY * -1;
260 241

  
261
    # Eine kurze Vorabfrage, um den Lagerplatz, Chargennummer und die Mindesthaltbarkeit zu bestimmen
262
    # Offen: Die Summe über alle Lagerplätze wird noch nicht gebildet
263
    # Gelöst: Wir haben vorher schon die Abfrage durchgeführt, ob wir fertigen können.
264
    # Noch besser gelöst: Wir laufen durch alle benötigten Waren zum Fertigen und geben eine Rückmeldung an den Benutzer was noch fehlt
265
    # und lösen den Rest dann so wie bei xplace im Barcode-Programm
266
    # S.a. Kommentar im bin/mozilla-Code mb übernimmt und macht das in ordentlich
267

  
268
    my $tempquery = qq|SELECT SUM(qty), bin_id, chargenumber, bestbefore   FROM inventory
269
                       WHERE warehouse_id = ? AND parts_id = ?  GROUP BY bin_id, chargenumber, bestbefore having SUM(qty)>0|;
270
    my $tempsth   = prepare_execute_query($form, $dbh, $tempquery, $currentPart_WH_ID, $currentPart_ID);
271

  
272
    # Alle Werte zu dem einzelnen Artikel, die wir später auslagern
273
    my $tmpPartsQTY = $partsQTY;
274

  
275
    while (my $temphash_ref = $tempsth->fetchrow_hashref()) {
276
      my $temppart_bin_id       = $temphash_ref->{bin_id}; # kann man hier den quelllagerplatz beim verbauen angeben?
277
      my $temppart_chargenumber = $temphash_ref->{chargenumber};
278
      my $temppart_bestbefore   = conv_date($temphash_ref->{bestbefore});
279
      my $temppart_qty          = $temphash_ref->{sum};
280

  
281
      if ($tmpPartsQTY > $temppart_qty) {  # wir haben noch mehr waren zum wegbuchen.
282
                                           # Wir buchen den kompletten Lagerplatzbestand und zählen die Hilfsvariable runter
283
        $tmpPartsQTY = $tmpPartsQTY - $temppart_qty;
284
        $temppart_qty = $temppart_qty * -1; # TODO beim analyiseren des sql-trace, war dieser wert positiv,
285
                                            # wenn * -1 als berechnung in der parameter-übergabe angegeben wird.
286
                                            # Dieser Wert IST und BLEIBT positiv!! Hilfe.
287
                                            # Liegt das daran, dass dieser Wert aus einem SQL-Statement stammt?
288
        do_statement($form, $sthTransferPartSQL, $transferPartSQL, $currentPart_ID, $currentPart_WH_ID,
289
                     $temppart_bin_id, $temppart_chargenumber, $temppart_bestbefore, 'Verbraucht für ' .
290
                     $self->get_part_description(parts_id => $params{assembly_id}), $params{login}, $temppart_qty);
291

  
292
        # hier ist noch ein fehler am besten mit definierten erzeugnissen debuggen 02/2009 jb
293
        # idee: ausbuch algorithmus mit rekursion lösen und an- und abschaltbar machen
294
        # das problem könnte sein, dass strict nicht an war und sth global eine andere zuweisung bekam
295
        # auf jeden fall war der internal-server-error nach aktivierung von strict und warnings plus ein paar my-definitionen weg
296
      } else { # okay, wir haben weniger oder gleich Waren die wir wegbuchen müssen, wir können also aufhören
297
        $tmpPartsQTY *=-1;
298 242
        do_statement($form, $sthTransferPartSQL, $transferPartSQL, $currentPart_ID, $currentPart_WH_ID,
299
                     $temppart_bin_id, $temppart_chargenumber, $temppart_bestbefore, 'Verbraucht für ' .
300
                     $self->get_part_description(parts_id => $params{assembly_id}), $params{login}, $tmpPartsQTY);
301
        last; # beendet die schleife (springt zum letzten element)
243
                       $temppart_bin_id, $temppart_chargenumber, $temppart_bestbefore, 'Verbraucht für ' .
244
                       $self->get_part_description(parts_id => $params{assembly_id}), $params{login}, $temppart_qty);
245
        next;
246
      }
247
      # Überprüfen, ob diese Anzahl gefertigt werden kann
248
      my $max_parts = $self->get_max_qty_parts(parts_id     => $currentPart_ID, # $self->method() == this.method()
249
                                               warehouse_id => $currentPart_WH_ID);
250

  
251
      if ($partsQTY  > $max_parts){
252
        # Gibt es hier ein Problem mit nicht "escapten" Zeichen?
253
        # 25.4.09 Antwort: Ja.  Aber erst wenn im Frontend die locales-Funktion aufgerufen wird
254

  
255
        $kannNichtFertigen .= "Zum Fertigen fehlen: " . abs($partsQTY - $max_parts) .
256
                              " Einheiten der Ware: " . $self->get_part_description(parts_id => $currentPart_ID) .
257
                              " im Lager: " . $warehouse_desc .
258
                              ", um das Erzeugnis herzustellen. <br>"; # Konnte die Menge nicht mit der aktuellen Anzahl der Waren fertigen
259
        next; # die weiteren Überprüfungen sind unnötig, daher das nächste elemente prüfen (genaue Ausgabe, was noch fehlt)
302 260
      }
303
    }  # ende while SELECT SUM(qty), bin_id, chargenumber, bestbefore   FROM inventory  WHERE warehouse_id
304
  } #ende while select parts_id,qty from assembly where id = ?
305

  
306
  if ($schleife_durchlaufen==0){  # falls die schleife nicht durchlaufen wurde, wurden auch
307
                                  # keine einzelteile definiert
308
      $kannNichtFertigen ="Für dieses Erzeugnis sind keine Einzelteile definiert.
309
                           Dementsprechend kann auch nichts hergestellt werden";
310
 }
311
  # gibt die Fehlermeldung zurück. A.) Keine Teile definiert
312
  #                                B.) Artikel und Anzahl der fehlenden Teile/Dienstleistungen
313
  if ($kannNichtFertigen) {
314
    return $kannNichtFertigen;
315
  }
316 261

  
317
  # soweit alles gut. Jetzt noch die wirkliche Lagerbewegung für das Erzeugnis ausführen ...
318
  my $transferAssemblySQL = qq|INSERT INTO inventory (parts_id, warehouse_id, bin_id, chargenumber, bestbefore,
319
                                                      comment, employee_id, qty, trans_id, trans_type_id, shippingdate)
320
                               VALUES (?, ?, ?, ?, ?, ?, (SELECT id FROM employee WHERE login = ?), ?, nextval('id'),
321
                               (SELECT id FROM transfer_type WHERE direction = 'in' AND description = 'assembled'),
322
                               (select current_date))|;
323
  my $sthTransferAssemblySQL   = prepare_query($form, $dbh, $transferAssemblySQL);
324
  do_statement($form, $sthTransferAssemblySQL, $transferAssemblySQL, $params{assembly_id}, $params{dst_warehouse_id},
325
               $params{dst_bin_id}, $params{chargenumber}, conv_date($params{bestbefore}), $params{comment}, $params{login}, $params{qty});
326
  $dbh->commit();
262
      # Eine kurze Vorabfrage, um den Lagerplatz, Chargennummer und die Mindesthaltbarkeit zu bestimmen
263
      # Offen: Die Summe über alle Lagerplätze wird noch nicht gebildet
264
      # Gelöst: Wir haben vorher schon die Abfrage durchgeführt, ob wir fertigen können.
265
      # Noch besser gelöst: Wir laufen durch alle benötigten Waren zum Fertigen und geben eine Rückmeldung an den Benutzer was noch fehlt
266
      # und lösen den Rest dann so wie bei xplace im Barcode-Programm
267
      # S.a. Kommentar im bin/mozilla-Code mb übernimmt und macht das in ordentlich
268

  
269
      my $tempquery = qq|SELECT SUM(qty), bin_id, chargenumber, bestbefore   FROM inventory
270
                         WHERE warehouse_id = ? AND parts_id = ?  GROUP BY bin_id, chargenumber, bestbefore having SUM(qty)>0|;
271
      my $tempsth   = prepare_execute_query($form, $dbh, $tempquery, $currentPart_WH_ID, $currentPart_ID);
272

  
273
      # Alle Werte zu dem einzelnen Artikel, die wir später auslagern
274
      my $tmpPartsQTY = $partsQTY;
275

  
276
      while (my $temphash_ref = $tempsth->fetchrow_hashref()) {
277
        my $temppart_bin_id       = $temphash_ref->{bin_id}; # kann man hier den quelllagerplatz beim verbauen angeben?
278
        my $temppart_chargenumber = $temphash_ref->{chargenumber};
279
        my $temppart_bestbefore   = conv_date($temphash_ref->{bestbefore});
280
        my $temppart_qty          = $temphash_ref->{sum};
281

  
282
        if ($tmpPartsQTY > $temppart_qty) {  # wir haben noch mehr waren zum wegbuchen.
283
                                             # Wir buchen den kompletten Lagerplatzbestand und zählen die Hilfsvariable runter
284
          $tmpPartsQTY = $tmpPartsQTY - $temppart_qty;
285
          $temppart_qty = $temppart_qty * -1; # TODO beim analyiseren des sql-trace, war dieser wert positiv,
286
                                              # wenn * -1 als berechnung in der parameter-übergabe angegeben wird.
287
                                              # Dieser Wert IST und BLEIBT positiv!! Hilfe.
288
                                              # Liegt das daran, dass dieser Wert aus einem SQL-Statement stammt?
289
          do_statement($form, $sthTransferPartSQL, $transferPartSQL, $currentPart_ID, $currentPart_WH_ID,
290
                       $temppart_bin_id, $temppart_chargenumber, $temppart_bestbefore, 'Verbraucht für ' .
291
                       $self->get_part_description(parts_id => $params{assembly_id}), $params{login}, $temppart_qty);
292

  
293
          # hier ist noch ein fehler am besten mit definierten erzeugnissen debuggen 02/2009 jb
294
          # idee: ausbuch algorithmus mit rekursion lösen und an- und abschaltbar machen
295
          # das problem könnte sein, dass strict nicht an war und sth global eine andere zuweisung bekam
296
          # auf jeden fall war der internal-server-error nach aktivierung von strict und warnings plus ein paar my-definitionen weg
297
        } else { # okay, wir haben weniger oder gleich Waren die wir wegbuchen müssen, wir können also aufhören
298
          $tmpPartsQTY *=-1;
299
          do_statement($form, $sthTransferPartSQL, $transferPartSQL, $currentPart_ID, $currentPart_WH_ID,
300
                       $temppart_bin_id, $temppart_chargenumber, $temppart_bestbefore, 'Verbraucht für ' .
301
                       $self->get_part_description(parts_id => $params{assembly_id}), $params{login}, $tmpPartsQTY);
302
          last; # beendet die schleife (springt zum letzten element)
303
        }
304
      }  # ende while SELECT SUM(qty), bin_id, chargenumber, bestbefore   FROM inventory  WHERE warehouse_id
305
    } #ende while select parts_id,qty from assembly where id = ?
306

  
307
    if ($schleife_durchlaufen==0){  # falls die schleife nicht durchlaufen wurde, wurden auch
308
                                    # keine einzelteile definiert
309
        $kannNichtFertigen ="Für dieses Erzeugnis sind keine Einzelteile definiert.
310
                             Dementsprechend kann auch nichts hergestellt werden";
311
   }
312
    # gibt die Fehlermeldung zurück. A.) Keine Teile definiert
313
    #                                B.) Artikel und Anzahl der fehlenden Teile/Dienstleistungen
314
    if ($kannNichtFertigen) {
315
      return 0;
316
    }
317

  
318
    # soweit alles gut. Jetzt noch die wirkliche Lagerbewegung für das Erzeugnis ausführen ...
319
    my $transferAssemblySQL = qq|INSERT INTO inventory (parts_id, warehouse_id, bin_id, chargenumber, bestbefore,
320
                                                        comment, employee_id, qty, trans_id, trans_type_id, shippingdate)
321
                                 VALUES (?, ?, ?, ?, ?, ?, (SELECT id FROM employee WHERE login = ?), ?, nextval('id'),
322
                                 (SELECT id FROM transfer_type WHERE direction = 'in' AND description = 'assembled'),
323
                                 (select current_date))|;
324
    my $sthTransferAssemblySQL   = prepare_query($form, $dbh, $transferAssemblySQL);
325
    do_statement($form, $sthTransferAssemblySQL, $transferAssemblySQL, $params{assembly_id}, $params{dst_warehouse_id},
326
                 $params{dst_bin_id}, $params{chargenumber}, conv_date($params{bestbefore}), $params{comment}, $params{login}, $params{qty});
327
    1;
328
  }) or do { return $kannNichtFertigen };
327 329

  
328 330
  $main::lxdebug->leave_sub();
329 331
  return 1; # Alles erfolgreich

Auch abrufbar als: Unified diff