ctkXnatSessionTest.cpp 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449
  1. /*=========================================================================
  2. Library: CTK
  3. Copyright (c) 2013 University College London, Centre for Medical Image Computing
  4. Licensed under the Apache License, Version 2.0 (the "License");
  5. you may not use this file except in compliance with the License.
  6. You may obtain a copy of the License at
  7. http://www.apache.org/licenses/LICENSE-2.0.txt
  8. Unless required by applicable law or agreed to in writing, software
  9. distributed under the License is distributed on an "AS IS" BASIS,
  10. WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  11. See the License for the specific language governing permissions and
  12. limitations under the License.
  13. =========================================================================*/
  14. #include "ctkXnatSessionTest.h"
  15. #include <QCoreApplication>
  16. #include <QCryptographicHash>
  17. #include <QDebug>
  18. #include <QDir>
  19. #include <QSignalSpy>
  20. #include <QStringList>
  21. #include <QTest>
  22. #include <QTime>
  23. #include <QTimer>
  24. #include <QUrl>
  25. #include <QUuid>
  26. #include <ctkXnatDataModel.h>
  27. #include <ctkXnatException.h>
  28. #include <ctkXnatFile.h>
  29. #include <ctkXnatLoginProfile.h>
  30. #include <ctkXnatProject.h>
  31. #include <ctkXnatResource.h>
  32. #include <ctkXnatResourceFolder.h>
  33. #include <ctkXnatSession.h>
  34. #include <ctkXnatSubject.h>
  35. class ctkXnatSessionTestCasePrivate
  36. {
  37. public:
  38. ctkXnatSession* Session;
  39. ctkXnatLoginProfile LoginProfile;
  40. QString Project;
  41. QString Subject;
  42. QString Experiment;
  43. QDateTime DateTime;
  44. };
  45. // --------------------------------------------------------------------------
  46. ctkXnatSessionTestCase::ctkXnatSessionTestCase()
  47. : d_ptr(new ctkXnatSessionTestCasePrivate())
  48. {
  49. }
  50. // --------------------------------------------------------------------------
  51. ctkXnatSessionTestCase::~ctkXnatSessionTestCase()
  52. {
  53. }
  54. // --------------------------------------------------------------------------
  55. void ctkXnatSessionTestCase::initTestCase()
  56. {
  57. Q_D(ctkXnatSessionTestCase);
  58. d->LoginProfile.setName("ctk");
  59. d->LoginProfile.setServerUrl(QString("https://central.xnat.org"));
  60. d->LoginProfile.setUserName("ctk");
  61. d->LoginProfile.setPassword("ctk-xnat2015");
  62. }
  63. // --------------------------------------------------------------------------
  64. void ctkXnatSessionTestCase::init()
  65. {
  66. Q_D(ctkXnatSessionTestCase);
  67. d->DateTime = QDateTime::currentDateTime();
  68. d->Session = new ctkXnatSession(d->LoginProfile);
  69. d->Session->open();
  70. }
  71. // --------------------------------------------------------------------------
  72. void ctkXnatSessionTestCase::cleanupTestCase()
  73. {
  74. }
  75. // --------------------------------------------------------------------------
  76. void ctkXnatSessionTestCase::cleanup()
  77. {
  78. Q_D(ctkXnatSessionTestCase);
  79. delete d->Session;
  80. d->Session = NULL;
  81. }
  82. // --------------------------------------------------------------------------
  83. void ctkXnatSessionTestCase::testProjectList()
  84. {
  85. Q_D(ctkXnatSessionTestCase);
  86. ctkXnatObject* dataModel = d->Session->dataModel();
  87. dataModel->fetch();
  88. QList<ctkXnatObject*> projects = dataModel->children();
  89. QVERIFY(projects.size() > 0);
  90. }
  91. // --------------------------------------------------------------------------
  92. void ctkXnatSessionTestCase::testResourceUri()
  93. {
  94. Q_D(ctkXnatSessionTestCase);
  95. ctkXnatObject* dataModel = d->Session->dataModel();
  96. QVERIFY(!dataModel->resourceUri().isNull());
  97. QVERIFY(dataModel->resourceUri().isEmpty());
  98. }
  99. // --------------------------------------------------------------------------
  100. void ctkXnatSessionTestCase::testParentChild()
  101. {
  102. Q_D(ctkXnatSessionTestCase);
  103. ctkXnatDataModel* dataModel = d->Session->dataModel();
  104. ctkXnatProject* project = new ctkXnatProject(dataModel);
  105. QVERIFY(project->parent() == dataModel);
  106. QVERIFY(dataModel->children().contains(project));
  107. dataModel->add(project);
  108. int numberOfOccurrences = 0;
  109. foreach (ctkXnatObject* serverProject, dataModel->children())
  110. {
  111. if (serverProject == project || serverProject->id() == project->id())
  112. {
  113. ++numberOfOccurrences;
  114. }
  115. }
  116. QVERIFY(numberOfOccurrences == 1);
  117. dataModel->remove(project);
  118. numberOfOccurrences = 0;
  119. foreach (ctkXnatObject* serverProject, dataModel->children())
  120. {
  121. if (serverProject == project || serverProject->id() == project->id())
  122. {
  123. ++numberOfOccurrences;
  124. }
  125. }
  126. QVERIFY(numberOfOccurrences == 0);
  127. delete project;
  128. }
  129. // --------------------------------------------------------------------------
  130. void ctkXnatSessionTestCase::testSession()
  131. {
  132. Q_D(ctkXnatSessionTestCase);
  133. QVERIFY(d->Session->isOpen());
  134. QDateTime expirationDate = d->Session->expirationDate();
  135. QVERIFY(d->DateTime < expirationDate);
  136. QTest::qSleep(2000);
  137. QUuid uuid = d->Session->httpGet("/data/version");
  138. QVERIFY(!uuid.isNull());
  139. d->Session->httpSync(uuid);
  140. QVERIFY(expirationDate < d->Session->expirationDate());
  141. try
  142. {
  143. d->Session->httpSync(uuid);
  144. QFAIL("Exception for unknown uuid expected");
  145. }
  146. catch(const ctkInvalidArgumentException&)
  147. {}
  148. d->Session->close();
  149. try
  150. {
  151. d->Session->dataModel();
  152. QFAIL("Exception for closed session expected");
  153. }
  154. catch(const ctkXnatInvalidSessionException&)
  155. {}
  156. }
  157. // --------------------------------------------------------------------------
  158. void ctkXnatSessionTestCase::testAuthenticationError()
  159. {
  160. ctkXnatLoginProfile loginProfile;
  161. loginProfile.setName("error");
  162. loginProfile.setServerUrl(QString("https://central.xnat.org"));
  163. loginProfile.setUserName("x");
  164. loginProfile.setPassword("y");
  165. ctkXnatSession session(loginProfile);
  166. try
  167. {
  168. session.open();
  169. QFAIL("Authenication error exception expected");
  170. }
  171. catch (const ctkXnatAuthenticationException&)
  172. {}
  173. }
  174. // --------------------------------------------------------------------------
  175. void ctkXnatSessionTestCase::testCreateProject()
  176. {
  177. Q_D(ctkXnatSessionTestCase);
  178. ctkXnatDataModel* dataModel = d->Session->dataModel();
  179. QString projectId = QString("CTK_") + QUuid::createUuid().toString().mid(1, 8);
  180. d->Project = projectId;
  181. ctkXnatProject* project = new ctkXnatProject(dataModel);
  182. project->setId(projectId);
  183. project->setName(projectId);
  184. project->setDescription("CTK_test_project");
  185. bool exists = d->Session->exists(project);
  186. QVERIFY(!exists);
  187. project->save();
  188. exists = d->Session->exists(project);
  189. QVERIFY(exists);
  190. d->Session->remove(project);
  191. exists = d->Session->exists(project);
  192. QVERIFY(!exists);
  193. }
  194. // --------------------------------------------------------------------------
  195. void ctkXnatSessionTestCase::testCreateSubject()
  196. {
  197. Q_D(ctkXnatSessionTestCase);
  198. ctkXnatDataModel* dataModel = d->Session->dataModel();
  199. QString projectId = QString("CTK_") + QUuid::createUuid().toString().mid(1, 8);
  200. d->Project = projectId;
  201. ctkXnatProject* project = new ctkXnatProject(dataModel);
  202. project->setId(projectId);
  203. project->setName(projectId);
  204. project->setDescription("CTK_test_project");
  205. QVERIFY(!project->exists());
  206. project->save();
  207. QVERIFY(project->exists());
  208. ctkXnatSubject* subject = new ctkXnatSubject(project);
  209. QString subjectName = QString("CTK_S") + QUuid::createUuid().toString().mid(1, 8);
  210. subject->setName(subjectName);
  211. subject->save();
  212. QVERIFY(!subject->id().isNull());
  213. subject->erase();
  214. QVERIFY(!subject->exists());
  215. project->erase();
  216. QVERIFY(!project->exists());
  217. }
  218. // --------------------------------------------------------------------------
  219. void ctkXnatSessionTestCase::testAddResourceFolder()
  220. {
  221. Q_D(ctkXnatSessionTestCase);
  222. ctkXnatDataModel* dataModel = d->Session->dataModel();
  223. QString projectId = QString("CTK_") + QUuid::createUuid().toString().mid(1, 8);
  224. d->Project = projectId;
  225. ctkXnatProject* project = new ctkXnatProject(dataModel);
  226. project->setId(projectId);
  227. project->setName(projectId);
  228. project->setDescription("CTK_test_project");
  229. QVERIFY(!project->exists());
  230. project->save();
  231. QVERIFY(project->exists());
  232. ctkXnatResource* resource = project->addResourceFolder("TestResource", "testFormat", "testContent", "testTag1,testTag2");
  233. QVERIFY(resource->exists());
  234. QVERIFY(resource->name() == "TestResource");
  235. QVERIFY(resource->format() == "testFormat");
  236. QVERIFY(resource->content() == "testContent");
  237. QVERIFY(resource->tags() == "testTag1,testTag2");
  238. ctkXnatResourceFolder* folder = dynamic_cast<ctkXnatResourceFolder*>(resource->parent());
  239. QVERIFY(folder->exists());
  240. QVERIFY(folder != 0);
  241. QVERIFY(folder->name() == "Resources");
  242. project->erase();
  243. QVERIFY(!project->exists());
  244. QVERIFY(!folder->exists());
  245. QVERIFY(!resource->exists());
  246. }
  247. // --------------------------------------------------------------------------
  248. void ctkXnatSessionTestCase::testUploadAndDownloadFile()
  249. {
  250. Q_D(ctkXnatSessionTestCase);
  251. ctkXnatDataModel* dataModel = d->Session->dataModel();
  252. QString projectId = QString("CTK_") + QUuid::createUuid().toString().mid(1, 8);
  253. d->Project = projectId;
  254. ctkXnatProject* project = new ctkXnatProject(dataModel);
  255. project->setId(projectId);
  256. project->setName(projectId);
  257. project->setDescription("CTK_test_project");
  258. QVERIFY(!project->exists());
  259. project->save();
  260. QVERIFY(project->exists());
  261. ctkXnatResource* resource = project->addResourceFolder("TestResourceContainingData");
  262. QVERIFY(resource->exists());
  263. QVERIFY(resource->name() == "TestResourceContainingData");
  264. QVERIFY(resource->format() == "");
  265. QVERIFY(resource->content() == "");
  266. QVERIFY(resource->tags() == "");
  267. QString tempDirPath = QDir::tempPath() + QUuid::createUuid().toString().mid(1, 8);
  268. QString uploadFileName = tempDirPath + "/ctk_xnat_upload_" + QUuid::createUuid().toString().mid(1, 8) + ".txt";
  269. QString downloadFileName = tempDirPath + "/ctk_xnat_download_" + QUuid::createUuid().toString().mid(1, 8) + ".txt";
  270. QDir tempDir;
  271. if (tempDir.mkdir(tempDirPath))
  272. {
  273. QFile uploadedFile(uploadFileName);
  274. if (uploadedFile.open(QFile::ReadWrite))
  275. {
  276. QTextStream stream( &uploadedFile );
  277. stream << "Hi, I am a CTK test file! ;-)" << endl;
  278. QFileInfo fileInfo;
  279. fileInfo.setFile(uploadFileName);
  280. // Create xnatFile object
  281. ctkXnatFile* xnatFile = new ctkXnatFile(resource);
  282. xnatFile->setLocalFilePath(fileInfo.filePath());
  283. xnatFile->setName(fileInfo.fileName());
  284. xnatFile->setFileFormat("some format");
  285. xnatFile->setFileContent("some content");
  286. xnatFile->setFileTags("some, tags");
  287. resource->add(xnatFile);
  288. // Actual file upload
  289. xnatFile->save();
  290. QVERIFY(xnatFile->exists());
  291. xnatFile->download(downloadFileName);
  292. QFile downloadedFile(downloadFileName);
  293. QVERIFY(downloadedFile.exists());
  294. uploadedFile.close();
  295. if (downloadedFile.open(QFile::ReadOnly) && uploadedFile.open(QFile::ReadOnly))
  296. {
  297. QCryptographicHash hashUploaded(QCryptographicHash::Md5);
  298. QCryptographicHash hashDownloaded(QCryptographicHash::Md5);
  299. #if QT_VERSION >= QT_VERSION_CHECK(5,0,0)
  300. hashUploaded.addData(&uploadedFile);
  301. hashDownloaded.addData(&downloadedFile);
  302. #else
  303. hashUploaded.addData(uploadedFile.readAll());
  304. hashDownloaded.addData(downloadedFile.readAll());
  305. #endif
  306. QString md5ChecksumUploaded(hashUploaded.result().toHex());
  307. QString md5ChecksumDownloaded(hashDownloaded.result().toHex());
  308. QVERIFY (md5ChecksumDownloaded == md5ChecksumUploaded);
  309. // Remove the data from XNAT
  310. project->erase();
  311. QVERIFY(!project->exists());
  312. QVERIFY(!resource->exists());
  313. QVERIFY(!xnatFile->exists());
  314. // Remove the local data
  315. uploadedFile.close();
  316. downloadedFile.close();
  317. uploadedFile.remove();
  318. downloadedFile.remove();
  319. tempDir.cdUp();
  320. tempDir.rmdir(tempDirPath);
  321. }
  322. else
  323. {
  324. qWarning()<<"Could not open files for validation! Could not finish test!";
  325. }
  326. }
  327. else
  328. {
  329. qWarning()<<"Could not create temporary file for upload! Could not finish test!";
  330. return;
  331. }
  332. }
  333. else
  334. {
  335. qWarning()<<"Could not create temporary directory! Could not finish test!";
  336. }
  337. }
  338. // --------------------------------------------------------------------------
  339. int ctkXnatSessionTest(int argc, char* argv[])
  340. {
  341. QCoreApplication app(argc, argv);
  342. ctkXnatSessionTestCase test;
  343. return QTest::qExec(&test, argc, argv);
  344. }