tilia-dav 3.1.0.pre.alpha2

Sign up to get free protection for your applications and to get access to all the features.
Files changed (499) hide show
  1. checksums.yaml +7 -0
  2. data/.database.travis.yml +6 -0
  3. data/.gitignore +25 -0
  4. data/.rubocop.yml +35 -0
  5. data/.simplecov +4 -0
  6. data/.travis.yml +10 -0
  7. data/CHANGELOG.sabre.md +2084 -0
  8. data/CONTRIBUTING.md +25 -0
  9. data/Gemfile +25 -0
  10. data/Gemfile.lock +103 -0
  11. data/LICENSE +27 -0
  12. data/LICENSE.sabre +27 -0
  13. data/README.md +40 -0
  14. data/Rakefile +18 -0
  15. data/database.sample.yml +6 -0
  16. data/examples/minimal.rb +25 -0
  17. data/lib/tilia/cal_dav.rb +27 -0
  18. data/lib/tilia/cal_dav/backend.rb +17 -0
  19. data/lib/tilia/cal_dav/backend/abstract_backend.rb +194 -0
  20. data/lib/tilia/cal_dav/backend/backend_interface.rb +250 -0
  21. data/lib/tilia/cal_dav/backend/notification_support.rb +38 -0
  22. data/lib/tilia/cal_dav/backend/scheduling_support.rb +57 -0
  23. data/lib/tilia/cal_dav/backend/sequel.rb +1118 -0
  24. data/lib/tilia/cal_dav/backend/sharing_support.rb +239 -0
  25. data/lib/tilia/cal_dav/backend/subscription_support.rb +79 -0
  26. data/lib/tilia/cal_dav/backend/sync_support.rb +75 -0
  27. data/lib/tilia/cal_dav/calendar.rb +426 -0
  28. data/lib/tilia/cal_dav/calendar_home.rb +335 -0
  29. data/lib/tilia/cal_dav/calendar_object.rb +219 -0
  30. data/lib/tilia/cal_dav/calendar_query_validator.rb +294 -0
  31. data/lib/tilia/cal_dav/calendar_root.rb +57 -0
  32. data/lib/tilia/cal_dav/exception.rb +7 -0
  33. data/lib/tilia/cal_dav/exception/invalid_component_type.rb +21 -0
  34. data/lib/tilia/cal_dav/i_calendar.rb +11 -0
  35. data/lib/tilia/cal_dav/i_calendar_object.rb +13 -0
  36. data/lib/tilia/cal_dav/i_calendar_object_container.rb +32 -0
  37. data/lib/tilia/cal_dav/i_shareable_calendar.rb +40 -0
  38. data/lib/tilia/cal_dav/i_shared_calendar.rb +28 -0
  39. data/lib/tilia/cal_dav/ics_export_plugin.rb +327 -0
  40. data/lib/tilia/cal_dav/notifications.rb +12 -0
  41. data/lib/tilia/cal_dav/notifications/collection.rb +131 -0
  42. data/lib/tilia/cal_dav/notifications/i_collection.rb +17 -0
  43. data/lib/tilia/cal_dav/notifications/i_node.rb +30 -0
  44. data/lib/tilia/cal_dav/notifications/node.rb +142 -0
  45. data/lib/tilia/cal_dav/notifications/plugin.rb +138 -0
  46. data/lib/tilia/cal_dav/plugin.rb +891 -0
  47. data/lib/tilia/cal_dav/principal.rb +12 -0
  48. data/lib/tilia/cal_dav/principal/collection.rb +21 -0
  49. data/lib/tilia/cal_dav/principal/i_proxy_read.rb +13 -0
  50. data/lib/tilia/cal_dav/principal/i_proxy_write.rb +13 -0
  51. data/lib/tilia/cal_dav/principal/proxy_read.rb +127 -0
  52. data/lib/tilia/cal_dav/principal/proxy_write.rb +127 -0
  53. data/lib/tilia/cal_dav/principal/user.rb +96 -0
  54. data/lib/tilia/cal_dav/schedule.rb +14 -0
  55. data/lib/tilia/cal_dav/schedule/i_inbox.rb +12 -0
  56. data/lib/tilia/cal_dav/schedule/i_mip_plugin.rb +156 -0
  57. data/lib/tilia/cal_dav/schedule/i_outbox.rb +12 -0
  58. data/lib/tilia/cal_dav/schedule/i_scheduling_object.rb +10 -0
  59. data/lib/tilia/cal_dav/schedule/inbox.rb +211 -0
  60. data/lib/tilia/cal_dav/schedule/outbox.rb +143 -0
  61. data/lib/tilia/cal_dav/schedule/plugin.rb +851 -0
  62. data/lib/tilia/cal_dav/schedule/scheduling_object.rb +126 -0
  63. data/lib/tilia/cal_dav/shareable_calendar.rb +54 -0
  64. data/lib/tilia/cal_dav/shared_calendar.rb +120 -0
  65. data/lib/tilia/cal_dav/sharing_plugin.rb +359 -0
  66. data/lib/tilia/cal_dav/subscriptions.rb +9 -0
  67. data/lib/tilia/cal_dav/subscriptions/i_subscription.rb +37 -0
  68. data/lib/tilia/cal_dav/subscriptions/plugin.rb +83 -0
  69. data/lib/tilia/cal_dav/subscriptions/subscription.rb +205 -0
  70. data/lib/tilia/cal_dav/xml.rb +10 -0
  71. data/lib/tilia/cal_dav/xml/filter.rb +12 -0
  72. data/lib/tilia/cal_dav/xml/filter/calendar_data.rb +64 -0
  73. data/lib/tilia/cal_dav/xml/filter/comp_filter.rb +79 -0
  74. data/lib/tilia/cal_dav/xml/filter/param_filter.rb +66 -0
  75. data/lib/tilia/cal_dav/xml/filter/prop_filter.rb +80 -0
  76. data/lib/tilia/cal_dav/xml/notification.rb +13 -0
  77. data/lib/tilia/cal_dav/xml/notification/invite.rb +253 -0
  78. data/lib/tilia/cal_dav/xml/notification/invite_reply.rb +167 -0
  79. data/lib/tilia/cal_dav/xml/notification/notification_interface.rb +41 -0
  80. data/lib/tilia/cal_dav/xml/notification/system_status.rb +139 -0
  81. data/lib/tilia/cal_dav/xml/property.rb +15 -0
  82. data/lib/tilia/cal_dav/xml/property/allowed_sharing_modes.rb +64 -0
  83. data/lib/tilia/cal_dav/xml/property/email_address_set.rb +60 -0
  84. data/lib/tilia/cal_dav/xml/property/invite.rb +207 -0
  85. data/lib/tilia/cal_dav/xml/property/schedule_calendar_transp.rb +108 -0
  86. data/lib/tilia/cal_dav/xml/property/supported_calendar_component_set.rb +100 -0
  87. data/lib/tilia/cal_dav/xml/property/supported_calendar_data.rb +50 -0
  88. data/lib/tilia/cal_dav/xml/property/supported_collation_set.rb +47 -0
  89. data/lib/tilia/cal_dav/xml/request.rb +14 -0
  90. data/lib/tilia/cal_dav/xml/request/calendar_multi_get_report.rb +99 -0
  91. data/lib/tilia/cal_dav/xml/request/calendar_query_report.rb +112 -0
  92. data/lib/tilia/cal_dav/xml/request/free_busy_query_report.rb +70 -0
  93. data/lib/tilia/cal_dav/xml/request/invite_reply.rb +110 -0
  94. data/lib/tilia/cal_dav/xml/request/mk_calendar.rb +67 -0
  95. data/lib/tilia/cal_dav/xml/request/share.rb +93 -0
  96. data/lib/tilia/card_dav.rb +17 -0
  97. data/lib/tilia/card_dav/address_book.rb +338 -0
  98. data/lib/tilia/card_dav/address_book_home.rb +192 -0
  99. data/lib/tilia/card_dav/address_book_root.rb +58 -0
  100. data/lib/tilia/card_dav/backend.rb +12 -0
  101. data/lib/tilia/card_dav/backend/abstract_backend.rb +30 -0
  102. data/lib/tilia/card_dav/backend/backend_interface.rb +175 -0
  103. data/lib/tilia/card_dav/backend/sequel.rb +476 -0
  104. data/lib/tilia/card_dav/backend/sync_support.rb +80 -0
  105. data/lib/tilia/card_dav/card.rb +193 -0
  106. data/lib/tilia/card_dav/i_address_book.rb +10 -0
  107. data/lib/tilia/card_dav/i_card.rb +11 -0
  108. data/lib/tilia/card_dav/i_directory.rb +14 -0
  109. data/lib/tilia/card_dav/plugin.rb +724 -0
  110. data/lib/tilia/card_dav/vcf_export_plugin.rb +122 -0
  111. data/lib/tilia/card_dav/xml.rb +9 -0
  112. data/lib/tilia/card_dav/xml/filter.rb +11 -0
  113. data/lib/tilia/card_dav/xml/filter/address_data.rb +50 -0
  114. data/lib/tilia/card_dav/xml/filter/param_filter.rb +71 -0
  115. data/lib/tilia/card_dav/xml/filter/prop_filter.rb +77 -0
  116. data/lib/tilia/card_dav/xml/property.rb +10 -0
  117. data/lib/tilia/card_dav/xml/property/supported_address_data.rb +67 -0
  118. data/lib/tilia/card_dav/xml/property/supported_collation_set.rb +38 -0
  119. data/lib/tilia/card_dav/xml/request.rb +10 -0
  120. data/lib/tilia/card_dav/xml/request/address_book_multi_get_report.rb +91 -0
  121. data/lib/tilia/card_dav/xml/request/address_book_query_report.rb +156 -0
  122. data/lib/tilia/dav.rb +94 -0
  123. data/lib/tilia/dav/auth.rb +8 -0
  124. data/lib/tilia/dav/auth/backend.rb +15 -0
  125. data/lib/tilia/dav/auth/backend/abstract_basic.rb +119 -0
  126. data/lib/tilia/dav/auth/backend/abstract_digest.rb +132 -0
  127. data/lib/tilia/dav/auth/backend/apache.rb +85 -0
  128. data/lib/tilia/dav/auth/backend/backend_interface.rb +61 -0
  129. data/lib/tilia/dav/auth/backend/basic_call_back.rb +46 -0
  130. data/lib/tilia/dav/auth/backend/file.rb +61 -0
  131. data/lib/tilia/dav/auth/backend/sequel.rb +46 -0
  132. data/lib/tilia/dav/auth/plugin.rb +157 -0
  133. data/lib/tilia/dav/browser.rb +12 -0
  134. data/lib/tilia/dav/browser/assets/favicon.ico +0 -0
  135. data/lib/tilia/dav/browser/assets/openiconic/ICON-LICENSE +21 -0
  136. data/lib/tilia/dav/browser/assets/openiconic/open-iconic.css +510 -0
  137. data/lib/tilia/dav/browser/assets/openiconic/open-iconic.eot +0 -0
  138. data/lib/tilia/dav/browser/assets/openiconic/open-iconic.otf +0 -0
  139. data/lib/tilia/dav/browser/assets/openiconic/open-iconic.svg +543 -0
  140. data/lib/tilia/dav/browser/assets/openiconic/open-iconic.ttf +0 -0
  141. data/lib/tilia/dav/browser/assets/openiconic/open-iconic.woff +0 -0
  142. data/lib/tilia/dav/browser/assets/sabredav.css +228 -0
  143. data/lib/tilia/dav/browser/assets/sabredav.png +0 -0
  144. data/lib/tilia/dav/browser/guess_content_type.rb +80 -0
  145. data/lib/tilia/dav/browser/html_output.rb +27 -0
  146. data/lib/tilia/dav/browser/html_output_helper.rb +86 -0
  147. data/lib/tilia/dav/browser/map_get_to_prop_find.rb +41 -0
  148. data/lib/tilia/dav/browser/plugin.rb +693 -0
  149. data/lib/tilia/dav/browser/prop_find_all.rb +95 -0
  150. data/lib/tilia/dav/client.rb +341 -0
  151. data/lib/tilia/dav/collection.rb +79 -0
  152. data/lib/tilia/dav/core_plugin.rb +824 -0
  153. data/lib/tilia/dav/exception.rb +59 -0
  154. data/lib/tilia/dav/exception/bad_request.rb +18 -0
  155. data/lib/tilia/dav/exception/conflict.rb +18 -0
  156. data/lib/tilia/dav/exception/conflicting_lock.rb +26 -0
  157. data/lib/tilia/dav/exception/forbidden.rb +18 -0
  158. data/lib/tilia/dav/exception/insufficient_storage.rb +18 -0
  159. data/lib/tilia/dav/exception/invalid_resource_type.rb +23 -0
  160. data/lib/tilia/dav/exception/invalid_sync_token.rb +26 -0
  161. data/lib/tilia/dav/exception/length_required.rb +18 -0
  162. data/lib/tilia/dav/exception/lock_token_matches_request_uri.rb +25 -0
  163. data/lib/tilia/dav/exception/locked.rb +48 -0
  164. data/lib/tilia/dav/exception/method_not_allowed.rb +29 -0
  165. data/lib/tilia/dav/exception/not_authenticated.rb +18 -0
  166. data/lib/tilia/dav/exception/not_found.rb +18 -0
  167. data/lib/tilia/dav/exception/not_implemented.rb +18 -0
  168. data/lib/tilia/dav/exception/payment_required.rb +18 -0
  169. data/lib/tilia/dav/exception/precondition_failed.rb +47 -0
  170. data/lib/tilia/dav/exception/report_not_supported.rb +21 -0
  171. data/lib/tilia/dav/exception/requested_range_not_satisfiable.rb +18 -0
  172. data/lib/tilia/dav/exception/service_unavailable.rb +18 -0
  173. data/lib/tilia/dav/exception/too_many_matches.rb +21 -0
  174. data/lib/tilia/dav/exception/unsupported_media_type.rb +18 -0
  175. data/lib/tilia/dav/file.rb +58 -0
  176. data/lib/tilia/dav/fs.rb +9 -0
  177. data/lib/tilia/dav/fs/directory.rb +119 -0
  178. data/lib/tilia/dav/fs/file.rb +69 -0
  179. data/lib/tilia/dav/fs/node.rb +57 -0
  180. data/lib/tilia/dav/fs_ext.rb +8 -0
  181. data/lib/tilia/dav/fs_ext/directory.rb +175 -0
  182. data/lib/tilia/dav/fs_ext/file.rb +118 -0
  183. data/lib/tilia/dav/i_collection.rb +65 -0
  184. data/lib/tilia/dav/i_extended_collection.rb +36 -0
  185. data/lib/tilia/dav/i_file.rb +70 -0
  186. data/lib/tilia/dav/i_move_target.rb +37 -0
  187. data/lib/tilia/dav/i_multi_get.rb +29 -0
  188. data/lib/tilia/dav/i_node.rb +33 -0
  189. data/lib/tilia/dav/i_properties.rb +39 -0
  190. data/lib/tilia/dav/i_quota.rb +19 -0
  191. data/lib/tilia/dav/locks.rb +9 -0
  192. data/lib/tilia/dav/locks/backend.rb +12 -0
  193. data/lib/tilia/dav/locks/backend/abstract_backend.rb +16 -0
  194. data/lib/tilia/dav/locks/backend/backend_interface.rb +41 -0
  195. data/lib/tilia/dav/locks/backend/file.rb +146 -0
  196. data/lib/tilia/dav/locks/backend/sequel.rb +154 -0
  197. data/lib/tilia/dav/locks/lock_info.rb +60 -0
  198. data/lib/tilia/dav/locks/plugin.rb +467 -0
  199. data/lib/tilia/dav/mk_col.rb +47 -0
  200. data/lib/tilia/dav/mount.rb +7 -0
  201. data/lib/tilia/dav/mount/plugin.rb +62 -0
  202. data/lib/tilia/dav/node.rb +36 -0
  203. data/lib/tilia/dav/partial_update.rb +8 -0
  204. data/lib/tilia/dav/partial_update/i_patch_support.rb +40 -0
  205. data/lib/tilia/dav/partial_update/plugin.rb +179 -0
  206. data/lib/tilia/dav/prop_find.rb +262 -0
  207. data/lib/tilia/dav/prop_patch.rb +278 -0
  208. data/lib/tilia/dav/property_storage.rb +8 -0
  209. data/lib/tilia/dav/property_storage/backend.rb +10 -0
  210. data/lib/tilia/dav/property_storage/backend/backend_interface.rb +69 -0
  211. data/lib/tilia/dav/property_storage/backend/sequel.rb +192 -0
  212. data/lib/tilia/dav/property_storage/plugin.rb +131 -0
  213. data/lib/tilia/dav/server.rb +1388 -0
  214. data/lib/tilia/dav/server_plugin.rb +81 -0
  215. data/lib/tilia/dav/simple_collection.rb +71 -0
  216. data/lib/tilia/dav/simple_file.rb +82 -0
  217. data/lib/tilia/dav/string_util.rb +68 -0
  218. data/lib/tilia/dav/sync.rb +8 -0
  219. data/lib/tilia/dav/sync/i_sync_collection.rb +80 -0
  220. data/lib/tilia/dav/sync/plugin.rb +225 -0
  221. data/lib/tilia/dav/temporary_file_filter_plugin.rb +248 -0
  222. data/lib/tilia/dav/tree.rb +270 -0
  223. data/lib/tilia/dav/uuid_util.rb +45 -0
  224. data/lib/tilia/dav/version.rb +9 -0
  225. data/lib/tilia/dav/xml.rb +11 -0
  226. data/lib/tilia/dav/xml/element.rb +10 -0
  227. data/lib/tilia/dav/xml/element/prop.rb +92 -0
  228. data/lib/tilia/dav/xml/element/response.rb +188 -0
  229. data/lib/tilia/dav/xml/property.rb +16 -0
  230. data/lib/tilia/dav/xml/property/complex.rb +76 -0
  231. data/lib/tilia/dav/xml/property/get_last_modified.rb +79 -0
  232. data/lib/tilia/dav/xml/property/href.rb +137 -0
  233. data/lib/tilia/dav/xml/property/lock_discovery.rb +89 -0
  234. data/lib/tilia/dav/xml/property/resource_type.rb +96 -0
  235. data/lib/tilia/dav/xml/property/supported_lock.rb +48 -0
  236. data/lib/tilia/dav/xml/property/supported_method_set.rb +101 -0
  237. data/lib/tilia/dav/xml/property/supported_report_set.rb +118 -0
  238. data/lib/tilia/dav/xml/request.rb +13 -0
  239. data/lib/tilia/dav/xml/request/lock.rb +67 -0
  240. data/lib/tilia/dav/xml/request/mk_col.rb +69 -0
  241. data/lib/tilia/dav/xml/request/prop_find.rb +70 -0
  242. data/lib/tilia/dav/xml/request/prop_patch.rb +101 -0
  243. data/lib/tilia/dav/xml/request/sync_collection_report.rb +102 -0
  244. data/lib/tilia/dav/xml/response.rb +9 -0
  245. data/lib/tilia/dav/xml/response/multi_status.rb +108 -0
  246. data/lib/tilia/dav/xml/service.rb +42 -0
  247. data/lib/tilia/dav_acl.rb +16 -0
  248. data/lib/tilia/dav_acl/abstract_principal_collection.rb +143 -0
  249. data/lib/tilia/dav_acl/exception.rb +11 -0
  250. data/lib/tilia/dav_acl/exception/ace_conflict.rb +21 -0
  251. data/lib/tilia/dav_acl/exception/need_privileges.rb +65 -0
  252. data/lib/tilia/dav_acl/exception/no_abstract.rb +21 -0
  253. data/lib/tilia/dav_acl/exception/not_recognized_principal.rb +21 -0
  254. data/lib/tilia/dav_acl/exception/not_supported_privilege.rb +21 -0
  255. data/lib/tilia/dav_acl/fs.rb +9 -0
  256. data/lib/tilia/dav_acl/fs/collection.rb +108 -0
  257. data/lib/tilia/dav_acl/fs/file.rb +87 -0
  258. data/lib/tilia/dav_acl/fs/home_collection.rb +148 -0
  259. data/lib/tilia/dav_acl/i_acl.rb +61 -0
  260. data/lib/tilia/dav_acl/i_principal.rb +63 -0
  261. data/lib/tilia/dav_acl/i_principal_collection.rb +52 -0
  262. data/lib/tilia/dav_acl/plugin.rb +1109 -0
  263. data/lib/tilia/dav_acl/principal.rb +213 -0
  264. data/lib/tilia/dav_acl/principal_backend.rb +11 -0
  265. data/lib/tilia/dav_acl/principal_backend/abstract_backend.rb +42 -0
  266. data/lib/tilia/dav_acl/principal_backend/backend_interface.rb +127 -0
  267. data/lib/tilia/dav_acl/principal_backend/create_principal_support.rb +27 -0
  268. data/lib/tilia/dav_acl/principal_backend/sequel.rb +313 -0
  269. data/lib/tilia/dav_acl/principal_collection.rb +117 -0
  270. data/lib/tilia/dav_acl/xml.rb +8 -0
  271. data/lib/tilia/dav_acl/xml/property.rb +13 -0
  272. data/lib/tilia/dav_acl/xml/property/acl.rb +222 -0
  273. data/lib/tilia/dav_acl/xml/property/acl_restrictions.rb +40 -0
  274. data/lib/tilia/dav_acl/xml/property/current_user_privilege_set.rb +125 -0
  275. data/lib/tilia/dav_acl/xml/property/principal.rb +149 -0
  276. data/lib/tilia/dav_acl/xml/property/supported_privilege_set.rb +135 -0
  277. data/lib/tilia/dav_acl/xml/request.rb +11 -0
  278. data/lib/tilia/dav_acl/xml/request/expand_property_report.rb +86 -0
  279. data/lib/tilia/dav_acl/xml/request/principal_property_search_report.rb +111 -0
  280. data/lib/tilia/dav_acl/xml/request/principal_search_property_set_report.rb +49 -0
  281. data/test/cal_dav/backend/abstract_sequel_test.rb +817 -0
  282. data/test/cal_dav/backend/abstract_test.rb +163 -0
  283. data/test/cal_dav/backend/mock.rb +169 -0
  284. data/test/cal_dav/backend/mock_scheduling.rb +84 -0
  285. data/test/cal_dav/backend/mock_sharing.rb +124 -0
  286. data/test/cal_dav/backend/mock_subscription_support.rb +123 -0
  287. data/test/cal_dav/backend/sequel_my_sql_test.rb +102 -0
  288. data/test/cal_dav/backend/sequel_sqlite_test.rb +105 -0
  289. data/test/cal_dav/calendar_home_notifications_test.rb +41 -0
  290. data/test/cal_dav/calendar_home_shared_calendars_test.rb +64 -0
  291. data/test/cal_dav/calendar_home_subscriptions_test.rb +67 -0
  292. data/test/cal_dav/calendar_home_test.rb +144 -0
  293. data/test/cal_dav/calendar_object_test.rb +317 -0
  294. data/test/cal_dav/calendar_query_v_alarm_test.rb +114 -0
  295. data/test/cal_dav/calendar_query_validator_test.rb +820 -0
  296. data/test/cal_dav/calendar_test.rb +203 -0
  297. data/test/cal_dav/expand_events_double_events_test.rb +94 -0
  298. data/test/cal_dav/expand_events_dtstar_tand_dten_dby_day_test.rb +94 -0
  299. data/test/cal_dav/expand_events_dtstar_tand_dtend_test.rb +100 -0
  300. data/test/cal_dav/expand_events_floating_time_test.rb +211 -0
  301. data/test/cal_dav/free_busy_report_test.rb +156 -0
  302. data/test/cal_dav/get_events_by_timerange_test.rb +74 -0
  303. data/test/cal_dav/ics_export_plugin_test.rb +638 -0
  304. data/test/cal_dav/issue166_test.rb +59 -0
  305. data/test/cal_dav/issue172_test.rb +139 -0
  306. data/test/cal_dav/issue203_test.rb +130 -0
  307. data/test/cal_dav/issue205_test.rb +89 -0
  308. data/test/cal_dav/issue211_test.rb +84 -0
  309. data/test/cal_dav/issue220_test.rb +94 -0
  310. data/test/cal_dav/issue228_test.rb +74 -0
  311. data/test/cal_dav/j_cal_transform_test.rb +244 -0
  312. data/test/cal_dav/notifications/collection_test.rb +67 -0
  313. data/test/cal_dav/notifications/node_test.rb +73 -0
  314. data/test/cal_dav/notifications/plugin_test.rb +144 -0
  315. data/test/cal_dav/plugin_test.rb +1049 -0
  316. data/test/cal_dav/principal/collection_test.rb +19 -0
  317. data/test/cal_dav/principal/proxy_read_test.rb +67 -0
  318. data/test/cal_dav/principal/proxy_write_test.rb +29 -0
  319. data/test/cal_dav/principal/user_test.rb +91 -0
  320. data/test/cal_dav/schedule/deliver_new_event_test.rb +81 -0
  321. data/test/cal_dav/schedule/free_busy_request_test.rb +565 -0
  322. data/test/cal_dav/schedule/i_mip/mock_plugin.rb +40 -0
  323. data/test/cal_dav/schedule/i_mip_plugin_test.rb +196 -0
  324. data/test/cal_dav/schedule/inbox_test.rb +150 -0
  325. data/test/cal_dav/schedule/outbox_post_test.rb +124 -0
  326. data/test/cal_dav/schedule/outbox_test.rb +76 -0
  327. data/test/cal_dav/schedule/plugin_basic_test.rb +39 -0
  328. data/test/cal_dav/schedule/plugin_properties_test.rb +96 -0
  329. data/test/cal_dav/schedule/plugin_properties_with_shared_calendar_test.rb +69 -0
  330. data/test/cal_dav/schedule/schedule_deliver_test.rb +605 -0
  331. data/test/cal_dav/schedule/scheduling_object_test.rb +327 -0
  332. data/test/cal_dav/shareable_calendar_test.rb +58 -0
  333. data/test/cal_dav/shared_calendar_test.rb +189 -0
  334. data/test/cal_dav/sharing_plugin_test.rb +373 -0
  335. data/test/cal_dav/subscriptions/create_subscription_test.rb +115 -0
  336. data/test/cal_dav/subscriptions/plugin_test.rb +46 -0
  337. data/test/cal_dav/subscriptions/subscription_test.rb +119 -0
  338. data/test/cal_dav/test_util.rb +164 -0
  339. data/test/cal_dav/validate_i_cal_test.rb +219 -0
  340. data/test/cal_dav/xml/notification/invite_reply_test.rb +136 -0
  341. data/test/cal_dav/xml/notification/invite_test.rb +225 -0
  342. data/test/cal_dav/xml/notification/system_status_test.rb +63 -0
  343. data/test/cal_dav/xml/property/allowed_sharing_modes_test.rb +34 -0
  344. data/test/cal_dav/xml/property/email_address_set_test.rb +35 -0
  345. data/test/cal_dav/xml/property/invite_test.rb +173 -0
  346. data/test/cal_dav/xml/property/schedule_calendar_transp_test.rb +96 -0
  347. data/test/cal_dav/xml/property/supported_calendar_component_set_test.rb +76 -0
  348. data/test/cal_dav/xml/property/supported_calendar_data_test.rb +32 -0
  349. data/test/cal_dav/xml/property/supported_collation_set_test.rb +33 -0
  350. data/test/cal_dav/xml/request/calendar_query_report_test.rb +339 -0
  351. data/test/cal_dav/xml/request/invite_reply_test.rb +68 -0
  352. data/test/cal_dav/xml/request/share_test.rb +79 -0
  353. data/test/card_dav/abstract_plugin_test.rb +24 -0
  354. data/test/card_dav/address_book_home_test.rb +128 -0
  355. data/test/card_dav/address_book_query_test.rb +303 -0
  356. data/test/card_dav/address_book_root_test.rb +26 -0
  357. data/test/card_dav/address_book_test.rb +166 -0
  358. data/test/card_dav/backend/abstract_sequel_test.rb +302 -0
  359. data/test/card_dav/backend/mock.rb +122 -0
  360. data/test/card_dav/backend/sequel_my_sql_test.rb +56 -0
  361. data/test/card_dav/backend/sequel_sqlite_test.rb +59 -0
  362. data/test/card_dav/card_test.rb +164 -0
  363. data/test/card_dav/i_directory_test.rb +22 -0
  364. data/test/card_dav/multi_get_test.rb +97 -0
  365. data/test/card_dav/plugin_test.rb +87 -0
  366. data/test/card_dav/sogo_strip_content_type_test.rb +63 -0
  367. data/test/card_dav/test_util.rb +51 -0
  368. data/test/card_dav/validate_filter_test.rb +210 -0
  369. data/test/card_dav/validate_v_card_test.rb +143 -0
  370. data/test/card_dav/vcf_export_test.rb +66 -0
  371. data/test/card_dav/xml/property/supported_address_data_test.rb +34 -0
  372. data/test/card_dav/xml/property/supported_collation_set_test.rb +34 -0
  373. data/test/card_dav/xml/request/address_book_query_report_test.rb +276 -0
  374. data/test/dav/abstract_server.rb +36 -0
  375. data/test/dav/auth/backend/abstract_basic_test.rb +74 -0
  376. data/test/dav/auth/backend/abstract_digest_test.rb +114 -0
  377. data/test/dav/auth/backend/abstract_sequel_test.rb +25 -0
  378. data/test/dav/auth/backend/apache_test.rb +60 -0
  379. data/test/dav/auth/backend/basic_call_back_test.rb +33 -0
  380. data/test/dav/auth/backend/file_test.rb +43 -0
  381. data/test/dav/auth/backend/mock.rb +73 -0
  382. data/test/dav/auth/backend/sequel_my_sql_test.rb +32 -0
  383. data/test/dav/auth/backend/sequel_sqlite_test.rb +21 -0
  384. data/test/dav/auth/plugin_test.rb +92 -0
  385. data/test/dav/basic_node_test.rb +143 -0
  386. data/test/dav/browser/guess_content_type_test.rb +44 -0
  387. data/test/dav/browser/map_get_to_prop_find_test.rb +37 -0
  388. data/test/dav/browser/plugin_test.rb +165 -0
  389. data/test/dav/browser/prop_find_all_test.rb +59 -0
  390. data/test/dav/client_mock.rb +24 -0
  391. data/test/dav/client_test.rb +231 -0
  392. data/test/dav/copy_test.rb +33 -0
  393. data/test/dav/exception/locked_test.rb +61 -0
  394. data/test/dav/exception/payment_required_test.rb +14 -0
  395. data/test/dav/exception/service_unavailable_test.rb +14 -0
  396. data/test/dav/exception/too_many_matches_test.rb +31 -0
  397. data/test/dav/exception_test.rb +24 -0
  398. data/test/dav/fs_ext/file_test.rb +72 -0
  399. data/test/dav/fs_ext/server_test.rb +251 -0
  400. data/test/dav/get_if_conditions_test.rb +299 -0
  401. data/test/dav/http_delete_test.rb +110 -0
  402. data/test/dav/http_get_test.rb +130 -0
  403. data/test/dav/http_head_test.rb +80 -0
  404. data/test/dav/http_move_test.rb +105 -0
  405. data/test/dav/http_prefer_parsing_test.rb +186 -0
  406. data/test/dav/http_put_test.rb +271 -0
  407. data/test/dav/issue33_test.rb +90 -0
  408. data/test/dav/locks/backend/abstract_test.rb +160 -0
  409. data/test/dav/locks/backend/file_test.rb +24 -0
  410. data/test/dav/locks/backend/mock.rb +82 -0
  411. data/test/dav/locks/backend/sequel_my_sql_test.rb +32 -0
  412. data/test/dav/locks/backend/sequel_test.rb +19 -0
  413. data/test/dav/locks/ms_word_test.rb +119 -0
  414. data/test/dav/locks/plugin2_test.rb +61 -0
  415. data/test/dav/locks/plugin_test.rb +896 -0
  416. data/test/dav/mock/collection.rb +113 -0
  417. data/test/dav/mock/file.rb +100 -0
  418. data/test/dav/mock/properties_collection.rb +80 -0
  419. data/test/dav/mock/streaming_file.rb +66 -0
  420. data/test/dav/mount/plugin_test.rb +48 -0
  421. data/test/dav/object_tree_test.rb +65 -0
  422. data/test/dav/partial_update/file_mock.rb +92 -0
  423. data/test/dav/partial_update/plugin_test.rb +125 -0
  424. data/test/dav/partial_update/specification_test.rb +77 -0
  425. data/test/dav/prop_find_test.rb +87 -0
  426. data/test/dav/prop_patch_test.rb +367 -0
  427. data/test/dav/property_storage/backend/abstract_sequel_test.rb +147 -0
  428. data/test/dav/property_storage/backend/mock.rb +96 -0
  429. data/test/dav/property_storage/backend/sequel_mysql_test.rb +32 -0
  430. data/test/dav/property_storage/backend/sequel_sqlite_test.rb +31 -0
  431. data/test/dav/property_storage/plugin_test.rb +90 -0
  432. data/test/dav/server_copy_move_test.rb +164 -0
  433. data/test/dav/server_events_test.rb +105 -0
  434. data/test/dav/server_mkcol_test.rb +337 -0
  435. data/test/dav/server_mock.rb +10 -0
  436. data/test/dav/server_plugin_test.rb +85 -0
  437. data/test/dav/server_precondition_test.rb +253 -0
  438. data/test/dav/server_props_infinite_depth_test.rb +144 -0
  439. data/test/dav/server_props_test.rb +182 -0
  440. data/test/dav/server_range_test.rb +262 -0
  441. data/test/dav/server_simple_test.rb +388 -0
  442. data/test/dav/server_update_properties_test.rb +93 -0
  443. data/test/dav/simple_file_test.rb +17 -0
  444. data/test/dav/string_util_test.rb +92 -0
  445. data/test/dav/sync/mock_sync_collection.rb +141 -0
  446. data/test/dav/sync/plugin_test.rb +491 -0
  447. data/test/dav/sync_token_property_test.rb +105 -0
  448. data/test/dav/temporary_file_filter_test.rb +179 -0
  449. data/test/dav/test_plugin.rb +24 -0
  450. data/test/dav/tree_test.rb +201 -0
  451. data/test/dav/uuid_util_test.rb +14 -0
  452. data/test/dav/xml/element/prop_test.rb +121 -0
  453. data/test/dav/xml/element/response_test.rb +202 -0
  454. data/test/dav/xml/property/href_test.rb +112 -0
  455. data/test/dav/xml/property/last_modified_test.rb +52 -0
  456. data/test/dav/xml/property/lock_discovery_test.rb +79 -0
  457. data/test/dav/xml/property/supported_method_set_test.rb +54 -0
  458. data/test/dav/xml/property/supported_report_set_test.rb +109 -0
  459. data/test/dav/xml/request/prop_find_test.rb +45 -0
  460. data/test/dav/xml/request/prop_patch_test.rb +47 -0
  461. data/test/dav/xml/request/sync_collection_test.rb +89 -0
  462. data/test/dav/xml/xml_tester.rb +35 -0
  463. data/test/dav_acl/acl_method_test.rb +299 -0
  464. data/test/dav_acl/allow_access_test.rb +94 -0
  465. data/test/dav_acl/block_access_test.rb +161 -0
  466. data/test/dav_acl/exception/ace_conflict_test.rb +33 -0
  467. data/test/dav_acl/exception/need_privileges_exception_test.rb +43 -0
  468. data/test/dav_acl/exception/no_abstract_test.rb +33 -0
  469. data/test/dav_acl/exception/not_recognized_principal_test.rb +33 -0
  470. data/test/dav_acl/exception/not_supported_privilege_test.rb +33 -0
  471. data/test/dav_acl/expand_properties_test.rb +265 -0
  472. data/test/dav_acl/fs/collection_test.rb +39 -0
  473. data/test/dav_acl/fs/file_test.rb +47 -0
  474. data/test/dav_acl/fs/home_collection_test.rb +82 -0
  475. data/test/dav_acl/mock_acl_node.rb +27 -0
  476. data/test/dav_acl/mock_principal.rb +27 -0
  477. data/test/dav_acl/plugin_admin_test.rb +60 -0
  478. data/test/dav_acl/plugin_properties_test.rb +346 -0
  479. data/test/dav_acl/plugin_update_properties_test.rb +82 -0
  480. data/test/dav_acl/principal_backend/abstract_sequel_test.rb +159 -0
  481. data/test/dav_acl/principal_backend/mock.rb +150 -0
  482. data/test/dav_acl/principal_backend/sequel_my_sql_test.rb +43 -0
  483. data/test/dav_acl/principal_backend/sequel_sqlite_test.rb +31 -0
  484. data/test/dav_acl/principal_collection_test.rb +44 -0
  485. data/test/dav_acl/principal_property_search_test.rb +354 -0
  486. data/test/dav_acl/principal_search_property_set_test.rb +125 -0
  487. data/test/dav_acl/principal_test.rb +181 -0
  488. data/test/dav_acl/simple_plugin_test.rb +320 -0
  489. data/test/dav_acl/xml/property/acl_restrictions_test.rb +28 -0
  490. data/test/dav_acl/xml/property/acl_test.rb +325 -0
  491. data/test/dav_acl/xml/property/current_user_privilege_set_test.rb +77 -0
  492. data/test/dav_acl/xml/property/principal_test.rb +158 -0
  493. data/test/dav_acl/xml/property/supported_privilege_set_test.rb +109 -0
  494. data/test/dav_server_test.rb +225 -0
  495. data/test/http/response_mock.rb +16 -0
  496. data/test/http/sapi_mock.rb +29 -0
  497. data/test/test_helper.rb +176 -0
  498. data/tilia-dav.gemspec +28 -0
  499. metadata +726 -0
@@ -0,0 +1,24 @@
1
+ require 'test_helper'
2
+
3
+ module Tilia
4
+ module Dav
5
+ module Locks
6
+ module Backend
7
+ class FileTest < Minitest::Test
8
+ include AbstractTest
9
+
10
+ def backend
11
+ @temp_dir = Dir.mktmpdir
12
+
13
+ backend = File.new("#{@temp_dir}/lockdb")
14
+ backend
15
+ end
16
+
17
+ def teardown
18
+ FileUtils.remove_entry @temp_dir
19
+ end
20
+ end
21
+ end
22
+ end
23
+ end
24
+ end
@@ -0,0 +1,82 @@
1
+ module Tilia
2
+ module Dav
3
+ module Locks
4
+ module Backend
5
+ # Locks Mock backend.
6
+ #
7
+ # This backend stores lock information in memory. Mainly useful for testing.
8
+ class Mock < AbstractBackend
9
+ # Returns a list of Sabre\DAV\Locks\LockInfo objects
10
+ #
11
+ # This method should return all the locks for a particular uri, including
12
+ # locks that might be set on a parent uri.
13
+ #
14
+ # If returnChildLocks is set to true, this method should also look for
15
+ # any locks in the subtree of the uri for locks.
16
+ #
17
+ # @param string uri
18
+ # @param bool return_child_locks
19
+ # @return array
20
+ def locks(uri, return_child_locks)
21
+ new_locks = []
22
+
23
+ @locks.each do |lock|
24
+ next unless lock.uri == uri ||
25
+ # deep locks on parents
26
+ (lock.depth != 0 && uri.index("#{lock.uri}/") == 0) ||
27
+
28
+ # locks on children
29
+ (return_child_locks && (lock.uri.index("#{uri}/") == 0))
30
+ new_locks << lock
31
+ end
32
+
33
+ # Checking if we can remove any of these locks
34
+ new_locks.delete_if do |lock|
35
+ Time.now.to_i > lock.timeout + lock.created
36
+ end
37
+ new_locks
38
+ end
39
+
40
+ # Locks a uri
41
+ #
42
+ # @param string uri
43
+ # @param LockInfo lock_info
44
+ # @return bool
45
+ def lock(uri, lock_info)
46
+ # We're making the lock timeout 30 minutes
47
+ lock_info.timeout = 1800
48
+ lock_info.created = Time.now.to_i
49
+ lock_info.uri = uri
50
+
51
+ @locks.delete_if do |lock|
52
+ (lock.token == lock_info.token) || (Time.now.to_i > lock.timeout + lock.created)
53
+ end
54
+
55
+ @locks << lock_info
56
+ true
57
+ end
58
+
59
+ # Removes a lock from a uri
60
+ #
61
+ # @param string uri
62
+ # @param LockInfo lock_info
63
+ # @return bool
64
+ def unlock(_uri, lock_info)
65
+ @locks.each_with_index do |lock, k|
66
+ if lock.token == lock_info.token
67
+ @locks.delete_at(k)
68
+ return true
69
+ end
70
+ end
71
+
72
+ false
73
+ end
74
+
75
+ def initialize
76
+ @locks = []
77
+ end
78
+ end
79
+ end
80
+ end
81
+ end
82
+ end
@@ -0,0 +1,32 @@
1
+ require 'test_helper'
2
+
3
+ module Tilia
4
+ module Dav
5
+ module Locks
6
+ module Backend
7
+ class SequelMySqlTest < Minitest::Test
8
+ include AbstractTest
9
+
10
+ def backend
11
+ db = TestUtil.mysql
12
+ db.run('DROP TABLE IF EXISTS locks')
13
+ db.run(<<QUERY
14
+ CREATE TABLE locks (
15
+ id INTEGER UNSIGNED NOT NULL PRIMARY KEY AUTO_INCREMENT,
16
+ owner VARCHAR(100),
17
+ timeout INTEGER UNSIGNED,
18
+ created INTEGER,
19
+ token VARCHAR(100),
20
+ scope TINYINT,
21
+ depth TINYINT,
22
+ uri VARCHAR(255)
23
+ ) #{TestUtil.mysql_engine}
24
+ QUERY
25
+ )
26
+ Sequel.new(db)
27
+ end
28
+ end
29
+ end
30
+ end
31
+ end
32
+ end
@@ -0,0 +1,19 @@
1
+ require 'test_helper'
2
+
3
+ module Tilia
4
+ module Dav
5
+ module Locks
6
+ module Backend
7
+ class SequelTest < Minitest::Test
8
+ include AbstractTest
9
+
10
+ def backend
11
+ db = TestUtil.sqlite
12
+ db.run('CREATE TABLE locks ( id integer primary key asc, owner text, timeout text, created integer, token text, scope integer, depth integer, uri text)')
13
+ Sequel.new(db)
14
+ end
15
+ end
16
+ end
17
+ end
18
+ end
19
+ end
@@ -0,0 +1,119 @@
1
+ require 'test_helper'
2
+
3
+ module Tilia
4
+ module Dav
5
+ module Locks
6
+ class MSWordTest < Minitest::Test
7
+ def setup
8
+ @temp_dir = Dir.mktmpdir
9
+ end
10
+
11
+ def teardown
12
+ FileUtils.remove_entry @temp_dir
13
+ end
14
+
15
+ def test_lock_etc
16
+ Dir.mkdir("#{@temp_dir}/mstest")
17
+ tree = Fs::Directory.new("#{@temp_dir}/mstest")
18
+
19
+ server = ServerMock.new(tree)
20
+ server.debug_exceptions = true
21
+ locks_backend = Backend::File.new("#{@temp_dir}/locksdb")
22
+ locks_plugin = Plugin.new(locks_backend)
23
+ server.add_plugin(locks_plugin)
24
+
25
+ response1 = Http::ResponseMock.new
26
+
27
+ server.http_request = lock_request
28
+ server.http_response = response1
29
+ server.sapi = Http::SapiMock.new
30
+ server.exec
31
+
32
+ assert_equal(201, server.http_response.status, "Full response body: #{response1.body_as_string}")
33
+ assert(server.http_response.header('Lock-Token'))
34
+ lock_token = server.http_response.header('Lock-Token')
35
+
36
+ # sleep(10)
37
+
38
+ response2 = Http::ResponseMock.new
39
+
40
+ server.http_request = lock_request2
41
+ server.http_response = response2
42
+ server.exec
43
+
44
+ assert_equal(201, server.http_response.status)
45
+ assert(server.http_response.header('Lock-Token'))
46
+
47
+ # sleep(10)
48
+
49
+ response3 = Http::ResponseMock.new
50
+ server.http_request = get_put_request(lock_token)
51
+ server.http_response = response3
52
+ server.exec
53
+
54
+ assert_equal(204, server.http_response.status)
55
+ end
56
+
57
+ def lock_request
58
+ request = Http::Sapi.create_from_server_array(
59
+ 'REQUEST_METHOD' => 'LOCK',
60
+ 'HTTP_CONTENT_TYPE' => 'application/xml',
61
+ 'HTTP_TIMEOUT' => 'Second-3600',
62
+ 'REQUEST_PATH' => '/Nouveau%20Microsoft%20Office%20Excel%20Worksheet.xlsx'
63
+ )
64
+
65
+ request.body = <<XML
66
+ <D:lockinfo xmlns:D="DAV:">
67
+ <D:lockscope>
68
+ <D:exclusive />
69
+ </D:lockscope>
70
+ <D:locktype>
71
+ <D:write />
72
+ </D:locktype>
73
+ <D:owner>
74
+ <D:href>PC-Vista\User</D:href>
75
+ </D:owner>
76
+ </D:lockinfo>
77
+ XML
78
+
79
+ request
80
+ end
81
+
82
+ def lock_request2
83
+ request = Http::Sapi.create_from_server_array(
84
+ 'REQUEST_METHOD' => 'LOCK',
85
+ 'HTTP_CONTENT_TYPE' => 'application/xml',
86
+ 'HTTP_TIMEOUT' => 'Second-3600',
87
+ 'REQUEST_PATH' => '/~$Nouveau%20Microsoft%20Office%20Excel%20Worksheet.xlsx'
88
+ )
89
+
90
+ request.body = <<XML
91
+ <D:lockinfo xmlns:D="DAV:">
92
+ <D:lockscope>
93
+ <D:exclusive />
94
+ </D:lockscope>
95
+ <D:locktype>
96
+ <D:write />
97
+ </D:locktype>
98
+ <D:owner>
99
+ <D:href>PC-Vista\User</D:href>
100
+ </D:owner>
101
+ </D:lockinfo>
102
+ XML
103
+
104
+ request
105
+ end
106
+
107
+ def get_put_request(lock_token)
108
+ request = Http::Sapi.create_from_server_array(
109
+ 'REQUEST_METHOD' => 'PUT',
110
+ 'REQUEST_PATH' => '/Nouveau%20Microsoft%20Office%20Excel%20Worksheet.xlsx',
111
+ 'HTTP_IF' => "If: (#{lock_token})"
112
+ )
113
+ request.body = 'FAKE BODY'
114
+ request
115
+ end
116
+ end
117
+ end
118
+ end
119
+ end
@@ -0,0 +1,61 @@
1
+ require 'test_helper'
2
+
3
+ module Tilia
4
+ module Dav
5
+ module Locks
6
+ class Plugin2Test < DavServerTest
7
+ def setup
8
+ @setup_locks = true
9
+ super
10
+ end
11
+
12
+ def set_up_tree
13
+ @tree = Fs::Directory.new(@temp_dir)
14
+ end
15
+
16
+ # This test first creates a file with LOCK and then deletes it.
17
+ #
18
+ # After deleting the file, the lock should no longer be in the lock
19
+ # backend.
20
+ #
21
+ # Reported in ticket #487
22
+ def test_unlock_after_delete
23
+ body = <<XML
24
+ <?xml version="1.0"?>
25
+ <D:lockinfo xmlns:D="DAV:">
26
+ <D:lockscope><D:exclusive/></D:lockscope>
27
+ <D:locktype><D:write/></D:locktype>
28
+ </D:lockinfo>
29
+ XML
30
+
31
+ request = Http::Request.new(
32
+ 'LOCK',
33
+ '/file.txt',
34
+ {},
35
+ body
36
+ )
37
+ response = request(request)
38
+ assert_equal(201, response.status, response.body_as_string)
39
+
40
+ assert_equal(
41
+ 1,
42
+ @locks_backend.locks('file.txt', true).size
43
+ )
44
+
45
+ request = Http::Request.new(
46
+ 'DELETE',
47
+ '/file.txt',
48
+ 'If' => "(#{response.header('Lock-Token')})"
49
+ )
50
+ response = request(request)
51
+ assert_equal(204, response.status, response.body_as_string)
52
+
53
+ assert_equal(
54
+ 0,
55
+ @locks_backend.locks('file.txt', true).size
56
+ )
57
+ end
58
+ end
59
+ end
60
+ end
61
+ end
@@ -0,0 +1,896 @@
1
+ require 'test_helper'
2
+
3
+ module Tilia
4
+ module Dav
5
+ module Locks
6
+ class PluginTest < AbstractServer
7
+ def setup
8
+ super
9
+
10
+ locks_backend = Backend::File.new("#{@temp_dir}/locksdb")
11
+ @locks_plugin = Plugin.new(locks_backend)
12
+ @server.add_plugin(@locks_plugin)
13
+ end
14
+
15
+ def test_get_info
16
+ assert_has_key('name', @locks_plugin.plugin_info)
17
+ end
18
+
19
+ def test_get_features
20
+ assert_equal([2], @locks_plugin.features)
21
+ end
22
+
23
+ def test_get_http_methods
24
+ assert_equal(['LOCK', 'UNLOCK'], @locks_plugin.http_methods(''))
25
+ end
26
+
27
+ def test_lock_no_body
28
+ request = Http::Request.new('LOCK', '/test.txt')
29
+ @server.http_request = request
30
+ @server.exec
31
+
32
+ assert_equal(
33
+ {
34
+ 'X-Sabre-Version' => [Version::VERSION],
35
+ 'Content-Type' => ['application/xml; charset=utf-8']
36
+ },
37
+ @response.headers
38
+ )
39
+
40
+ assert_equal(400, @response.status)
41
+ end
42
+
43
+ def test_lock
44
+ request = Http::Request.new('LOCK', '/test.txt')
45
+ request.body = <<XML
46
+ <?xml version="1.0"?>
47
+ <D:lockinfo xmlns:D="DAV:">
48
+ <D:lockscope><D:exclusive/></D:lockscope>
49
+ <D:locktype><D:write/></D:locktype>
50
+ <D:owner>
51
+ <D:href>http://example.org/~ejw/contact.html</D:href>
52
+ </D:owner>
53
+ </D:lockinfo>
54
+ XML
55
+
56
+ @server.http_request = request
57
+ @server.exec
58
+
59
+ assert_equal('application/xml; charset=utf-8', @response.header('Content-Type'))
60
+ assert(@response.header('Lock-Token') =~ /^<opaquelocktoken:(.*)>$/, "We did not get a valid Locktoken back (#{@response.header('Lock-Token')})")
61
+
62
+ assert_equal(200, @response.status, "Got an incorrect status back. Response body: #{@response.body_as_string}")
63
+
64
+ body = @response.body.gsub(/xmlns(:[A-Za-z0-9_])?=("|')DAV:("|')/) { "xmlns#{Regexp.last_match(1)}=\"urn:DAV\"" }
65
+
66
+ xml = LibXML::XML::Document.string(body)
67
+
68
+ elements = [
69
+ '/d:prop',
70
+ '/d:prop/d:lockdiscovery',
71
+ '/d:prop/d:lockdiscovery/d:activelock',
72
+ '/d:prop/d:lockdiscovery/d:activelock/d:locktype',
73
+ '/d:prop/d:lockdiscovery/d:activelock/d:lockroot',
74
+ '/d:prop/d:lockdiscovery/d:activelock/d:lockroot/d:href',
75
+ '/d:prop/d:lockdiscovery/d:activelock/d:locktype/d:write',
76
+ '/d:prop/d:lockdiscovery/d:activelock/d:lockscope',
77
+ '/d:prop/d:lockdiscovery/d:activelock/d:lockscope/d:exclusive',
78
+ '/d:prop/d:lockdiscovery/d:activelock/d:depth',
79
+ '/d:prop/d:lockdiscovery/d:activelock/d:owner',
80
+ '/d:prop/d:lockdiscovery/d:activelock/d:timeout',
81
+ '/d:prop/d:lockdiscovery/d:activelock/d:locktoken',
82
+ '/d:prop/d:lockdiscovery/d:activelock/d:locktoken/d:href'
83
+ ]
84
+
85
+ elements.each do |elem|
86
+ data = xml.find(elem)
87
+ assert_equal(1, data.size, "We expected 1 match for the xpath expression \"#{elem}\". #{data.size} were found. Full response body: #{@response.body_as_string}")
88
+ end
89
+
90
+ depth = xml.find('/d:prop/d:lockdiscovery/d:activelock/d:depth')
91
+ assert_equal('infinity', depth[0].content)
92
+
93
+ token = xml.find('/d:prop/d:lockdiscovery/d:activelock/d:locktoken/d:href')
94
+ assert_equal(@response.header('Lock-Token'), "<#{token[0].content}>", 'Token in response body didn\'t match token in response header.')
95
+ end
96
+
97
+ def test_double_lock
98
+ request = Http::Request.new('LOCK', '/test.txt')
99
+ request.body = <<XML
100
+ <?xml version="1.0"?>
101
+ <D:lockinfo xmlns:D="DAV:">
102
+ <D:lockscope><D:exclusive/></D:lockscope>
103
+ <D:locktype><D:write/></D:locktype>
104
+ <D:owner>
105
+ <D:href>http://example.org/~ejw/contact.html</D:href>
106
+ </D:owner>
107
+ </D:lockinfo>
108
+ XML
109
+
110
+ @server.http_request = request
111
+ @server.exec
112
+
113
+ @response = Http::ResponseMock.new
114
+ @server.http_response = @response
115
+
116
+ @server.exec
117
+
118
+ assert_equal('application/xml; charset=utf-8', @response.header('Content-Type'))
119
+
120
+ assert_equal(423, @response.status, "Full response: #{@response.body_as_string}")
121
+ end
122
+
123
+ def test_lock_refresh
124
+ request = Http::Request.new('LOCK', '/test.txt')
125
+ request.body = <<XML
126
+ <?xml version="1.0"?>
127
+ <D:lockinfo xmlns:D="DAV:">
128
+ <D:lockscope><D:exclusive/></D:lockscope>
129
+ <D:locktype><D:write/></D:locktype>
130
+ <D:owner>
131
+ <D:href>http://example.org/~ejw/contact.html</D:href>
132
+ </D:owner>
133
+ </D:lockinfo>
134
+ XML
135
+
136
+ @server.http_request = request
137
+ @server.exec
138
+
139
+ lock_token = @response.header('Lock-Token')
140
+
141
+ @response = Http::ResponseMock.new
142
+ @server.http_response = @response
143
+
144
+ request = Http::Request.new('LOCK', '/test.txt', 'If' => "(#{lock_token})")
145
+ request.body = ''
146
+
147
+ @server.http_request = request
148
+ @server.exec
149
+
150
+ assert_equal('application/xml; charset=utf-8', @response.header('Content-Type'))
151
+
152
+ assert_equal(200, @response.status, "We received an incorrect status code. Full response body: #{@response.body_as_string}")
153
+ end
154
+
155
+ def test_lock_refresh_bad_token
156
+ request = Http::Request.new('LOCK', '/test.txt')
157
+ request.body = <<XML
158
+ <?xml version="1.0"?>
159
+ <D:lockinfo xmlns:D="DAV:">
160
+ <D:lockscope><D:exclusive/></D:lockscope>
161
+ <D:locktype><D:write/></D:locktype>
162
+ <D:owner>
163
+ <D:href>http://example.org/~ejw/contact.html</D:href>
164
+ </D:owner>
165
+ </D:lockinfo>
166
+ XML
167
+
168
+ @server.http_request = request
169
+ @server.exec
170
+
171
+ lock_token = @response.header('Lock-Token')
172
+
173
+ @response = Http::ResponseMock.new
174
+ @server.http_response = @response
175
+
176
+ request = Http::Request.new('LOCK', '/test.txt', 'If' => "(#{lock_token}foobar) (<opaquelocktoken:anotherbadtoken>)")
177
+ request.body = ''
178
+
179
+ @server.http_request = request
180
+ @server.exec
181
+
182
+ assert_equal('application/xml; charset=utf-8', @response.header('Content-Type'))
183
+
184
+ assert_equal(423, @response.status, "We received an incorrect status code. Full response body: #{@response.body_as_string}")
185
+ end
186
+
187
+ def test_lock_no_file
188
+ request = Http::Request.new('LOCK', '/notfound.txt')
189
+ request.body = <<XML
190
+ <?xml version="1.0"?>
191
+ <D:lockinfo xmlns:D="DAV:">
192
+ <D:lockscope><D:exclusive/></D:lockscope>
193
+ <D:locktype><D:write/></D:locktype>
194
+ <D:owner>
195
+ <D:href>http://example.org/~ejw/contact.html</D:href>
196
+ </D:owner>
197
+ </D:lockinfo>
198
+ XML
199
+
200
+ @server.http_request = request
201
+ @server.exec
202
+
203
+ assert_equal('application/xml; charset=utf-8', @response.header('Content-Type'))
204
+ assert(@response.header('Lock-Token') =~ /^<opaquelocktoken:(.*)>$/, "We did not get a valid Locktoken back (#{@response.header('Lock-Token')})")
205
+
206
+ assert_equal(201, @response.status)
207
+ end
208
+
209
+ def test_unlock_no_token
210
+ request = Http::Request.new('UNLOCK', '/test.txt')
211
+ @server.http_request = request
212
+ @server.exec
213
+
214
+ assert_equal(
215
+ {
216
+ 'X-Sabre-Version' => [Version::VERSION],
217
+ 'Content-Type' => ['application/xml; charset=utf-8']
218
+ },
219
+ @response.headers
220
+ )
221
+
222
+ assert_equal(400, @response.status)
223
+ end
224
+
225
+ def test_unlock_bad_token
226
+ request = Http::Request.new('UNLOCK', '/test.txt', 'Lock-Token' => '<opaquelocktoken:blablabla>')
227
+ @server.http_request = request
228
+ @server.exec
229
+
230
+ assert_equal(
231
+ {
232
+ 'X-Sabre-Version' => [Version::VERSION],
233
+ 'Content-Type' => ['application/xml; charset=utf-8']
234
+ },
235
+ @response.headers
236
+ )
237
+
238
+ assert_equal(409, @response.status, "Got an incorrect status code. Full response body: #{@response.body_as_string}")
239
+ end
240
+
241
+ def test_lock_put_no_token
242
+ request = Http::Request.new('LOCK', '/test.txt')
243
+ request.body = <<XML
244
+ <?xml version="1.0"?>
245
+ <D:lockinfo xmlns:D="DAV:">
246
+ <D:lockscope><D:exclusive/></D:lockscope>
247
+ <D:locktype><D:write/></D:locktype>
248
+ <D:owner>
249
+ <D:href>http://example.org/~ejw/contact.html</D:href>
250
+ </D:owner>
251
+ </D:lockinfo>
252
+ XML
253
+
254
+ @server.http_request = request
255
+ @server.exec
256
+
257
+ assert_equal('application/xml; charset=utf-8', @response.header('Content-Type'))
258
+ assert(@response.header('Lock-Token') =~ /^<opaquelocktoken:(.*)>$/, "We did not get a valid Locktoken back (#{@response.header('Lock-Token')})")
259
+
260
+ assert_equal(200, @response.status)
261
+
262
+ request = Http::Request.new('PUT', '/test.txt')
263
+ request.body = 'newbody'
264
+ @server.http_request = request
265
+ @server.exec
266
+
267
+ assert_equal('application/xml; charset=utf-8', @response.header('Content-Type'))
268
+ assert(@response.header('Lock-Token') =~ /^<opaquelocktoken:(.*)>$/, "We did not get a valid Locktoken back (#{@response.header('Lock-Token')})")
269
+
270
+ assert_equal(423, @response.status)
271
+ end
272
+
273
+ def test_unlock
274
+ request = Http::Request.new('LOCK', '/test.txt')
275
+ @server.http_request = request
276
+
277
+ request.body = <<XML
278
+ <?xml version="1.0"?>
279
+ <D:lockinfo xmlns:D="DAV:">
280
+ <D:lockscope><D:exclusive/></D:lockscope>
281
+ <D:locktype><D:write/></D:locktype>
282
+ <D:owner>
283
+ <D:href>http://example.org/~ejw/contact.html</D:href>
284
+ </D:owner>
285
+ </D:lockinfo>
286
+ XML
287
+
288
+ @server.invoke_method(request, @server.http_response)
289
+ lock_token = @server.http_response.header('Lock-Token')
290
+
291
+ request = Http::Request.new('UNLOCK', '/test.txt', 'Lock-Token' => lock_token)
292
+ @server.http_request = request
293
+ @server.http_response = Http::ResponseMock.new
294
+ @server.invoke_method(request, @server.http_response)
295
+
296
+ assert_equal(204, @server.http_response.status, "Got an incorrect status code. Full response body: #{@response.body_as_string}")
297
+ assert_equal(
298
+ {
299
+ 'X-Sabre-Version' => [Version::VERSION],
300
+ 'Content-Length' => ['0']
301
+ },
302
+ @server.http_response.headers
303
+ )
304
+ end
305
+
306
+ def test_unlock_windows_bug
307
+ request = Http::Request.new('LOCK', '/test.txt')
308
+ @server.http_request = request
309
+
310
+ request.body = <<XML
311
+ <?xml version="1.0"?>
312
+ <D:lockinfo xmlns:D="DAV:">
313
+ <D:lockscope><D:exclusive/></D:lockscope>
314
+ <D:locktype><D:write/></D:locktype>
315
+ <D:owner>
316
+ <D:href>http://example.org/~ejw/contact.html</D:href>
317
+ </D:owner>
318
+ </D:lockinfo>
319
+ XML
320
+
321
+ @server.invoke_method(request, @server.http_response)
322
+ lock_token = @server.http_response.header('Lock-Token')
323
+
324
+ # See Issue 123
325
+ lock_token = lock_token.gsub(/^(<>)+|(<>)+$/, '')
326
+
327
+ request = Http::Request.new('UNLOCK', '/test.txt', 'Lock-Token' => lock_token)
328
+ @server.http_request = request
329
+ @server.http_response = Http::ResponseMock.new
330
+ @server.invoke_method(request, @server.http_response)
331
+
332
+ assert_equal(204, @server.http_response.status, "Got an incorrect status code. Full response body: #{@response.body_as_string}")
333
+ assert_equal(
334
+ {
335
+ 'X-Sabre-Version' => [Version::VERSION],
336
+ 'Content-Length' => ['0']
337
+ },
338
+ @server.http_response.headers
339
+ )
340
+ end
341
+
342
+ def test_lock_retain_owner
343
+ request = Http::Sapi.create_from_server_array(
344
+ 'REQUEST_PATH' => '/test.txt',
345
+ 'REQUEST_METHOD' => 'LOCK'
346
+ )
347
+ @server.http_request = request
348
+
349
+ request.body = <<XML
350
+ <?xml version="1.0"?>
351
+ <D:lockinfo xmlns:D="DAV:">
352
+ <D:lockscope><D:exclusive/></D:lockscope>
353
+ <D:locktype><D:write/></D:locktype>
354
+ <D:owner>Evert</D:owner>
355
+ </D:lockinfo>
356
+ XML
357
+
358
+ @server.invoke_method(request, @server.http_response)
359
+ lock_token = @server.http_response.header('Lock-Token')
360
+
361
+ locks = @locks_plugin.locks('test.txt')
362
+ assert_equal(1, locks.size)
363
+ assert_equal('Evert', locks[0].owner)
364
+ end
365
+
366
+ def test_lock_put_bad_token
367
+ server_vars = {
368
+ 'REQUEST_PATH' => '/test.txt',
369
+ 'REQUEST_METHOD' => 'LOCK'
370
+ }
371
+
372
+ request = Http::Sapi.create_from_server_array(server_vars)
373
+ request.body = <<XML
374
+ <?xml version="1.0"?>
375
+ <D:lockinfo xmlns:D="DAV:">
376
+ <D:lockscope><D:exclusive/></D:lockscope>
377
+ <D:locktype><D:write/></D:locktype>
378
+ <D:owner>
379
+ <D:href>http://example.org/~ejw/contact.html</D:href>
380
+ </D:owner>
381
+ </D:lockinfo>
382
+ XML
383
+
384
+ @server.http_request = request
385
+ @server.exec
386
+
387
+ assert_equal('application/xml; charset=utf-8', @response.header('Content-Type'))
388
+ assert(@response.header('Lock-Token') =~ /^<opaquelocktoken:(.*)>$/, "We did not get a valid Locktoken back (#{@response.header('Lock-Token')})")
389
+
390
+ assert_equal(200, @response.status)
391
+
392
+ server_vars = {
393
+ 'REQUEST_PATH' => '/test.txt',
394
+ 'REQUEST_METHOD' => 'PUT',
395
+ 'HTTP_IF' => '(<opaquelocktoken:token1>)'
396
+ }
397
+
398
+ request = Http::Sapi.create_from_server_array(server_vars)
399
+ request.body = 'newbody'
400
+ @server.http_request = request
401
+ @server.exec
402
+
403
+ assert_equal('application/xml; charset=utf-8', @response.header('Content-Type'))
404
+ assert(@response.header('Lock-Token') =~ /^<opaquelocktoken:(.*)>$/, "We did not get a valid Locktoken back (#{@response.header('Lock-Token')})")
405
+
406
+ # assert_equal('412 Precondition failed',@response.status)
407
+ assert_equal(423, @response.status)
408
+ end
409
+
410
+ def test_lock_delete_parent
411
+ server_vars = {
412
+ 'REQUEST_PATH' => '/dir/child.txt',
413
+ 'REQUEST_METHOD' => 'LOCK'
414
+ }
415
+
416
+ request = Http::Sapi.create_from_server_array(server_vars)
417
+ request.body = <<XML
418
+ <?xml version="1.0"?>
419
+ <D:lockinfo xmlns:D="DAV:">
420
+ <D:lockscope><D:exclusive/></D:lockscope>
421
+ <D:locktype><D:write/></D:locktype>
422
+ <D:owner>
423
+ <D:href>http://example.org/~ejw/contact.html</D:href>
424
+ </D:owner>
425
+ </D:lockinfo>
426
+ XML
427
+
428
+ @server.http_request = request
429
+ @server.exec
430
+
431
+ assert_equal('application/xml; charset=utf-8', @response.header('Content-Type'))
432
+ assert(@response.header('Lock-Token') =~ /^<opaquelocktoken:(.*)>$/, "We did not get a valid Locktoken back (#{@response.header('Lock-Token')})")
433
+
434
+ assert_equal(200, @response.status)
435
+
436
+ server_vars = {
437
+ 'REQUEST_PATH' => '/dir',
438
+ 'REQUEST_METHOD' => 'DELETE'
439
+ }
440
+
441
+ request = Http::Sapi.create_from_server_array(server_vars)
442
+ @server.http_request = request
443
+ @server.exec
444
+
445
+ assert_equal(423, @response.status)
446
+ assert_equal('application/xml; charset=utf-8', @response.header('Content-Type'))
447
+ end
448
+
449
+ def test_lock_delete_succeed
450
+ server_vars = {
451
+ 'REQUEST_PATH' => '/dir/child.txt',
452
+ 'REQUEST_METHOD' => 'LOCK'
453
+ }
454
+
455
+ request = Http::Sapi.create_from_server_array(server_vars)
456
+ request.body = <<XML
457
+ <?xml version="1.0"?>
458
+ <D:lockinfo xmlns:D="DAV:">
459
+ <D:lockscope><D:exclusive/></D:lockscope>
460
+ <D:locktype><D:write/></D:locktype>
461
+ <D:owner>
462
+ <D:href>http://example.org/~ejw/contact.html</D:href>
463
+ </D:owner>
464
+ </D:lockinfo>
465
+ XML
466
+
467
+ @server.http_request = request
468
+ @server.exec
469
+
470
+ assert_equal('application/xml; charset=utf-8', @response.header('Content-Type'))
471
+ assert(@response.header('Lock-Token') =~ /^<opaquelocktoken:(.*)>$/, "We did not get a valid Locktoken back (#{@response.header('Lock-Token')})")
472
+
473
+ assert_equal(200, @response.status)
474
+
475
+ server_vars = {
476
+ 'REQUEST_PATH' => '/dir/child.txt',
477
+ 'REQUEST_METHOD' => 'DELETE',
478
+ 'HTTP_IF' => "(#{@response.header('Lock-Token')})"
479
+ }
480
+
481
+ request = Http::Sapi.create_from_server_array(server_vars)
482
+ @server.http_request = request
483
+ @server.exec
484
+
485
+ assert_equal(204, @response.status)
486
+ assert_equal('application/xml; charset=utf-8', @response.header('Content-Type'))
487
+ end
488
+
489
+ def test_lock_copy_lock_source
490
+ server_vars = {
491
+ 'REQUEST_PATH' => '/dir/child.txt',
492
+ 'REQUEST_METHOD' => 'LOCK'
493
+ }
494
+
495
+ request = Http::Sapi.create_from_server_array(server_vars)
496
+ request.body = <<XML
497
+ <?xml version="1.0"?>
498
+ <D:lockinfo xmlns:D="DAV:">
499
+ <D:lockscope><D:exclusive/></D:lockscope>
500
+ <D:locktype><D:write/></D:locktype>
501
+ <D:owner>
502
+ <D:href>http://example.org/~ejw/contact.html</D:href>
503
+ </D:owner>
504
+ </D:lockinfo>
505
+ XML
506
+
507
+ @server.http_request = request
508
+ @server.exec
509
+
510
+ assert_equal('application/xml; charset=utf-8', @response.header('Content-Type'))
511
+ assert(@response.header('Lock-Token') =~ /^<opaquelocktoken:(.*)>$/, "We did not get a valid Locktoken back (#{@response.header('Lock-Token')})")
512
+
513
+ assert_equal(200, @response.status)
514
+
515
+ server_vars = {
516
+ 'REQUEST_PATH' => '/dir/child.txt',
517
+ 'REQUEST_METHOD' => 'COPY',
518
+ 'HTTP_DESTINATION' => '/dir/child2.txt'
519
+ }
520
+
521
+ request = Http::Sapi.create_from_server_array(server_vars)
522
+ @server.http_request = request
523
+ @server.exec
524
+
525
+ assert_equal(201, @response.status, 'Copy must succeed if only the source is locked, but not the destination')
526
+ assert_equal('application/xml; charset=utf-8', @response.header('Content-Type'))
527
+ end
528
+
529
+ def test_lock_copy_lock_destination
530
+ server_vars = {
531
+ 'REQUEST_PATH' => '/dir/child2.txt',
532
+ 'REQUEST_METHOD' => 'LOCK'
533
+ }
534
+
535
+ request = Http::Sapi.create_from_server_array(server_vars)
536
+ request.body = <<XML
537
+ <?xml version="1.0"?>
538
+ <D:lockinfo xmlns:D="DAV:">
539
+ <D:lockscope><D:exclusive/></D:lockscope>
540
+ <D:locktype><D:write/></D:locktype>
541
+ <D:owner>
542
+ <D:href>http://example.org/~ejw/contact.html</D:href>
543
+ </D:owner>
544
+ </D:lockinfo>
545
+ XML
546
+
547
+ @server.http_request = request
548
+ @server.exec
549
+
550
+ assert_equal('application/xml; charset=utf-8', @response.header('Content-Type'))
551
+ assert(@response.header('Lock-Token') =~ /^<opaquelocktoken:(.*)>$/, "We did not get a valid Locktoken back (#{@response.header('Lock-Token')})")
552
+
553
+ assert_equal(201, @response.status)
554
+
555
+ server_vars = {
556
+ 'REQUEST_PATH' => '/dir/child.txt',
557
+ 'REQUEST_METHOD' => 'COPY',
558
+ 'HTTP_DESTINATION' => '/dir/child2.txt'
559
+ }
560
+
561
+ request = Http::Sapi.create_from_server_array(server_vars)
562
+ @server.http_request = request
563
+ @server.exec
564
+
565
+ assert_equal(423, @response.status, 'Copy must succeed if only the source is locked, but not the destination')
566
+ assert_equal('application/xml; charset=utf-8', @response.header('Content-Type'))
567
+ end
568
+
569
+ def test_lock_move_lock_source_locked
570
+ server_vars = {
571
+ 'REQUEST_PATH' => '/dir/child.txt',
572
+ 'REQUEST_METHOD' => 'LOCK'
573
+ }
574
+
575
+ request = Http::Sapi.create_from_server_array(server_vars)
576
+ request.body = <<XML
577
+ <?xml version="1.0"?>
578
+ <D:lockinfo xmlns:D="DAV:">
579
+ <D:lockscope><D:exclusive/></D:lockscope>
580
+ <D:locktype><D:write/></D:locktype>
581
+ <D:owner>
582
+ <D:href>http://example.org/~ejw/contact.html</D:href>
583
+ </D:owner>
584
+ </D:lockinfo>
585
+ XML
586
+
587
+ @server.http_request = request
588
+ @server.exec
589
+
590
+ assert_equal('application/xml; charset=utf-8', @response.header('Content-Type'))
591
+ assert(@response.header('Lock-Token') =~ /^<opaquelocktoken:(.*)>$/, "We did not get a valid Locktoken back (#{@response.header('Lock-Token')})")
592
+
593
+ assert_equal(200, @response.status)
594
+
595
+ server_vars = {
596
+ 'REQUEST_PATH' => '/dir/child.txt',
597
+ 'REQUEST_METHOD' => 'MOVE',
598
+ 'HTTP_DESTINATION' => '/dir/child2.txt'
599
+ }
600
+
601
+ request = Http::Sapi.create_from_server_array(server_vars)
602
+ @server.http_request = request
603
+ @server.exec
604
+
605
+ assert_equal(423, @response.status, 'Copy must succeed if only the source is locked, but not the destination')
606
+ assert_equal('application/xml; charset=utf-8', @response.header('Content-Type'))
607
+ end
608
+
609
+ def test_lock_move_lock_source_succeed
610
+ server_vars = {
611
+ 'REQUEST_PATH' => '/dir/child.txt',
612
+ 'REQUEST_METHOD' => 'LOCK'
613
+ }
614
+
615
+ request = Http::Sapi.create_from_server_array(server_vars)
616
+ request.body = <<XML
617
+ <?xml version="1.0"?>
618
+ <D:lockinfo xmlns:D="DAV:">
619
+ <D:lockscope><D:exclusive/></D:lockscope>
620
+ <D:locktype><D:write/></D:locktype>
621
+ <D:owner>
622
+ <D:href>http://example.org/~ejw/contact.html</D:href>
623
+ </D:owner>
624
+ </D:lockinfo>
625
+ XML
626
+
627
+ @server.http_request = request
628
+ @server.exec
629
+
630
+ assert_equal('application/xml; charset=utf-8', @response.header('Content-Type'))
631
+ assert(@response.header('Lock-Token') =~ /^<opaquelocktoken:(.*)>$/, "We did not get a valid Locktoken back (#{@response.header('Lock-Token')})")
632
+
633
+ assert_equal(200, @response.status)
634
+
635
+ server_vars = {
636
+ 'REQUEST_PATH' => '/dir/child.txt',
637
+ 'REQUEST_METHOD' => 'MOVE',
638
+ 'HTTP_DESTINATION' => '/dir/child2.txt',
639
+ 'HTTP_IF' => "(#{@response.header('Lock-Token')})"
640
+ }
641
+
642
+ request = Http::Sapi.create_from_server_array(server_vars)
643
+ @server.http_request = request
644
+ @server.exec
645
+
646
+ assert_equal(201, @response.status, "A valid lock-token was provided for the source, so this MOVE operation must succeed. Full response body: #{@response.body_as_string}")
647
+ end
648
+
649
+ def test_lock_move_lock_destination
650
+ server_vars = {
651
+ 'REQUEST_PATH' => '/dir/child2.txt',
652
+ 'REQUEST_METHOD' => 'LOCK'
653
+ }
654
+
655
+ request = Http::Sapi.create_from_server_array(server_vars)
656
+ request.body = <<XML
657
+ <?xml version="1.0"?>
658
+ <D:lockinfo xmlns:D="DAV:">
659
+ <D:lockscope><D:exclusive/></D:lockscope>
660
+ <D:locktype><D:write/></D:locktype>
661
+ <D:owner>
662
+ <D:href>http://example.org/~ejw/contact.html</D:href>
663
+ </D:owner>
664
+ </D:lockinfo>
665
+ XML
666
+
667
+ @server.http_request = request
668
+ @server.exec
669
+
670
+ assert_equal('application/xml; charset=utf-8', @response.header('Content-Type'))
671
+ assert(@response.header('Lock-Token') =~ /^<opaquelocktoken:(.*)>$/, "We did not get a valid Locktoken back (#{@response.header('Lock-Token')})")
672
+
673
+ assert_equal(201, @response.status)
674
+
675
+ server_vars = {
676
+ 'REQUEST_PATH' => '/dir/child.txt',
677
+ 'REQUEST_METHOD' => 'MOVE',
678
+ 'HTTP_DESTINATION' => '/dir/child2.txt'
679
+ }
680
+
681
+ request = Http::Sapi.create_from_server_array(server_vars)
682
+ @server.http_request = request
683
+ @server.exec
684
+
685
+ assert_equal(423, @response.status, 'Copy must succeed if only the source is locked, but not the destination')
686
+ assert_equal('application/xml; charset=utf-8', @response.header('Content-Type'))
687
+ end
688
+
689
+ def test_lock_move_lock_parent
690
+ server_vars = {
691
+ 'REQUEST_PATH' => '/dir',
692
+ 'REQUEST_METHOD' => 'LOCK',
693
+ 'HTTP_DEPTH' => 'infinite'
694
+ }
695
+
696
+ request = Http::Sapi.create_from_server_array(server_vars)
697
+ request.body = <<XML
698
+ <?xml version="1.0"?>
699
+ <D:lockinfo xmlns:D="DAV:">
700
+ <D:lockscope><D:exclusive/></D:lockscope>
701
+ <D:locktype><D:write/></D:locktype>
702
+ <D:owner>
703
+ <D:href>http://example.org/~ejw/contact.html</D:href>
704
+ </D:owner>
705
+ </D:lockinfo>
706
+ XML
707
+
708
+ @server.http_request = request
709
+ @server.exec
710
+
711
+ assert_equal('application/xml; charset=utf-8', @response.header('Content-Type'))
712
+ assert(@response.header('Lock-Token') =~ /^<opaquelocktoken:(.*)>$/, "We did not get a valid Locktoken back (#{@response.header('Lock-Token')})")
713
+
714
+ assert_equal(200, @response.status)
715
+
716
+ server_vars = {
717
+ 'REQUEST_PATH' => '/dir/child.txt',
718
+ 'REQUEST_METHOD' => 'MOVE',
719
+ 'HTTP_DESTINATION' => '/dir/child2.txt',
720
+ 'HTTP_IF' => "</dir> (#{@response.header('Lock-Token')})"
721
+ }
722
+
723
+ request = Http::Sapi.create_from_server_array(server_vars)
724
+ @server.http_request = request
725
+ @server.exec
726
+
727
+ assert_equal(201, @response.status, 'We locked the parent of both the source and destination, but the move didn\'t succeed.')
728
+ assert_equal('application/xml; charset=utf-8', @response.header('Content-Type'))
729
+ end
730
+
731
+ def test_lock_put_good_token
732
+ server_vars = {
733
+ 'REQUEST_PATH' => '/test.txt',
734
+ 'REQUEST_METHOD' => 'LOCK'
735
+ }
736
+
737
+ request = Http::Sapi.create_from_server_array(server_vars)
738
+ request.body = <<XML
739
+ <?xml version="1.0"?>
740
+ <D:lockinfo xmlns:D="DAV:">
741
+ <D:lockscope><D:exclusive/></D:lockscope>
742
+ <D:locktype><D:write/></D:locktype>
743
+ <D:owner>
744
+ <D:href>http://example.org/~ejw/contact.html</D:href>
745
+ </D:owner>
746
+ </D:lockinfo>
747
+ XML
748
+
749
+ @server.http_request = request
750
+ @server.exec
751
+
752
+ assert_equal('application/xml; charset=utf-8', @response.header('Content-Type'))
753
+ assert(@response.header('Lock-Token') =~ /^<opaquelocktoken:(.*)>$/, "We did not get a valid Locktoken back (#{@response.header('Lock-Token')})")
754
+
755
+ assert_equal(200, @response.status)
756
+
757
+ server_vars = {
758
+ 'REQUEST_PATH' => '/test.txt',
759
+ 'REQUEST_METHOD' => 'PUT',
760
+ 'HTTP_IF' => "(#{@response.header('Lock-Token')})"
761
+ }
762
+
763
+ request = Http::Sapi.create_from_server_array(server_vars)
764
+ request.body = 'newbody'
765
+ @server.http_request = request
766
+ @server.exec
767
+
768
+ assert_equal('application/xml; charset=utf-8', @response.header('Content-Type'))
769
+ assert(@response.header('Lock-Token') =~ /^<opaquelocktoken:(.*)>$/, "We did not get a valid Locktoken back (#{@response.header('Lock-Token')})")
770
+
771
+ assert_equal(204, @response.status)
772
+ end
773
+
774
+ def test_lock_put_unrelated_token
775
+ request = Http::Request.new('LOCK', '/unrelated.txt')
776
+ request.body = <<XML
777
+ <?xml version="1.0"?>
778
+ <D:lockinfo xmlns:D="DAV:">
779
+ <D:lockscope><D:exclusive/></D:lockscope>
780
+ <D:locktype><D:write/></D:locktype>
781
+ <D:owner>
782
+ <D:href>http://example.org/~ejw/contact.html</D:href>
783
+ </D:owner>
784
+ </D:lockinfo>
785
+ XML
786
+
787
+ @server.http_request = request
788
+ @server.exec
789
+
790
+ assert_equal('application/xml; charset=utf-8', @response.header('Content-Type'))
791
+ assert(@response.header('Lock-Token') =~ /^<opaquelocktoken:(.*)>$/, "We did not get a valid Locktoken back (#{@response.header('Lock-Token')})")
792
+
793
+ assert_equal(201, @response.status)
794
+
795
+ request = Http::Request.new(
796
+ 'PUT',
797
+ '/test.txt',
798
+ 'If' => "</unrelated.txt> (#{@response.header('Lock-Token')})"
799
+ )
800
+ request.body = 'newbody'
801
+ @server.http_request = request
802
+ @server.exec
803
+
804
+ assert_equal('application/xml; charset=utf-8', @response.header('Content-Type'))
805
+ assert(@response.header('Lock-Token') =~ /^<opaquelocktoken:(.*)>$/, "We did not get a valid Locktoken back (#{@response.header('Lock-Token')})")
806
+
807
+ assert_equal(204, @response.status)
808
+ end
809
+
810
+ def test_put_with_incorrect_etag
811
+ server_vars = {
812
+ 'REQUEST_PATH' => '/test.txt',
813
+ 'REQUEST_METHOD' => 'PUT',
814
+ 'HTTP_IF' => '(["etag1"])'
815
+ }
816
+
817
+ request = Http::Sapi.create_from_server_array(server_vars)
818
+ request.body = 'newbody'
819
+ @server.http_request = request
820
+ @server.exec
821
+ assert_equal(412, @response.status)
822
+ end
823
+
824
+ def test_put_with_correct_etag
825
+ # We need an ETag-enabled file node.
826
+ tree = Tree.new(FsExt::Directory.new(@temp_dir))
827
+ @server.tree = tree
828
+
829
+ filename = "#{@temp_dir}/test.txt"
830
+ stat = ::File.stat(filename)
831
+ etag = Digest::SHA1.hexdigest(stat.ino.to_s + stat.size.to_s + stat.mtime.to_s)
832
+ server_vars = {
833
+ 'REQUEST_PATH' => '/test.txt',
834
+ 'REQUEST_METHOD' => 'PUT',
835
+ 'HTTP_IF' => "([\"#{etag}\"])"
836
+ }
837
+
838
+ request = Http::Sapi.create_from_server_array(server_vars)
839
+ request.body = 'newbody'
840
+ @server.http_request = request
841
+ @server.exec
842
+ assert_equal(204, @response.status, "Incorrect status received. Full response body:#{@response.body_as_string}")
843
+ end
844
+
845
+ def test_delete_with_etag_on_collection
846
+ server_vars = {
847
+ 'REQUEST_PATH' => '/dir',
848
+ 'REQUEST_METHOD' => 'DELETE',
849
+ 'HTTP_IF' => '(["etag1"])'
850
+ }
851
+ request = Http::Sapi.create_from_server_array(server_vars)
852
+
853
+ @server.http_request = request
854
+ @server.exec
855
+ assert_equal(412, @response.status)
856
+ end
857
+
858
+ def test_get_timeout_header
859
+ request = Http::Sapi.create_from_server_array(
860
+ 'HTTP_TIMEOUT' => 'second-100'
861
+ )
862
+
863
+ @server.http_request = request
864
+ assert_equal(100, @locks_plugin.timeout_header)
865
+ end
866
+
867
+ def test_get_timeout_header_two_items
868
+ request = Http::Sapi.create_from_server_array(
869
+ 'HTTP_TIMEOUT' => 'second-5, infinite'
870
+ )
871
+
872
+ @server.http_request = request
873
+ assert_equal(5, @locks_plugin.timeout_header)
874
+ end
875
+
876
+ def test_get_timeout_header_infinite
877
+ request = Http::Sapi.create_from_server_array(
878
+ 'HTTP_TIMEOUT' => 'infinite, second-5'
879
+ )
880
+
881
+ @server.http_request = request
882
+ assert_equal(LockInfo::TIMEOUT_INFINITE, @locks_plugin.timeout_header)
883
+ end
884
+
885
+ def test_get_timeout_header_invalid
886
+ request = Http::Sapi.create_from_server_array(
887
+ 'HTTP_TIMEOUT' => 'yourmom'
888
+ )
889
+
890
+ @server.http_request = request
891
+ assert_raises(Exception::BadRequest) { @locks_plugin.timeout_header }
892
+ end
893
+ end
894
+ end
895
+ end
896
+ end