diff --git a/VERSIONS_WIP/50215 b/VERSIONS_WIP/50215
new file mode 100644
index 0000000000000000000000000000000000000000..2b313a9c4cfcd66fa23acb88b570652d2aa675bd
--- /dev/null
+++ b/VERSIONS_WIP/50215
@@ -0,0 +1 @@
+ - ticket #50215 : Ressource Numériques : Moissonnage Bibliondemand
\ No newline at end of file
diff --git a/application/modules/admin/views/scripts/bibnum/index.phtml b/application/modules/admin/views/scripts/bibnum/index.phtml
index bddf56a1aa7b4dff0193df5ed2b9b4f5a90abe43..f03933bd1723b72c870fa1464e99e411963f60d0 100644
--- a/application/modules/admin/views/scripts/bibnum/index.phtml
+++ b/application/modules/admin/views/scripts/bibnum/index.phtml
@@ -7,7 +7,6 @@ echo $this->_("Si vous souhaitez ajouter une ressource à votre portail, il vous
 <?php
 $datas = (new Class_WebService_BibNumerique_Connectors())->getDescription();
 $datas = json_decode(json_encode($datas));
-
 ?>
 <table class="digital_connectors">
   <tbody>
@@ -54,9 +53,9 @@ $datas = json_decode(json_encode($datas));
           <?php } ?>
           <?php if ($contact) { ?>
             <br/><br/>
-            <?php echo $this->tag('strong', $this->_('Contact : ')) . implode(', ', $contact);?>
+            <?php echo $this->tag('strong', $this->_('Contact : ')) . implode(', ', $contact);}?>
             <br/><br/>
-            <?php if ($connector->enabled)
+            <?php if ($connector->enabled) {
               echo $this->tag('div', $this->_('Activé'), ['class' => 'enabled']);?>
           <?php } ?>
         </td>
diff --git a/application/modules/opac/controllers/ModulesController.php b/application/modules/opac/controllers/ModulesController.php
index 6a3bf1c9df4e795f9f97f0989d193c231c9bfe7c..1fe1e1f449d34dfbcef43a3d420b41fc9beb7919 100644
--- a/application/modules/opac/controllers/ModulesController.php
+++ b/application/modules/opac/controllers/ModulesController.php
@@ -88,6 +88,25 @@ class ModulesController extends ZendAfi_Controller_Action {
   }
 
 
+  public function ssoAction() {
+    if (!$record = Class_Notice::find($this->_getParam('id', 0))) {
+      $this->_helper->notify($this->_('Document non trouvé'));
+      $this->_redirectToReferer();
+      return;
+    }
+
+    $sso_record = new Class_Notice_Sso($record);
+    if(!$module_menu = $sso_record->getModuleMenu()) {
+      $this->_helper->notify($this->_('Oups, l\'adresse de consultation de ce document est indeterminée'));
+      $this->_redirectToReferer();
+      return;
+    }
+
+    $module_menu->setRecord($sso_record);
+    $this->simple($module_menu);
+  }
+
+
   protected function simple($link) {
     $this->willRedirectToMe($link)
          ->checkNotifyMessage($link, $link->getDynamiqueUrl());
diff --git a/cosmogramme/cosmozend/tests/application/modules/cosmo/controllers/DataProfileControllerDublinCoreTest.php b/cosmogramme/cosmozend/tests/application/modules/cosmo/controllers/DataProfileControllerDublinCoreTest.php
index d4e3da21f233b517b11e92a62f48ecdf966b66a8..a08f7c6ac41cdf42cf54dfb6f8998b2bff2689ce 100644
--- a/cosmogramme/cosmozend/tests/application/modules/cosmo/controllers/DataProfileControllerDublinCoreTest.php
+++ b/cosmogramme/cosmozend/tests/application/modules/cosmo/controllers/DataProfileControllerDublinCoreTest.php
@@ -41,6 +41,7 @@ abstract class Cosmo_DataProfileControllerDublinCoreProfileTestCase extends Cosm
 
 
 
+
 class Cosmo_DataProfileControllerDublinCoreTest extends Cosmo_DataProfileControllerDublinCoreProfileTestCase {
   public function setUp() {
     parent::setUp();
@@ -48,28 +49,35 @@ class Cosmo_DataProfileControllerDublinCoreTest extends Cosmo_DataProfileControl
   }
 
 
-/** @test */
+  /** @test */
   public function fileFormatShouldBeDublinCore() {
     $this->assertXPath('//select/option[@label="Dublin Core"][@value="9"][@selected]');
   }
 
 
-/** @test */
+  /** @test */
   public function fieldsetDocTypesShouldBePresent() {
     $this->assertXPathContentContains('//form//fieldset/legend',
                                       'Type de documents');
   }
 
 
-/** @test */
+  /** @test */
   public function typeDocColumnLabelsShouldBeTypeAndFormat() {
     $this->assertXPathContentContains('//script',
                                       'fields:[{"name":"1_type","label":"dc:Type"},{"name":"1_format","label":"dc:Format"}]');
   }
+
+
+  /** @test */
+  public function docTypeBibliondemandShouldNotBePresent() {
+    $this->assertNotXPathContentContains('//script', 'Bibliondemand', $this->_response->getBody());
+  }
 }
 
 
 
+
 class Cosmo_DataProfileControllerDublinCorePostTest extends Cosmo_DataProfileControllerDublinCoreProfileTestCase {
   public function setUp() {
     parent::setUp();
diff --git a/cosmogramme/cosmozend/tests/application/modules/cosmo/controllers/DataProfileControllerTest.php b/cosmogramme/cosmozend/tests/application/modules/cosmo/controllers/DataProfileControllerTest.php
index 7821df274d0d1403fc35e033f1cbb52b6c74426e..4fd85e564af187095fef43c48777742f1e0e9233 100644
--- a/cosmogramme/cosmozend/tests/application/modules/cosmo/controllers/DataProfileControllerTest.php
+++ b/cosmogramme/cosmozend/tests/application/modules/cosmo/controllers/DataProfileControllerTest.php
@@ -216,6 +216,12 @@ class Cosmo_DataProfileControllerEditUnimarcKohaTest extends Cosmo_DataProfileCo
   }
 
 
+  /** @test */
+  public function formatBibliondemandShouldBeAvailable() {
+    $this->assertXPathContentContains('//form//select[@name="format"]/option[@value="11"]', 'Dublin Core Bibliondemand');
+  }
+
+
   /** @test */
   public function fieldNouveauteJoursShouldBePresent() {
     $this->assertXPath('//form//input[@id="nouveaute_jours"]');
diff --git a/cosmogramme/php/classes/classe_notice_integration.php b/cosmogramme/php/classes/classe_notice_integration.php
index 063ba09d37e45ee4eee4aa775ad886c0e9d0584b..183fc572f1438187660c09c54bc91a0295468b03 100644
--- a/cosmogramme/php/classes/classe_notice_integration.php
+++ b/cosmogramme/php/classes/classe_notice_integration.php
@@ -131,6 +131,10 @@ class notice_integration {
         $this->analyseur = new Class_Cosmogramme_Integration_Record_DublinCore();
         break;
 
+      case Class_IntProfilDonnees::FORMAT_BIBLIONDEMAND:
+        $this->analyseur = new Class_Cosmogramme_Integration_Record_Bibliondemand();
+        break;
+
       default:
         require_once("classe_notice_ascii.php");
         $this->analyseur = new notice_ascii();
diff --git a/library/Class/Cosmogramme/Integration.php b/library/Class/Cosmogramme/Integration.php
index 68af4e7ef25bbc9edaf3574a908bf280bd16ed73..44efcd02906af1cfaeba2df9faf1ffbb08ad3872 100644
--- a/library/Class/Cosmogramme/Integration.php
+++ b/library/Class/Cosmogramme/Integration.php
@@ -87,6 +87,11 @@ class Class_Cosmogramme_Integration extends Storm_Model_Abstract {
   }
 
 
+  public function isBibliondemand() {
+    return $this->getProfilDonnees()->isBibliondemand();
+  }
+
+
   public function fileExists() {
     return $this->getFileSystem()->file_exists($this->getFilePath());
   }
diff --git a/library/Class/Cosmogramme/Integration/DataSource.php b/library/Class/Cosmogramme/Integration/DataSource.php
index 937f01f83004535f45e619cc7590c730a354b38f..7a80273a0221d83ecdc664c7ec7bc60ccf8ca6e4 100644
--- a/library/Class/Cosmogramme/Integration/DataSource.php
+++ b/library/Class/Cosmogramme/Integration/DataSource.php
@@ -26,7 +26,8 @@ abstract class Class_Cosmogramme_Integration_DataSource {
 
 
   public static function on($integration) {
-    if ($integration->isSourceHTTP() && $integration->isFormatDublinCore())
+    if ($integration->isSourceHTTP()
+        && ($integration->isFormatDublinCore() || $integration->isBibliondemand()))
       return new Class_Cosmogramme_Integration_DataSourceOAIDublinCore($integration);
 
     if ($integration->isSourceHTTP())
@@ -249,10 +250,14 @@ class Class_Cosmogramme_Integration_DataSourceOAIMarc21 extends Class_Cosmogramm
 
 class Class_Cosmogramme_Integration_DataSourceOAIDublinCore extends Class_Cosmogramme_Integration_DataSourceOAI {
   public function newClient() {
+    $parser = $this->_integration->isBibliondemand()
+      ? new Class_WebService_OAI_DublinCoreParser_Bibliondemand()
+      : new Class_WebService_OAI_DublinCoreParser();
+
     return (new Class_WebService_OAI())
       ->setOAIHandler($this->_uri)
       ->setMetadataPrefix('oai_dc')
-      ->setParser(new Class_WebService_OAI_DublinCoreParser());
+      ->setParser($parser);
   }
 
 
diff --git a/library/Class/Cosmogramme/Integration/Record/Bibliondemand.php b/library/Class/Cosmogramme/Integration/Record/Bibliondemand.php
new file mode 100644
index 0000000000000000000000000000000000000000..4cd5ae5d08f21bdbe565f518402eb65b06c503f7
--- /dev/null
+++ b/library/Class/Cosmogramme/Integration/Record/Bibliondemand.php
@@ -0,0 +1,45 @@
+<?php
+/**
+ * Copyright (c) 2012-2017, Agence Française Informatique (AFI). All rights reserved.
+ *
+ * BOKEH is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU AFFERO GENERAL PUBLIC LICENSE as published by
+ * the Free Software Foundation.
+ *
+ * There are special exceptions to the terms and conditions of the AGPL as it
+ * is applied to this software (see README file).
+ *
+ * BOKEH is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ * GNU AFFERO GENERAL PUBLIC LICENSE for more details.
+ *
+ * You should have received a copy of the GNU AFFERO GENERAL PUBLIC LICENSE
+ * along with BOKEH; if not, write to the Free Software
+ * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301  USA
+ */
+
+
+class Class_Cosmogramme_Integration_Record_Bibliondemand extends Class_Cosmogramme_Integration_Record_DublinCore {
+  protected function _dublinCoreToUnimarc() {
+    $visitor = parent::_dublinCoreToUnimarc();
+    $visitor
+      ->visitThumbnail($this->_datas['thumbnail'])
+      ->visitPrivateUrl($this->_datas['url'])
+      ->visitCatalogAgency('Bibliondemand')
+      ->visitLocations($this->_datas['source']);
+
+    return $visitor;
+  }
+
+
+  protected function _achieveUnimarc($unimarc_visitor) {
+    return $unimarc_visitor;
+  }
+
+
+  protected function _getItem() {
+    return [['f', $this->_datas['id_oai']],
+            ['r', implode(';', $this->_types)]];
+  }
+}
diff --git a/library/Class/Cosmogramme/Integration/Record/DublinCore.php b/library/Class/Cosmogramme/Integration/Record/DublinCore.php
index da9b91a32c187c8febf8b881aee9bddbcdd93d82..d79e9d5efe51f89df87732f14764f6ea4f5aa447 100644
--- a/library/Class/Cosmogramme/Integration/Record/DublinCore.php
+++ b/library/Class/Cosmogramme/Integration/Record/DublinCore.php
@@ -36,16 +36,12 @@ class Class_Cosmogramme_Integration_Record_DublinCore extends notice_unimarc {
     $this->_formats = $data['format'];
     $this->_datas = $data;
 
-    $unimarc = $this->_dublinCoreToUnimarc();
+    $unimarc = $this->_dublinCoreToUnimarc()->getUnimarc();
     return parent::ouvrirNotice($unimarc, $id_data_profile, $sigb, $type_doc_force);
   }
 
 
   protected function _dublinCoreToUnimarc() {
-    $item = [['f', md5($this->_datas['id_oai'])],
-             ['r', implode(';', $this->_types + $this->_formats)],
-             ['u', $this->_datas['id_oai']]];
-
     $authors = $this->_authorsFrom($this->_datas['auteur']);
 
     $unimarc_visitor = new Class_Indexation_PseudoNotice_UnimarcVisitor();
@@ -57,14 +53,26 @@ class Class_Cosmogramme_Integration_Record_DublinCore extends notice_unimarc {
       ->visitDescriptions($this->_datas['rights'])
       ->visitDescriptions($this->_datas['relation'])
       ->visitYear($this->_datas['date'])
-      ->visitLanguageId($this->_datas['language'][0])
-      ->visitLabelledUrl($this->_('Consulter la ressource en ligne'), $this->_datas['id_oai'])
+      ->visitLanguageId(reset($this->_datas['language']))
       ->visitMatiere(implode(';', $this->_datas['matiere']))
       ->visitEditors($this->_datas['editeur'])
-      ->visitItems($item)
+      ->visitItems($this->_getItem())
       ->visitAuteur($authors);
 
-    return $unimarc_visitor->getUnimarc();
+    return $this->_achieveUnimarc($unimarc_visitor);
+  }
+
+
+  protected function _achieveUnimarc($unimarc_visitor) {
+    return $unimarc_visitor
+      ->visitLabelledUrl($this->_('Consulter la ressource en ligne'), $this->_datas['id_oai']);
+  }
+
+
+  protected function _getItem() {
+    return [['f', md5($this->_datas['id_oai'])],
+            ['r', implode(';', $this->_types + $this->_formats)],
+            ['u', $this->_datas['id_oai']]];
   }
 
 
@@ -120,13 +128,13 @@ class Class_Cosmogramme_Integration_Record_DublinCore extends notice_unimarc {
     $profile_doc_type_prefs = $this->_profil->getItemDocTypesPrefs();
 
     foreach($profile_doc_type_prefs as $profile_doc_type) {
-      $types = explode(';', $profile_doc_type['type']);
-      $formats = explode(';', $profile_doc_type['format']);
+      $types = array_filter(explode(';', $profile_doc_type['type']));
+      $formats = array_filter(explode(';', $profile_doc_type['format']));
 
       $types_found = array_intersect($types, $this->_types);
       $formats_found = array_intersect($formats, $this->_formats);
 
-      if(   (!empty($types_found) && empty($formats))
+      if ((!empty($types_found) && empty($formats))
          || (!empty($types_found) && !empty($formats_found))
          || (empty($types) && !empty($formats_found)))
         return $profile_doc_type[$key];
diff --git a/library/Class/DigitalResource.php b/library/Class/DigitalResource.php
index f0467c08f008e1f8741925be1fe9421735f8c129..7adf793e87e089193aab521566a10a4eebd1a539 100644
--- a/library/Class/DigitalResource.php
+++ b/library/Class/DigitalResource.php
@@ -87,6 +87,15 @@ class Class_DigitalResource extends Class_Entity {
   }
 
 
+  public function getPluginForSsoAction($action) {
+    return $this->getPlugins()
+                ->detect(function($config) use ($action)
+                         {
+                           return $config->getSsoAction() == strtolower($action);
+                         });
+  }
+
+
   public function getPluginsDescription() {
     return $this->pluginsByName(function($config)
                                 {
@@ -180,7 +189,10 @@ class Class_DigitalResource extends Class_Entity {
                 ->injectInto([],
                              function($types, $config)
                              {
-                               $types[$this->getDocType($config->getName())] = $config->getDocTypeLabel();
+                               if(!$label = $config->getDocTypeLabel())
+                                 return $types;
+
+                               $types[$this->getDocType($config->getName())] = $label;
                                return $types;
                              }
                 );
@@ -196,6 +208,15 @@ class Class_DigitalResource extends Class_Entity {
   }
 
 
+  public function getModuleMenuFor($name) {
+    $modules = $this->getModulesMenu();
+    if(!isset($modules[$name]))
+      return null;
+
+    return $modules[$name];
+  }
+
+
   public function isPluginDocType($type) {
     if(!$type)
       return false;
diff --git a/library/Class/DigitalResource/Config.php b/library/Class/DigitalResource/Config.php
index c6fda696acadf69537b2f970a64fac5d68806cb4..7472c3d21b931538c0847d54bfdfb11edf9606c5 100644
--- a/library/Class/DigitalResource/Config.php
+++ b/library/Class/DigitalResource/Config.php
@@ -45,13 +45,8 @@ class Class_DigitalResource_Config extends Class_Entity {
 
 
   public function __construct($digital_resource) {
-    $this->updateAttributes(
-                            array_merge(
-                                        [static::DIGITAL_RESOURCE => $digital_resource],
-                                        $this->getConfig()
-                            )
-    );
-
+    $this->setDigitalResource($digital_resource);
+    $this->updateAttributes($this->getConfig());
     $this->_ensurePermission();
   }
 
@@ -115,7 +110,8 @@ class Class_DigitalResource_Config extends Class_Entity {
   public function getFeatures() {
     $features = [];
 
-    if($this->getBatch())
+    if($this->getBatch()
+       || $this->getHarvesting())
       $features[] = static::HARVEST;
 
     if($this->getSsoAction())
@@ -154,4 +150,16 @@ class Class_DigitalResource_Config extends Class_Entity {
 
     return $this;
   }
+
+
+  public function getSsoAction() {
+    return $this->get('SsoAction', false)
+      ? strtolower($this->getName())
+      : null;
+  }
+
+
+  public function withNameSpace($value) {
+    return $this->getDigitalResource()->withNameSpace($value);
+  }
 }
\ No newline at end of file
diff --git a/library/Class/DigitalResource/ModuleMenu.php b/library/Class/DigitalResource/ModuleMenu.php
index e2c1ce6bf44eaca6b5d50f8036bb96076f309e90..0fbe9c988a5aa868423df78b3628ebc5228283e7 100644
--- a/library/Class/DigitalResource/ModuleMenu.php
+++ b/library/Class/DigitalResource/ModuleMenu.php
@@ -48,9 +48,14 @@ class Class_DigitalResource_ModuleMenu extends Class_Systeme_ModulesMenu_SSOAbst
     if ($user
         && $this->_config->hasRightAccess($user)
         && $this->_config->isEnabled())
-      return $this->_config->getSsoUrl($user);
+      return $this->getSsoUrl($user);
 
     $this->setMessage($this->_config->getNotAllowedMessage());
     return '';
   }
+
+
+  protected function getSsoUrl($user) {
+    return $this->_config->getSsoUrl($user);
+  }
 }
diff --git a/library/Class/Indexation/PseudoNotice/UnimarcVisitor.php b/library/Class/Indexation/PseudoNotice/UnimarcVisitor.php
index d645bbe21029804906e10b3878e452ead7fd81a2..1c894bf691358448b4c3687bd914495bdc499fcb 100644
--- a/library/Class/Indexation/PseudoNotice/UnimarcVisitor.php
+++ b/library/Class/Indexation/PseudoNotice/UnimarcVisitor.php
@@ -82,6 +82,9 @@ class Class_Indexation_PseudoNotice_UnimarcVisitor extends Class_Indexation_Pseu
 
 
   public function visitLanguageId($languageId) {
+    if (!$languageId)
+      return $this;
+
     $this->unimarc->add_field('101', '0 ', 'a' . $languageId);
     return $this;
   }
@@ -209,6 +212,32 @@ class Class_Indexation_PseudoNotice_UnimarcVisitor extends Class_Indexation_Pseu
   }
 
 
+  public function visitThumbnail($data) {
+    $this->unimarc->add_field('959', '  ', [['u', $data]]);
+    return $this;
+  }
+
+
+  public function visitPrivateUrl($data) {
+    $this->unimarc->add_field('956', '  ', [['u', $data]]);
+    return $this;
+  }
+
+
+  public function visitCatalogAgency($data) {
+    $this->unimarc->add_field('801', '  ', [['b', $data]]);
+    return $this;
+  }
+
+
+  public function visitLocations($datas) {
+    foreach($datas as $location)
+      $this->unimarc->add_field('852', '  ', [['a', $location]]);
+
+    return $this;
+  }
+
+
   public function getUnimarc() {
     $this->unimarc->update();
     return $this->unimarc->getFullRecord();
diff --git a/library/Class/IntProfilDonnees.php b/library/Class/IntProfilDonnees.php
index 7f38359c8c05af98956d9ad25d26b155839382ec..561ff66f09afa3e5a89bd0d4a0c06fbfa9ef4d6b 100644
--- a/library/Class/IntProfilDonnees.php
+++ b/library/Class/IntProfilDonnees.php
@@ -73,7 +73,8 @@ class IntProfilDonneesLoader extends Storm_Model_Loader {
             Class_IntProfilDonnees::FORMAT_MARC21 => $this->_('MARC 21'),
             Class_IntProfilDonnees::FORMAT_UNIMARC_XML => $this->_('Unimarc XML'),
             Class_IntProfilDonnees::FORMAT_AVENIO => $this->_('Avenio'),
-            Class_IntProfilDonnees::FORMAT_DUBLIN_CORE => $this->_('Dublin Core')];
+            Class_IntProfilDonnees::FORMAT_DUBLIN_CORE => $this->_('Dublin Core'),
+            Class_IntProfilDonnees::FORMAT_BIBLIONDEMAND => $this->_('Dublin Core Bibliondemand')];
   }
 
 
@@ -114,7 +115,8 @@ class IntProfilDonneesLoader extends Storm_Model_Loader {
                                 Class_IntProfilDonnees::FORMAT_MARC21 => '',
                                 Class_IntProfilDonnees::FORMAT_UNIMARC_XML => '',
                                 Class_IntProfilDonnees::FORMAT_AVENIO => '',
-                                Class_IntProfilDonnees::FORMAT_DUBLIN_CORE => '']);
+                                Class_IntProfilDonnees::FORMAT_DUBLIN_CORE => '',
+                                Class_IntProfilDonnees::FORMAT_BIBLIONDEMAND => '']);
   }
 
 
@@ -241,6 +243,7 @@ class Class_IntProfilDonnees extends Storm_Model_Abstract {
     FORMAT_AVENIO = 8,
     FORMAT_DUBLIN_CORE = 9,
     FORMAT_CG68_ARCHIVES = 10,
+    FORMAT_BIBLIONDEMAND = 11,
     SERIAL_FORMAT_NONE = 0,
     SERIAL_FORMAT_PERGAME = 1,
     SERIAL_FORMAT_ALOES_INDEXPRESS = 2,
@@ -853,6 +856,11 @@ class Class_IntProfilDonnees extends Storm_Model_Abstract {
   }
 
 
+  public function isBibliondemand() {
+    return self::FORMAT_BIBLIONDEMAND == $this->getFormat();
+  }
+
+
   public function getSeparator() {
     switch ($this->getFormat()) {
       case self::FORMAT_SEMI_COLON_ASCII : return ';';break;
diff --git a/library/Class/Notice/Sso.php b/library/Class/Notice/Sso.php
new file mode 100644
index 0000000000000000000000000000000000000000..2ab654de7a53433aa65d8006a60cad22ec9d6c32
--- /dev/null
+++ b/library/Class/Notice/Sso.php
@@ -0,0 +1,59 @@
+<?php
+/**
+ * Copyright (c) 2012-2017, Agence Française Informatique (AFI). All rights reserved.
+ *
+ * BOKEH is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU AFFERO GENERAL PUBLIC LICENSE as published by
+ * the Free Software Foundation.
+ *
+ * There are special exceptions to the terms and conditions of the AGPL as it
+ * is applied to this software (see README file).
+ *
+ * BOKEH is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ * GNU AFFERO GENERAL PUBLIC LICENSE for more details.
+ *
+ * You should have received a copy of the GNU AFFERO GENERAL PUBLIC LICENSE
+ * along with BOKEH; if not, write to the Free Software
+ * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301  USA
+ */
+
+
+class Class_Notice_Sso extends Class_Entity {
+  public function __construct($record) {
+    $this->setRecord($record);
+  }
+
+
+  public function isValid() {
+    if(!$this->getRecord())
+      return false;
+
+    if(!$catalog_agency = $this->getFirstCatalogAgency())
+      return false;
+
+    return $catalog_agency == 'Bibliondemand';
+  }
+
+
+  protected function getFirstCatalogAgency() {
+    $agencies = $this->getRecord()->get_subfield('801', 'b');
+    return reset($agencies);
+  }
+
+
+  public function getFirstPrivateUrl() {
+    $urls = $this->getRecord()->get_subfield('956', 'u');
+    return reset($urls);
+  }
+
+
+  public function getModuleMenu() {
+    if (!$this->isValid())
+      return null;
+
+    return Class_DigitalResource::getInstance()
+      ->getModuleMenuFor($this->getFirstCatalogAgency());
+  }
+}
diff --git a/library/Class/ProfileSerializer.php b/library/Class/ProfileSerializer.php
index 2a5f9c095f3973f2dd3d695bc93eb51700816418..5a5263a50095a1f850e71f3177e3c9659d501e4c 100644
--- a/library/Class/ProfileSerializer.php
+++ b/library/Class/ProfileSerializer.php
@@ -45,7 +45,8 @@ class Class_ProfileSerializer {
       return new Class_ProfileSerializer_UnimarcRecord();
 
     if($this->_datas[Class_IntProfilDonnees::PROFILE_FILE_TYPE] == Class_IntProfilDonnees::FT_RECORDS
-          && $this->_datas[Class_IntProfilDonnees::PROFILE_FILE_FORMAT] == Class_IntProfilDonnees::FORMAT_DUBLIN_CORE)
+       && in_array($this->_datas[Class_IntProfilDonnees::PROFILE_FILE_FORMAT], [Class_IntProfilDonnees::FORMAT_DUBLIN_CORE,
+                                                                                Class_IntProfilDonnees::FORMAT_BIBLIONDEMAND]))
       return new Class_ProfileSerializer_DublinCoreRecords();
 
     if($this->_datas[Class_IntProfilDonnees::PROFILE_FILE_TYPE] == Class_IntProfilDonnees::FT_RECORDS
diff --git a/library/Class/WebService/OAI/DublinCoreParser.php b/library/Class/WebService/OAI/DublinCoreParser.php
index 77a2491ae863416a2b51cc142cf42cda2739444d..3fd61b791bde5ccf379ae8103a310231d66ed6ed 100644
--- a/library/Class/WebService/OAI/DublinCoreParser.php
+++ b/library/Class/WebService/OAI/DublinCoreParser.php
@@ -118,6 +118,11 @@ class Class_WebService_OAI_DublinCoreParser extends Class_WebService_OAI_ParserA
   }
 
 
+  public function enddc_source($data) {
+    $this->_record['source'][] = $data;
+  }
+
+
   public function getLastRecord($data) {
     return $this->_record;
   }
diff --git a/library/Class/WebService/OAI/DublinCoreParser/Bibliondemand.php b/library/Class/WebService/OAI/DublinCoreParser/Bibliondemand.php
new file mode 100644
index 0000000000000000000000000000000000000000..2592906897ae38eb30abcc19e911957799e5c40d
--- /dev/null
+++ b/library/Class/WebService/OAI/DublinCoreParser/Bibliondemand.php
@@ -0,0 +1,49 @@
+<?php
+/**
+ * Copyright (c) 2012, Agence Française Informatique (AFI). All rights reserved.
+ *
+ * BOKEH is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU AFFERO GENERAL PUBLIC LICENSE as published by
+ * the Free Software Foundation.
+ *
+ * There are special exceptions to the terms and conditions of the AGPL as it
+ * is applied to this software (see README file).
+ *
+ * BOKEH is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ * GNU AFFERO GENERAL PUBLIC LICENSE for more details.
+ *
+ * You should have received a copy of the GNU AFFERO GENERAL PUBLIC LICENSE
+ * along with BOKEH; if not, write to the Free Software
+ * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301  USA
+ */
+
+class Class_WebService_OAI_DublinCoreParser_Bibliondemand extends Class_WebService_OAI_DublinCoreParser {
+
+  public function endIdentifier($data) {
+    if(!$this->parser->getParent() == 'header')
+      return;
+
+    $this->_record['id_oai'] = $data;
+  }
+
+
+  public function enddc_identifier($data) {
+    $this->_prefixData('Vignette : ', $data, 'thumbnail');
+    $this->_prefixData('Url : ', $data, 'url');
+  }
+
+
+  protected function _prefixData($prefix, $data, $field) {
+    if (0 === strpos($data, $prefix))
+      $this->_record[$field] = str_replace($prefix, '', $data);
+  }
+
+
+  public function newRecord($attributes) {
+    return $this->_record = array_merge(parent::newRecord($attributes),
+                                        ['url' => '',
+                                         'thumbnail' => '']);
+  }
+}
\ No newline at end of file
diff --git a/library/ZendAfi/Controller/Action.php b/library/ZendAfi/Controller/Action.php
index a7d0fb38aa7d742113058d1538c7a97f09fef93a..61fe3a8358d14f612c07507b1a77c407fe03b6bb 100644
--- a/library/ZendAfi/Controller/Action.php
+++ b/library/ZendAfi/Controller/Action.php
@@ -30,11 +30,13 @@ class ZendAfi_Controller_Action extends Zend_Controller_Action {
     if ('Action' != substr($method_name, -6))
       return parent::__call($method_name, $args);
 
-    $plugins = $this->_plugins
-      ->select(function($plugin) use ($method_name)
-               {
-                 return $plugin->isActionDefined($method_name);
-               });
+    $plugins = new Storm_Collection();
+    if ($this->_plugins)
+      $plugins = $this->_plugins
+        ->select(function($plugin) use ($method_name)
+                 {
+                   return $plugin->isActionDefined($method_name);
+                 });
 
     if($plugins->isEmpty())
       return parent::__call($method_name, $args);
diff --git a/library/ZendAfi/View/Helper/Notice/ExemplairesTable.php b/library/ZendAfi/View/Helper/Notice/ExemplairesTable.php
index d3881a23abdd80e37efe53a34a2fd02431f9abed..e79914121d441db56895641c16d6c176eab13d81 100644
--- a/library/ZendAfi/View/Helper/Notice/ExemplairesTable.php
+++ b/library/ZendAfi/View/Helper/Notice/ExemplairesTable.php
@@ -254,6 +254,15 @@ class ZendAfi_View_Helper_Notice_Exemplaires_Dispo extends ZendAfi_View_Helper_N
       $libelle = $this->view->tagAnchor($exemplaire->getUrl(),
                                         $this->view->_('Description en ligne'));
 
+    if((new Class_Notice_Sso($exemplaire->getNotice()))->isValid())
+      $libelle = $this->view->tagAnchor($this->view->url(['module' => 'opac',
+                                                          'controller' => 'modules',
+                                                          'action' => 'sso',
+                                                          'id' => $exemplaire->getIdNotice()],
+                                                         null,
+                                                         true),
+                                        $this->view->_('Consulter en ligne'));
+
     if ($exemplaire->getNbResas() > 0)
       $libelle.=
         '<span>'
diff --git a/library/digital_resources/Assimil/Config.php b/library/digital_resources/Assimil/Config.php
index eea26e0a0d7e06e97686caef106fd201fc0f20cf..22b65e007e8cce3a1243c513171bb660e6306d11 100644
--- a/library/digital_resources/Assimil/Config.php
+++ b/library/digital_resources/Assimil/Config.php
@@ -40,7 +40,7 @@ class Assimil_Config extends Class_DigitalResource_Config {
                             'Assimil' => Class_AdminVar_Meta::newOnOff($this->_('Activer ou désactiver la ressource numérique Assimil'))->bePrivate()
             ],
 
-            'SsoAction' => 'assimil',
+            'SsoAction' => true,
             'SsoUrl' => 'http://biblio.assimil.com/assimilweb',
 
             'Batch' => 'Assimil_Batch',
diff --git a/library/digital_resources/Bibliondemand/Config.php b/library/digital_resources/Bibliondemand/Config.php
new file mode 100644
index 0000000000000000000000000000000000000000..d630edb866be187c8b42d39ecbbab934910487b0
--- /dev/null
+++ b/library/digital_resources/Bibliondemand/Config.php
@@ -0,0 +1,57 @@
+<?php
+/**
+ * Copyright (c) 2012-2017, Agence Française Informatique (AFI). All rights reserved.
+ *
+ * BOKEH is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU AFFERO GENERAL PUBLIC LICENSE as published by
+ * the Free Software Foundation.
+ *
+ * There are special exceptions to the terms and conditions of the AGPL as it
+ * is applied to this software (see README file).
+ *
+ * BOKEH is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ * GNU AFFERO GENERAL PUBLIC LICENSE for more details.
+ *
+ * You should have received a copy of the GNU AFFERO GENERAL PUBLIC LICENSE
+ * along with BOKEH; if not, write to the Free Software
+ * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301  USA
+ */
+
+
+class Bibliondemand_Config extends Class_DigitalResource_Config {
+  public function getConfig() {
+    return ['Introduction' => $this->_('Bibliondemand, solution pour bibliothèque numérique.'),
+            'AdminVars' => ['SSO_URL' => Class_AdminVar_Meta::newDefault($this->_('URL SSO des ressources Bibliondemand'))
+                            ->bePrivate()],
+
+            'SsoAction' => true,
+            'Harvesting' => true,
+
+            'HelpLink' => 'http://wiki.bokeh-library-portal.org/index.php/Bibliondemand',
+            'Url' => 'http://www.bibliondemand.com/',
+            'Icon' => 'http://www.bibliondemand.com/ui/skins/catalogue/images/logo_bod.png',
+
+            'MenuLabel' => $this->_('Lien vers Bibliondemand'),
+            'ModuleMenu' => $this->withNameSpace('ModuleMenu'),
+            'NotAllowedMessage' => $this->_('Vous devez être abonné pour accéder à cette ressource.'),
+    ];
+  }
+
+
+  public function hasRightAccess($user) {
+    return $user && $user->isAbonnementValid();
+  }
+
+
+  public function getSsoUrl($user) {
+    return $this->getAdminVar('SSO_URL');
+  }
+
+
+  public function isEnabled() {
+    return '' != $this->getAdminVar('SSO_URL');
+  }
+}
+?>
\ No newline at end of file
diff --git a/library/digital_resources/Bibliondemand/ModuleMenu.php b/library/digital_resources/Bibliondemand/ModuleMenu.php
new file mode 100644
index 0000000000000000000000000000000000000000..ec498e46052901ae00fe7c5596c403ef0f07fcb1
--- /dev/null
+++ b/library/digital_resources/Bibliondemand/ModuleMenu.php
@@ -0,0 +1,41 @@
+<?php
+/**
+ * Copyright (c) 2012-2017, Agence Française Informatique (AFI). All rights reserved.
+ *
+ * BOKEH is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU AFFERO GENERAL PUBLIC LICENSE as published by
+ * the Free Software Foundation.
+ *
+ * There are special exceptions to the terms and conditions of the AGPL as it
+ * is applied to this software (see README file).
+ *
+ * BOKEH is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ * GNU AFFERO GENERAL PUBLIC LICENSE for more details.
+ *
+ * You should have received a copy of the GNU AFFERO GENERAL PUBLIC LICENSE
+ * along with BOKEH; if not, write to the Free Software
+ * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301  USA
+ */
+
+
+class Bibliondemand_ModuleMenu extends Class_DigitalResource_ModuleMenu {
+  protected $_record;
+
+  protected function getSsoUrl($user) {
+    $url = parent::getSsoUrl($user);
+
+    if (!$this->_record
+       || (!$record_url = $this->_record->getFirstPrivateUrl()))
+      return $url;
+
+    return $url . '&' . http_build_query(['returnUrl' => $record_url]);
+  }
+
+
+  public function setRecord($record) {
+    $this->_record = $record;
+    return $this;
+  }
+}
\ No newline at end of file
diff --git a/library/digital_resources/Bibliondemand/controllers/IndexController.php b/library/digital_resources/Bibliondemand/controllers/IndexController.php
new file mode 100644
index 0000000000000000000000000000000000000000..f3b3a6cb909d45470738be014c9db09d308157eb
--- /dev/null
+++ b/library/digital_resources/Bibliondemand/controllers/IndexController.php
@@ -0,0 +1,34 @@
+<?php
+/**
+ * Copyright (c) 2012-2017, Agence Française Informatique (AFI). All rights reserved.
+ *
+ * BOKEH is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU AFFERO GENERAL PUBLIC LICENSE as published by
+ * the Free Software Foundation.
+ *
+ * There are special exceptions to the terms and conditions of the AGPL as it
+ * is applied to this software (see README file).
+ *
+ * BOKEH is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ * GNU AFFERO GENERAL PUBLIC LICENSE for more details.
+ *
+ * You should have received a copy of the GNU AFFERO GENERAL PUBLIC LICENSE
+ * along with BOKEH; if not, write to the Free Software
+ * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301  USA
+ */
+
+
+class Bibliondemand_Plugin_IndexController extends Class_DigitalResource_Controller {
+  public function ssoAction() {
+    $module = $this->_config->getModuleMenu();
+    $module = new $module($this->_config);
+
+    if(!$url = $module->urlForUser(Class_Users::getIdentity()))
+      return $this->willRedirectToMe($module)
+                  ->checkNotifyMessage($module, $module->getDynamiqueUrl());
+
+    return $this->checkNotifyMessage($module, $url);
+  }
+}
\ No newline at end of file
diff --git a/library/digital_resources/Bibliondemand/tests/BibliondemandTest.php b/library/digital_resources/Bibliondemand/tests/BibliondemandTest.php
new file mode 100644
index 0000000000000000000000000000000000000000..41c58b936ee000cb1d1b4596b1777e1879786e96
--- /dev/null
+++ b/library/digital_resources/Bibliondemand/tests/BibliondemandTest.php
@@ -0,0 +1,55 @@
+<?php
+/**
+ * Copyright (c) 2012-2017, Agence Française Informatique (AFI). All rights reserved.
+ *
+ * BOKEH is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU AFFERO GENERAL PUBLIC LICENSE as published by
+ * the Free Software Foundation.
+ *
+ * There are special exceptions to the terms and conditions of the AGPL as it
+ * is applied to this software (see README file).
+ *
+ * BOKEH is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ * GNU AFFERO GENERAL PUBLIC LICENSE for more details.
+ *
+ * You should have received a copy of the GNU AFFERO GENERAL PUBLIC LICENSE
+ * along with BOKEH; if not, write to the Free Software
+ * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301  USA
+ */
+
+
+class BibliondemandModulesControllerTest extends AbstractControllerTestCase {
+
+  /** @test */
+  public function shouldHaveAccessForbiddenMessage() {
+    ZendAfi_Auth::getInstance()->logUser($this->fixture('Class_Users',
+                                                        ['id' => 1,
+                                                         'login' => 'Tom',
+                                                         'password' => 'pwd']));
+
+    $this->dispatch('/opac/modules/bibliondemand', true);
+    $this->assertFlashMessengerContentContains('Vous devez être abonné pour accéder à cette ressource.');
+  }
+
+
+  /** @test */
+  public function shouldContainsRedirectToBibliondemand() {
+    $this->fixture('Class_AdminVar',
+                   ['id' => 'Bibliondemand_SSO_URL',
+                    'valeur' => 'https://biblio.org']);
+
+    ZendAfi_Auth::getInstance()->logUser(
+                                         $this->fixture('Class_Users',
+                                                        ['id' => 1,
+                                                         'login' => 'Tom',
+                                                         'password' => 'pwd'])
+                                         ->beAbonneSIGB());
+
+    $this->dispatch('/opac/modules/bibliondemand', true);
+
+    $this->assertXpathContentContains('//script', 'document.location.href="https://biblio.org";');
+  }
+}
+?>
\ No newline at end of file
diff --git a/library/digital_resources/Lekiosk/Config.php b/library/digital_resources/Lekiosk/Config.php
index 869dfc42e5f476dd6447ab8e297983fd18d7f51a..95fc72311cbbb16e719da952062586f72d3181ee 100644
--- a/library/digital_resources/Lekiosk/Config.php
+++ b/library/digital_resources/Lekiosk/Config.php
@@ -45,7 +45,7 @@ class Lekiosk_Config extends Class_DigitalResource_Config {
                             'FTP_PASSWORD' => Class_AdminVar_Meta::newDefault($this->_('Mot de passe du compte FTP fournit par LeKiosk'))->bePrivate(),
             ],
 
-            'SsoAction' => 'lekiosk',
+            'SsoAction' => true,
             'Batch' => 'Lekiosk_Batch',
             'Service' => 'Lekiosk_Service',
             'ModuleMenu' => 'Lekiosk_ModuleMenu',
diff --git a/tests/application/modules/admin/controllers/BibnumControllerTest.php b/tests/application/modules/admin/controllers/BibnumControllerTest.php
index 3bf169440b6008748460ee777b381299e2b19c1f..4b3df0cc9bf77c9fb30f7b8ea9b859f976d07853 100644
--- a/tests/application/modules/admin/controllers/BibnumControllerTest.php
+++ b/tests/application/modules/admin/controllers/BibnumControllerTest.php
@@ -24,6 +24,9 @@ class Admin_BibnumControllerIndexTest extends Admin_AbstractControllerTestCase {
     parent::setUp();
     RessourcesNumeriquesFixtures::activate1Dtouch();
     RessourcesNumeriquesFixtures::deactivateJamendo();
+    $this->fixture('Class_AdminVar',
+                   ['id' => 'Bibliondemand_SSO_URL',
+                    'valeur' => 'http://une-url.fr']);
     $this->dispatch('/admin/bibnum', true);
   }
 
@@ -78,6 +81,24 @@ class Admin_BibnumControllerIndexTest extends Admin_AbstractControllerTestCase {
     $this->assertNotXPath('//table[@class="digital_connectors"]//tr[@data-code="jamendo"]/td//div[@class="enabled"]');
   }
 
+
+  /** @test */
+  public function bibliondemandShouldBeSsoReady() {
+    $this->assertXPathContentContains('//table[@class="digital_connectors"]//tr[@data-code="bibliondemand"]/td', 'SSO');
+  }
+
+
+  /** @test */
+  public function bibliondemandShouldBeHarvestReady() {
+    $this->assertXPathContentContains('//table[@class="digital_connectors"]//tr[@data-code="bibliondemand"]/td', 'Recherche fédérée');
+  }
+
+
+  /** @test */
+  public function biliondemandShouldBeEnabled() {
+    $this->assertXPathContentContains('//table[@class="digital_connectors"]//tr[@data-code="bibliondemand"]/td//div[@class="enabled"]',
+                                      'Activé');
+  }
 }
 
 
diff --git a/tests/application/modules/opac/controllers/ModulesControllerTest.php b/tests/application/modules/opac/controllers/ModulesControllerTest.php
index 4cee8f6e64550c0b23b8823d3efe245c854c7e09..628ef8e985b5ededeea7f4f1d45f93429c8a147d 100644
--- a/tests/application/modules/opac/controllers/ModulesControllerTest.php
+++ b/tests/application/modules/opac/controllers/ModulesControllerTest.php
@@ -233,3 +233,51 @@ class ModulesControllerOrthodidacteUserWithGroupWithRightCasTest extends Abstrac
                           $this->_response->getBody());
   }
 }
+
+
+
+
+class ModulesControllerBibliondemandSsoTest extends AbstractControllerTestCase {
+  protected $_storm_default_to_volatile = true;
+
+  public function setUp() {
+    parent::setUp();
+
+    $this->fixture('Class_AdminVar',
+                   ['id' => 'Bibliondemand_SSO_URL',
+                    'valeur' => 'http://numerique-pasdecalais.bibliondemand.com/logon.aspx?provider=SsoCas&sso-id=cg62-saintomer']);
+
+    Class_Users::getIdentity()
+      ->beAbonneSIGB()
+      ->setPassword('passe')
+      ->setIdabon('4343')
+      ->setBib(Class_Bib::find(1))
+      ->assertSave();
+
+    $unimarc = (new Class_Indexation_PseudoNotice_UnimarcVisitor())
+      ->visitCatalogAgency('Bibliondemand')
+      ->visitLocations(['1DTOUCH'])
+      ->visitPrivateUrl('http://music.1dtouch.com/users/auth/assa?dest=albums/137962&bibid=CG62')
+      ->getUnimarc();
+
+    $item = $this->fixture('Class_Exemplaire', ['id' => 56, 'cote' => '789456456']);
+
+    $record = $this->fixture('Class_Notice', ['id' => 21,
+                                              'exemplaires' => [$item],
+                                              'unimarc' => $unimarc]);
+  }
+
+
+  /** @test */
+  public function unknownRecordshouldRedirectToReferer() {
+    $this->dispatch('/modules/sso/id/44', true);
+    $this->assertRedirect();
+  }
+
+
+  /** @test */
+  public function record21ShouldRedirectToBibliondemandSso() {
+    $this->dispatch('/modules/sso/id/21', true);
+    $this->assertXPathContentContains('//script', 'document.location.href="http://numerique-pasdecalais.bibliondemand.com/logon.aspx?provider=SsoCas&sso-id=cg62-saintomer&returnUrl='. urlencode('http://music.1dtouch.com/users/auth/assa?dest=albums/137962&bibid=CG62').'"', $this->_response->getBody());
+  }
+}
diff --git a/tests/application/modules/opac/controllers/NoticeAjaxControllerBibliondemandTest.php b/tests/application/modules/opac/controllers/NoticeAjaxControllerBibliondemandTest.php
new file mode 100644
index 0000000000000000000000000000000000000000..53757a99d66257d94b27bde9424e4efb8355daa5
--- /dev/null
+++ b/tests/application/modules/opac/controllers/NoticeAjaxControllerBibliondemandTest.php
@@ -0,0 +1,56 @@
+<?php
+/**
+ * Copyright (c) 2012-2017, Agence Française Informatique (AFI). All rights reserved.
+ *
+ * BOKEH is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU AFFERO GENERAL PUBLIC LICENSE as published by
+ * the Free Software Foundation.
+ *
+ * There are special exceptions to the terms and conditions of the AGPL as it
+ * is applied to this software (see README file).
+ *
+ * BOKEH is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ * GNU AFFERO GENERAL PUBLIC LICENSE for more details.
+ *
+ * You should have received a copy of the GNU AFFERO GENERAL PUBLIC LICENSE
+ * along with BOKEH; if not, write to the Free Software
+ * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301  USA
+ */
+
+
+class NoticeAjaxControllerBibliondemandItemsTest extends AbstractControllerTestCase {
+  protected $_storm_default_to_volatile = true;
+
+  public function setUp() {
+    parent::setUp();
+
+    ZendAfi_Auth::getInstance()->clearIdentity();
+
+    $unimarc = (new Class_Indexation_PseudoNotice_UnimarcVisitor())
+      ->visitCatalogAgency('Bibliondemand')
+      ->visitLocations(['1DTOUCH'])
+      ->getUnimarc();
+
+    $item = $this->fixture('Class_Exemplaire',
+                           ['id' => 56,
+                            'cote' => '789456456']);
+
+    $record = $this->fixture('Class_Notice',
+                             ['id' => 21,
+                              'exemplaires' => [$item],
+                              'unimarc' => $unimarc]);
+
+    $this->dispatch('/opac/noticeajax/exemplaires/id_notice/21', true);
+  }
+
+
+  /** @test */
+  public function shouldContainsLinkToModulesSso() {
+    $this->assertXPathContentContains('//table//td/a[contains(@href, "/modules/sso/id/21")]',
+                                      'Consulter en ligne',
+                                      $this->_response->getBody());
+  }
+}
+?>
\ No newline at end of file
diff --git a/tests/library/Class/Cosmogramme/Integration/Bibliondemand.xml b/tests/library/Class/Cosmogramme/Integration/Bibliondemand.xml
new file mode 100644
index 0000000000000000000000000000000000000000..0710412b9cd85e8cff0538cd593142b62f7019d4
--- /dev/null
+++ b/tests/library/Class/Cosmogramme/Integration/Bibliondemand.xml
@@ -0,0 +1 @@
+<?xml version="1.0" encoding="UTF-8"?><OAI-PMH xmlns="http://www.openarchives.org/OAI/2.0/" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://www.openarchives.org/OAI/2.0/ http://www.openarchives.org/OAI/2.0/OAI-PMH.xsd"><responseDate>2017-02-14T09:28:12Z</responseDate><request verb="ListRecords">oai.bibliondemand.com/oaiserver.ashx</request><ListRecords><record><header><identifier>CG62_5532b9e4e47a7d41fe81d125d3ba7863</identifier><datestamp>2016-12-23T08:42:33Z</datestamp></header><metadata><oai_dc:dc xmlns:oai_dc="http://www.openarchives.org/OAI/2.0/oai_dc/" xmlns:dc="http://purl.org/dc/elements/1.1/"><dc:identifier>Url : http://music.1dtouch.com/users/auth/assa?dest=albums/137962&amp;bibid=CG62</dc:identifier><dc:identifier>Vignette : http://play-assets-production.1dtouch.com/cargo/medium_8017297007164.jpg</dc:identifier><dc:title>Piero Soffici Quartet</dc:title><dc:type>Document sonore</dc:type><dc:creator>Piero Soffici Quartet</dc:creator><dc:source>1DTOUCH</dc:source><dc:publisher /></oai_dc:dc></metadata></record><record><header><identifier>CG62_516fbc7cfe8fe5e7b6a3c5577b9894dc</identifier><datestamp>2016-12-23T08:42:33Z</datestamp></header><metadata><oai_dc:dc xmlns:oai_dc="http://www.openarchives.org/OAI/2.0/oai_dc/" xmlns:dc="http://purl.org/dc/elements/1.1/"><dc:identifier>Url : http://music.1dtouch.com/users/auth/assa?dest=albums/109814&amp;bibid=CG62</dc:identifier><dc:identifier>Vignette : http://play-assets-production.1dtouch.com/cargo/medium_5051142090607.jpg</dc:identifier><dc:title>Island Row</dc:title><dc:type>Document sonore</dc:type><dc:creator>Capitol K</dc:creator><dc:source>1DTOUCH</dc:source><dc:publisher /></oai_dc:dc></metadata></record><record><header><identifier>CG62_0856eef7ca0440685bd51ac04d5258ab</identifier><datestamp>2016-12-23T08:42:33Z</datestamp></header><metadata><oai_dc:dc xmlns:oai_dc="http://www.openarchives.org/OAI/2.0/oai_dc/" xmlns:dc="http://purl.org/dc/elements/1.1/"><dc:identifier>Url : http://music.1dtouch.com/users/auth/assa?dest=albums/121552&amp;bibid=CG62</dc:identifier><dc:identifier>Vignette : http://play-assets-production.1dtouch.com/cargo/medium_3614970852384.jpg</dc:identifier><dc:title>Tunos de Compostela</dc:title><dc:type>Document sonore</dc:type><dc:creator>Tunos De Compostela</dc:creator><dc:source>1DTOUCH</dc:source><dc:publisher /></oai_dc:dc></metadata></record><record><header><identifier>CG62_432ebd04ac7e70f70ecb65a34ad86a79</identifier><datestamp>2016-12-23T08:42:33Z</datestamp></header><metadata><oai_dc:dc xmlns:oai_dc="http://www.openarchives.org/OAI/2.0/oai_dc/" xmlns:dc="http://purl.org/dc/elements/1.1/"><dc:identifier>Url : http://music.1dtouch.com/users/auth/assa?dest=albums/20206&amp;bibid=CG62</dc:identifier><dc:identifier>Vignette : http://play-assets-production.1dtouch.com/cargo/medium_3700738801098.jpg</dc:identifier><dc:title>Fantasy Classic, No. 1: Atmosphere Synthetic Harpsichord</dc:title><dc:type>Document sonore</dc:type><dc:creator>Christian Levitan</dc:creator><dc:source>1DTOUCH</dc:source><dc:publisher>LEVITAN</dc:publisher></oai_dc:dc></metadata></record><record><header><identifier>CG62_0925032</identifier><datestamp>2017-02-09T07:44:56Z</datestamp></header><metadata><oai_dc:dc xmlns:oai_dc="http://www.openarchives.org/OAI/2.0/oai_dc/" xmlns:dc="http://purl.org/dc/elements/1.1/"><dc:identifier>Url : http://media.citedelamusique.fr/medias/logon/62-pasdecalais-md?backUrl=http://media.citedelamusique.fr/medias/doc/EXTRANET/CIMU/0925032/DETAIL</dc:identifier><dc:identifier>Vignette : http://media.citedelamusique.fr/images/DocThumbs/CMVI000139300.jpg</dc:identifier><dc:title>We Want Miles. Birth of the Cool Suite</dc:title><dc:type>Document vidéo</dc:type><dc:creator>Joe Lovano</dc:creator><dc:creator>Ralph Lalama</dc:creator><dc:creator>Steve Slagle</dc:creator><dc:creator>Gary Smulyan</dc:creator><dc:creator>Larry Farrell</dc:creator><dc:creator>Barry Ries</dc:creator><dc:creator>James Weidman</dc:creator><dc:creator>Cameron Brown</dc:creator><dc:creator>Lewis Nash</dc:creator><dc:creator>Joe Lovano Nonet</dc:creator><dc:subject>Jazz</dc:subject><dc:source>CIMU</dc:source><dc:publisher>Cité de la musique</dc:publisher></oai_dc:dc></metadata></record><record><header><identifier>CG62_0980101</identifier><datestamp>2017-02-09T07:44:56Z</datestamp></header><metadata><oai_dc:dc xmlns:oai_dc="http://www.openarchives.org/OAI/2.0/oai_dc/" xmlns:dc="http://purl.org/dc/elements/1.1/"><dc:identifier>Url : http://media.citedelamusique.fr/medias/logon/62-pasdecalais-md?backUrl=http://media.citedelamusique.fr/medias/doc/EXTRANET/CIMU/0980101/DETAIL</dc:identifier><dc:identifier>Vignette : http://media.citedelamusique.fr/images/DocThumbs/CMVI000020202_02.jpg</dc:identifier><dc:title>Musiques pour harpes et pluriarc des Fang du Gabon</dc:title><dc:type>Document vidéo</dc:type><dc:subject>Musiques traditionnelles</dc:subject><dc:source>CIMU</dc:source><dc:publisher>Cité de la musique</dc:publisher></oai_dc:dc></metadata></record><record><header><identifier>CG62_0935471</identifier><datestamp>2017-02-09T07:44:56Z</datestamp></header><metadata><oai_dc:dc xmlns:oai_dc="http://www.openarchives.org/OAI/2.0/oai_dc/" xmlns:dc="http://purl.org/dc/elements/1.1/"><dc:identifier>Url : http://media.citedelamusique.fr/medias/logon/62-pasdecalais-md?backUrl=http://media.citedelamusique.fr/medias/doc/EXTRANET/CIMU/0935471/DETAIL</dc:identifier><dc:identifier>Vignette : http://media.citedelamusique.fr/images/DocThumbs/CMVI000144001_02.jpg</dc:identifier><dc:title>Salve Regina, cantate en la majeur pour mezzo-soprano et cordes</dc:title><dc:type>Document vidéo</dc:type><dc:creator>Johann Adolf Hasse</dc:creator><dc:creator>Bernarda Fink</dc:creator><dc:creator>Berliner Barock Solisten</dc:creator><dc:subject>Musique baroque</dc:subject><dc:source>CIMU</dc:source><dc:publisher>Cité de la musique</dc:publisher></oai_dc:dc></metadata></record><record><header><identifier>CG62_804</identifier><datestamp>2017-02-09T07:45:32Z</datestamp></header><metadata><oai_dc:dc xmlns:oai_dc="http://www.openarchives.org/OAI/2.0/oai_dc/" xmlns:dc="http://purl.org/dc/elements/1.1/"><dc:identifier>Url : http://www.mediatheque-numerique.com/films/situation-of-the-street</dc:identifier><dc:identifier>Vignette : http://media.universcine.com/34/3d/343d4726-7c56-11de-85a1-871b2e208cb2.jpg</dc:identifier><dc:title>Situation of the street</dc:title><dc:type>Document vidéo</dc:type><dc:creator>Gogola, Jan (Réalisateur)</dc:creator><dc:contributor>Holubová, Eva (Acteur)</dc:contributor><dc:contributor>Pollert, Lukáš (Acteur)</dc:contributor><dc:contributor>Stivínová, Zuzana (Acteur)</dc:contributor><dc:contributor>Marek, Petr (Acteur)</dc:contributor><dc:contributor>Saudek, Jan (Acteur)</dc:contributor><dc:contributor>Sobota, Luděk (Acteur)</dc:contributor><dc:contributor>Kratochvílová, Jana (Acteur)</dc:contributor><dc:contributor>Krejčík, Jiří (Acteur)</dc:contributor><dc:contributor>Dušek, Jaroslav (Acteur)</dc:contributor><dc:subject>Allégorie</dc:subject><dc:subject>Documentaire</dc:subject><dc:subject>Documentaire</dc:subject><dc:subject>Documentaire</dc:subject><dc:subject>Engagé</dc:subject><dc:subject>Etrange</dc:subject><dc:subject>Historique</dc:subject><dc:subject>Politique</dc:subject><dc:subject>Politique</dc:subject><dc:subject>Ville</dc:subject><dc:source>MEDIATHEQUE_NUMERIQUE</dc:source><dc:language>czech</dc:language><dc:description>A travers l'histoire de la célèbre rue Nationale (Národní třída), à Prague, un témoignage historique, radical et polémique, capté ici et maintenant, sur la nation Tchèque. Charlot et Hitler en témoins alors qu'une fille nue comme un fantôme fend la foule en courant... Un documentaire mariant une approche expérimentale et le questionnement politique du monde.  Sous-titre du film : "La mer Tchèque prise dans 18 tsunamis"</dc:description></oai_dc:dc></metadata></record></ListRecords></OAI-PMH>
diff --git a/tests/library/Class/Cosmogramme/Integration/RecordPhaseBibliondemandTest.php b/tests/library/Class/Cosmogramme/Integration/RecordPhaseBibliondemandTest.php
new file mode 100644
index 0000000000000000000000000000000000000000..46c3962d701247a379e88311c6a84008d3ac167c
--- /dev/null
+++ b/tests/library/Class/Cosmogramme/Integration/RecordPhaseBibliondemandTest.php
@@ -0,0 +1,212 @@
+<?php
+/**
+ * Copyright (c) 2012-2014, Agence Française Informatique (AFI). All rights reserved.
+ *
+ * BOKEH is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU AFFERO GENERAL PUBLIC LICENSE as published by
+ * the Free Software Foundation.
+ *
+ * There are special exceptions to the terms and conditions of the AGPL as it
+ * is applied to this software (see README file).
+ *
+ * BOKEH is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ * GNU AFFERO GENERAL PUBLIC LICENSE for more details.
+ *
+ * You should have received a copy of the GNU AFFERO GENERAL PUBLIC LICENSE
+ * along with BOKEH; if not, write to the Free Software
+ * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301  USA
+ */
+
+require_once('cosmogramme/php/fonctions/variables.php');
+require_once(__DIR__.'/PhaseNoticeTestCase.php');
+
+abstract class AbstractRecordPhaseBibliondemandTestCase extends PhaseNoticeTestCase {
+  protected $_storm_default_to_volatile = true;
+
+  protected function _prepareFixtures() {
+    parent::_prepareFixtures();
+
+    Class_TypeDoc::newInstanceWithId(15,
+                                     ['label' => 'Film']);
+
+    Class_TypeDoc::newInstanceWithId(16,
+                                     ['label' => 'Sonore']);
+
+    $settings = ['format' => Class_IntProfilDonnees::FORMAT_BIBLIONDEMAND,
+                 'type_fichier' => Class_IntProfilDonnees::FT_RECORDS,
+                 '0_type' => [''],
+                 '0_format' => [''],
+                 '15_type' => ['Document vidéo'],
+                 '15_format' => [''],
+                 '16_type' => ['Document sonore'],
+                 '16_format' => [''],];
+
+    $serialized_settings = (new Class_ProfileSerializer($settings))->serializeDatas();
+
+    Class_IntProfilDonnees::find(102)
+      ->setTypeFichier(Class_IntProfilDonnees::FT_RECORDS)
+      ->setFormat(Class_IntProfilDonnees::FORMAT_BIBLIONDEMAND)
+      ->setAttributs($serialized_settings)
+      ->save();
+
+    $this->fixture('Class_IntBib',
+                   ['id' => 3,
+                    'nom_court' => 'Bibliondemand']);
+
+    Class_Cosmogramme_Integration::find(999)
+      ->setIdBib(3)
+      ->setFichier('http://oai.bibliondemand.com/oaiserver.ashx?verb=ListRecords&metadataPrefix=oai_dc&set=CG62')
+      ->save();
+
+    $xml = file_get_contents(__DIR__.'/Bibliondemand.xml');
+    $webclient = $this->mock();
+    $webclient->whenCalled('open_url')
+              ->with('http://oai.bibliondemand.com/oaiserver.ashx?verb=ListRecords&metadataPrefix=oai_dc&set=CG62')
+              ->answers($xml)
+              ->beStrict();
+
+    Class_WebService_OAI::setDefaultHttpClient($webclient);
+  }
+
+
+  public function tearDown() {
+    Class_WebService_OAI::setDefaultHttpClient(null);
+    parent::tearDown();
+  }
+}
+
+
+
+class RecordPhaseBibliondemandSimpleTest extends AbstractRecordPhaseBibliondemandTestCase {
+  /** @test */
+  public function logShouldNotContainsError() {
+    $this->assertNotLogContains('Erreur');
+  }
+
+
+  /** @test */
+  public function totalNumberOfRecordsFromOAIShouldBeEight() {
+    $this->assertEquals(8, Class_Notice::count());
+  }
+
+
+  /**
+   * @test
+   */
+  public function firstRecordTitleShoudBePieroSofficiQuartet() {
+    $this->assertContains('Piero Soffici Quartet', Class_Notice::find(1)->getTitrePrincipal());
+  }
+
+
+  /**
+   * @test
+   */
+  public function firstRecordAuthorShouldBePieroSofficiQuartet() {
+    $this->assertEquals('Piero Soffici Quartet', Class_Notice::find(1)->getAuteurPrincipal());
+  }
+
+
+  /**
+   * @test
+   */
+  public function situationOfTheStreetShouldHaveSummary() {
+    $this->assertNotEmpty(Class_Notice::find(8)->getResume());
+  }
+
+
+  /**
+   * @test
+   */
+  public function firstRecordShouldHaveOneItem() {
+    $this->assertCount(1, Class_Notice::find(1)->getExemplaires());
+  }
+
+
+  /**
+   * @test
+   */
+  public function firstRecordDocTypeShouldBeShouldSonore() {
+    $this->assertEquals('Sonore', Class_Notice::find(1)->getTypeDocLabel());
+  }
+
+
+  /**
+   * @test
+   */
+  public function firstRecordItemBarcodeShouldBeOaiIdentifier() {
+    $item = Class_Notice::find(1)->getExemplaires()[0];
+    $this->assertEquals('CG62_5532b9e4e47a7d41fe81d125d3ba7863', $item->getCodeBarres());
+  }
+
+
+  /**
+   * @test
+   */
+  public function firstRecordItemLibraryShouldBeBNFGallicaAtlas() {
+    $item = Class_Notice::find(1)->getExemplaires()[0];
+    $this->assertEquals('Bibliondemand', $item->getIntBib()->getNomCourt());
+  }
+
+
+  /** @test */
+  public function lastRecordDocTypeShouldBeShouldFilm() {
+    $this->assertEquals('Film', Class_Notice::find(8)->getTypeDocLabel());
+  }
+
+
+  /** @test */
+  public function firstRecordIdShouldBeGC62() {
+    $this->assertEquals('CG62_5532b9e4e47a7d41fe81d125d3ba7863',
+                        Class_Notice::find(1)->get_subfield('001')[0]);
+  }
+
+
+  /** @test */
+  public function firstRecordThumbShouldBeMusicDot1dtouchUrl() {
+    $this->assertEquals('http://play-assets-production.1dtouch.com/cargo/medium_8017297007164.jpg',
+                        Class_Notice::find(1)->get_subfield('959', 'u')[0]);
+  }
+
+
+  /** @test */
+  public function firstRecordResourceUrlShouldBeIn956u() {
+    $this->assertEquals('http://music.1dtouch.com/users/auth/assa?dest=albums/137962&bibid=CG62',
+                        Class_Notice::find(1)->get_subfield('956', 'u')[0]);
+  }
+
+
+  /** @test */
+  public function firstRecordProviderShouldBeBibliondemand() {
+    $this->assertEquals('Bibliondemand',
+                        Class_Notice::find(1)->get_subfield('801', 'b')[0]);
+  }
+
+
+  /** @test */
+  public function firstRecordSourceShould1dtouch() {
+    $this->assertEquals('1DTOUCH',
+                        Class_Notice::find(1)->get_subfield('852', 'a')[0]);
+  }
+
+
+  /** @test */
+  public function eigthRecordSourceShouldMediathequeNumerique() {
+    $this->assertEquals('MEDIATHEQUE_NUMERIQUE',
+                        Class_Notice::find(8)->get_subfield('852', 'a')[0]);
+  }
+
+
+  /** @test */
+  public function firstRecordShouldNotHave856u() {
+    $this->assertEmpty(Class_Notice::find(1)->get_subfield('856', 'u'));
+  }
+
+
+  /** @test */
+  public function firstRecordItemUrlShouldBeEmpty() {
+    $item = Class_Notice::find(1)->getExemplaires()[0];
+    $this->assertEmpty($item->getUrl());
+  }
+}