diff --git a/CMakeLists.txt b/CMakeLists.txt
index 7d311e83f2f4f96da836cc88463ee6eb66ab70a1..b5ac62eb0b17bebb2c3092f79b269cea1021fb6d 100644
--- a/CMakeLists.txt
+++ b/CMakeLists.txt
@@ -29,8 +29,11 @@ set(CMAKE_DISABLE_SOURCE_CHANGES ON)
 set(CMAKE_DISABLE_IN_SOURCE_BUILD ON)
 list(APPEND CMAKE_MODULE_PATH ${PROJECT_SOURCE_DIR}/cmake)
 
+include_directories(${CMAKE_SOURCE_DIR})
+
 add_subdirectory(libs)
 add_subdirectory(objectstore)
+add_subdirectory(objectstore_middletier)
 add_subdirectory(tests)
 add_subdirectory(xroot_clients)
 add_subdirectory(xroot_plugins)
diff --git a/objectstore/AdminUsersList.cpp b/objectstore/AdminUsersList.cpp
new file mode 100644
index 0000000000000000000000000000000000000000..9b3676f348a9645af300980fa0407e4639a4a774
--- /dev/null
+++ b/objectstore/AdminUsersList.cpp
@@ -0,0 +1,31 @@
+#include "AdminUsersList.hpp"
+
+namespace cta { namespace objectstore {
+  
+AdminUsersList::AdminUsersList(Backend& os):
+  ObjectOps<serializers::AdminUsersList>(os) {}
+
+AdminUsersList::AdminUsersList(const std::string & name, Backend& os):
+  ObjectOps<serializers::AdminUsersList>(os, name) {}
+
+void AdminUsersList::add(const cta::AdminUser& adminUser) {
+  // Check that the admin user is not already present
+  ::google::protobuf::RepeatedPtrField<serializers::AdminUser>* list =
+      m_payload.mutable_element();
+  for (size_t i=0; i<(size_t)list->size(); i++) {
+    if (adminUser.getUser().getUid() == list->Get(i).user().uid()) {
+      throw cta::exception::Exception("Attempt to duplicate entry in AdminUsersList");
+    }
+  }
+  // Insert the new admin user
+  serializers::AdminUser * newEntry = list->Add();
+  // Set the content of the new entry
+  newEntry->mutable_user()->set_uid(adminUser.getUser().getUid());
+  newEntry->mutable_user()->set_gid(adminUser.getUser().getUid());
+  newEntry->mutable_creator()->set_uid(adminUser.getCreator().getUid());
+  newEntry->mutable_creator()->set_gid(adminUser.getCreator().getGid());
+  newEntry->set_creationtime(adminUser.getCreationTime());
+  newEntry->set_comment(adminUser.getComment());
+}
+
+}}
\ No newline at end of file
diff --git a/objectstore/AdminUsersList.hpp b/objectstore/AdminUsersList.hpp
new file mode 100644
index 0000000000000000000000000000000000000000..cfbf0ac42b7db0cf1a24aaa6dd188c16e2f811b9
--- /dev/null
+++ b/objectstore/AdminUsersList.hpp
@@ -0,0 +1,22 @@
+#pragma once
+
+#include "ObjectOps.hpp"
+#include "objectstore/cta.pb.h"
+#include "libs/middletier/AdminUser.hpp"
+
+namespace cta { namespace objectstore {
+
+/**
+ * Class containing the list of admin users
+ */
+
+class AdminUsersList: public ObjectOps<serializers::AdminUsersList> {
+public:
+  AdminUsersList(Backend & os);
+  
+  AdminUsersList(const std::string & name, Backend & os);
+  
+  void add(const cta::AdminUser & adminUser);
+};
+
+}}
\ No newline at end of file
diff --git a/objectstore/BackendTest.cpp b/objectstore/BackendTest.cpp
index 7f8b1062fdc6e4321b54853b844ace45952419a6..ccf819629e180b3bc675a43255cfd928e882921b 100644
--- a/objectstore/BackendTest.cpp
+++ b/objectstore/BackendTest.cpp
@@ -3,6 +3,8 @@
 #include "BackendRados.hpp"
 #include "exception/Exception.hpp"
 
+namespace unitTests {
+
 TEST_P(BackendAbstractTest, BasicReadWrite) {
   std::cout << "Type=" << m_os->typeName() << std::endl;
   const std::string testValue = "1234";
@@ -63,4 +65,6 @@ cta::objectstore::BackendRados osRados("tapetest", "tapetest");
 INSTANTIATE_TEST_CASE_P(BackendTest, BackendAbstractTest, ::testing::Values(&osVFS, &osRados));
 #else
 INSTANTIATE_TEST_CASE_P(BackendTest, BackendAbstractTest, ::testing::Values((cta::objectstore::Backend*)&osVFS));
-#endif
\ No newline at end of file
+#endif
+
+}
diff --git a/objectstore/BackendTest.hpp b/objectstore/BackendTest.hpp
index 94acd8eca9c1ef24d902fc8ab256531c7eee4776..7a97838705751bd78abc93169a01e5efa5542b76 100644
--- a/objectstore/BackendTest.hpp
+++ b/objectstore/BackendTest.hpp
@@ -3,6 +3,8 @@
 #include <gtest/gtest.h>
 #include "Backend.hpp"
 
+namespace unitTests {
+
 class BackendAbstractTest: public ::testing::TestWithParam<cta::objectstore::Backend *> {
 protected:
   BackendAbstractTest() {}
@@ -11,3 +13,6 @@ protected:
   }
   cta::objectstore::Backend * m_os;
 };
+
+}
+
diff --git a/objectstore/CMakeLists.txt b/objectstore/CMakeLists.txt
index 8cfc03e97d26acbb3f8249881203ad370ad756ba..f24a7b4a1283b8117bd80f6a0c873e1ed1a26feb 100644
--- a/objectstore/CMakeLists.txt
+++ b/objectstore/CMakeLists.txt
@@ -16,6 +16,7 @@ add_library (CTAObjectStore
   Agent.cpp
   AgentRegister.cpp
   AgentWatchdog.cpp
+  AdminUsersList.cpp
   BackendVFS.cpp
   BackendRados.cpp
   ObjectOps.cpp
diff --git a/objectstore/FIFOTest.cpp b/objectstore/FIFOTest.cpp
index d088cca03a1f973d8eef0c4a24d0c89f09a657aa..a8ceb281c729313616c938651ef52540e98681a5 100644
--- a/objectstore/FIFOTest.cpp
+++ b/objectstore/FIFOTest.cpp
@@ -4,6 +4,8 @@
 #include "FIFO.hpp"
 #include "Agent.hpp"
 
+namespace unitTests {
+
 TEST(FIFO, BasicFuctionnality) {
   cta::objectstore::BackendVFS be;
   cta::objectstore::Agent agent(be);
@@ -68,4 +70,6 @@ TEST(FIFO, BasicFuctionnality) {
   ASSERT_EQ(0, ff.size());
   ff.remove();
   ASSERT_EQ(false, ff.exists());
-}
\ No newline at end of file
+}
+
+}
diff --git a/objectstore/GarbageCollectorTest.cpp b/objectstore/GarbageCollectorTest.cpp
index d5bf4ef1c54b71a006091aa06d47d5b7f098f349..8f3d0900fc1b39c7178dc760540c6e8d2e2912ca 100644
--- a/objectstore/GarbageCollectorTest.cpp
+++ b/objectstore/GarbageCollectorTest.cpp
@@ -7,6 +7,8 @@
 #include "AgentRegister.hpp"
 #include "RootEntry.hpp"
 
+namespace unitTests {
+
 TEST(GarbageCollector, BasicFuctionnality) {
   cta::objectstore::BackendVFS be;
   cta::objectstore::Agent agent(be);
@@ -80,4 +82,6 @@ TEST(GarbageCollector, BasicFuctionnality) {
   ffLock.lock(ff);
   ff.fetch();
   ASSERT_EQ(100, ff.size());
-}
\ No newline at end of file
+}
+
+}
diff --git a/objectstore/RootEntry.hpp b/objectstore/RootEntry.hpp
index 1ac27450abc79a16a68147b303bc9e621c20a539..00c2a131999275f25507290298f419fbd9da3d7f 100644
--- a/objectstore/RootEntry.hpp
+++ b/objectstore/RootEntry.hpp
@@ -34,6 +34,12 @@ public:
   // Get the name of a (possibly freshly created) job pool
   std::string allocateOrGetJobPool(Agent & agent);
   
+  // Get the name of the AdminUsersList (or exception if not available)
+  std::string getAdminUsersList();
+  
+  // Get the name of a (possibly freshly created) AdminUsersList
+  std::string allocateOrGetAdminUsersList(Agent & agent);
+  
 private:
   void addIntendedAgentRegistry(const std::string & name);
   
diff --git a/objectstore/RootEntryTest.cpp b/objectstore/RootEntryTest.cpp
index 7548a46899332410a27f4c6504f49f8919acb322..6f70cf61fba3e722b960f66bc0fc93a09cd41c21 100644
--- a/objectstore/RootEntryTest.cpp
+++ b/objectstore/RootEntryTest.cpp
@@ -4,6 +4,8 @@
 #include "RootEntry.hpp"
 #include "Agent.hpp"
 
+namespace unitTests {
+
 TEST(RootEntry, BasicAccess) {
   cta::objectstore::BackendVFS be;
   { 
@@ -39,4 +41,6 @@ TEST(RootEntry, BasicAccess) {
   cta::objectstore::ScopedExclusiveLock lock(re);
   re.remove();
   ASSERT_EQ(false, re.exists());
-}
\ No newline at end of file
+}
+
+}
diff --git a/objectstore/cta.proto b/objectstore/cta.proto
index d2815b227e8be01d935e04480355c0a10b78b785..790e4e3a82b43b2a4e717f85d587683ac4e6f51e 100644
--- a/objectstore/cta.proto
+++ b/objectstore/cta.proto
@@ -12,6 +12,7 @@ enum ObjectType {
   RecallJob_t = 6;
   Counter_t = 7;
   FIFO_t = 8;
+  AdminUsersList_t = 9;
   GenericObject_t = 1000;
 }
 
@@ -68,6 +69,8 @@ message RootEntry {
   repeated string agentregisterintentlog = 92;
   optional string jobpool = 93;
   repeated string jobpoolintentlog = 94;
+  optional string adminuserslist = 95;
+  repeated string adminuserslistintentlog = 96;
 }
 
 // The registers (simple name arrays)
@@ -150,4 +153,20 @@ message RecallFIFO {}
 
 message MigrationFIFO {}
 
+message UserIdentity {
+  required uint32 uid = 8001;
+  required uint32 gid = 8002;
+}
+
+message AdminUser  {
+  required UserIdentity user = 8010;
+  required UserIdentity creator = 8011;
+  required uint64 creationTime = 8012;
+  required string comment = 8013;
+}
+
+message AdminUsersList {
+  repeated AdminUser element = 8020;
+}
+
 
diff --git a/objectstore/exception/Exception.hpp b/objectstore/exception/Exception.hpp
index b1ca73e7ad578bbdf246d39c2f1eb2caf91bea09..55412a305a592b76a9a00cca9ffa03465e67e121 100644
--- a/objectstore/exception/Exception.hpp
+++ b/objectstore/exception/Exception.hpp
@@ -24,7 +24,7 @@
 
 #pragma once
 
-#include "exception/Backtrace.hpp"
+#include "objectstore/exception/Backtrace.hpp"
 
 #include <exception>
 #include <sstream>
diff --git a/objectstore/tapeResourceManagerTest.cpp b/objectstore/tapeResourceManagerTest.cpp
index 8ca607ec6a7b605b2df1a786d29428f4b293ef2d..df0e13f2aaa3dc3ea43e37965d5cbd4cbc10f660 100644
--- a/objectstore/tapeResourceManagerTest.cpp
+++ b/objectstore/tapeResourceManagerTest.cpp
@@ -15,7 +15,7 @@
 #include "Counter.hpp"
 #include <math.h>
 
-
+namespace systemTests { 
 
 class jobExecutorThread: public cta::threading::Thread {
 public:
@@ -239,3 +239,5 @@ int main(void){
   
   return 0;
 }
+
+}
diff --git a/objectstore_middletier/CMakeLists.txt b/objectstore_middletier/CMakeLists.txt
new file mode 100644
index 0000000000000000000000000000000000000000..6df1a3271c69b6f9abdf5d2e83027b1ff8658030
--- /dev/null
+++ b/objectstore_middletier/CMakeLists.txt
@@ -0,0 +1,8 @@
+cmake_minimum_required (VERSION 2.6)
+
+include_directories(${CMAKE_CURRENT_SOURCE_DIR})
+include_directories(${CMAKE_BINARY_DIR})
+
+add_library (CTAObjectStoreMiddleTier
+  ObjectStoreMiddleTier.cpp
+)
\ No newline at end of file
diff --git a/objectstore_middletier/ObjectStoreMiddleTier.cpp b/objectstore_middletier/ObjectStoreMiddleTier.cpp
new file mode 100644
index 0000000000000000000000000000000000000000..5affbd7aac39e84753ed3decde0747117e4f382b
--- /dev/null
+++ b/objectstore_middletier/ObjectStoreMiddleTier.cpp
@@ -0,0 +1,47 @@
+#include "ObjectStoreMiddleTier.hpp"
+#include "objectstore/Backend.hpp"
+#include "objectstore/RootEntry.hpp"
+#include "objectstore/AdminUsersList.hpp"
+#include "../libs/middletier/Exception.hpp"
+
+namespace cta {
+
+//------------------------------------------------------------------------------
+// constructor
+//------------------------------------------------------------------------------
+OStoreMiddleTierAdmin::OStoreMiddleTierAdmin(objectstore::Backend& backend):
+  m_backend(backend) {
+  // check that we can at least access the root entry
+  objectstore::RootEntry re(m_backend);
+  objectstore::ScopedSharedLock reLock(re);
+  re.fetch();
+}
+
+//------------------------------------------------------------------------------
+// destructor
+//------------------------------------------------------------------------------
+OStoreMiddleTierAdmin::~OStoreMiddleTierAdmin() throw() {
+}
+
+//------------------------------------------------------------------------------
+// createAdminUser
+//------------------------------------------------------------------------------
+void OStoreMiddleTierAdmin::createAdminUser(
+  const SecurityIdentity &requester,
+  const UserIdentity &user,
+  const std::string &comment) {
+  // TODO: authz is needed here!
+  // Find the admin users list from the root entry.
+  objectstore::RootEntry re(m_backend);
+  objectstore::ScopedSharedLock reLock(re);
+  re.fetch();
+  objectstore::AdminUsersList aul(re.getAdminUsersList(), m_backend);
+  reLock.release();
+  objectstore::ScopedExclusiveLock auLock(aul);
+  aul.fetch();
+  AdminUser au(user, requester.user, time(NULL), comment);
+  aul.add(au);
+  aul.commit();
+}
+
+}
diff --git a/objectstore_middletier/ObjectStoreMiddleTier.hpp b/objectstore_middletier/ObjectStoreMiddleTier.hpp
new file mode 100644
index 0000000000000000000000000000000000000000..1683594ecb02ece5884bacc70afba90be0d72016
--- /dev/null
+++ b/objectstore_middletier/ObjectStoreMiddleTier.hpp
@@ -0,0 +1,293 @@
+#pragma once
+
+#include "../libs/middletier/MiddleTierAdmin.hpp"
+
+namespace cta {
+
+namespace objectstore {
+  class Backend;
+}
+  
+/**
+ * The administration API of the the middle-tier.
+ * ObjectStore based implementation
+ */
+class OStoreMiddleTierAdmin: public MiddleTierAdmin {
+public:
+  /**
+   * Contructor
+   */
+  OStoreMiddleTierAdmin(objectstore::Backend & backend);
+  
+  /**
+   * Destructor
+   */
+  virtual ~OStoreMiddleTierAdmin() throw();
+
+  /**
+   * Creates the specified administrator.
+   *
+   * @param requester The identity of the user requesting the creation of the
+   * administrator.
+   * @param user The identity of the administrator.
+   * @param comment The comment describing the sministrator.
+   */
+  virtual void createAdminUser(
+    const SecurityIdentity &requester,
+    const UserIdentity &user,
+    const std::string &comment);
+
+  /**
+   * Deletes the specified administrator.
+   *
+   * @param requester The identity of the user requesting the deletion of the
+   * administrator.
+   * @param user The identity of the administrator.
+   */
+  virtual void deleteAdminUser(
+    const SecurityIdentity &requester,
+    const UserIdentity &user);
+
+  /**
+   * Returns the current list of administrators in lexicographical order.
+   *
+   * @param requester The identity of the user requesting the list.
+   * @return The current list of administrators in lexicographical order.
+   */
+  virtual std::list<AdminUser> getAdminUsers(const SecurityIdentity &requester)
+   const;
+
+  /**
+   * Creates the specified administration host.
+   *
+   * @param requester The identity of the user requesting the creation of the
+   * administration host.
+   * @param hostName The network name of the administration host.
+   * @param comment The comment describing the administration host.
+   */
+  virtual void createAdminHost(
+    const SecurityIdentity &requester,
+    const std::string &hostName,
+    const std::string &comment);
+
+  /**
+   * Deletes the specified administration host.
+   *
+   * @param requester The identity of the user requesting the deletion of the
+   * administration host.
+   * @param hostName The network name of the administration host.
+   * @param comment The comment describing the administration host.
+   */
+  virtual void deleteAdminHost(
+    const SecurityIdentity &requester,
+    const std::string &hostName);
+
+  /**
+   * Returns the current list of administration hosts in lexicographical order.
+   *
+   * @param requester The identity of the user requesting the list.
+   * @return The current list of administration hosts in lexicographical order.
+   */
+  virtual std::list<AdminHost> getAdminHosts(const SecurityIdentity &requester)
+   const;
+
+  /**
+   * Creates the specified storage class.
+   *
+   * @param requester The identity of the user requesting the creation of the
+   * storage class.
+   * @param name The name of the storage class.
+   * @param nbCopies The number of copies a file associated with this storage
+   * class should have on tape.
+   * @param comment The comment describing the storage class.
+   */
+  virtual void createStorageClass(
+    const SecurityIdentity &requester,
+    const std::string &name,
+    const uint8_t nbCopies,
+    const std::string &comment);
+
+  /**
+   * Deletes the specified storage class.
+   *
+   * @param requester The identity of the user requesting the deletion of the
+   * storage class.
+   * @param name The name of the storage class.
+   */
+  virtual void deleteStorageClass(
+    const SecurityIdentity &requester,
+    const std::string &name);
+
+  /**
+   * Gets the current list of storage classes in lexicographical order.
+   *
+   * @param requester The identity of the user requesting the list.
+   * @return The current list of storage classes in lexicographical order.
+   */
+  virtual std::list<StorageClass> getStorageClasses(
+    const SecurityIdentity &requester) const;
+
+  /**
+   * Creates a tape pool with the specifed name.
+   *
+   * @param requester The identity of the user requesting the creation of the
+   * tape pool.
+   * @param name The name of the tape pool.
+   * @param nbDrives The maximum number of drives that can be concurrently
+   * assigned to this pool independent of whether they are archiving or
+   * retrieving files.
+   * @param nbPartialTapes The maximum number of tapes that can be partially
+   * full at any moment in time.
+   * @param comment The comment describing the tape pool.
+   */
+  virtual void createTapePool(
+    const SecurityIdentity &requester,
+    const std::string &name,
+    const uint16_t nbDrives,
+    const uint32_t nbPartialTapes,
+    const std::string &comment);
+
+  /**
+   * Delete the tape pool with the specifed name.
+   *
+   * @param requester The identity of the user requesting the deletion of the
+   * tape pool.
+   * @param name The name of the tape pool.
+   */
+  virtual void deleteTapePool(
+    const SecurityIdentity &requester,
+    const std::string &name);
+
+  /**
+   * Gets the current list of tape pools in lexicographical order.
+   *
+   * @param requester The identity of the user requesting the list.
+   * @return The current list of tape pools in lexicographical order.
+   */
+  virtual std::list<TapePool> getTapePools(
+    const SecurityIdentity &requester) const;
+
+  /**
+   * Creates the specified archive route.
+   *
+   * @param requester The identity of the user requesting the creation of the
+   * archive route.
+   * @param storageClassName The name of the storage class that identifies the
+   * source disk files.
+   * @param copyNb The tape copy number.
+   * @param tapePoolName The name of the destination tape pool.
+   * @param comment The comment describing the archive route.
+   */
+  virtual void createArchiveRoute(
+    const SecurityIdentity &requester,
+    const std::string &storageClassName,
+    const uint8_t copyNb,
+    const std::string &tapePoolName,
+    const std::string &comment);
+
+  /**
+   * Deletes the specified archive route.
+   *
+   * @param requester The identity of the user requesting the deletion of the
+   * archive route.
+   * @param storageClassName The name of the storage class that identifies the
+   * source disk files.
+   * @param copyNb The tape copy number.
+   */
+  virtual void deleteArchiveRoute(
+    const SecurityIdentity &requester,
+    const std::string &storageClassName,
+    const uint8_t copyNb);
+
+  /**
+   * Gets the current list of archive routes.
+   *
+   * @param requester The identity of the user requesting the list.
+   */
+  virtual std::list<ArchiveRoute> getArchiveRoutes(
+    const SecurityIdentity &requester) const;
+
+  /**
+   * Creates a logical library with the specified name.
+   *
+   * @param requester The identity of the user requesting the creation of the
+   * logical library.
+   * @param name The name of the logical library.
+   * @param comment The comment describing the logical library.
+   */
+  virtual void createLogicalLibrary(
+    const SecurityIdentity &requester,
+    const std::string &name,
+    const std::string &comment);
+
+  /**
+   * Deletes the logical library with the specified name.
+   *
+   * @param requester The identity of the user requesting the deletion of the
+   * logical library.
+   * @param name The name of the logical library.
+   */
+  virtual void deleteLogicalLibrary(
+    const SecurityIdentity &requester,
+    const std::string &name);
+
+  /**
+   * Returns the current list of libraries in lexicographical order.
+   *
+   * @param requester The identity of the user requesting the list.
+   * @return The current list of libraries in lexicographical order.
+   */
+  virtual std::list<LogicalLibrary> getLogicalLibraries(
+    const SecurityIdentity &requester) const;
+
+  /**
+   * Creates a tape.
+   *
+   * @param requester The identity of the user requesting the creation of the
+   * tape.
+   * @param vid The volume identifier of the tape.
+   * @param logicalLibrary The name of the logical library to which the tape
+   * belongs.
+   * @param tapePoolName The name of the tape pool to which the tape belongs.
+   * @param capacityInBytes The capacity of the tape.
+   * @param comment The comment describing the logical library.
+   */
+  virtual void createTape(
+    const SecurityIdentity &requester,
+    const std::string &vid,
+    const std::string &logicalLibraryName,
+    const std::string &tapePoolName,
+    const uint64_t capacityInBytes,
+    const std::string &comment);
+
+  /**
+   * Deletes the tape with the specified volume identifier.
+   *
+   * @param requester The identity of the user requesting the deletion of the
+   * tape.
+   * @param vid The volume identifier of the tape.
+   */
+  virtual void deleteTape(
+    const SecurityIdentity &requester,
+    const std::string &vid);
+
+  /**
+   * Returns the current list of tapes in the lexicographical order of their
+   * volume identifiers.
+   *
+   * @param requester The identity of the user requesting the list.
+   * @return The current list of tapes in the lexicographical order of their
+   * volume identifiers.
+   */
+  virtual std::list<Tape> getTapes(
+    const SecurityIdentity &requester) const;
+  
+private:
+  /**
+   * Reference to the backend used for storing objects
+   */
+  objectstore::Backend & m_backend;
+};
+
+}
+