| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442 | 
							- /*=============================================================================
 
-   Library: XNAT/Core
 
-   Copyright (c) University College London,
 
-     Centre for Medical Image Computing
 
-   Licensed under the Apache License, Version 2.0 (the "License");
 
-   you may not use this file except in compliance with the License.
 
-   You may obtain a copy of the License at
 
-     http://www.apache.org/licenses/LICENSE-2.0
 
-   Unless required by applicable law or agreed to in writing, software
 
-   distributed under the License is distributed on an "AS IS" BASIS,
 
-   WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 
-   See the License for the specific language governing permissions and
 
-   limitations under the License.
 
- =============================================================================*/
 
- #include "ctkXnatObject.h"
 
- #include "ctkXnatObjectPrivate.h"
 
- #include "ctkXnatDataModel.h"
 
- #include "ctkXnatDefaultSchemaTypes.h"
 
- #include "ctkXnatException.h"
 
- #include "ctkXnatResource.h"
 
- #include "ctkXnatResourceFolder.h"
 
- #include "ctkXnatSession.h"
 
- #include <QDateTime>
 
- #include <QDebug>
 
- #include <QStringList>
 
- #include <QVariant>
 
- const QString ctkXnatObject::ID = "ID";
 
- const QString ctkXnatObject::NAME = "name";
 
- const QString ctkXnatObject::LABEL = "label";
 
- //----------------------------------------------------------------------------
 
- ctkXnatObject::ctkXnatObject(const ctkXnatObject&)
 
- {
 
-   throw ctkRuntimeException("Copy constructor not implemented");
 
- }
 
- //----------------------------------------------------------------------------
 
- ctkXnatObject::ctkXnatObject(ctkXnatObject* parent, const QString& schemaType)
 
- : d_ptr(new ctkXnatObjectPrivate())
 
- {
 
-   this->setParent(parent);
 
-   this->setSchemaType(schemaType);
 
- }
 
- //----------------------------------------------------------------------------
 
- ctkXnatObject::ctkXnatObject(ctkXnatObjectPrivate& dd, ctkXnatObject* parent, const QString& schemaType)
 
- : d_ptr(&dd)
 
- {
 
-   this->setParent(parent);
 
-   this->setSchemaType(schemaType);
 
- }
 
- //----------------------------------------------------------------------------
 
- ctkXnatObject::~ctkXnatObject()
 
- {
 
-   Q_D(ctkXnatObject);
 
-   foreach (ctkXnatObject* child, d->children)
 
-   {
 
-     delete child;
 
-   }
 
- }
 
- //----------------------------------------------------------------------------
 
- QString ctkXnatObject::id() const
 
- {
 
-   return this->property(ID);
 
- }
 
- //----------------------------------------------------------------------------
 
- void ctkXnatObject::setId(const QString& id)
 
- {
 
-   this->setProperty(ID, id);
 
- }
 
- //----------------------------------------------------------------------------
 
- QString ctkXnatObject::name() const
 
- {
 
-   return this->property(NAME);
 
- }
 
- //----------------------------------------------------------------------------
 
- void ctkXnatObject::setName(const QString& name)
 
- {
 
-   this->setProperty(NAME, name);
 
- }
 
- //----------------------------------------------------------------------------
 
- QString ctkXnatObject::description() const
 
- {
 
-   Q_D(const ctkXnatObject);
 
-   return d->description;
 
- }
 
- //----------------------------------------------------------------------------
 
- void ctkXnatObject::setDescription(const QString& description)
 
- {
 
-   Q_D(ctkXnatObject);
 
-   d->description = description;
 
- }
 
- //----------------------------------------------------------------------------
 
- QString ctkXnatObject::childDataType() const
 
- {
 
-   return "Resources";
 
- }
 
- QDateTime ctkXnatObject::lastModifiedTimeOnServer()
 
- {
 
-   Q_D(ctkXnatObject);
 
-   QUuid queryId = this->session()->httpHead(this->resourceUri());
 
-   QMap<QByteArray, QByteArray> header = this->session()->httpHeadSync(queryId);
 
-   QVariant lastModifiedHeader = header.value("Last-Modified");
 
-   QDateTime lastModifiedTime;
 
-   if (lastModifiedHeader.isValid())
 
-   {
 
-     QStringList dateformates;
 
-     // In case http date formate RFC 822 ( "Sun, 06 Nov 1994 08:49:37 GMT" )
 
-     dateformates<<"ddd, dd MMM yyyy HH:mm:ss";
 
-     // In case http date formate ANSI ( "Sun Nov  6 08:49:37 1994" )
 
-     dateformates<<"ddd MMM  d HH:mm:ss yyyy";
 
-     // In case http date formate RFC 850 ( "Sunday, 06-Nov-94 08:49:37 GMT" )
 
-     dateformates<<"dddd, dd-MMM-yy HH:mm:ss";
 
-     QString dateText = lastModifiedHeader.toString();
 
-     // Remove "GMT" addition at the end of the http timestamp
 
-     if (dateText.indexOf("GMT") != -1)
 
-     {
 
-       dateText = dateText.left(dateText.length()-4);
 
-     }
 
-     foreach (QString format, dateformates)
 
-     {
 
-       lastModifiedTime = QDateTime::fromString(dateText, format);
 
-       if (lastModifiedTime.isValid())
 
-         break;
 
-     }
 
-   }
 
-   return lastModifiedTime;
 
- }
 
- void ctkXnatObject::setLastModifiedTime(const QDateTime &lastModifiedTime)
 
- {
 
-   Q_D(ctkXnatObject);
 
-   if (d->lastModifiedTime < lastModifiedTime)
 
-   {
 
-     d->lastModifiedTime = lastModifiedTime;
 
-   }
 
- }
 
- //----------------------------------------------------------------------------
 
- QString ctkXnatObject::property(const QString& name) const
 
- {
 
-   Q_D(const ctkXnatObject);
 
-   ctkXnatObjectPrivate::PropertyMapConstInterator iter = d->properties.find(name);
 
-   if (iter != d->properties.end())
 
-   {
 
-     return iter.value();
 
-   }
 
-   return QString::null;
 
- }
 
- //----------------------------------------------------------------------------
 
- void ctkXnatObject::setProperty(const QString& name, const QVariant& value)
 
- {
 
-   Q_D(ctkXnatObject);
 
-   if (d->properties[name] != value)
 
-   {
 
-     d->properties.insert(name, value.toString());
 
-   }
 
- }
 
- //----------------------------------------------------------------------------
 
- const QMap<QString, QString>& ctkXnatObject::properties() const
 
- {
 
-   Q_D(const ctkXnatObject);
 
-   return d->properties;
 
- }
 
- //----------------------------------------------------------------------------
 
- ctkXnatObject* ctkXnatObject::parent() const
 
- {
 
-   Q_D(const ctkXnatObject);
 
-   return d->parent;
 
- }
 
- //----------------------------------------------------------------------------
 
- void ctkXnatObject::setParent(ctkXnatObject* parent)
 
- {
 
-   Q_D(ctkXnatObject);
 
-   if (d->parent != parent)
 
-   {
 
-     if (d->parent)
 
-     {
 
-       d->parent->remove(this);
 
-     }
 
-     if (parent)
 
-     {
 
-       parent->add(this);
 
-     }
 
-   }
 
- }
 
- //----------------------------------------------------------------------------
 
- QList<ctkXnatObject*> ctkXnatObject::children() const
 
- {
 
-   Q_D(const ctkXnatObject);
 
-   return d->children;
 
- }
 
- //----------------------------------------------------------------------------
 
- void ctkXnatObject::add(ctkXnatObject* child)
 
- {
 
-   Q_D(ctkXnatObject);
 
-   if (child->parent() != this)
 
-   {
 
-     child->d_func()->parent = this;
 
-   }
 
-   bool childExists (false);
 
-   QList<ctkXnatObject*>::iterator iter;
 
-   for (iter = d->children.begin(); iter != d->children.end(); ++iter)
 
-   {
 
-     if (((*iter)->id().length() != 0 && (*iter)->id() == child->id()) ||
 
-         ((*iter)->id().length() == 0 && (*iter)->name() == child->name()))
 
-     {
 
-       *iter = child;
 
-       childExists = true;
 
-     }
 
-   }
 
-   if (!childExists)
 
-   {
 
-     d->children.push_back(child);
 
-   }
 
- }
 
- //----------------------------------------------------------------------------
 
- void ctkXnatObject::remove(ctkXnatObject* child)
 
- {
 
-   Q_D(ctkXnatObject);
 
-   if (!d->children.removeOne(child))
 
-   {
 
-     qWarning() << "ctkXnatObject::remove(): Child does not exist";
 
-   }
 
- }
 
- //----------------------------------------------------------------------------
 
- void ctkXnatObject::reset()
 
- {
 
-   Q_D(ctkXnatObject);
 
-   // d->properties.clear();
 
-   d->children.clear();
 
-   d->fetched = false;
 
- }
 
- //----------------------------------------------------------------------------
 
- bool ctkXnatObject::isFetched() const
 
- {
 
-   Q_D(const ctkXnatObject);
 
-   return d->fetched;
 
- }
 
- //----------------------------------------------------------------------------
 
- QString ctkXnatObject::schemaType() const
 
- {
 
-   return this->property("xsiType");
 
- }
 
- //----------------------------------------------------------------------------
 
- void ctkXnatObject::setSchemaType(const QString& schemaType)
 
- {
 
-   this->setProperty("xsiType", schemaType);
 
- }
 
- //----------------------------------------------------------------------------
 
- void ctkXnatObject::fetch(bool forceFetch)
 
- {
 
-   Q_D(ctkXnatObject);
 
-   if (!d->fetched || forceFetch)
 
-   {
 
-     this->fetchImpl();
 
-     d->fetched = true;
 
-   }
 
- }
 
- //----------------------------------------------------------------------------
 
- ctkXnatSession* ctkXnatObject::session() const
 
- {
 
-   const ctkXnatObject* xnatObject = this;
 
-   while (ctkXnatObject* parent = xnatObject->parent())
 
-   {
 
-     xnatObject = parent;
 
-   }
 
-   const ctkXnatDataModel* dataModel = dynamic_cast<const ctkXnatDataModel*>(xnatObject);
 
-   return dataModel ? dataModel->session() : NULL;
 
- }
 
- //----------------------------------------------------------------------------
 
- void ctkXnatObject::download(const QString& filename)
 
- {
 
-   this->downloadImpl(filename);
 
- }
 
- //----------------------------------------------------------------------------
 
- void ctkXnatObject::save(bool overwrite)
 
- {
 
-   Q_D(ctkXnatObject);
 
-   this->saveImpl(overwrite);
 
- }
 
- //----------------------------------------------------------------------------
 
- ctkXnatResource* ctkXnatObject::addResourceFolder(QString foldername, QString format,
 
-                                    QString content, QString tags)
 
- {
 
-   if (foldername.size() == 0)
 
-   {
 
-     throw ctkXnatException("Error creating resource! Foldername must not be empty!");
 
-   }
 
-   ctkXnatResourceFolder* resFolder = 0;
 
-   QList<ctkXnatObject*> children = this->children();
 
-   for (unsigned int i = 0; i < children.size(); ++i)
 
-   {
 
-     resFolder = dynamic_cast<ctkXnatResourceFolder*>(children.at(i));
 
-     if (resFolder)
 
-     {
 
-       break;
 
-     }
 
-   }
 
-   if (!resFolder)
 
-   {
 
-     resFolder = new ctkXnatResourceFolder();
 
-     this->add(resFolder);
 
-   }
 
-   ctkXnatResource* resource = new ctkXnatResource();
 
-   resource->setName(foldername);
 
-   if (format.size() != 0)
 
-     resource->setFormat(format);
 
-   if (content.size() != 0)
 
-     resource->setContent(content);
 
-   if (tags.size() != 0)
 
-     resource->setTags(tags);
 
-   resFolder->add(resource);
 
-   if (!resource->exists())
 
-     resource->save();
 
-   else
 
-     qDebug()<<"Not adding resource folder. Folder already exists!";
 
-   return resource;
 
- }
 
- //----------------------------------------------------------------------------
 
- bool ctkXnatObject::exists() const
 
- {
 
-   return this->session()->exists(this);
 
- }
 
- //----------------------------------------------------------------------------
 
- void ctkXnatObject::saveImpl(bool /*overwrite*/)
 
- {
 
-   Q_D(ctkXnatObject);
 
-   ctkXnatSession::UrlParameters urlParams;
 
-   urlParams["xsi:type"] = this->schemaType();
 
-   // Just do this if there is already a valid last-modification-time,
 
-   // otherwise the object is not yet on the server!
 
-   QDateTime remoteModTime;
 
-   if (d->lastModifiedTime.isValid())
 
-   {
 
-     // TODO Overwrite this for e.g. project and subject which already support modification time!
 
-     remoteModTime = this->lastModifiedTimeOnServer();
 
-     // If the object has been modified on the server, perform an update
 
-     if (d->lastModifiedTime < remoteModTime)
 
-     {
 
-       qWarning()<<"Uploaded object maybe overwritten on server!";
 
-       // TODO update from server, since modification time is not really supported
 
-       // by xnat right now this is not of high priority
 
-       // something like this->updateImpl + setLastModifiedTime()
 
-     }
 
-   }
 
-   const QMap<QString, QString>& properties = this->properties();
 
-   QMapIterator<QString, QString> itProperties(properties);
 
-   while (itProperties.hasNext())
 
-   {
 
-     itProperties.next();
 
-     if (itProperties.key() == "ID")
 
-       continue;
 
-     urlParams[itProperties.key()] = itProperties.value();
 
-   }
 
-   // Execute the update
 
-   QUuid queryID = this->session()->httpPut(this->resourceUri(), urlParams);
 
-   const QList<QVariantMap> results = this->session()->httpSync(queryID);
 
-   // If this xnat object did not exist before on the server set the ID returned by Xnat
 
-   if (results.size() == 1 && results[0].size() == 1)
 
-   {
 
-     QVariant id = results[0][ID];
 
-     if (!id.isNull())
 
-     {
 
-       this->setId(id.toString());
 
-     }
 
-   }
 
-   // Finally update the modification time on the server
 
-   remoteModTime = this->lastModifiedTimeOnServer();
 
-   d->lastModifiedTime = remoteModTime;
 
- }
 
- //----------------------------------------------------------------------------
 
- void ctkXnatObject::fetchResources(const QString& path)
 
- {
 
-   ctkXnatResourceFolder* resFolder = new ctkXnatResourceFolder();
 
-   this->add(resFolder);
 
- }
 
- //----------------------------------------------------------------------------
 
- void ctkXnatObject::erase()
 
- {
 
-   this->session()->remove(this);
 
-   this->parent()->remove(this);
 
- }
 
 
  |