From acb42de61cf3633fa935dac6135d5c9e18c05544 Mon Sep 17 00:00:00 2001
From: Patrick Barroca <pbarroca@sandbox.pergame.net>
Date: Thu, 28 Jan 2016 17:47:54 +0100
Subject: [PATCH] rel #37078 : generation screen to cosmozend work in progress

---
 .../controllers/IntegrationControllerTest.php | 203 ++++++++-
 library/Class/Cosmogramme/Generator.php       | 404 ++++++++++++++++--
 .../Class/Cosmogramme/LandingDirectory.php    |  24 +-
 library/Class/Log.php                         |   7 +
 4 files changed, 604 insertions(+), 34 deletions(-)

diff --git a/cosmogramme/cosmozend/tests/application/modules/cosmo/controllers/IntegrationControllerTest.php b/cosmogramme/cosmozend/tests/application/modules/cosmo/controllers/IntegrationControllerTest.php
index 167f56930ef..ad48675d98b 100644
--- a/cosmogramme/cosmozend/tests/application/modules/cosmo/controllers/IntegrationControllerTest.php
+++ b/cosmogramme/cosmozend/tests/application/modules/cosmo/controllers/IntegrationControllerTest.php
@@ -203,16 +203,40 @@ abstract class Cosmo_IntegrationControllerGenerateActionTestCase extends CosmoCo
                         ->with('../ftp/transfert/library1/etalon/sections.txt')
                         ->answers(true)
 
+                        ->whenCalled('is_readable')
+                        ->with('../ftp/transfert/library1/etalon/emplacements.txt')
+                        ->answers(true)
+
                         ->whenCalled('file')
                         ->with('../ftp/transfert/library1/etalon/annexes.txt')
                         ->answers(['BIB_SPS_UTT|ID_SITE|LIBELLE',
                                    '20|library1',
-                                   '12|Caden'])
+                                   '12|library2'])
+
+                        ->whenCalled('file')
+                        ->with('../ftp/transfert/library1/etalon/sections.txt')
+                        ->answers(['BIB_C_SECTION|CODE|LIBELLE',
+                                   '1|Adulte',
+                                   '2|Jeunesse'])
+
+                        ->whenCalled('file')
+                        ->with('../ftp/transfert/library1/etalon/emplacements.txt')
+                        ->answers(['BIB_C_EMPLACEMENT|CODE|LIBELLE',
+                                   '4|Coin des tout-petits',
+                                   '5|Livres CD',
+                                   '6|Bacs imagiers',
+                                   '7|Albums'])
 
                         ->beStrict();
 
     Class_Cosmogramme_LandingDirectory::setFileSystem($file_system);
   }
+
+
+  public function tearDown() {
+    Class_Log::getInstance()->reset();
+    parent::tearDown();
+  }
 }
 
 
@@ -274,7 +298,9 @@ class Cosmo_IntegrationControllerGenerateActionTest extends Cosmo_IntegrationCon
 
 
 
-class Cosmo_IntegrationControllerGenerateActionPostTest extends Cosmo_IntegrationControllerGenerateActionTestCase {
+class Cosmo_IntegrationControllerGenerateActionNanookPostTest
+  extends Cosmo_IntegrationControllerGenerateActionTestCase {
+
   public function setUp() {
     parent::setUp();
 
@@ -288,14 +314,25 @@ class Cosmo_IntegrationControllerGenerateActionPostTest extends Cosmo_Integratio
                     'libelle' => 'Abonné Nanook',
                     'type_fichier' => 1]);
 
+    Class_Cosmogramme_Generator::setTimeSource($this->mock()
+                                               ->whenCalled('time')->answers(strtotime('2016-01-28'))
+                                               ->beStrict());
+
     $this->postDispatch('cosmo/integration/generate',
                         ['path_ftp' => 'library1',
                          'type_sigb' => 13,
+                         'service_nanook' => 'http://nanook-ws.net/ilsdi/',
                          'creer_annexes' => 1,
                          'synchro' => 1]);
   }
 
 
+  public function tearDown() {
+    Class_Cosmogramme_Generator::setTimeSource(null);
+    parent::tearDown();
+  }
+
+
   /** @test */
   public function shouldNotContainsForm() {
     $this->assertNotXPath('//form');
@@ -315,7 +352,165 @@ class Cosmo_IntegrationControllerGenerateActionPostTest extends Cosmo_Integratio
 
 
   /** @test */
-  public function shouldCreateLibrary1() {
-    $this->assertNotNull(Class_Bib::findFirstBy(['id_site'=> 20]));
+  public function shouldHaveCreated2Libraries() {
+    $this->assertEquals(2, Class_Bib::count());
+  }
+
+
+  /** @test */
+  public function library1ShouldBeCreated() {
+    $this->assertNotNull(Class_Bib::findFirstBy(['id_site' => 20,
+                                                 'libelle' => 'library1',
+                                                 'ville' => 'library1',
+                                                 'id_zone' => 1,
+                                                 'visibilite' => Class_Bib::V_DATA]),
+                         $this->_response->getBody());
+  }
+
+
+  /** @test */
+  public function library1IntegrationShouldBeCreated() {
+    $this->assertNotNull($bib = Class_IntBib::findFirstBy(['id_bib' => 20,
+                                                           'nom' => 'library1',
+                                                           'nom_court' => 'library1',
+                                                           'qualite' => 5,
+                                                           'sigb' => Class_IntBib::SIGB_NANOOK,
+                                                           'planif_mode' => 'r',
+                                                           'planif_jours' => '1111111',
+                                                           'comm_sigb' => Class_IntBib::COM_NANOOK]),
+                         $this->_response->getBody());
+    return $bib;
+  }
+
+
+  /**
+   * @test
+   * @depends library1IntegrationShouldBeCreated
+   */
+  public function library1IntegrationCommParamsShouldContainsNanookWebservice($bib) {
+    $this->assertContains('http://nanook-ws.net/ilsdi/', $bib->getCommParams());
+  }
+
+
+  /** @test */
+  public function library1BranchShouldBeCreated() {
+    $this->assertNotNull(Class_CodifAnnexe::findFirstBy(['id_bib' => 20,
+                                                         'code' => 20,
+                                                         'libelle' => 'library1',
+                                                         'invisible' => 0]),
+                         $this->_response->getBody());
+  }
+
+
+  /** @test */
+  public function barCodeProfileShouldBeCreated() {
+    $this->assertNotNull($profile = Class_IntProfilDonnees::findFirstBy(['libelle' => 'Liste de codes-barres']));
+    return $profile;
+  }
+
+
+  /**
+   * @test
+   * @depends barCodeProfileShouldBeCreated
+   */
+  public function library1ShouldHeveItemDeletionPlanned($profile) {
+    $this->assertNotNull(Class_IntMajAuto::findFirstBy(['id_bib' => 20,
+                                                        'profil' => $profile->getId(),
+                                                        'type_operation' => Class_IntMajAuto::OP_ITEMS_DELETION,
+                                                        'nom_fichier' => '../ftp/transfert/library1/site20/suppressions.txt']));
+  }
+
+
+  /** @test */
+  public function library1ShouldHaveRecordsTotalPlanned() {
+    $this->assertNotNull(Class_IntMajAuto::findFirstBy(['id_bib' => 20,
+                                                        'profil' => 104,
+                                                        'type_operation' => Class_IntMajAuto::OP_FULL_IMPORT,
+                                                        'nom_fichier' => '../ftp/transfert/library1/site20/notices_total.txt']));
+  }
+
+
+  /** @test */
+  public function library1ShouldHaveRecordsPartialPlanned() {
+    $this->assertNotNull(Class_IntMajAuto::findFirstBy(['id_bib' => 20,
+                                                        'profil' => 104,
+                                                        'type_operation' => Class_IntMajAuto::OP_PARTIAL_IMPORT,
+                                                        'nom_fichier' => '../ftp/transfert/library1/site20/notices.txt']));
+  }
+
+
+  /** @test */
+  public function library1ShouldHaveUsersPlanned() {
+    $this->assertNotNull(Class_IntMajAuto::findFirstBy(['id_bib' => 20,
+                                                        'profil' => 105,
+                                                        'type_operation' => Class_IntMajAuto::OP_FULL_IMPORT,
+                                                        'nom_fichier' => '../ftp/transfert/library1/site20/abonnes.txt']));
+  }
+
+
+  /** @test */
+  public function library1ShouldHaveLoansPlanned() {
+    $this->assertNotNull(Class_IntMajAuto::findFirstBy(['id_bib' => 20,
+                                                        'profil' => 102,
+                                                        'type_operation' => Class_IntMajAuto::OP_FULL_IMPORT,
+                                                        'nom_fichier' => '../ftp/transfert/library1/site20/prets.txt']));
+  }
+
+
+  /** @test */
+  public function library1ShouldNotHaveHoldsPlanned() {
+    $this->assertNull(Class_IntMajAuto::findFirstBy(['id_bib' => 20,
+                                                     'profil' => 103,
+                                                     'type_operation' => Class_IntMajAuto::OP_FULL_IMPORT,
+                                                     'nom_fichier' => '../ftp/transfert/library1/site20/reservations.txt']));
+  }
+
+
+  /** @test */
+  public function basketsProfileShouldBeCreated() {
+    $profile = Class_IntProfilDonnees::findFirstBy(['accents' => Class_IntProfilDonnees::ENCODING_UTF8,
+                                                    'type_fichier' => Class_IntProfilDonnees::FT_BASKETS,
+                                                    'format' => Class_IntProfilDonnees::FORMAT_PIPED_ASCII]);
+    $this->assertNotNull($profile);
+    return $profile;
+  }
+
+
+
+  /**
+   * @test
+   * @depends basketsProfileShouldBeCreated
+   */
+  public function library1ShouldHaveBasketsFullPlanned($profile) {
+    $this->assertNotNull(Class_IntMajAuto::findFirstBy(['id_bib' => 20,
+                                                        'profil' => $profile->getId(),
+                                                        'type_operation' => Class_IntMajAuto::OP_FULL_IMPORT,
+                                                        'nom_fichier' => '../ftp/transfert/library1/site20/paniers_total.txt']));
+  }
+
+
+  /**
+   * @test
+   * @depends basketsProfileShouldBeCreated
+   */
+  public function library1ShouldHaveBasketsPartialPlanned($profile) {
+    $this->assertNotNull(Class_IntMajAuto::findFirstBy(['id_bib' => 20,
+                                                        'profil' => $profile->getId(),
+                                                        'type_operation' => Class_IntMajAuto::OP_PARTIAL_IMPORT,
+                                                        'nom_fichier' => '../ftp/transfert/library1/site20/paniers.txt']));
+  }
+
+
+  /** @test */
+  public function adulteSectionShouldBeCreated() {
+    $this->assertNotNull(Class_CodifSection::findFirstBy(['libelle' => 'Adulte',
+                                                          'regles' => '995$9=1']));
+  }
+
+
+  /** @test */
+  public function coinLocationShouldBeCreated() {
+    $this->assertNotNull(Class_CodifEmplacement::findFirstBy(['libelle' => 'Coin des tout-petits',
+                                                              'regles' => '995$6=4']));
   }
 }
\ No newline at end of file
diff --git a/library/Class/Cosmogramme/Generator.php b/library/Class/Cosmogramme/Generator.php
index 18456865402..51696735b65 100644
--- a/library/Class/Cosmogramme/Generator.php
+++ b/library/Class/Cosmogramme/Generator.php
@@ -21,7 +21,7 @@
 
 
 class Class_Cosmogramme_Generator {
-  use Trait_Translator, Trait_Loggable;
+  use Trait_Translator, Trait_Loggable, Trait_TimeSource;
 
   protected $_params = [], $_landing_directory;
 
@@ -33,8 +33,9 @@ class Class_Cosmogramme_Generator {
     if (!$this->isValid())
       return false;
 
-    $this->_generateLibraries();
-    return true;
+    return $this->_generateLibraries()
+      && $this->_generateSections()
+      && $this->_generateLocations();
   }
 
 
@@ -66,25 +67,31 @@ class Class_Cosmogramme_Generator {
 
 
   protected function _generateLibraries() {
-    $this->log('<h3>' . $this->_('1 - Création des bibliothèques') . '</h3>');
+    if (!$libraries = $this->_landing_directory->getLibrariesOf($this->_params['path_ftp'])) {
+      $this->log($this->_('Étalon des bibliothèques vide'));
+      return false;
+    }
+
+    $this->_generateLibrariesEntities($libraries);
+    $this->_generateLibrariesBranches($libraries);
+    $this->_generatePlannedIntegrations($libraries);
 
+    return true;
+  }
+
+
+  protected function _generateLibrariesEntities($libraries) {
+    $this->logTitle($this->_('1 - Création des bibliothèques'));
     $html = '';
-    foreach ($this->_landing_directory->getLibrariesOf($this->_params['path_ftp']) as $data)
+    foreach ($libraries as $data)
       $html .= $this->_generateLibrary($data);
-
     $this->log('<div class="liste"><table class="blank">' . $html . '</table></div>');
   }
 
 
   protected function _generateLibrary($data) {
-    xdebug_break();
-    if (!$data)
-      return;
-
-		$data = utf8_encode($data);
-		$elem = explode('|', $data);
-		if ($elem[0] == 'BIB_SPS_UTT')
-      return;
+    if (!$elem = $this->_extractLibraryData($data))
+      return '';
 
     if (!$bib = Class_Bib::findFirstBy(['id_site' => $elem[0]]))
       $bib = new Class_Bib();
@@ -93,7 +100,7 @@ class Class_Cosmogramme_Generator {
                             'libelle' => trim($elem[1]),
                             'ville' => $this->_params['path_ftp'],
                             'id_zone' => 1,
-                            'visibilite' => 2])
+                            'visibilite' => Class_Bib::V_DATA])
         ->save();
 
     Class_IntBib::deleteBy(['id_bib' => $bib->getIdSite()]);
@@ -120,21 +127,178 @@ class Class_Cosmogramme_Generator {
   }
 
 
-  protected function _getBarCodeProfile() {
-    $label = 'Liste de codes-barres';
-    if ($profile = Class_IntProfilDonnees::findFirstBy(['libelle' => $label]))
-      return $profile;
+  protected function _generateLibrariesBranches($libraries) {
+    $this->logTitle($this->_('2 - Création des annexes'));
+    if ('1' != $this->_params['creer_annexes']) {
+      $this->log($this->_('Non demandée'));
+      return;
+    }
+
+    $deleted = Class_CodifAnnexe::deleteBy([]);
+    $html = '<tr><td class="blank" colspan="2">' . $this->_('%d annexe(s) supprimée(s)', $deleted) . '</td></tr>';
+    foreach ($libraries as $data)
+      $html .= $this->_generateBranch($data);
+    $this->log('<div class="liste"><table class="blank">' . $html . '</table></div>');
+  }
 
 
-    $attributs='a:6:{i:0;a:7:{s:8:"type_doc";a:11:{i:0;a:3:{s:4:"code";s:1:"0";s:5:"label";s:0:"";s:8:"zone_995";s:0:"";}i:1;a:3:{s:4:"code";s:1:"1";s:5:"label";s:5:"am;na";s:8:"zone_995";s:0:"";}i:2;a:3:{s:4:"code";s:1:"2";s:5:"label";s:2:"as";s:8:"zone_995";s:0:"";}i:3;a:3:{s:4:"code";s:1:"3";s:5:"label";s:3:"i;j";s:8:"zone_995";s:0:"";}i:4;a:3:{s:4:"code";s:1:"4";s:5:"label";s:1:"g";s:8:"zone_995";s:0:"";}i:5;a:3:{s:4:"code";s:1:"5";s:5:"label";s:3:"l;m";s:8:"zone_995";s:0:"";}i:6;a:3:{s:4:"code";s:1:"8";s:5:"label";s:0:"";s:8:"zone_995";s:0:"";}i:7;a:3:{s:4:"code";s:1:"9";s:5:"label";s:0:"";s:8:"zone_995";s:0:"";}i:8;a:3:{s:4:"code";s:2:"10";s:5:"label";s:0:"";s:8:"zone_995";s:0:"";}i:9;a:3:{s:4:"code";s:3:"100";s:5:"label";s:0:"";s:8:"zone_995";s:0:"";}i:10;a:3:{s:4:"code";s:2:"10";s:5:"label";s:0:"";s:8:"zone_995";s:0:"";}}s:17:"champ_code_barres";s:1:"f";s:10:"champ_cote";s:1:"k";s:11:"champ_genre";s:0:"";s:13:"champ_section";s:1:"q";s:17:"champ_emplacement";s:1:"u";s:12:"champ_annexe";s:1:"a";}i:1;a:1:{s:6:"champs";s:11:"code_barres";}i:2;a:1:{s:6:"champs";s:11:"code_barres";}i:3;a:1:{s:6:"champs";s:11:"code_barres";}i:5;a:3:{s:6:"champs";s:11:"code_barres";s:17:"xml_balise_abonne";s:0:"";s:17:"xml_champs_abonne";a:10:{s:6:"IDABON";s:0:"";s:9:"ORDREABON";s:0:"";s:3:"NOM";s:0:"";s:6:"PRENOM";s:0:"";s:9:"NAISSANCE";s:0:"";s:8:"PASSWORD";s:0:"";s:4:"MAIL";s:0:"";s:10:"DATE_DEBUT";s:0:"";s:8:"DATE_FIN";s:0:"";s:7:"ID_SIGB";s:0:"";}}i:4;a:5:{s:4:"zone";s:0:"";s:5:"champ";s:0:"";s:6:"format";s:0:"";s:5:"jours";s:0:"";s:7:"valeurs";s:0:"";}}';
+  protected function _generateBranch($data) {
+    if (!$elem = $this->_extractLibraryData($data))
+      return '';
 
-    $profile = Class_IntProfilDonnees::newInstance(['libelle' => $label,
-                                                    'accents' => Class_IntProfilDonnees::ENCODING_UTF8,
-                                                    'type_fichier' => Class_IntProfilDonnees::FT_RECORDS,
-                                                    'format' => Class_IntProfilDonnees::FORMAT_TABBED_ASCII,
-                                                    'attributs' => $attributs]);
-    $profile->save();
-    return $profile;
+    $branch = Class_CodifAnnexe::newInstance(['id_bib' => $elem[0],
+                                              'code' => $elem[0],
+                                              'libelle' => trim($elem[1]),
+                                              'invisible' => 0]);
+    $branch->save();
+
+    return '<tr><td class="blank">Site n° '. $branch->getIdBib() .'</td><td class="blank">'. $branch->getLibelle() . '</td></tr>';
+  }
+
+
+  protected function _generatePlannedIntegrations($libraries) {
+    $this->logTitle($this->_('3 - Programmation des intégrations'));
+    $html = '';
+    foreach ($libraries as $data)
+      $html .= $this->_generatePlannedIntegration($data);
+    $this->log('<div class="liste"><table class="blank">' . $html . '</table></div>');
+  }
+
+
+  protected function _generatePlannedIntegration($data) {
+    if (!$elem = $this->_extractLibraryData($data))
+      return '';
+
+    $id_bib = $elem[0];
+    Class_IntMajAuto::deleteBy(['id_bib' => $id_bib]);
+    $base_path = $this->_landing_directory->getSubdirPath($this->_params['path_ftp']);
+    return $this->getPlannedGenerator()->plan($id_bib, $base_path);
+  }
+
+
+  protected function getPlannedGenerator() {
+    return (Class_IntBib::SIGB_NANOOK == $this->_params['type_sigb'])
+      ? new Class_Cosmogramme_GeneratorPlannedNanook()
+      : new Class_Cosmogramme_GeneratorPlannedPergame();
+  }
+
+
+  protected function logTitle($title) {
+    $this->log('<h3>' . $title . '</h3>');
+  }
+
+
+  protected function _extractLibraryData($data) {
+    $data = utf8_encode($data);
+		$elem = explode('|', $data);
+
+		return ($elem[0] == 'BIB_SPS_UTT') ? [] : $elem;
+  }
+
+
+  protected function _generateSections() {
+    if (!$sections = $this->_landing_directory->getSectionsOf($this->_params['path_ftp'])) {
+      $this->log($this->_('Étalon des sections vide'));
+      return false;
+    }
+
+    $this->logTitle($this->_('4 - Création des sections'));
+    $date = $this->getCurrentDate();
+
+    $html = '';
+    foreach($sections as $data)
+      $html .= $this->_generateSection($data, $date);
+
+    $this->log('<div class="liste"><table class="blank">' . $html . '</table></div>');
+
+    $this
+      ->_directQuery('delete from codif_section where date_maj != "' . $date. '"')
+      ->_directQuery('update codif_section set date_maj=""');
+
+    return true;
+  }
+
+
+  protected function _generateSection($data, $date) {
+    $data = utf8_encode($data);
+    $elems = explode('|', $data);
+    if ('BIB_C_SECTION' == $elems[0] || !trim($elems[0]))
+      return '';
+
+    $label = trim($elems[1]);
+    if (!$section = Class_CodifSection::findFirstBy(['libelle' => $label]))
+      $section = Class_CodifSection::newInstance(['libelle' => $label,
+                                                  'date_maj' => $date,
+                                                  'regles' => $this->_getSectionRuleFor($elems[0])]);
+
+    if (!$section->isNew() && $section->getDateMaj() == $date)
+      $section->setRegles($section->getRegles() . ';' . strtolower($elems[0]));
+
+    $section->save();
+
+    return '<tr><td class="blank">' . $elems[0] . '</td><td class="blank">' . $elems[1] . '</td></tr>';
+  }
+
+
+  protected function _getSectionRuleFor($code) {
+    $field = null;
+    if (Class_IntBib::SIGB_NANOOK == $this->_params['type_sigb'])
+      $field = '9';
+    if (Class_IntBib::SIGB_PERGAME == $this->_params['type_sigb'])
+      $field = 'q';
+
+    if ($field)
+      return '995$' . $field . '=' . strtolower($code);
+  }
+
+
+  protected function _generateLocations() {
+    if (!$locations = $this->_landing_directory->getLocationsOf($this->_params['path_ftp'])) {
+      $this->log($this->_('Étalon des emplacements vide'));
+      return false;
+    }
+
+    $this->logTitle($this->_('5 - Création des emplacements'));
+    $date = $this->getCurrentDate();
+
+    $html = '';
+    foreach($locations as $data)
+      $html .= $this->_generateLocation($data, $date);
+
+    $this->log('<div class="liste"><table class="blank">' . $html . '</table></div>');
+
+    $this
+      ->_directQuery('delete from codif_emplacement where date_maj !="' . $date . '"')
+      ->_directQuery('update codif_emplacement set date_maj =""');
+
+    return true;
+  }
+
+
+  protected function _generateLocation($data, $date) {
+    $data = utf8_encode($data);
+    $elems = explode('|', $data);
+    if ('BIB_' == substr($elems[0], 0, 4) || !trim($elems[0]))
+      return '';
+
+    $label = trim($elems[1]);
+    if (!$location = Class_CodifEmplacement::findFirstBy(['libelle' => $label]))
+      $location = Class_CodifEmplacement::newInstance(['libelle' => $label,
+                                                       'date_maj' => $date,
+                                                       'regles' => '995$6=' . strtolower($elems[0])]);
+
+    if (!$location->isNew() && $location->getDateMaj() == $date)
+      $location->setRegles($location->getRegles() . ';' . strtolower($elems[0]));
+
+    $location->save();
+
+    return '<tr><td class="blank">' . $elems[0] . '</td><td class="blank">' . $elems[1] . '</td></tr>';
+  }
+
+
+  protected function _directQuery($sql) {
+    Zend_Db_Table::getDefaultAdapter()->query($sql);
+    return $this;
   }
 }
 
@@ -166,4 +330,190 @@ class Class_Cosmogramme_GeneratorProfilesNanookValidator {
 
     return true;
   }
+}
+
+
+
+class Class_Cosmogramme_GeneratorPlannedAbstract {
+  use Trait_Translator, Trait_Loggable;
+
+  protected
+    $_id_bib,
+    $_id_prog,
+    $_base_path,
+    $_html = '',
+    $_records_profile,
+    $_users_profile,
+    $_loans_profile = 102,
+    $_holds_profile = 103,
+    $_item_delete = false,
+    $_holds = false,
+    $_baskets = false;
+
+  public function plan($id_bib, $base_path) {
+    $this->_id_bib = $id_bib;
+    $this->_id_prog = $id_bib * 100;
+    $this->_base_path = $base_path . DIRECTORY_SEPARATOR . 'site' . $id_bib . DIRECTORY_SEPARATOR;
+
+    $this
+      ->itemDeletion()
+      ->recordsTotal()
+      ->recordsPartial()
+      ->users()
+      ->loans()
+      ->holds()
+      ->basketsTotal()
+      ->basketsPartial();
+
+    return $this->_html;
+  }
+
+
+  protected function itemDeletion() {
+    if (!$this->_item_delete)
+      return $this;
+
+    $profile = $this->_getBarCodeProfile();
+
+    return $this->planWith('Notices - suppression d\'exemplaires',
+                           $profile->getId(),
+                           Class_IntMajAuto::OP_ITEMS_DELETION,
+                           'suppressions.txt');
+  }
+
+
+  protected function recordsTotal() {
+    return $this->planWith('Notices - import total',
+                           $this->_records_profile,
+                           Class_IntMajAuto::OP_FULL_IMPORT,
+                           'notices_total.txt');
+  }
+
+
+  protected function recordsPartial() {
+    return $this->planWith('Notices - import incrémentiel',
+                           $this->_records_profile,
+                           Class_IntMajAuto::OP_PARTIAL_IMPORT,
+                           'notices.txt');
+  }
+
+
+  protected function users() {
+    return $this->planWith('Abonnés',
+                           $this->_users_profile,
+                           Class_IntMajAuto::OP_FULL_IMPORT,
+                           'abonnes.txt');
+  }
+
+
+  protected function loans() {
+    return $this->planWith('Prêts',
+                           $this->_loans_profile,
+                           Class_IntMajAuto::OP_FULL_IMPORT,
+                           'prets.txt');
+  }
+
+
+  protected function holds() {
+    return $this->_holds
+      ? $this->planWith('Réservations', $this->_holds_profile,
+                        Class_IntMajAuto::OP_FULL_IMPORT, 'reservations.txt')
+      : $this;
+  }
+
+
+  protected function basketsTotal() {
+    if (!$this->_baskets)
+      return $this;
+
+    $profile = $this->_getBasketProfile();
+
+    return $this->planWith('Paniers - import total',
+                           $profile->getId(),
+                           Class_IntMajAuto::OP_FULL_IMPORT,
+                           'paniers_total.txt');
+  }
+
+
+  protected function basketsPartial() {
+    if (!$this->_baskets)
+      return $this;
+
+    $profile = $this->_getBasketProfile();
+
+    return $this->planWith('Paniers - import incrémentiel',
+                           $profile->getId(),
+                           Class_IntMajAuto::OP_PARTIAL_IMPORT,
+                           'paniers.txt');
+  }
+
+
+  protected function planWith($name, $profile, $type, $file) {
+    $this->_html .= '<tr><td class="blank">Site n° '. $this->_id_bib . '</td><td class="blank">' . $name . '</td></tr>';
+    Class_IntMajAuto::newInstance(['id_bib' => $this->_id_bib,
+                                   'id_prog' => $this->_id_prog++,
+                                   'libelle' => $name,
+                                   'profil' => $profile,
+                                   'type_operation' => $type,
+                                   'nom_fichier' => $this->_base_path . $file,
+                                   'rang' => $id])
+      ->save();
+
+    return $this;
+  }
+
+
+  protected function _getBarCodeProfile() {
+    $label = 'Liste de codes-barres';
+    if ($profile = Class_IntProfilDonnees::findFirstBy(['libelle' => $label]))
+      return $profile;
+
+
+    $attributs = 'a:6:{i:0;a:7:{s:8:"type_doc";a:11:{i:0;a:3:{s:4:"code";s:1:"0";s:5:"label";s:0:"";s:8:"zone_995";s:0:"";}i:1;a:3:{s:4:"code";s:1:"1";s:5:"label";s:5:"am;na";s:8:"zone_995";s:0:"";}i:2;a:3:{s:4:"code";s:1:"2";s:5:"label";s:2:"as";s:8:"zone_995";s:0:"";}i:3;a:3:{s:4:"code";s:1:"3";s:5:"label";s:3:"i;j";s:8:"zone_995";s:0:"";}i:4;a:3:{s:4:"code";s:1:"4";s:5:"label";s:1:"g";s:8:"zone_995";s:0:"";}i:5;a:3:{s:4:"code";s:1:"5";s:5:"label";s:3:"l;m";s:8:"zone_995";s:0:"";}i:6;a:3:{s:4:"code";s:1:"8";s:5:"label";s:0:"";s:8:"zone_995";s:0:"";}i:7;a:3:{s:4:"code";s:1:"9";s:5:"label";s:0:"";s:8:"zone_995";s:0:"";}i:8;a:3:{s:4:"code";s:2:"10";s:5:"label";s:0:"";s:8:"zone_995";s:0:"";}i:9;a:3:{s:4:"code";s:3:"100";s:5:"label";s:0:"";s:8:"zone_995";s:0:"";}i:10;a:3:{s:4:"code";s:2:"10";s:5:"label";s:0:"";s:8:"zone_995";s:0:"";}}s:17:"champ_code_barres";s:1:"f";s:10:"champ_cote";s:1:"k";s:11:"champ_genre";s:0:"";s:13:"champ_section";s:1:"q";s:17:"champ_emplacement";s:1:"u";s:12:"champ_annexe";s:1:"a";}i:1;a:1:{s:6:"champs";s:11:"code_barres";}i:2;a:1:{s:6:"champs";s:11:"code_barres";}i:3;a:1:{s:6:"champs";s:11:"code_barres";}i:5;a:3:{s:6:"champs";s:11:"code_barres";s:17:"xml_balise_abonne";s:0:"";s:17:"xml_champs_abonne";a:10:{s:6:"IDABON";s:0:"";s:9:"ORDREABON";s:0:"";s:3:"NOM";s:0:"";s:6:"PRENOM";s:0:"";s:9:"NAISSANCE";s:0:"";s:8:"PASSWORD";s:0:"";s:4:"MAIL";s:0:"";s:10:"DATE_DEBUT";s:0:"";s:8:"DATE_FIN";s:0:"";s:7:"ID_SIGB";s:0:"";}}i:4;a:5:{s:4:"zone";s:0:"";s:5:"champ";s:0:"";s:6:"format";s:0:"";s:5:"jours";s:0:"";s:7:"valeurs";s:0:"";}}';
+
+    $profile = Class_IntProfilDonnees::newInstance(['libelle' => $label,
+                                                    'accents' => Class_IntProfilDonnees::ENCODING_UTF8,
+                                                    'type_fichier' => Class_IntProfilDonnees::FT_RECORDS,
+                                                    'format' => Class_IntProfilDonnees::FORMAT_TABBED_ASCII,
+                                                    'attributs' => $attributs]);
+    $profile->save();
+    return $profile;
+  }
+
+
+  protected function _getBasketProfile() {
+    if ($profile = Class_IntProfilDonnees::findFirstBy(['accents' => Class_IntProfilDonnees::ENCODING_UTF8,
+                                                        'type_fichier' => Class_IntProfilDonnees::FT_BASKETS,
+                                                        'format' => Class_IntProfilDonnees::FORMAT_PIPED_ASCII]))
+      return $profile;
+
+    $profile = Class_IntProfilDonnees::newInstance(['libelle' => 'Paniers Nanook',
+                                                    'accents' => Class_IntProfilDonnees::ENCODING_UTF8,
+                                                    'rejet_periodiques' => 0,
+                                                    'id_article_periodique' => 4,
+                                                    'type_fichier' => Class_IntProfilDonnees::FT_BASKETS,
+                                                    'format' => Class_IntProfilDonnees::FORMAT_PIPED_ASCII,
+                                                    'attributs' => 'a:7:{i:0;a:9:{s:8:"type_doc";a:35:{i:0;a:3:{s:4:"code";s:1:"0";s:5:"label";s:0:"";s:8:"zone_995";s:0:"";}i:1;a:3:{s:4:"code";s:1:"1";s:5:"label";s:8:"am;na;mn";s:8:"zone_995";s:9:"LIV;MS;uu";}i:2;a:3:{s:4:"code";s:1:"2";s:5:"label";s:2:"as";s:8:"zone_995";s:3:"PER";}i:3;a:3:{s:4:"code";s:1:"3";s:5:"label";s:3:"i;j";s:8:"zone_995";s:17:"CD;LIVCD;LIVK7;K7";}i:4;a:3:{s:4:"code";s:1:"4";s:5:"label";s:1:"g";s:8:"zone_995";s:20:"DIAPO;DVD;VHS;VHD;VD";}i:5;a:3:{s:4:"code";s:1:"5";s:5:"label";s:6:"l;m;mm";s:8:"zone_995";s:3:"CDR";}i:6;a:3:{s:4:"code";s:1:"8";s:5:"label";s:0:"";s:8:"zone_995";s:1:"D";}i:7;a:3:{s:4:"code";s:1:"9";s:5:"label";s:0:"";s:8:"zone_995";s:0:"";}i:8;a:3:{s:4:"code";s:1:"8";s:5:"label";s:0:"";s:8:"zone_995";s:3:"DOS";}i:9;a:3:{s:4:"code";s:1:"9";s:5:"label";s:0:"";s:8:"zone_995";s:0:"";}i:10;a:3:{s:4:"code";s:2:"10";s:5:"label";s:0:"";s:8:"zone_995";s:6:"WEB;MF";}i:11;a:3:{s:4:"code";s:3:"102";s:5:"label";s:0:"";s:8:"zone_995";s:0:"";}i:12;a:3:{s:4:"code";s:3:"103";s:5:"label";s:0:"";s:8:"zone_995";s:0:"";}i:13;a:3:{s:4:"code";s:3:"104";s:5:"label";s:0:"";s:8:"zone_995";s:0:"";}i:14;a:3:{s:4:"code";s:3:"105";s:5:"label";s:0:"";s:8:"zone_995";s:0:"";}i:15;a:3:{s:4:"code";s:3:"106";s:5:"label";s:0:"";s:8:"zone_995";s:0:"";}i:16;a:3:{s:4:"code";s:3:"107";s:5:"label";s:0:"";s:8:"zone_995";s:0:"";}i:17;a:3:{s:4:"code";s:3:"108";s:5:"label";s:0:"";s:8:"zone_995";s:0:"";}i:18;a:3:{s:4:"code";s:3:"100";s:5:"label";s:0:"";s:8:"zone_995";s:0:"";}i:19;a:3:{s:4:"code";s:3:"101";s:5:"label";s:0:"";s:8:"zone_995";s:0:"";}i:20;a:3:{s:4:"code";s:3:"102";s:5:"label";s:0:"";s:8:"zone_995";s:0:"";}i:21;a:3:{s:4:"code";s:3:"103";s:5:"label";s:0:"";s:8:"zone_995";s:0:"";}i:22;a:3:{s:4:"code";s:3:"104";s:5:"label";s:0:"";s:8:"zone_995";s:0:"";}i:23;a:3:{s:4:"code";s:3:"105";s:5:"label";s:0:"";s:8:"zone_995";s:0:"";}i:24;a:3:{s:4:"code";s:3:"106";s:5:"label";s:0:"";s:8:"zone_995";s:0:"";}i:25;a:3:{s:4:"code";s:3:"107";s:5:"label";s:0:"";s:8:"zone_995";s:0:"";}i:26;a:3:{s:4:"code";s:3:"108";s:5:"label";s:0:"";s:8:"zone_995";s:0:"";}i:27;a:3:{s:4:"code";s:3:"109";s:5:"label";s:0:"";s:8:"zone_995";s:0:"";}i:28;a:3:{s:4:"code";s:3:"111";s:5:"label";s:0:"";s:8:"zone_995";s:0:"";}i:29;a:3:{s:4:"code";s:3:"110";s:5:"label";s:0:"";s:8:"zone_995";s:0:"";}i:30;a:3:{s:4:"code";s:3:"112";s:5:"label";s:0:"";s:8:"zone_995";s:0:"";}i:31;a:3:{s:4:"code";s:3:"114";s:5:"label";s:0:"";s:8:"zone_995";s:0:"";}i:32;a:3:{s:4:"code";s:3:"113";s:5:"label";s:0:"";s:8:"zone_995";s:0:"";}i:33;a:3:{s:4:"code";s:3:"115";s:5:"label";s:0:"";s:8:"zone_995";s:0:"";}i:34;a:3:{s:4:"code";s:3:"116";s:5:"label";s:0:"";s:8:"zone_995";s:0:"";}}s:17:"champ_code_barres";s:1:"a";s:10:"champ_cote";s:1:"f";s:14:"champ_type_doc";s:0:"";s:11:"champ_genre";s:1:"#";s:13:"champ_section";s:1:"w";s:17:"champ_emplacement";s:1:"x";s:12:"champ_annexe";s:1:"h";s:9:"champ_url";a:2:{s:4:"zone";s:0:"";s:5:"champ";s:0:"";}}i:1;a:1:{s:6:"champs";s:47:"ID_SIGB;LIBELLE;IDABON;MAIL;ROLE;ID_NOTICE_SIGB";}i:2;a:1:{s:6:"champs";s:47:"ID_SIGB;LIBELLE;IDABON;MAIL;ROLE;ID_NOTICE_SIGB";}i:3;a:1:{s:6:"champs";s:47:"ID_SIGB;LIBELLE;IDABON;MAIL;ROLE;ID_NOTICE_SIGB";}i:5;a:3:{s:6:"champs";s:47:"ID_SIGB;LIBELLE;IDABON;MAIL;ROLE;ID_NOTICE_SIGB";s:17:"xml_balise_abonne";s:0:"";s:17:"xml_champs_abonne";a:11:{s:6:"IDABON";s:0:"";s:9:"ORDREABON";s:0:"";s:3:"NOM";s:0:"";s:6:"PRENOM";s:0:"";s:9:"NAISSANCE";s:0:"";s:8:"PASSWORD";s:0:"";s:4:"MAIL";s:0:"";s:10:"DATE_DEBUT";s:0:"";s:8:"DATE_FIN";s:0:"";s:7:"ID_SIGB";s:0:"";s:9:"NUM_CARTE";s:0:"";}}i:4;a:5:{s:4:"zone";s:3:"995";s:5:"champ";s:1:"5";s:6:"format";s:1:"3";s:5:"jours";s:0:"";s:7:"valeurs";s:1:"1";}i:6;a:2:{s:4:"zone";s:0:"";s:5:"champ";s:0:"";}}']);
+    $profile->save();
+    return $profile;
+  }
+}
+
+
+
+class Class_Cosmogramme_GeneratorPlannedNanook extends Class_Cosmogramme_GeneratorPlannedAbstract{
+  protected
+    $_records_profile = 104,
+    $_users_profile = 105,
+    $_item_delete = true,
+    $_baskets = true;
+}
+
+
+
+class Class_Cosmogramme_GeneratorPlannedPergame extends Class_Cosmogramme_GeneratorPlannedAbstract{
+  protected
+    $_records_profile = 100,
+    $_users_profile = 101,
+    $_holds = true;
 }
\ No newline at end of file
diff --git a/library/Class/Cosmogramme/LandingDirectory.php b/library/Class/Cosmogramme/LandingDirectory.php
index c8188bf7814..4b508f2b8ed 100644
--- a/library/Class/Cosmogramme/LandingDirectory.php
+++ b/library/Class/Cosmogramme/LandingDirectory.php
@@ -30,7 +30,8 @@ class Class_Cosmogramme_LandingDirectory {
     $_path,
     $_required_files = ['libraries' => 'annexes.txt',
                         'kinds' => 'genres.txt',
-                        'sections' => 'sections.txt'];
+                        'sections' => 'sections.txt',
+                        'locations' => 'emplacements.txt'];
 
 
   public function __construct() {
@@ -76,10 +77,15 @@ class Class_Cosmogramme_LandingDirectory {
   }
 
 
-  protected function _getStandardDirPath($subdir) {
+  public function getSubdirPath($subdir) {
     return $this->_path
       . (DIRECTORY_SEPARATOR != substr($this->_path, -1) ? DIRECTORY_SEPARATOR : '')
-      . $subdir . DIRECTORY_SEPARATOR . static::STANDARD_NAME;
+      . $subdir;
+  }
+
+
+  protected function _getStandardDirPath($subdir) {
+    return $this->getSubdirPath($subdir) . DIRECTORY_SEPARATOR . static::STANDARD_NAME;
   }
 
 
@@ -94,6 +100,18 @@ class Class_Cosmogramme_LandingDirectory {
   }
 
 
+  public function getSectionsOf($subdir) {
+    $file = $this->_getStandardFilePath($subdir, $this->_required_files['sections']);
+    return $this->getFileSystem()->file($file);
+  }
+
+
+  public function getLocationsOf($subdir) {
+    $file = $this->_getStandardFilePath($subdir, $this->_required_files['locations']);
+    return $this->getFileSystem()->file($file);
+  }
+
+
   public function getSubdirectories() {
     if (!$this->isValid())
       return [];
diff --git a/library/Class/Log.php b/library/Class/Log.php
index 7b8a374cfbe..b47bc1cad0b 100644
--- a/library/Class/Log.php
+++ b/library/Class/Log.php
@@ -27,6 +27,7 @@ class Class_Log {
 
   public function log($message) {
     $this->_messages[] = $message;
+    return $this;
   }
 
 
@@ -38,4 +39,10 @@ class Class_Log {
   public function getMessages() {
     return $this->_messages;
   }
+
+
+  public function reset() {
+    $this->_messages = [];
+    return $this;
+  }
 }
\ No newline at end of file
-- 
GitLab