123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422 |
- /*=============================================================================
- 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);
- 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;
- }
- if (!d->children.contains(child))
- {
- d->children.push_back(child);
- }
- else
- {
- qWarning() << "ctkXnatObject::add(): Child already exists";
- }
- }
- //----------------------------------------------------------------------------
- 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()
- {
- Q_D(ctkXnatObject);
- if (!d->fetched)
- {
- 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()
- {
- this->saveImpl();
- }
- //----------------------------------------------------------------------------
- void ctkXnatObject::addResource(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);
- resource->save();
- }
- //----------------------------------------------------------------------------
- bool ctkXnatObject::exists() const
- {
- return this->session()->exists(this);
- }
- //----------------------------------------------------------------------------
- void ctkXnatObject::saveImpl()
- {
- Q_D(ctkXnatObject);
- QString query = this->resourceUri();
- // 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)
- {
- qDebug()<<"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()
- }
- }
- // Creating the update query
- query.append(QString("?%1=%2").arg("xsi:type", this->schemaType()));
- const QMap<QString, QString>& properties = this->properties();
- QMapIterator<QString, QString> itProperties(properties);
- while (itProperties.hasNext())
- {
- itProperties.next();
- if (itProperties.key() == "ID")
- continue;
- query.append(QString("&%1=%2").arg(itProperties.key(), itProperties.value()));
- }
- // Execute the update
- QUuid queryID = this->session()->httpPut(query);
- 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);
- }
|