test_base.py 36 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804
  1. # This file is part of Radicale Server - Calendar Server
  2. # Copyright © 2012-2016 Guillaume Ayoub
  3. #
  4. # This library is free software: you can redistribute it and/or modify
  5. # it under the terms of the GNU General Public License as published by
  6. # the Free Software Foundation, either version 3 of the License, or
  7. # (at your option) any later version.
  8. #
  9. # This library is distributed in the hope that it will be useful,
  10. # but WITHOUT ANY WARRANTY; without even the implied warranty of
  11. # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  12. # GNU General Public License for more details.
  13. #
  14. # You should have received a copy of the GNU General Public License
  15. # along with Radicale. If not, see <http://www.gnu.org/licenses/>.
  16. """
  17. Radicale tests with simple requests.
  18. """
  19. import logging
  20. import os
  21. import posixpath
  22. import shutil
  23. import tempfile
  24. from radicale import Application, config
  25. from . import BaseTest
  26. from .helpers import get_file_content
  27. class BaseRequestsMixIn:
  28. """Tests with simple requests."""
  29. def test_root(self):
  30. """GET request at "/"."""
  31. status, headers, answer = self.request("GET", "/")
  32. assert status == 200
  33. assert "Radicale works!" in answer
  34. # Test the creation of the collection
  35. self.request("MKCOL", "/calendar.ics/")
  36. self.request(
  37. "PUT", "/calendar.ics/", "BEGIN:VCALENDAR\r\nEND:VCALENDAR")
  38. status, headers, answer = self.request("GET", "/calendar.ics/")
  39. assert "BEGIN:VCALENDAR" in answer
  40. assert "END:VCALENDAR" in answer
  41. def test_add_event(self):
  42. """Add an event."""
  43. self.request("MKCOL", "/calendar.ics/")
  44. self.request(
  45. "PUT", "/calendar.ics/", "BEGIN:VCALENDAR\r\nEND:VCALENDAR")
  46. event = get_file_content("event1.ics")
  47. path = "/calendar.ics/event1.ics"
  48. status, headers, answer = self.request("PUT", path, event)
  49. assert status == 201
  50. status, headers, answer = self.request("GET", path)
  51. assert "ETag" in headers.keys()
  52. assert status == 200
  53. assert "VEVENT" in answer
  54. assert "Event" in answer
  55. assert "UID:event" in answer
  56. def test_add_todo(self):
  57. """Add a todo."""
  58. self.request("MKCOL", "/calendar.ics/")
  59. self.request(
  60. "PUT", "/calendar.ics/", "BEGIN:VCALENDAR\r\nEND:VCALENDAR")
  61. todo = get_file_content("todo1.ics")
  62. path = "/calendar.ics/todo1.ics"
  63. status, headers, answer = self.request("PUT", path, todo)
  64. assert status == 201
  65. status, headers, answer = self.request("GET", path)
  66. assert "ETag" in headers.keys()
  67. assert "VTODO" in answer
  68. assert "Todo" in answer
  69. assert "UID:todo" in answer
  70. def test_update(self):
  71. """Update an event."""
  72. self.request("MKCOL", "/calendar.ics/")
  73. self.request(
  74. "PUT", "/calendar.ics/", "BEGIN:VCALENDAR\r\nEND:VCALENDAR")
  75. event = get_file_content("event1.ics")
  76. path = "/calendar.ics/event1.ics"
  77. status, headers, answer = self.request("PUT", path, event)
  78. assert status == 201
  79. status, headers, answer = self.request("GET", path)
  80. assert "ETag" in headers.keys()
  81. assert status == 200
  82. assert "VEVENT" in answer
  83. assert "Event" in answer
  84. assert "UID:event" in answer
  85. assert "DTSTART;TZID=Europe/Paris:20130901T180000" in answer
  86. assert "DTEND;TZID=Europe/Paris:20130901T190000" in answer
  87. # Then we send another PUT request
  88. event = get_file_content("event1-prime.ics")
  89. status, headers, answer = self.request("PUT", path, event)
  90. assert status == 201
  91. status, headers, answer = self.request("GET", "/calendar.ics/")
  92. assert answer.count("BEGIN:VEVENT") == 1
  93. status, headers, answer = self.request("GET", path)
  94. assert "ETag" in headers.keys()
  95. assert status == 200
  96. assert "VEVENT" in answer
  97. assert "Event" in answer
  98. assert "UID:event" in answer
  99. assert "DTSTART;TZID=Europe/Paris:20130901T180000" not in answer
  100. assert "DTEND;TZID=Europe/Paris:20130901T190000" not in answer
  101. assert "DTSTART;TZID=Europe/Paris:20140901T180000" in answer
  102. assert "DTEND;TZID=Europe/Paris:20140901T210000" in answer
  103. def test_put_whole_collection(self):
  104. """Create and overwrite a whole collection."""
  105. event = get_file_content("event1.ics")
  106. status, headers, answer = self.request("PUT", "/calendar.ics/", event)
  107. assert status == 201
  108. status, headers, answer = self.request(
  109. "PUT", "/calendar.ics/event1.ics", event)
  110. assert status == 201
  111. # Overwrite
  112. status, headers, answer = self.request("PUT", "/calendar.ics/", event)
  113. assert status == 201
  114. status, headers, answer = self.request(
  115. "GET", "/calendar.ics/event1.ics")
  116. assert status == 404
  117. def test_delete(self):
  118. """Delete an event."""
  119. self.request("MKCOL", "/calendar.ics/")
  120. self.request(
  121. "PUT", "/calendar.ics/", "BEGIN:VCALENDAR\r\nEND:VCALENDAR")
  122. event = get_file_content("event1.ics")
  123. path = "/calendar.ics/event1.ics"
  124. status, headers, answer = self.request("PUT", path, event)
  125. # Then we send a DELETE request
  126. status, headers, answer = self.request("DELETE", path)
  127. assert status == 200
  128. assert "href>%s</" % path in answer
  129. status, headers, answer = self.request("GET", "/calendar.ics/")
  130. assert "VEVENT" not in answer
  131. def test_mkcalendar(self):
  132. """Make a calendar."""
  133. self.request("MKCALENDAR", "/calendar.ics/")
  134. status, headers, answer = self.request("GET", "/calendar.ics/")
  135. assert status == 200
  136. def test_move(self):
  137. """Move a item."""
  138. self.request("MKCALENDAR", "/calendar.ics/")
  139. event = get_file_content("event1.ics")
  140. path1 = "/calendar.ics/event1.ics"
  141. path2 = "/calendar.ics/event2.ics"
  142. status, headers, answer = self.request("PUT", path1, event)
  143. status, headers, answer = self.request(
  144. "MOVE", path1, HTTP_DESTINATION=path2, HTTP_HOST="")
  145. assert status == 201
  146. status, headers, answer = self.request("GET", path1)
  147. assert status == 404
  148. status, headers, answer = self.request("GET", path2)
  149. assert status == 200
  150. def test_head(self):
  151. status, headers, answer = self.request("HEAD", "/")
  152. assert status == 200
  153. def test_options(self):
  154. status, headers, answer = self.request("OPTIONS", "/")
  155. assert status == 200
  156. assert "DAV" in headers
  157. def test_delete_collection(self):
  158. """Delete a collection."""
  159. self.request("MKCOL", "/calendar.ics/")
  160. event = get_file_content("event1.ics")
  161. self.request("PUT", "/calendar.ics/event1.ics", event)
  162. status, headers, answer = self.request("DELETE", "/calendar.ics/")
  163. assert status == 200
  164. assert "href>/calendar.ics/</" in answer
  165. status, headers, answer = self.request("GET", "/calendar.ics/")
  166. assert status == 404
  167. def test_delete_root_collection(self):
  168. """Delete the root collection."""
  169. self.request("MKCOL", "/calendar.ics/")
  170. event = get_file_content("event1.ics")
  171. self.request("PUT", "/event1.ics", event)
  172. self.request("PUT", "/calendar.ics/event1.ics", event)
  173. status, headers, answer = self.request("DELETE", "/")
  174. assert status == 200
  175. assert "href>/</" in answer
  176. status, headers, answer = self.request("GET", "/calendar.ics/")
  177. assert status == 404
  178. status, headers, answer = self.request("GET", "/event1.ics")
  179. assert status == 404
  180. def test_propfind(self):
  181. calendar_path = "/calendar.ics/"
  182. self.request("MKCALENDAR", calendar_path)
  183. event = get_file_content("event1.ics")
  184. event_path = posixpath.join(calendar_path, "event.ics")
  185. self.request("PUT", event_path, event)
  186. status, headers, answer = self.request("PROPFIND", "/", HTTP_DEPTH="1")
  187. assert status == 207
  188. assert "href>/</" in answer
  189. assert "href>%s</" % calendar_path in answer
  190. status, headers, answer = self.request("PROPFIND", calendar_path, HTTP_DEPTH="1")
  191. assert status == 207
  192. assert "href>%s</" % calendar_path in answer
  193. assert "href>%s</" % event_path in answer
  194. def test_proppatch(self):
  195. """Write a property and read it back."""
  196. self.request("MKCALENDAR", "/calendar.ics/")
  197. proppatch = get_file_content("proppatch1.xml")
  198. status, headers, answer = self.request(
  199. "PROPPATCH", "/calendar.ics/", proppatch)
  200. assert status == 207
  201. assert "calendar-color" in answer
  202. assert "200 OK</status" in answer
  203. # Read property back
  204. propfind = get_file_content("propfind1.xml")
  205. status, headers, answer = self.request(
  206. "PROPFIND", "/calendar.ics/", propfind)
  207. assert status == 207
  208. assert ":calendar-color>#BADA55</" in answer
  209. assert "200 OK</status" in answer
  210. def test_multiple_events_with_same_uid(self):
  211. """Add two events with the same UID."""
  212. self.request("MKCOL", "/calendar.ics/")
  213. self.request("PUT", "/calendar.ics/", get_file_content("event2.ics"))
  214. status, headers, answer = self.request(
  215. "REPORT", "/calendar.ics/",
  216. """<?xml version="1.0" encoding="utf-8" ?>
  217. <C:calendar-query xmlns:C="urn:ietf:params:xml:ns:caldav">
  218. <D:prop xmlns:D="DAV:"><D:getetag/></D:prop>
  219. </C:calendar-query>""")
  220. assert answer.count("<getetag>") == 1
  221. status, headers, answer = self.request("GET", "/calendar.ics/")
  222. assert answer.count("BEGIN:VEVENT") == 2
  223. def _test_filter(self, filters, kind="event", items=1):
  224. filters_text = "".join(
  225. "<C:filter>%s</C:filter>" % filter_ for filter_ in filters)
  226. self.request("MKCOL", "/calendar.ics/")
  227. self.request(
  228. "PUT", "/calendar.ics/", "BEGIN:VCALENDAR\r\nEND:VCALENDAR")
  229. for i in range(items):
  230. filename = "{}{}.ics".format(kind, i + 1)
  231. event = get_file_content(filename)
  232. self.request("PUT", "/calendar.ics/{}".format(filename), event)
  233. status, headers, answer = self.request(
  234. "REPORT", "/calendar.ics",
  235. """<?xml version="1.0" encoding="utf-8" ?>
  236. <C:calendar-query xmlns:C="urn:ietf:params:xml:ns:caldav">
  237. <D:prop xmlns:D="DAV:">
  238. <D:getetag/>
  239. </D:prop>
  240. %s
  241. </C:calendar-query>""" % filters_text)
  242. return answer
  243. def test_calendar_tag_filter(self):
  244. """Report request with tag-based filter on calendar."""
  245. assert "href>/calendar.ics/event1.ics</" in self._test_filter(["""
  246. <C:comp-filter name="VCALENDAR"></C:comp-filter>"""])
  247. def test_item_tag_filter(self):
  248. """Report request with tag-based filter on an item."""
  249. assert "href>/calendar.ics/event1.ics</" in self._test_filter(["""
  250. <C:comp-filter name="VCALENDAR">
  251. <C:comp-filter name="VEVENT"></C:comp-filter>
  252. </C:comp-filter>"""])
  253. assert "href>/calendar.ics/event1.ics</" not in self._test_filter(["""
  254. <C:comp-filter name="VCALENDAR">
  255. <C:comp-filter name="VTODO"></C:comp-filter>
  256. </C:comp-filter>"""])
  257. def test_item_not_tag_filter(self):
  258. """Report request with tag-based is-not filter on an item."""
  259. assert "href>/calendar.ics/event1.ics</" not in self._test_filter(["""
  260. <C:comp-filter name="VCALENDAR">
  261. <C:comp-filter name="VEVENT">
  262. <C:is-not-defined />
  263. </C:comp-filter>
  264. </C:comp-filter>"""])
  265. assert "href>/calendar.ics/event1.ics</" in self._test_filter(["""
  266. <C:comp-filter name="VCALENDAR">
  267. <C:comp-filter name="VTODO">
  268. <C:is-not-defined />
  269. </C:comp-filter>
  270. </C:comp-filter>"""])
  271. def test_item_prop_filter(self):
  272. """Report request with prop-based filter on an item."""
  273. assert "href>/calendar.ics/event1.ics</" in self._test_filter(["""
  274. <C:comp-filter name="VCALENDAR">
  275. <C:comp-filter name="VEVENT">
  276. <C:prop-filter name="SUMMARY"></C:prop-filter>
  277. </C:comp-filter>
  278. </C:comp-filter>"""])
  279. assert "href>/calendar.ics/event1.ics</" not in self._test_filter(["""
  280. <C:comp-filter name="VCALENDAR">
  281. <C:comp-filter name="VEVENT">
  282. <C:prop-filter name="UNKNOWN"></C:prop-filter>
  283. </C:comp-filter>
  284. </C:comp-filter>"""])
  285. def test_item_not_prop_filter(self):
  286. """Report request with prop-based is-not filter on an item."""
  287. assert "href>/calendar.ics/event1.ics</" not in self._test_filter(["""
  288. <C:comp-filter name="VCALENDAR">
  289. <C:comp-filter name="VEVENT">
  290. <C:prop-filter name="SUMMARY">
  291. <C:is-not-defined />
  292. </C:prop-filter>
  293. </C:comp-filter>
  294. </C:comp-filter>"""])
  295. assert "href>/calendar.ics/event1.ics</" in self._test_filter(["""
  296. <C:comp-filter name="VCALENDAR">
  297. <C:comp-filter name="VEVENT">
  298. <C:prop-filter name="UNKNOWN">
  299. <C:is-not-defined />
  300. </C:prop-filter>
  301. </C:comp-filter>
  302. </C:comp-filter>"""])
  303. def test_mutiple_filters(self):
  304. """Report request with multiple filters on an item."""
  305. assert "href>/calendar.ics/event1.ics</" not in self._test_filter(["""
  306. <C:comp-filter name="VCALENDAR">
  307. <C:comp-filter name="VEVENT">
  308. <C:prop-filter name="SUMMARY">
  309. <C:is-not-defined />
  310. </C:prop-filter>
  311. </C:comp-filter>
  312. </C:comp-filter>""", """
  313. <C:comp-filter name="VCALENDAR">
  314. <C:comp-filter name="VEVENT">
  315. <C:prop-filter name="UNKNOWN">
  316. <C:is-not-defined />
  317. </C:prop-filter>
  318. </C:comp-filter>
  319. </C:comp-filter>"""])
  320. assert "href>/calendar.ics/event1.ics</" in self._test_filter(["""
  321. <C:comp-filter name="VCALENDAR">
  322. <C:comp-filter name="VEVENT">
  323. <C:prop-filter name="SUMMARY"></C:prop-filter>
  324. </C:comp-filter>
  325. </C:comp-filter>""", """
  326. <C:comp-filter name="VCALENDAR">
  327. <C:comp-filter name="VEVENT">
  328. <C:prop-filter name="UNKNOWN">
  329. <C:is-not-defined />
  330. </C:prop-filter>
  331. </C:comp-filter>
  332. </C:comp-filter>"""])
  333. assert "href>/calendar.ics/event1.ics</" in self._test_filter(["""
  334. <C:comp-filter name="VCALENDAR">
  335. <C:comp-filter name="VEVENT">
  336. <C:prop-filter name="SUMMARY"></C:prop-filter>
  337. <C:prop-filter name="UNKNOWN">
  338. <C:is-not-defined />
  339. </C:prop-filter>
  340. </C:comp-filter>
  341. </C:comp-filter>"""])
  342. def test_text_match_filter(self):
  343. """Report request with text-match filter on calendar."""
  344. assert "href>/calendar.ics/event1.ics</" in self._test_filter(["""
  345. <C:comp-filter name="VCALENDAR">
  346. <C:comp-filter name="VEVENT">
  347. <C:prop-filter name="SUMMARY">
  348. <C:text-match>event</C:text-match>
  349. </C:prop-filter>
  350. </C:comp-filter>
  351. </C:comp-filter>"""])
  352. assert "href>/calendar.ics/event1.ics</" not in self._test_filter(["""
  353. <C:comp-filter name="VCALENDAR">
  354. <C:comp-filter name="VEVENT">
  355. <C:prop-filter name="UNKNOWN">
  356. <C:text-match>event</C:text-match>
  357. </C:prop-filter>
  358. </C:comp-filter>
  359. </C:comp-filter>"""])
  360. assert "href>/calendar.ics/event1.ics</" not in self._test_filter(["""
  361. <C:comp-filter name="VCALENDAR">
  362. <C:comp-filter name="VEVENT">
  363. <C:prop-filter name="SUMMARY">
  364. <C:text-match>unknown</C:text-match>
  365. </C:prop-filter>
  366. </C:comp-filter>
  367. </C:comp-filter>"""])
  368. assert "href>/calendar.ics/event1.ics</" not in self._test_filter(["""
  369. <C:comp-filter name="VCALENDAR">
  370. <C:comp-filter name="VEVENT">
  371. <C:prop-filter name="SUMMARY">
  372. <C:text-match negate-condition="yes">event</C:text-match>
  373. </C:prop-filter>
  374. </C:comp-filter>
  375. </C:comp-filter>"""])
  376. def test_param_filter(self):
  377. """Report request with param-filter on calendar."""
  378. assert "href>/calendar.ics/event1.ics</" in self._test_filter(["""
  379. <C:comp-filter name="VCALENDAR">
  380. <C:comp-filter name="VEVENT">
  381. <C:prop-filter name="ATTENDEE">
  382. <C:param-filter name="PARTSTAT">
  383. <C:text-match collation="i;ascii-casemap"
  384. >ACCEPTED</C:text-match>
  385. </C:param-filter>
  386. </C:prop-filter>
  387. </C:comp-filter>
  388. </C:comp-filter>"""])
  389. assert "href>/calendar.ics/event1.ics</" not in self._test_filter(["""
  390. <C:comp-filter name="VCALENDAR">
  391. <C:comp-filter name="VEVENT">
  392. <C:prop-filter name="ATTENDEE">
  393. <C:param-filter name="PARTSTAT">
  394. <C:text-match collation="i;ascii-casemap"
  395. >UNKNOWN</C:text-match>
  396. </C:param-filter>
  397. </C:prop-filter>
  398. </C:comp-filter>
  399. </C:comp-filter>"""])
  400. assert "href>/calendar.ics/event1.ics</" not in self._test_filter(["""
  401. <C:comp-filter name="VCALENDAR">
  402. <C:comp-filter name="VEVENT">
  403. <C:prop-filter name="ATTENDEE">
  404. <C:param-filter name="PARTSTAT">
  405. <C:is-not-defined />
  406. </C:param-filter>
  407. </C:prop-filter>
  408. </C:comp-filter>
  409. </C:comp-filter>"""])
  410. assert "href>/calendar.ics/event1.ics</" in self._test_filter(["""
  411. <C:comp-filter name="VCALENDAR">
  412. <C:comp-filter name="VEVENT">
  413. <C:prop-filter name="ATTENDEE">
  414. <C:param-filter name="UNKNOWN">
  415. <C:is-not-defined />
  416. </C:param-filter>
  417. </C:prop-filter>
  418. </C:comp-filter>
  419. </C:comp-filter>"""])
  420. def test_time_range_filter_events(self):
  421. """Report request with time-range filter on events."""
  422. answer = self._test_filter(["""
  423. <C:comp-filter name="VCALENDAR">
  424. <C:comp-filter name="VEVENT">
  425. <C:time-range start="20130801T000000Z" end="20131001T000000Z"/>
  426. </C:comp-filter>
  427. </C:comp-filter>"""], "event", items=5)
  428. assert "href>/calendar.ics/event1.ics</" in answer
  429. assert "href>/calendar.ics/event2.ics</" in answer
  430. assert "href>/calendar.ics/event3.ics</" in answer
  431. assert "href>/calendar.ics/event4.ics</" in answer
  432. assert "href>/calendar.ics/event5.ics</" in answer
  433. answer = self._test_filter(["""
  434. <C:comp-filter name="VCALENDAR">
  435. <C:comp-filter name="VEVENT">
  436. <C:prop-filter name="ATTENDEE">
  437. <C:param-filter name="PARTSTAT">
  438. <C:is-not-defined />
  439. </C:param-filter>
  440. </C:prop-filter>
  441. <C:time-range start="20130801T000000Z" end="20131001T000000Z"/>
  442. </C:comp-filter>
  443. </C:comp-filter>"""], items=5)
  444. assert "href>/calendar.ics/event1.ics</" not in answer
  445. assert "href>/calendar.ics/event2.ics</" not in answer
  446. assert "href>/calendar.ics/event3.ics</" not in answer
  447. assert "href>/calendar.ics/event4.ics</" not in answer
  448. assert "href>/calendar.ics/event5.ics</" not in answer
  449. answer = self._test_filter(["""
  450. <C:comp-filter name="VCALENDAR">
  451. <C:comp-filter name="VEVENT">
  452. <C:time-range start="20130902T000000Z" end="20131001T000000Z"/>
  453. </C:comp-filter>
  454. </C:comp-filter>"""], items=5)
  455. assert "href>/calendar.ics/event1.ics</" not in answer
  456. assert "href>/calendar.ics/event2.ics</" in answer
  457. assert "href>/calendar.ics/event3.ics</" in answer
  458. assert "href>/calendar.ics/event4.ics</" in answer
  459. assert "href>/calendar.ics/event5.ics</" in answer
  460. answer = self._test_filter(["""
  461. <C:comp-filter name="VCALENDAR">
  462. <C:comp-filter name="VEVENT">
  463. <C:time-range start="20130903T000000Z" end="20130908T000000Z"/>
  464. </C:comp-filter>
  465. </C:comp-filter>"""], items=5)
  466. assert "href>/calendar.ics/event1.ics</" not in answer
  467. assert "href>/calendar.ics/event2.ics</" not in answer
  468. assert "href>/calendar.ics/event3.ics</" in answer
  469. assert "href>/calendar.ics/event4.ics</" in answer
  470. assert "href>/calendar.ics/event5.ics</" in answer
  471. answer = self._test_filter(["""
  472. <C:comp-filter name="VCALENDAR">
  473. <C:comp-filter name="VEVENT">
  474. <C:time-range start="20130903T000000Z" end="20130904T000000Z"/>
  475. </C:comp-filter>
  476. </C:comp-filter>"""], items=5)
  477. assert "href>/calendar.ics/event1.ics</" not in answer
  478. assert "href>/calendar.ics/event2.ics</" not in answer
  479. assert "href>/calendar.ics/event3.ics</" in answer
  480. assert "href>/calendar.ics/event4.ics</" not in answer
  481. assert "href>/calendar.ics/event5.ics</" not in answer
  482. answer = self._test_filter(["""
  483. <C:comp-filter name="VCALENDAR">
  484. <C:comp-filter name="VEVENT">
  485. <C:time-range start="20130805T000000Z" end="20130810T000000Z"/>
  486. </C:comp-filter>
  487. </C:comp-filter>"""], items=5)
  488. assert "href>/calendar.ics/event1.ics</" not in answer
  489. assert "href>/calendar.ics/event2.ics</" not in answer
  490. assert "href>/calendar.ics/event3.ics</" not in answer
  491. assert "href>/calendar.ics/event4.ics</" not in answer
  492. assert "href>/calendar.ics/event5.ics</" not in answer
  493. def test_time_range_filter_events_rrule(self):
  494. """Report request with time-range filter on events with rrules."""
  495. answer = self._test_filter(["""
  496. <C:comp-filter name="VCALENDAR">
  497. <C:comp-filter name="VEVENT">
  498. <C:time-range start="20130801T000000Z" end="20131001T000000Z"/>
  499. </C:comp-filter>
  500. </C:comp-filter>"""], "event", items=2)
  501. assert "href>/calendar.ics/event1.ics</" in answer
  502. assert "href>/calendar.ics/event2.ics</" in answer
  503. answer = self._test_filter(["""
  504. <C:comp-filter name="VCALENDAR">
  505. <C:comp-filter name="VEVENT">
  506. <C:time-range start="20140801T000000Z" end="20141001T000000Z"/>
  507. </C:comp-filter>
  508. </C:comp-filter>"""], "event", items=2)
  509. assert "href>/calendar.ics/event1.ics</" not in answer
  510. assert "href>/calendar.ics/event2.ics</" in answer
  511. answer = self._test_filter(["""
  512. <C:comp-filter name="VCALENDAR">
  513. <C:comp-filter name="VEVENT">
  514. <C:time-range start="20120801T000000Z" end="20121001T000000Z"/>
  515. </C:comp-filter>
  516. </C:comp-filter>"""], "event", items=2)
  517. assert "href>/calendar.ics/event1.ics</" not in answer
  518. assert "href>/calendar.ics/event2.ics</" not in answer
  519. answer = self._test_filter(["""
  520. <C:comp-filter name="VCALENDAR">
  521. <C:comp-filter name="VEVENT">
  522. <C:time-range start="20130903T000000Z" end="20130907T000000Z"/>
  523. </C:comp-filter>
  524. </C:comp-filter>"""], "event", items=2)
  525. assert "href>/calendar.ics/event1.ics</" not in answer
  526. assert "href>/calendar.ics/event2.ics</" not in answer
  527. def test_time_range_filter_todos(self):
  528. """Report request with time-range filter on todos."""
  529. answer = self._test_filter(["""
  530. <C:comp-filter name="VCALENDAR">
  531. <C:comp-filter name="VTODO">
  532. <C:time-range start="20130801T000000Z" end="20131001T000000Z"/>
  533. </C:comp-filter>
  534. </C:comp-filter>"""], "todo", items=8)
  535. assert "href>/calendar.ics/todo1.ics</" in answer
  536. assert "href>/calendar.ics/todo2.ics</" in answer
  537. assert "href>/calendar.ics/todo3.ics</" in answer
  538. assert "href>/calendar.ics/todo4.ics</" in answer
  539. assert "href>/calendar.ics/todo5.ics</" in answer
  540. assert "href>/calendar.ics/todo6.ics</" in answer
  541. assert "href>/calendar.ics/todo7.ics</" in answer
  542. assert "href>/calendar.ics/todo8.ics</" in answer
  543. answer = self._test_filter(["""
  544. <C:comp-filter name="VCALENDAR">
  545. <C:comp-filter name="VTODO">
  546. <C:time-range start="20130901T160000Z" end="20130901T183000Z"/>
  547. </C:comp-filter>
  548. </C:comp-filter>"""], "todo", items=8)
  549. assert "href>/calendar.ics/todo1.ics</" not in answer
  550. assert "href>/calendar.ics/todo2.ics</" in answer
  551. assert "href>/calendar.ics/todo3.ics</" in answer
  552. assert "href>/calendar.ics/todo4.ics</" not in answer
  553. assert "href>/calendar.ics/todo5.ics</" not in answer
  554. assert "href>/calendar.ics/todo6.ics</" not in answer
  555. assert "href>/calendar.ics/todo7.ics</" in answer
  556. assert "href>/calendar.ics/todo8.ics</" in answer
  557. answer = self._test_filter(["""
  558. <C:comp-filter name="VCALENDAR">
  559. <C:comp-filter name="VTODO">
  560. <C:time-range start="20130903T160000Z" end="20130901T183000Z"/>
  561. </C:comp-filter>
  562. </C:comp-filter>"""], "todo", items=8)
  563. assert "href>/calendar.ics/todo2.ics</" not in answer
  564. answer = self._test_filter(["""
  565. <C:comp-filter name="VCALENDAR">
  566. <C:comp-filter name="VTODO">
  567. <C:time-range start="20130903T160000Z" end="20130901T173000Z"/>
  568. </C:comp-filter>
  569. </C:comp-filter>"""], "todo", items=8)
  570. assert "href>/calendar.ics/todo2.ics</" not in answer
  571. answer = self._test_filter(["""
  572. <C:comp-filter name="VCALENDAR">
  573. <C:comp-filter name="VTODO">
  574. <C:time-range start="20130903T160000Z" end="20130903T173000Z"/>
  575. </C:comp-filter>
  576. </C:comp-filter>"""], "todo", items=8)
  577. assert "href>/calendar.ics/todo3.ics</" not in answer
  578. answer = self._test_filter(["""
  579. <C:comp-filter name="VCALENDAR">
  580. <C:comp-filter name="VTODO">
  581. <C:time-range start="20130903T160000Z" end="20130803T203000Z"/>
  582. </C:comp-filter>
  583. </C:comp-filter>"""], "todo", items=8)
  584. assert "href>/calendar.ics/todo7.ics</" in answer
  585. def test_time_range_filter_todos_rrule(self):
  586. """Report request with time-range filter on todos with rrules."""
  587. answer = self._test_filter(["""
  588. <C:comp-filter name="VCALENDAR">
  589. <C:comp-filter name="VTODO">
  590. <C:time-range start="20130801T000000Z" end="20131001T000000Z"/>
  591. </C:comp-filter>
  592. </C:comp-filter>"""], "todo", items=2)
  593. assert "href>/calendar.ics/todo1.ics</" in answer
  594. assert "href>/calendar.ics/todo2.ics</" in answer
  595. answer = self._test_filter(["""
  596. <C:comp-filter name="VCALENDAR">
  597. <C:comp-filter name="VTODO">
  598. <C:time-range start="20140801T000000Z" end="20141001T000000Z"/>
  599. </C:comp-filter>
  600. </C:comp-filter>"""], "todo", items=2)
  601. assert "href>/calendar.ics/todo1.ics</" not in answer
  602. assert "href>/calendar.ics/todo2.ics</" in answer
  603. answer = self._test_filter(["""
  604. <C:comp-filter name="VCALENDAR">
  605. <C:comp-filter name="VTODO">
  606. <C:time-range start="20140902T000000Z" end="20140903T000000Z"/>
  607. </C:comp-filter>
  608. </C:comp-filter>"""], "todo", items=2)
  609. assert "href>/calendar.ics/todo1.ics</" not in answer
  610. assert "href>/calendar.ics/todo2.ics</" in answer
  611. answer = self._test_filter(["""
  612. <C:comp-filter name="VCALENDAR">
  613. <C:comp-filter name="VTODO">
  614. <C:time-range start="20140904T000000Z" end="20140914T000000Z"/>
  615. </C:comp-filter>
  616. </C:comp-filter>"""], "todo", items=2)
  617. assert "href>/calendar.ics/todo1.ics</" not in answer
  618. assert "href>/calendar.ics/todo2.ics</" not in answer
  619. def test_time_range_filter_journals(self):
  620. """Report request with time-range filter on journals."""
  621. answer = self._test_filter(["""
  622. <C:comp-filter name="VCALENDAR">
  623. <C:comp-filter name="VJOURNAL">
  624. <C:time-range start="19991229T000000Z" end="20000202T000000Z"/>
  625. </C:comp-filter>
  626. </C:comp-filter>"""], "journal", items=3)
  627. assert "href>/calendar.ics/journal1.ics</" not in answer
  628. assert "href>/calendar.ics/journal2.ics</" in answer
  629. assert "href>/calendar.ics/journal3.ics</" in answer
  630. answer = self._test_filter(["""
  631. <C:comp-filter name="VCALENDAR">
  632. <C:comp-filter name="VJOURNAL">
  633. <C:time-range start="19991229T000000Z" end="20000202T000000Z"/>
  634. </C:comp-filter>
  635. </C:comp-filter>"""], "journal", items=3)
  636. assert "href>/calendar.ics/journal1.ics</" not in answer
  637. assert "href>/calendar.ics/journal2.ics</" in answer
  638. assert "href>/calendar.ics/journal3.ics</" in answer
  639. answer = self._test_filter(["""
  640. <C:comp-filter name="VCALENDAR">
  641. <C:comp-filter name="VJOURNAL">
  642. <C:time-range start="19981229T000000Z" end="19991012T000000Z"/>
  643. </C:comp-filter>
  644. </C:comp-filter>"""], "journal", items=3)
  645. assert "href>/calendar.ics/journal1.ics</" not in answer
  646. assert "href>/calendar.ics/journal2.ics</" not in answer
  647. assert "href>/calendar.ics/journal3.ics</" not in answer
  648. answer = self._test_filter(["""
  649. <C:comp-filter name="VCALENDAR">
  650. <C:comp-filter name="VJOURNAL">
  651. <C:time-range start="20131229T000000Z" end="21520202T000000Z"/>
  652. </C:comp-filter>
  653. </C:comp-filter>"""], "journal", items=3)
  654. assert "href>/calendar.ics/journal1.ics</" not in answer
  655. assert "href>/calendar.ics/journal2.ics</" in answer
  656. assert "href>/calendar.ics/journal3.ics</" not in answer
  657. answer = self._test_filter(["""
  658. <C:comp-filter name="VCALENDAR">
  659. <C:comp-filter name="VJOURNAL">
  660. <C:time-range start="20000101T000000Z" end="20000202T000000Z"/>
  661. </C:comp-filter>
  662. </C:comp-filter>"""], "journal", items=3)
  663. assert "href>/calendar.ics/journal1.ics</" not in answer
  664. assert "href>/calendar.ics/journal2.ics</" in answer
  665. assert "href>/calendar.ics/journal3.ics</" in answer
  666. def test_time_range_filter_journals_rrule(self):
  667. """Report request with time-range filter on journals with rrules."""
  668. answer = self._test_filter(["""
  669. <C:comp-filter name="VCALENDAR">
  670. <C:comp-filter name="VJOURNAL">
  671. <C:time-range start="19991229T000000Z" end="20000202T000000Z"/>
  672. </C:comp-filter>
  673. </C:comp-filter>"""], "journal", items=2)
  674. assert "href>/calendar.ics/journal1.ics</" not in answer
  675. assert "href>/calendar.ics/journal2.ics</" in answer
  676. answer = self._test_filter(["""
  677. <C:comp-filter name="VCALENDAR">
  678. <C:comp-filter name="VJOURNAL">
  679. <C:time-range start="20051229T000000Z" end="20060202T000000Z"/>
  680. </C:comp-filter>
  681. </C:comp-filter>"""], "journal", items=2)
  682. assert "href>/calendar.ics/journal1.ics</" not in answer
  683. assert "href>/calendar.ics/journal2.ics</" in answer
  684. answer = self._test_filter(["""
  685. <C:comp-filter name="VCALENDAR">
  686. <C:comp-filter name="VJOURNAL">
  687. <C:time-range start="20060102T000000Z" end="20060202T000000Z"/>
  688. </C:comp-filter>
  689. </C:comp-filter>"""], "journal", items=2)
  690. assert "href>/calendar.ics/journal1.ics</" not in answer
  691. assert "href>/calendar.ics/journal2.ics</" not in answer
  692. def test_principal_collection_creation(self):
  693. """Verify existence of the principal collection."""
  694. status, headers, answer = self.request(
  695. "GET", "/user/", REMOTE_USER="user")
  696. assert status == 200
  697. def test_existence_of_root_collections(self):
  698. """Verify that the root collection always exists."""
  699. # Use PROPFIND because GET returns message
  700. status, headers, answer = self.request("PROPFIND", "/")
  701. assert status == 207
  702. # it should still exist after deletion
  703. self.request("DELETE", "/")
  704. status, headers, answer = self.request("PROPFIND", "/")
  705. assert status == 207
  706. def test_fsync(self):
  707. """Create a directory and file with syncing enabled."""
  708. self.configuration.set("storage", "fsync", "True")
  709. status, headers, answer = self.request("MKCALENDAR", "/calendar.ics/")
  710. assert status == 201
  711. def test_hook(self):
  712. """Run hook."""
  713. self.configuration.set(
  714. "storage", "hook", "mkdir %s" % os.path.join("collection-root",
  715. "created_by_hook"))
  716. status, headers, answer = self.request("MKCOL", "/calendar.ics/")
  717. assert status == 201
  718. status, headers, answer = self.request("GET", "/created_by_hook/")
  719. assert status == 200
  720. def test_hook_read_access(self):
  721. """Verify that hook is not run for read accesses."""
  722. self.configuration.set(
  723. "storage", "hook", "mkdir %s" % os.path.join("collection-root",
  724. "created_by_hook"))
  725. status, headers, answer = self.request("GET", "/")
  726. assert status == 200
  727. status, headers, answer = self.request("GET", "/created_by_hook/")
  728. assert status == 404
  729. class BaseFileSystemTest(BaseTest):
  730. """Base class for filesystem backend tests."""
  731. storage_type = None
  732. def setup(self):
  733. self.configuration = config.load()
  734. self.configuration.set("storage", "type", self.storage_type)
  735. self.logger = logging.getLogger("radicale_test")
  736. self.colpath = tempfile.mkdtemp()
  737. self.configuration.set("storage", "filesystem_folder", self.colpath)
  738. # Disable syncing to disk for better performance
  739. self.configuration.set("storage", "fsync", "False")
  740. self.application = Application(self.configuration, self.logger)
  741. def teardown(self):
  742. shutil.rmtree(self.colpath)
  743. class TestMultiFileSystem(BaseFileSystemTest, BaseRequestsMixIn):
  744. """Test BaseRequests on multifilesystem."""
  745. storage_type = "multifilesystem"
  746. class TestCustomStorageSystem(BaseFileSystemTest):
  747. """Test custom backend loading."""
  748. storage_type = "tests.custom.storage"
  749. def test_root(self):
  750. """A simple test to verify that the custom backend works."""
  751. BaseRequestsMixIn.test_root(self)