spontaneous 0.2.0.beta1 → 0.2.0.beta2

Sign up to get free protection for your applications and to get access to all the features.
Files changed (335) hide show
  1. checksums.yaml +7 -0
  2. data/.gitignore +24 -0
  3. data/.locat +42 -0
  4. data/.travis/gemfiles/Gemfile.empty +7 -0
  5. data/.travis.yml +18 -0
  6. data/Gemfile +12 -8
  7. data/LICENSE +1 -1
  8. data/Rakefile +15 -157
  9. data/Readme.markdown +1 -1
  10. data/application/css/core.css.scss +22 -146
  11. data/application/css/definitions.css.scss +7 -3
  12. data/application/css/dialogue.css.scss +26 -1
  13. data/application/css/editing.css.scss +70 -28
  14. data/application/css/font.css.scss +1 -1
  15. data/application/css/popover.css.scss +2 -0
  16. data/application/css/top.css.scss +231 -0
  17. data/application/js/add_alias_dialogue.js +1 -1
  18. data/application/js/add_home_dialogue.js +1 -1
  19. data/application/js/ajax.js +61 -31
  20. data/application/js/box.js +4 -4
  21. data/application/js/conflicted_field_dialogue.js +1 -1
  22. data/application/js/content.js +5 -5
  23. data/application/js/dom.js +5 -0
  24. data/application/js/edit_panel.js +1 -0
  25. data/application/js/editing.js +1 -1
  26. data/application/js/extensions.js +8 -0
  27. data/application/js/field/boolean.js +31 -0
  28. data/application/js/field/file.js +32 -4
  29. data/application/js/field/image.js +24 -9
  30. data/application/js/field/markdown.js +87 -59
  31. data/application/js/field/select.js +1 -1
  32. data/application/js/field/webvideo.js +6 -1
  33. data/application/js/init.js +2 -2
  34. data/application/js/jquery-selection-position.js +130 -0
  35. data/application/js/location.js +4 -25
  36. data/application/js/meta_view/user_admin.js +2 -2
  37. data/application/js/metadata.js +2 -2
  38. data/application/js/page_browser.js +1 -1
  39. data/application/js/panel/root_menu.js +0 -1
  40. data/application/js/popover.js +27 -12
  41. data/application/js/popover_view.js +20 -4
  42. data/application/js/preview.js +31 -16
  43. data/application/js/progress.js +22 -21
  44. data/application/js/publish.js +18 -7
  45. data/application/js/sharded_upload.js +9 -6
  46. data/application/js/spontaneous.js +3 -1
  47. data/application/js/top_bar.js +264 -173
  48. data/application/js/upload.js +12 -5
  49. data/application/js/upload_manager.js +4 -3
  50. data/application/js/user.js +1 -2
  51. data/application/js/views/box_view.js +1 -1
  52. data/application/js/views/page_view.js +16 -5
  53. data/application/js/views/piece_view.js +5 -4
  54. data/application/static/font/fontawesome-webfont-1c66a4738b40ef0f6b1abca0ba9a796d.ttf +0 -0
  55. data/application/views/index.erb +6 -14
  56. data/application/views/login.erb +6 -25
  57. data/application/views/schema_modification_error.html.erb +3 -7
  58. data/db/migrations/20130114120000_create_revision_tables.rb +2 -2
  59. data/db/migrations/20130813111009_increase_path_length.rb +14 -0
  60. data/gem-public_cert.pem +20 -0
  61. data/lib/spontaneous/asset/app_compiler.rb +44 -0
  62. data/lib/spontaneous/asset/environment.rb +225 -0
  63. data/lib/spontaneous/asset.rb +2 -67
  64. data/lib/spontaneous/box.rb +0 -1
  65. data/lib/spontaneous/capistrano/deploy.rb +2 -2
  66. data/lib/spontaneous/capistrano/sync.rb +1 -1
  67. data/lib/spontaneous/cli/init.rb +36 -13
  68. data/lib/spontaneous/cli/server.rb +0 -1
  69. data/lib/spontaneous/cli/site.rb +2 -1
  70. data/lib/spontaneous/cli.rb +3 -1
  71. data/lib/spontaneous/collections/entry_set.rb +4 -12
  72. data/lib/spontaneous/collections/hash_with_fallback.rb +20 -0
  73. data/lib/spontaneous/collections/prototype_set.rb +6 -5
  74. data/lib/spontaneous/crypt.rb +2 -2
  75. data/lib/spontaneous/data_mapper/content_model/associations.rb +115 -63
  76. data/lib/spontaneous/data_mapper.rb +1 -1
  77. data/lib/spontaneous/errors.rb +6 -0
  78. data/lib/spontaneous/extensions/object_space.rb +6 -0
  79. data/lib/spontaneous/facet.rb +1 -0
  80. data/lib/spontaneous/field/base.rb +86 -13
  81. data/lib/spontaneous/field/boolean.rb +65 -0
  82. data/lib/spontaneous/field/file.rb +17 -6
  83. data/lib/spontaneous/field/html.rb +13 -0
  84. data/lib/spontaneous/field/image/size.rb +76 -0
  85. data/lib/spontaneous/field/image.rb +99 -414
  86. data/lib/spontaneous/field/tags.rb +36 -0
  87. data/lib/spontaneous/field/update.rb +1 -1
  88. data/lib/spontaneous/field/webvideo/fallback.rb +41 -0
  89. data/lib/spontaneous/field/webvideo/vimeo.rb +113 -0
  90. data/lib/spontaneous/field/webvideo/vine.rb +94 -0
  91. data/lib/spontaneous/field/webvideo/youtube.rb +133 -0
  92. data/lib/spontaneous/field/webvideo.rb +100 -250
  93. data/lib/spontaneous/field.rb +1 -1
  94. data/lib/spontaneous/generators/site/Gemfile.tt +5 -14
  95. data/lib/spontaneous/generators/site/assets/README.md +20 -0
  96. data/lib/spontaneous/generators/site/assets/css/site.scss +8 -0
  97. data/lib/spontaneous/generators/site/assets/js/site.js +6 -0
  98. data/lib/spontaneous/generators/site/config/deploy.rb.tt +9 -0
  99. data/lib/spontaneous/generators/site/config/user_levels.yml +14 -3
  100. data/lib/spontaneous/generators/site/public/README.md +12 -0
  101. data/lib/spontaneous/generators/site/templates/layouts/standard.html.cut.tt +2 -2
  102. data/lib/spontaneous/generators/site.rb +77 -35
  103. data/lib/spontaneous/layout.rb +6 -7
  104. data/lib/spontaneous/loader.rb +21 -13
  105. data/lib/spontaneous/media/file.rb +22 -9
  106. data/lib/spontaneous/media/image/attributes.rb +33 -0
  107. data/lib/spontaneous/media/image/format/gif.rb +4 -0
  108. data/lib/spontaneous/media/image/format/jpg.rb +17 -0
  109. data/lib/spontaneous/media/image/format/png.rb +4 -0
  110. data/lib/spontaneous/media/image/format/webp.rb +26 -0
  111. data/lib/spontaneous/media/image/format.rb +79 -0
  112. data/lib/spontaneous/media/image/optimizer.rb +69 -0
  113. data/lib/spontaneous/media/image/processor.rb +17 -0
  114. data/lib/spontaneous/media/image/renderable.rb +52 -0
  115. data/lib/spontaneous/media/image/skeptick.rb +70 -0
  116. data/lib/spontaneous/media/image.rb +50 -0
  117. data/lib/spontaneous/media/temp_file.rb +4 -0
  118. data/lib/spontaneous/media.rb +1 -0
  119. data/lib/spontaneous/model/core/aliases.rb +14 -8
  120. data/lib/spontaneous/model/core/boxes.rb +5 -2
  121. data/lib/spontaneous/model/core/entries.rb +4 -0
  122. data/lib/spontaneous/model/core/entry.rb +1 -0
  123. data/lib/spontaneous/model/core/fields.rb +5 -2
  124. data/lib/spontaneous/model/core/locks.rb +16 -0
  125. data/lib/spontaneous/model/core/media.rb +1 -15
  126. data/lib/spontaneous/model/core.rb +31 -1
  127. data/lib/spontaneous/model/page/controllers.rb +2 -2
  128. data/lib/spontaneous/model/page/formats.rb +1 -4
  129. data/lib/spontaneous/model/page/layouts.rb +6 -2
  130. data/lib/spontaneous/model/page/locks.rb +8 -2
  131. data/lib/spontaneous/model/page/page_tree.rb +2 -2
  132. data/lib/spontaneous/model/page/paths.rb +74 -9
  133. data/lib/spontaneous/model/page.rb +11 -3
  134. data/lib/spontaneous/model.rb +6 -6
  135. data/lib/spontaneous/output/context/render_cache.rb +23 -0
  136. data/lib/spontaneous/output/context.rb +56 -30
  137. data/lib/spontaneous/output/helpers/script_helper.rb +9 -53
  138. data/lib/spontaneous/output/helpers/stylesheet_helper.rb +8 -40
  139. data/lib/spontaneous/output/template/renderer.rb +17 -5
  140. data/lib/spontaneous/output.rb +0 -1
  141. data/lib/spontaneous/paths.rb +6 -2
  142. data/lib/spontaneous/permissions/access_key.rb +18 -0
  143. data/lib/spontaneous/permissions/user.rb +1 -1
  144. data/lib/spontaneous/permissions.rb +4 -1
  145. data/lib/spontaneous/plugins/application/state.rb +19 -12
  146. data/lib/spontaneous/prototypes/field_prototype.rb +14 -8
  147. data/lib/spontaneous/published_revision.rb +7 -0
  148. data/lib/spontaneous/publishing/immediate.rb +43 -34
  149. data/lib/spontaneous/publishing/revision.rb +9 -6
  150. data/lib/spontaneous/rack/asset_server.rb +20 -0
  151. data/lib/spontaneous/rack/back/alias.rb +46 -0
  152. data/lib/spontaneous/rack/back/application_assets.rb +28 -0
  153. data/lib/spontaneous/rack/back/base.rb +34 -0
  154. data/lib/spontaneous/rack/back/changes.rb +19 -0
  155. data/lib/spontaneous/rack/back/content.rb +54 -0
  156. data/lib/spontaneous/rack/back/events.rb +38 -0
  157. data/lib/spontaneous/rack/back/field.rb +37 -0
  158. data/lib/spontaneous/rack/back/file.rb +118 -0
  159. data/lib/spontaneous/rack/back/helpers.rb +71 -0
  160. data/lib/spontaneous/rack/back/index.rb +16 -0
  161. data/lib/spontaneous/rack/back/login.rb +47 -0
  162. data/lib/spontaneous/rack/back/map.rb +24 -0
  163. data/lib/spontaneous/rack/back/page.rb +46 -0
  164. data/lib/spontaneous/rack/back/preview.rb +43 -0
  165. data/lib/spontaneous/rack/back/schema.rb +30 -0
  166. data/lib/spontaneous/rack/back/site.rb +25 -0
  167. data/lib/spontaneous/rack/back/site_assets.rb +13 -0
  168. data/lib/spontaneous/rack/back/unsupported_browser.rb +7 -0
  169. data/lib/spontaneous/rack/{user_admin.rb → back/user_admin.rb} +2 -5
  170. data/lib/spontaneous/rack/back.rb +85 -764
  171. data/lib/spontaneous/rack/cacheable_file.rb +3 -3
  172. data/lib/spontaneous/rack/front.rb +16 -9
  173. data/lib/spontaneous/rack/middleware/authenticate.rb +65 -0
  174. data/lib/spontaneous/rack/middleware/csrf.rb +66 -0
  175. data/lib/spontaneous/rack/middleware/reloader.rb +52 -0
  176. data/lib/spontaneous/rack/middleware/scope.rb +60 -0
  177. data/lib/spontaneous/rack/middleware.rb +6 -0
  178. data/lib/spontaneous/rack/page_controller.rb +18 -5
  179. data/lib/spontaneous/rack/public.rb +17 -11
  180. data/lib/spontaneous/rack.rb +34 -24
  181. data/lib/spontaneous/revision.rb +29 -2
  182. data/lib/spontaneous/schema/uid.rb +4 -3
  183. data/lib/spontaneous/schema/uid_map.rb +5 -24
  184. data/lib/spontaneous/schema.rb +1 -0
  185. data/lib/spontaneous/search/database.rb +8 -0
  186. data/lib/spontaneous/search/field.rb +1 -1
  187. data/lib/spontaneous/search/index.rb +3 -5
  188. data/lib/spontaneous/server.rb +1 -1
  189. data/lib/spontaneous/simultaneous.rb +1 -1
  190. data/lib/spontaneous/site/features.rb +4 -5
  191. data/lib/spontaneous/site/helpers.rb +22 -5
  192. data/lib/spontaneous/site/instance.rb +2 -2
  193. data/lib/spontaneous/site/selectors.rb +22 -3
  194. data/lib/spontaneous/storage/cloud.rb +13 -9
  195. data/lib/spontaneous/storage/local.rb +11 -6
  196. data/lib/spontaneous/style.rb +40 -23
  197. data/lib/spontaneous/utils/database/mysql_dumper.rb +1 -1
  198. data/lib/spontaneous/utils/smush_it.rb +1 -1
  199. data/lib/spontaneous/version.rb +1 -1
  200. data/lib/spontaneous.rb +35 -33
  201. data/spontaneous.gemspec +53 -787
  202. data/test/experimental/test_crypt.rb +56 -56
  203. data/test/experimental/test_features.rb +16 -27
  204. data/test/fixtures/assets/public1/css/data.css.scss +3 -0
  205. data/test/fixtures/assets/public1/css/image1.css.scss +4 -0
  206. data/test/fixtures/assets/public1/css/import.css.scss +1 -0
  207. data/test/fixtures/assets/public1/css/urlhash.css.scss +3 -0
  208. data/test/fixtures/assets/public1/js/a.js +1 -1
  209. data/test/fixtures/assets/public1/js/all.js +4 -0
  210. data/test/fixtures/assets/public1/js/{m.coffee → m.js.coffee} +1 -0
  211. data/test/fixtures/assets/public1/x.js +1 -0
  212. data/test/fixtures/assets/public2/css/all.css +4 -0
  213. data/test/fixtures/assets/public2/css/missing.css.scss +3 -0
  214. data/test/fixtures/assets/public2/i/y.png +0 -0
  215. data/test/fixtures/assets/public2/js/b.js +1 -1
  216. data/test/fixtures/assets/public2/js/c.js +1 -1
  217. data/test/fixtures/images/size.extended.webp +0 -0
  218. data/test/fixtures/images/size.lossless.webp +0 -0
  219. data/test/fixtures/images/size.lossy.webp +0 -0
  220. data/test/fixtures/schema/before.yml +4 -4
  221. data/test/fixtures/schema/schema.yml +1 -1
  222. data/test/fixtures/templates/aliases/aaa.html.cut +0 -0
  223. data/test/fixtures/templates/extended/partial_with_renderer.html.cut +1 -0
  224. data/test/fixtures/templates/extended/with_includes_and_renderer.html.cut +2 -0
  225. data/test/functional/test_application.rb +108 -106
  226. data/test/functional/test_back.rb +924 -930
  227. data/test/functional/test_front.rb +285 -238
  228. data/test/functional/test_user_manager.rb +75 -100
  229. data/test/integration/test_installation.rb +1 -1
  230. data/test/support/matchers.rb +12 -0
  231. data/test/support/minitest.rb +121 -0
  232. data/test/support/rack.rb +45 -0
  233. data/test/support/test_start_finish.rb +103 -0
  234. data/test/test_helper.rb +21 -68
  235. data/test/test_integration_helper.rb +1 -3
  236. data/test/unit/test_alias.rb +432 -408
  237. data/test/unit/test_asset_bundler.rb +58 -58
  238. data/test/unit/test_assets.rb +485 -155
  239. data/test/unit/test_async.rb +16 -37
  240. data/test/unit/test_authentication.rb +425 -457
  241. data/test/unit/test_boxes.rb +191 -191
  242. data/test/unit/test_changesets.rb +244 -254
  243. data/test/unit/test_config.rb +128 -142
  244. data/test/unit/test_content.rb +313 -359
  245. data/test/unit/test_content_inheritance.rb +29 -30
  246. data/test/unit/test_datamapper.rb +1205 -1080
  247. data/test/unit/test_datamapper_content.rb +49 -51
  248. data/test/unit/test_extensions.rb +23 -23
  249. data/test/unit/test_fields.rb +1488 -1180
  250. data/test/unit/test_formats.rb +158 -158
  251. data/test/unit/test_generators.rb +98 -40
  252. data/test/unit/test_helpers.rb +73 -76
  253. data/test/unit/test_image_size.rb +53 -22
  254. data/test/unit/test_images.rb +164 -165
  255. data/test/unit/test_layouts.rb +133 -122
  256. data/test/unit/test_logger.rb +14 -17
  257. data/test/unit/test_media.rb +69 -84
  258. data/test/unit/test_modifications.rb +513 -525
  259. data/test/unit/test_page.rb +462 -361
  260. data/test/unit/test_permissions.rb +379 -364
  261. data/test/unit/test_piece.rb +67 -75
  262. data/test/unit/test_plugins.rb +82 -89
  263. data/test/unit/test_prototype_set.rb +215 -216
  264. data/test/unit/test_prototypes.rb +114 -124
  265. data/test/unit/test_publishing.rb +252 -289
  266. data/test/unit/test_render.rb +167 -115
  267. data/test/unit/test_revisions.rb +436 -444
  268. data/test/unit/test_schema.rb +339 -309
  269. data/test/unit/test_search.rb +577 -574
  270. data/test/unit/test_serialisation.rb +136 -147
  271. data/test/unit/test_site.rb +252 -227
  272. data/test/unit/test_skeptick.rb +130 -0
  273. data/test/unit/test_storage.rb +46 -40
  274. data/test/unit/test_structure.rb +57 -66
  275. data/test/unit/test_styles.rb +104 -104
  276. data/test/unit/test_templates.rb +72 -57
  277. data/test/unit/test_type_hierarchy.rb +15 -16
  278. data/test/unit/test_visibility.rb +239 -257
  279. metadata +455 -326
  280. data/application/js/vendor/JS.Class-2.1.5/CHANGELOG +0 -283
  281. data/application/js/vendor/JS.Class-2.1.5/MIT-LICENSE +0 -30
  282. data/application/js/vendor/JS.Class-2.1.5/README +0 -30
  283. data/application/js/vendor/JS.Class-2.1.5/min/command.js +0 -1
  284. data/application/js/vendor/JS.Class-2.1.5/min/comparable.js +0 -1
  285. data/application/js/vendor/JS.Class-2.1.5/min/constant_scope.js +0 -1
  286. data/application/js/vendor/JS.Class-2.1.5/min/decorator.js +0 -1
  287. data/application/js/vendor/JS.Class-2.1.5/min/enumerable.js +0 -1
  288. data/application/js/vendor/JS.Class-2.1.5/min/forwardable.js +0 -1
  289. data/application/js/vendor/JS.Class-2.1.5/min/hash.js +0 -1
  290. data/application/js/vendor/JS.Class-2.1.5/min/linked_list.js +0 -1
  291. data/application/js/vendor/JS.Class-2.1.5/min/loader.js +0 -1
  292. data/application/js/vendor/JS.Class-2.1.5/min/method_chain.js +0 -1
  293. data/application/js/vendor/JS.Class-2.1.5/min/observable.js +0 -1
  294. data/application/js/vendor/JS.Class-2.1.5/min/package.js +0 -1
  295. data/application/js/vendor/JS.Class-2.1.5/min/proxy.js +0 -1
  296. data/application/js/vendor/JS.Class-2.1.5/min/ruby.js +0 -1
  297. data/application/js/vendor/JS.Class-2.1.5/min/set.js +0 -1
  298. data/application/js/vendor/JS.Class-2.1.5/min/stack_trace.js +0 -1
  299. data/application/js/vendor/JS.Class-2.1.5/min/state.js +0 -1
  300. data/application/js/vendor/JS.Class-2.1.5/min/stdlib.js +0 -16
  301. data/application/js/vendor/jquery-1.6.2.min.js +0 -18
  302. data/application/js/vendor/jquery-ui-1.8.16.custom.min.js +0 -791
  303. data/application/js/vendor/jquery-ui-1.8.9.custom.min.js +0 -415
  304. data/application/static/font/fontawesome-webfont-5c5c21100a346972a82c34c5e96ffcfe.ttf +0 -0
  305. data/application/static/select-arrow-6e7dd3745b00e934b0d7a3250c46558b.png +0 -0
  306. data/bin/limit-upload +0 -5
  307. data/bin/unlimit-upload +0 -3
  308. data/lib/spontaneous/asset/file.rb +0 -25
  309. data/lib/spontaneous/asset/source.rb +0 -28
  310. data/lib/spontaneous/image_size.rb +0 -123
  311. data/lib/spontaneous/output/assets/compression.rb +0 -58
  312. data/lib/spontaneous/output/assets.rb +0 -32
  313. data/lib/spontaneous/rack/around_back.rb +0 -20
  314. data/lib/spontaneous/rack/around_front.rb +0 -27
  315. data/lib/spontaneous/rack/around_preview.rb +0 -22
  316. data/lib/spontaneous/rack/assets.rb +0 -126
  317. data/lib/spontaneous/rack/authentication.rb +0 -20
  318. data/lib/spontaneous/rack/cookie_authentication.rb +0 -38
  319. data/lib/spontaneous/rack/helpers.rb +0 -52
  320. data/lib/spontaneous/rack/http.rb +0 -18
  321. data/lib/spontaneous/rack/media.rb +0 -30
  322. data/lib/spontaneous/rack/query_authentication.rb +0 -35
  323. data/lib/spontaneous/rack/reloader.rb +0 -45
  324. data/lib/spontaneous/rack/user_helpers.rb +0 -28
  325. /data/{README → application/js/field/markdown/text_command.js} +0 -0
  326. /data/application/js/vendor/{JS.Class-2.1.5/min/core.js → js.class-2.1.5.min.js} +0 -0
  327. /data/test/fixtures/assets/public1/css/{a.scss → a.css.scss} +0 -0
  328. /data/{lib/spontaneous/generators/site/public/css/site.scss → test/fixtures/assets/public1/x.css} +0 -0
  329. /data/{lib/spontaneous/generators/site/public/js/.empty_directory → test/fixtures/assets/public1/x.png} +0 -0
  330. /data/test/fixtures/assets/public2/css/{b.scss → b.css.scss} +0 -0
  331. /data/test/fixtures/assets/public2/js/{n.coffee → n.js.coffee} +0 -0
  332. /data/test/fixtures/back/{public → assets}/css/sass_include.scss +0 -0
  333. /data/test/fixtures/back/{public → assets}/css/sass_template.scss +0 -0
  334. /data/test/fixtures/back/{public → assets}/js/coffeescript.coffee +0 -0
  335. /data/{lib/spontaneous/generators/site/public/js/site.js → test/fixtures/templates/aliases/aa_alias.html.cut} +0 -0
@@ -0,0 +1,20 @@
1
+ module Spontaneous::Rack
2
+ # Sprockets (2.9.0) doesn't include charset information in the content-type
3
+ # header. This wrapper class proxies all requests to a Sprockets enviroment
4
+ # and adds in a charset setting to the content-type header of all responses
5
+ class AssetServer
6
+ def initialize(environment, charset = "UTF-8")
7
+ @environment, @charset = environment, charset
8
+ end
9
+
10
+ def call(env)
11
+ force_encoding(*@environment.call(env))
12
+ end
13
+
14
+ def force_encoding(status, headers, body)
15
+ content_type = headers["Content-Type"]
16
+ headers.update("Content-Type" => "#{content_type}; charset=#{@charset}")
17
+ [status, headers, body]
18
+ end
19
+ end
20
+ end
@@ -0,0 +1,46 @@
1
+ module Spontaneous::Rack::Back
2
+ class Alias < Base
3
+ get '/:schema_id/:id/:box_id' do
4
+ klass = content_model.schema.to_class(params[:schema_id])
5
+ if klass.alias?
6
+ content_for_request do |content, box|
7
+ options = {}
8
+ if (query = params[:query])
9
+ options[:search] = Regexp.new(query, Regexp::IGNORECASE)
10
+ end
11
+ targets = klass.targets(content, box, options).map { |t|
12
+ { :id => t.id,
13
+ :title => t.alias_title,
14
+ :icon => t.exported_alias_icon }
15
+ }
16
+ json({
17
+ :pages => 1,
18
+ :total => targets.length,
19
+ :page => 1,
20
+ :targets => targets
21
+ })
22
+ end
23
+ end
24
+ end
25
+
26
+ post '/:id/:box_id' do
27
+ content_for_request(true) do |content, box|
28
+ type = content_model.schema.to_class(params[:alias_id])
29
+ position = (params[:position] || 0).to_i
30
+ if box.writable?(user, type)
31
+ instance = type.for_target(params[:target_id])
32
+ if instance
33
+ box.insert(position, instance)
34
+ content.save
35
+ json({
36
+ :position => position,
37
+ :entry => instance.entry.export(user)
38
+ })
39
+ end
40
+ else
41
+ forbidden!
42
+ end
43
+ end
44
+ end
45
+ end
46
+ end
@@ -0,0 +1,28 @@
1
+ require 'sprockets'
2
+
3
+ module Spontaneous::Rack::Back
4
+ class ApplicationAssets < Base
5
+ def initialize(app, charset = "UTF-8")
6
+ css, js = %w(css js).map { |d| build_asset_handler(d, charset) }
7
+ assets = Spontaneous::Rack::CacheableFile.new(Spontaneous.root / "public/@spontaneous/assets")
8
+ @app = ::Rack::Builder.app do
9
+ use Spontaneous::Rack::Static, :root => Spontaneous.application_dir, :urls => %W(/static)
10
+ map("/assets") { run assets }
11
+ map("/css") { run css }
12
+ map("/js") { run js }
13
+ run app
14
+ end
15
+ end
16
+
17
+ def call(env)
18
+ @app.call(env)
19
+ end
20
+
21
+ def build_asset_handler(dir, charset)
22
+ environment = ::Sprockets::Environment.new(Spontaneous.application_dir ) do |env|
23
+ env.append_path("#{dir}")
24
+ end
25
+ Spontaneous::Rack::AssetServer.new(environment, charset)
26
+ end
27
+ end
28
+ end
@@ -0,0 +1,34 @@
1
+ module Spontaneous::Rack
2
+ module Back
3
+ class Base < ServerBase
4
+ helpers Helpers
5
+
6
+ set :views, Proc.new { Spontaneous.application_dir + '/views' }
7
+
8
+ def content_model
9
+ Spontaneous::Content
10
+ end
11
+
12
+ def update_fields(model, field_data)
13
+ return unless field_data
14
+ Spontaneous::Field.update_asynchronously(model, field_data, user)
15
+ json(model)
16
+ end
17
+
18
+ def content_for_request(lock = false)
19
+ content_model.db.transaction do
20
+ dataset = lock ? content_model.for_update : content_model
21
+ content = dataset.get(params[:id])
22
+ halt 404 if content.nil?
23
+ content.current_editor = user
24
+ if box_id = content_model.schema.uids[params[:box_id]]
25
+ box = content.boxes.detect { |b| b.schema_id == box_id }
26
+ yield(content, box)
27
+ else
28
+ yield(content)
29
+ end
30
+ end
31
+ end
32
+ end
33
+ end
34
+ end
@@ -0,0 +1,19 @@
1
+ module Spontaneous::Rack::Back
2
+ class Changes < Base
3
+ before do
4
+ forbidden! unless user.level.can_publish?
5
+ end
6
+
7
+ get '/?' do
8
+ json(Spontaneous::Change)
9
+ end
10
+
11
+ post '/?' do
12
+ ids = params[:page_ids]
13
+ halt 400 if ids.blank? || ids.empty?
14
+ pages = ids.map(&:to_i)
15
+ Spontaneous::Site.publish_pages(pages)
16
+ json({})
17
+ end
18
+ end
19
+ end
@@ -0,0 +1,54 @@
1
+ module Spontaneous::Rack::Back
2
+ class Content < Base
3
+ put '/:id/?:box_id?' do
4
+ content_for_request(true) do |content, box|
5
+ target = content
6
+ if box
7
+ forbidden! unless box.writable?(user)
8
+ target = box
9
+ end
10
+ update_fields(target, params[:field])
11
+ end
12
+ end
13
+
14
+ post '/:id/:box_id/:type_name' do
15
+ content_for_request(true) do |content, box|
16
+ position = (params[:position] || 0).to_i
17
+ type = content_model.schema.to_class(params[:type_name])#.constantize
18
+
19
+ forbidden! unless box.writable?(user, type)
20
+ instance = type.new(:created_by => user)
21
+ box.insert(position, instance)
22
+ content.save
23
+ json({
24
+ :position => position,
25
+ :entry => instance.entry.export(user)
26
+ })
27
+ end
28
+ end
29
+
30
+ delete '/:id' do
31
+ content_for_request(true) do |content|
32
+ forbidden! unless content.box.writable?(user)
33
+ content.destroy
34
+ json({})
35
+ end
36
+ end
37
+
38
+ patch '/:id/toggle' do
39
+ content_for_request(true) do |content|
40
+ forbidden! unless content.box && content.box.writable?(user)
41
+ content.toggle_visibility!
42
+ json({:id => content.id, :hidden => (content.hidden? ? true : false) })
43
+ end
44
+ end
45
+
46
+ patch '/:id/position/:position' do
47
+ content_for_request(true) do |content|
48
+ forbidden! unless content.box.writable?(user)
49
+ content.update_position(params[:position].to_i)
50
+ json( {:message => 'OK'} )
51
+ end
52
+ end
53
+ end
54
+ end
@@ -0,0 +1,38 @@
1
+ require 'spontaneous/simultaneous'
2
+
3
+ module Spontaneous::Rack::Back
4
+ class Events < Base
5
+ def self.messenger
6
+ @messenger ||= build_messenger
7
+ end
8
+
9
+ def self.build_messenger
10
+ messenger = ::Spontaneous::Rack::EventSource.new
11
+ # Find a way to move this into a more de-centralised place
12
+ # at some point we are going to want to have some configurable, extendable
13
+ # list of event handlers
14
+ begin
15
+ ::Simultaneous.on_event("publish_progress") { |event|
16
+ messenger.deliver_event(event)
17
+ }
18
+ ::Simultaneous.on_event("page_lock_status") { |event|
19
+ messenger.deliver_event(event)
20
+ }
21
+ rescue => e
22
+ logger.error "Error while connecting to Simultaneous server"
23
+ logger.error e.inspect
24
+ end
25
+ messenger
26
+ end
27
+
28
+ get '/?', :provides => 'text/event-stream' do
29
+ headers 'X-Accel-Buffering' => 'no'
30
+ stream(:keep_open) do |out|
31
+ messenger = self.class.messenger
32
+ out.errback { messenger.delete(out) }
33
+ out.callback { messenger.delete(out) }
34
+ messenger << out
35
+ end
36
+ end
37
+ end
38
+ end
@@ -0,0 +1,37 @@
1
+ module Spontaneous::Rack::Back
2
+ class Field < Base
3
+ # Should be a GET as it changes no state but the request can need a lot
4
+ # of data which is best done outside of the query string
5
+ post '/conflicts/:id/?:box_id?' do
6
+ content_for_request(true) do |content, box|
7
+ generate_conflict_list(box || content)
8
+ end
9
+ end
10
+
11
+ get '/options/:field_sid/:id/?:box_id?' do
12
+ content_for_request do |content, box|
13
+ field = (box || content).fields.sid(params[:field_sid])
14
+ json(field.option_list)
15
+ end
16
+ end
17
+
18
+ def generate_conflict_list(content)
19
+ field_versions = params[:fields]
20
+ conflicts = []
21
+ field_versions.each do |schema_id, version|
22
+ field = content.fields.sid(schema_id)
23
+ if field.matches_version?(version.to_i)
24
+ conflicts << field
25
+ end
26
+ end
27
+ if conflicts.empty?
28
+ 200
29
+ else
30
+ errors = conflicts.map do |field|
31
+ [field.schema_id.to_s, [field.version, field.conflicted_value]]
32
+ end
33
+ [409, json(Hash[errors])]
34
+ end
35
+ end
36
+ end
37
+ end
@@ -0,0 +1,118 @@
1
+ module Spontaneous::Rack::Back
2
+ module File
3
+ class Simple < Base
4
+ put '/:id/?:box_id?' do
5
+ content_for_request(true) do |content, box|
6
+ target = box || content
7
+ file = params[:file]
8
+ field = target.fields.sid(params['field'])
9
+ forbidden! unless target.field_writable?(user, field.name)
10
+ # version = params[:version].to_i
11
+ # if version == field.version
12
+ Spontaneous::Field.set_asynchronously(field, file, user)
13
+ json(field.export(user))
14
+ # else
15
+ # errors = [[field.schema_id.to_s, [field.version, field.conflicted_value]]]
16
+ # [409, json(Hash[errors])]
17
+ # end
18
+ end
19
+ end
20
+
21
+ post '/:id/:box_id' do
22
+ content_for_request(true) do |content, box|
23
+ file = params['file']
24
+ type = box.type_for_mime_type(file[:type])
25
+ if type
26
+ forbidden! unless box.writable?(user, type)
27
+ position = 0
28
+ instance = type.new
29
+ box.insert(position, instance)
30
+ field = instance.field_for_mime_type(file[:type])
31
+ Spontaneous::Field.set_asynchronously(field, file, user)
32
+ content.save
33
+ json({
34
+ :position => position,
35
+ :entry => instance.entry.export(user)
36
+ })
37
+ end
38
+ end
39
+ end
40
+ end
41
+
42
+ class Sharded < Base
43
+ get '/:sha1' do
44
+ shard = Spontaneous.shard_path(params[:sha1])
45
+ if ::File.file?(shard)
46
+ # touch the shard file so that clean up routines can delete unmodified files
47
+ # without affecting any uploads in progresss
48
+ FileUtils.touch(shard)
49
+ 200
50
+ else
51
+ 404
52
+ end
53
+ end
54
+
55
+ post '/:sha1' do
56
+ file = params[:file]
57
+ uploaded_hash = Spontaneous::Media.digest(file[:tempfile].path)
58
+ if uploaded_hash == params[:sha1] # rand(10000) % 2 == 0 # use to test shard re-uploading
59
+ shard_path = Spontaneous.shard_path(params[:sha1])
60
+ FileUtils.mv(file[:tempfile].path, shard_path)
61
+ 200
62
+ else
63
+ ::Rack::Utils.status_code(:conflict) #409
64
+ end
65
+ end
66
+
67
+ put '/:id/?:box_id?' do
68
+ content_for_request(true) do |content, box|
69
+ target = box || content
70
+ replace_with_shard(target, content.id)
71
+ end
72
+ end
73
+
74
+ def replace_with_shard(target, target_id)
75
+ field = target.fields.sid(params[:field])
76
+ forbidden! unless target.field_writable?(user, field.name)
77
+ # version = params[:version].to_i
78
+ # if version == field.version
79
+ Spontaneous::Media.combine_shards(params[:shards]) do |combined|
80
+ Spontaneous::Field.set_asynchronously(field, {
81
+ :filename => params[:filename],
82
+ :tempfile => combined
83
+ }, user)
84
+ end
85
+ json(field.export(user))
86
+ end
87
+
88
+ # TODO: remove duplication here
89
+ post '/:id/:box_id' do
90
+ content_for_request(true) do |content, box|
91
+ type = box.type_for_mime_type(params[:mime_type])
92
+ if type
93
+ forbidden! unless box.writable?(user, type)
94
+ position = 0
95
+ instance = type.new
96
+ box.insert(position, instance)
97
+ field = instance.field_for_mime_type(params[:mime_type])
98
+ Spontaneous::Media.combine_shards(params[:shards]) do |combined|
99
+ Spontaneous::Field.set_asynchronously(field, {
100
+ :filename => params[:filename],
101
+ :tempfile => combined
102
+ }, user)
103
+ content.save
104
+ # field.value = {
105
+ # :filename => params[:filename],
106
+ # :tempfile => combined
107
+ # }
108
+ end
109
+ json({
110
+ :position => position,
111
+ :entry => instance.entry.export(user)
112
+ })
113
+ end
114
+ end
115
+ end
116
+ end
117
+ end
118
+ end
@@ -0,0 +1,71 @@
1
+ module Spontaneous::Rack::Back
2
+ module TemplateHelpers
3
+ include Spontaneous::Rack::Constants
4
+
5
+ def application_assets
6
+ @application_assets_compiler ||= Spontaneous::Asset::AppCompiler.new(Spontaneous.gem_dir, Spontaneous.root)
7
+ end
8
+
9
+ def style_url(style)
10
+ asset_url(style, "css")
11
+ end
12
+
13
+ def script_url(script)
14
+ asset_url(script, "js")
15
+ end
16
+
17
+ def asset_url(file, type)
18
+ file = "#{file}.#{type}" unless file =~ /\.#{type}$/
19
+
20
+
21
+ if (compiled_asset = application_assets.manifest.assets[file])
22
+ return "#{NAMESPACE}/assets/#{compiled_asset}"
23
+ end
24
+ # TODO: use the sprockets environment to append a modification time to the non-compiled URL
25
+ "#{NAMESPACE}/#{type}/#{file}"
26
+ end
27
+
28
+ def script_list(scripts)
29
+ scripts.map do |script|
30
+ script = "#{script}.js" unless script =~ /\.js$/
31
+ src = script_url(script)
32
+ size = 0
33
+ if (asset = application_assets.manifest.files[::File.basename(src)])
34
+ size = asset["size"]
35
+ else
36
+ src = "#{NAMESPACE}/js/#{script}"
37
+ path = Spontaneous.application_dir("/js/#{script}")
38
+ size = ::File.size(path)
39
+ end
40
+ [src, size]
41
+ end.to_json
42
+ end
43
+ end
44
+
45
+ module Helpers
46
+ include Spontaneous::Rack::Constants
47
+
48
+ def json(response)
49
+ content_type 'application/json', :charset => 'utf-8'
50
+ response.serialise_http(user)
51
+ end
52
+
53
+
54
+ def forbidden!
55
+ halt 403#, "You do not have the necessary permissions to update the '#{name}' field"
56
+ end
57
+
58
+
59
+ def api_key
60
+ request.cookies[AUTH_COOKIE]
61
+ end
62
+
63
+ def user
64
+ env[ACTIVE_USER]
65
+ end
66
+
67
+ def show_login_page(locals = {})
68
+ halt(401, erb(:login, :views => Spontaneous.application_dir('/views'), :locals => locals))
69
+ end
70
+ end
71
+ end
@@ -0,0 +1,16 @@
1
+ module Spontaneous::Rack::Back
2
+ class Index < Base
3
+ include TemplateHelpers
4
+
5
+ # Matches either:
6
+ # /@spontaneous
7
+ # /@spontaneous/
8
+ # /@spontaneous/xxx/edit
9
+ # /@spontaneous/xxx/preview
10
+ #
11
+ # where xxx is a numeric id
12
+ get %r{\A(/?|/(\d+/?.*)?)\z} do
13
+ erb :index
14
+ end
15
+ end
16
+ end
@@ -0,0 +1,47 @@
1
+ module Spontaneous::Rack::Back
2
+ class Login < Base
3
+ include TemplateHelpers
4
+
5
+ def set_authentication_cookie(key)
6
+ response.set_cookie(AUTH_COOKIE, {
7
+ :value => key.key_id,
8
+ :path => '/',
9
+ :secure => request.ssl?,
10
+ :httponly => true,
11
+ :expires => (Time.now + SESSION_LIFETIME)
12
+ })
13
+ end
14
+
15
+ def unset_authentication_cookie
16
+ response.delete_cookie(AUTH_COOKIE, {
17
+ :path => '/',
18
+ :secure => request.ssl?,
19
+ :httponly => true
20
+ })
21
+ end
22
+
23
+ post "/login" do
24
+ login = params[:user][:login]
25
+ password = params[:user][:password]
26
+ origin = "#{NAMESPACE}#{params[:origin]}"
27
+ if key = Spontaneous::Permissions::User.authenticate(login, password, env["REMOTE_ADDR"])
28
+ set_authentication_cookie(key)
29
+ if request.xhr?
30
+ json({
31
+ :key => key.key_id,
32
+ :redirect => origin
33
+ })
34
+ else
35
+ redirect origin, 302
36
+ end
37
+ else
38
+ show_login_page( :login => login, :failed => true )
39
+ end
40
+ end
41
+
42
+ delete "/logout" do
43
+ unset_authentication_cookie
44
+ 401
45
+ end
46
+ end
47
+ end
@@ -0,0 +1,24 @@
1
+ module Spontaneous::Rack::Back
2
+ class Map < Base
3
+ get '/?:id?' do
4
+ last_modified(Spontaneous::Site.modified_at)
5
+ map = Spontaneous::Site.map(params[:id])
6
+ if map
7
+ json(map)
8
+ else
9
+ 404
10
+ end
11
+ end
12
+
13
+ get '/path*' do
14
+ last_modified(Spontaneous::Site.modified_at)
15
+ if content_model::Page.count == 0
16
+ 406
17
+ else
18
+ path = params[:splat].first
19
+ page = Spontaneous::Site[path]
20
+ json Spontaneous::Site.map(page.id)
21
+ end
22
+ end
23
+ end
24
+ end
@@ -0,0 +1,46 @@
1
+ module Spontaneous::Rack::Back
2
+ class Page < Base
3
+ get '/:id' do
4
+ content_for_request { |content| json(content)}
5
+ end
6
+
7
+ put '/:id/slug' do
8
+ content_for_request(true) do |content|
9
+ if params[:slug].nil? or params[:slug].empty?
10
+ 406 # Not Acceptable
11
+ else
12
+ content.slug = params[:slug]
13
+ if content.siblings.detect { |s| s.slug == content.slug }
14
+ 409 # Conflict
15
+ else
16
+ content.save
17
+ json({:path => content.path, :slug => content.slug })
18
+ end
19
+ end
20
+ end
21
+ end
22
+
23
+ get '/:id/slug/unavailable' do
24
+ content_for_request do |content|
25
+ json(content.siblings.map { |c| c.slug })
26
+ end
27
+ end
28
+
29
+ put '/:id/slug/sync' do
30
+ content_for_request do |page|
31
+ page.sync_slug_to_title
32
+ page.save
33
+ json({:path => page.path, :slug => page.slug })
34
+ end
35
+ end
36
+
37
+ put '/:id/uid' do
38
+ forbidden! unless user.developer?
39
+ content_for_request(true) do |content|
40
+ content.uid = params[:uid]
41
+ content.save
42
+ json({:uid => content.uid })
43
+ end
44
+ end
45
+ end
46
+ end
@@ -0,0 +1,43 @@
1
+ module Spontaneous::Rack::Back
2
+ class Preview < Base
3
+ include Spontaneous::Rack::Public
4
+
5
+
6
+ # In preview mode we want to find pages even if they're
7
+ # invisible.
8
+ def find_page_by_path(path)
9
+ Spontaneous::Content.scope do
10
+ Spontaneous::Site.by_path(path)
11
+ end
12
+ end
13
+
14
+ # Redirect to the edit UI if a preview page is being accessed directly
15
+ def ensure_edit_preview(path)
16
+ referer = env['HTTP_REFERER']
17
+ return true if Spontaneous.development? || referer || params.key?('preview')
18
+ home = find_page_by_path(path)
19
+ redirect "#{NAMESPACE}/#{home.id}/preview"
20
+ false
21
+ end
22
+
23
+ # Forward all GETs to the page resolution method
24
+ get '*' do
25
+ path = params[:splat][0]
26
+ ensure_edit_preview(path) && render_path(path)
27
+ end
28
+
29
+ # Forward all POSTs to the page resolution method
30
+ post '*' do
31
+ render_path(params[:splat][0])
32
+ end
33
+
34
+ # Override the S::Rack::Public method to add in some cache-busting headers
35
+ def render_page(page, format = :html, local_params = {})
36
+ now = Time.now.to_formatted_s(:rfc822)
37
+ response.headers[HTTP_EXPIRES] = now
38
+ response.headers[HTTP_LAST_MODIFIED] = now
39
+ response.headers[HTTP_CACHE_CONTROL] = HTTP_NO_CACHE
40
+ super
41
+ end
42
+ end # Preview
43
+ end
@@ -0,0 +1,30 @@
1
+ module Spontaneous::Rack::Back
2
+ class Schema < Base
3
+ # Use the CSRF verification step separately from the main app
4
+ # because it conflicts with the Index application:
5
+ # The index application needs to load without CSRF validation
6
+ # and the Schema app needs CSRF validation
7
+ # but the Schema app has be before Index in the stack
8
+ use Spontaneous::Rack::Middleware::CSRF::Verification
9
+
10
+ def schema
11
+ Spontaneous.schema
12
+ end
13
+
14
+ post "/delete" do
15
+ begin
16
+ schema.apply_fix(:delete, params[:uid])
17
+ rescue Spot::SchemaModificationError # ignore remaining errors - they will be fixed later
18
+ end
19
+ redirect(params[:origin])
20
+ end
21
+
22
+ post "/rename" do
23
+ begin
24
+ schema.apply_fix(:rename, params[:uid], params[:ref])
25
+ rescue Spot::SchemaModificationError => e # ignore remaining errors - they will be fixed later
26
+ end
27
+ redirect(params[:origin])
28
+ end
29
+ end
30
+ end