middleman-core 4.0.0.rc.1 → 4.0.0.rc.2

Sign up to get free protection for your applications and to get access to all the features.
Files changed (415) hide show
  1. checksums.yaml +4 -4
  2. data/.rspec +1 -0
  3. data/Rakefile +0 -1
  4. data/cucumber.yml +2 -0
  5. data/features/asset_hash.feature +25 -16
  6. data/features/asset_host.feature +5 -1
  7. data/features/chained_templates.feature +8 -5
  8. data/features/clean_build.feature +18 -0
  9. data/features/cli/preview_server-hook.feature +17 -0
  10. data/features/cli/preview_server.feature +533 -0
  11. data/features/cli_init.feature +3 -3
  12. data/features/collections.feature +5 -3
  13. data/features/content_type.feature +3 -0
  14. data/features/default-layout.feature +90 -0
  15. data/features/extension_api_deprecations.feature +10 -0
  16. data/features/front-matter.feature +6 -1
  17. data/features/i18n_link_to.feature +209 -0
  18. data/features/import_files.feature +17 -0
  19. data/features/javascript-testing.feature +18 -0
  20. data/features/markdown_redcarpet.feature +0 -1
  21. data/features/markdown_redcarpet_in_haml.feature +0 -1
  22. data/features/minify_css.feature +87 -5
  23. data/features/minify_javascript.feature +55 -8
  24. data/features/move_files.feature +44 -0
  25. data/features/redirects.feature +1 -0
  26. data/features/relative_assets.feature +13 -2
  27. data/features/slim.feature +1 -1
  28. data/features/stylus.feature +0 -1
  29. data/features/support/env.rb +5 -1
  30. data/features/support/preserve_mime_types.rb +7 -0
  31. data/fixtures/asset-hash-app/source/fonts/fontawesome-webfont.woff +0 -0
  32. data/fixtures/asset-hash-app/source/fonts/fontawesome-webfont.woff2 +0 -0
  33. data/fixtures/asset-hash-app/source/stylesheets/site.css.scss +2 -2
  34. data/fixtures/asset-hash-app/source/stylesheets/uses_fonts.css +4 -0
  35. data/fixtures/asset-host-app/source/asset_host.html.erb +5 -1
  36. data/fixtures/chained-app/source/test.erb.combobreaker.str.erb +8 -0
  37. data/fixtures/clean-app/config-hidden-dir-after.rb +5 -0
  38. data/fixtures/clean-app/config-hidden-dir-before.rb +1 -0
  39. data/fixtures/content-type-app/source/index.php +1 -0
  40. data/fixtures/extension-api-deprecations-app/config.rb +11 -0
  41. data/fixtures/extension-api-deprecations-app/source/index.html.erb +1 -0
  42. data/fixtures/extension-api-deprecations-app/source/layouts/layout.erb +3 -0
  43. data/fixtures/frontmatter-app/config.rb +1 -0
  44. data/fixtures/frontmatter-app/source/front-matter-haml.html.haml +6 -0
  45. data/fixtures/frontmatter-neighbor-app/config.rb +1 -1
  46. data/fixtures/frontmatter-settings-neighbor-app/config.rb +1 -1
  47. data/fixtures/i18n-test-app/source/localizable/partials/index.html.erb +3 -1
  48. data/fixtures/import-app/bower.json +21 -0
  49. data/fixtures/import-app/bower_components/jquery/.bower.json +39 -0
  50. data/fixtures/import-app/bower_components/jquery/MIT-LICENSE.txt +21 -0
  51. data/fixtures/import-app/bower_components/jquery/bower.json +28 -0
  52. data/fixtures/import-app/bower_components/jquery/dist/jquery.js +9210 -0
  53. data/fixtures/import-app/bower_components/jquery/dist/jquery.min.js +5 -0
  54. data/fixtures/import-app/bower_components/jquery/dist/jquery.min.map +1 -0
  55. data/fixtures/import-app/bower_components/jquery/src/ajax.js +786 -0
  56. data/fixtures/import-app/bower_components/jquery/src/ajax/jsonp.js +89 -0
  57. data/fixtures/import-app/bower_components/jquery/src/ajax/load.js +75 -0
  58. data/fixtures/import-app/bower_components/jquery/src/ajax/parseJSON.js +13 -0
  59. data/fixtures/import-app/bower_components/jquery/src/ajax/parseXML.js +28 -0
  60. data/fixtures/import-app/bower_components/jquery/src/ajax/script.js +64 -0
  61. data/fixtures/import-app/bower_components/jquery/src/ajax/var/nonce.js +5 -0
  62. data/fixtures/import-app/bower_components/jquery/src/ajax/var/rquery.js +3 -0
  63. data/fixtures/import-app/bower_components/jquery/src/ajax/xhr.js +136 -0
  64. data/fixtures/import-app/bower_components/jquery/src/attributes.js +11 -0
  65. data/fixtures/import-app/bower_components/jquery/src/attributes/attr.js +141 -0
  66. data/fixtures/import-app/bower_components/jquery/src/attributes/classes.js +158 -0
  67. data/fixtures/import-app/bower_components/jquery/src/attributes/prop.js +94 -0
  68. data/fixtures/import-app/bower_components/jquery/src/attributes/support.js +35 -0
  69. data/fixtures/import-app/bower_components/jquery/src/attributes/val.js +161 -0
  70. data/fixtures/import-app/bower_components/jquery/src/callbacks.js +205 -0
  71. data/fixtures/import-app/bower_components/jquery/src/core.js +502 -0
  72. data/fixtures/import-app/bower_components/jquery/src/core/access.js +60 -0
  73. data/fixtures/import-app/bower_components/jquery/src/core/init.js +123 -0
  74. data/fixtures/import-app/bower_components/jquery/src/core/parseHTML.js +39 -0
  75. data/fixtures/import-app/bower_components/jquery/src/core/ready.js +97 -0
  76. data/fixtures/import-app/bower_components/jquery/src/core/var/rsingleTag.js +4 -0
  77. data/fixtures/import-app/bower_components/jquery/src/css.js +450 -0
  78. data/fixtures/import-app/bower_components/jquery/src/css/addGetHookIf.js +22 -0
  79. data/fixtures/import-app/bower_components/jquery/src/css/curCSS.js +57 -0
  80. data/fixtures/import-app/bower_components/jquery/src/css/defaultDisplay.js +70 -0
  81. data/fixtures/import-app/bower_components/jquery/src/css/hiddenVisibleSelectors.js +15 -0
  82. data/fixtures/import-app/bower_components/jquery/src/css/support.js +96 -0
  83. data/fixtures/import-app/bower_components/jquery/src/css/swap.js +28 -0
  84. data/fixtures/import-app/bower_components/jquery/src/css/var/cssExpand.js +3 -0
  85. data/fixtures/import-app/bower_components/jquery/src/css/var/getStyles.js +12 -0
  86. data/fixtures/import-app/bower_components/jquery/src/css/var/isHidden.js +13 -0
  87. data/fixtures/import-app/bower_components/jquery/src/css/var/rmargin.js +3 -0
  88. data/fixtures/import-app/bower_components/jquery/src/css/var/rnumnonpx.js +5 -0
  89. data/fixtures/import-app/bower_components/jquery/src/data.js +178 -0
  90. data/fixtures/import-app/bower_components/jquery/src/data/Data.js +181 -0
  91. data/fixtures/import-app/bower_components/jquery/src/data/accepts.js +20 -0
  92. data/fixtures/import-app/bower_components/jquery/src/data/var/data_priv.js +5 -0
  93. data/fixtures/import-app/bower_components/jquery/src/data/var/data_user.js +5 -0
  94. data/fixtures/import-app/bower_components/jquery/src/deferred.js +149 -0
  95. data/fixtures/import-app/bower_components/jquery/src/deprecated.js +13 -0
  96. data/fixtures/import-app/bower_components/jquery/src/dimensions.js +50 -0
  97. data/fixtures/import-app/bower_components/jquery/src/effects.js +648 -0
  98. data/fixtures/import-app/bower_components/jquery/src/effects/Tween.js +114 -0
  99. data/fixtures/import-app/bower_components/jquery/src/effects/animatedSelector.js +13 -0
  100. data/fixtures/import-app/bower_components/jquery/src/event.js +868 -0
  101. data/fixtures/import-app/bower_components/jquery/src/event/ajax.js +13 -0
  102. data/fixtures/import-app/bower_components/jquery/src/event/alias.js +39 -0
  103. data/fixtures/import-app/bower_components/jquery/src/event/support.js +9 -0
  104. data/fixtures/import-app/bower_components/jquery/src/exports/amd.js +24 -0
  105. data/fixtures/import-app/bower_components/jquery/src/exports/global.js +32 -0
  106. data/fixtures/import-app/bower_components/jquery/src/intro.js +44 -0
  107. data/fixtures/import-app/bower_components/jquery/src/jquery.js +37 -0
  108. data/fixtures/import-app/bower_components/jquery/src/manipulation.js +580 -0
  109. data/fixtures/import-app/bower_components/jquery/src/manipulation/_evalUrl.js +18 -0
  110. data/fixtures/import-app/bower_components/jquery/src/manipulation/support.js +32 -0
  111. data/fixtures/import-app/bower_components/jquery/src/manipulation/var/rcheckableType.js +3 -0
  112. data/fixtures/import-app/bower_components/jquery/src/offset.js +207 -0
  113. data/fixtures/import-app/bower_components/jquery/src/outro.js +1 -0
  114. data/fixtures/import-app/bower_components/jquery/src/queue.js +142 -0
  115. data/fixtures/import-app/bower_components/jquery/src/queue/delay.js +22 -0
  116. data/fixtures/import-app/bower_components/jquery/src/selector-native.js +172 -0
  117. data/fixtures/import-app/bower_components/jquery/src/selector-sizzle.js +14 -0
  118. data/fixtures/import-app/bower_components/jquery/src/selector.js +1 -0
  119. data/fixtures/import-app/bower_components/jquery/src/serialize.js +111 -0
  120. data/fixtures/import-app/bower_components/jquery/src/sizzle/dist/sizzle.js +2067 -0
  121. data/fixtures/import-app/bower_components/jquery/src/sizzle/dist/sizzle.min.js +3 -0
  122. data/fixtures/import-app/bower_components/jquery/src/sizzle/dist/sizzle.min.map +1 -0
  123. data/fixtures/import-app/bower_components/jquery/src/traversing.js +199 -0
  124. data/fixtures/import-app/bower_components/jquery/src/traversing/findFilter.js +100 -0
  125. data/fixtures/import-app/bower_components/jquery/src/traversing/var/rneedsContext.js +6 -0
  126. data/fixtures/import-app/bower_components/jquery/src/var/arr.js +3 -0
  127. data/fixtures/import-app/bower_components/jquery/src/var/class2type.js +4 -0
  128. data/fixtures/import-app/bower_components/jquery/src/var/concat.js +5 -0
  129. data/fixtures/import-app/bower_components/jquery/src/var/hasOwn.js +5 -0
  130. data/fixtures/import-app/bower_components/jquery/src/var/indexOf.js +5 -0
  131. data/fixtures/import-app/bower_components/jquery/src/var/pnum.js +3 -0
  132. data/fixtures/import-app/bower_components/jquery/src/var/push.js +5 -0
  133. data/fixtures/import-app/bower_components/jquery/src/var/rnotwhite.js +3 -0
  134. data/fixtures/import-app/bower_components/jquery/src/var/slice.js +5 -0
  135. data/fixtures/import-app/bower_components/jquery/src/var/strundefined.js +3 -0
  136. data/fixtures/import-app/bower_components/jquery/src/var/support.js +4 -0
  137. data/fixtures/import-app/bower_components/jquery/src/var/toString.js +5 -0
  138. data/fixtures/import-app/bower_components/jquery/src/wrap.js +79 -0
  139. data/fixtures/import-app/config.rb +7 -0
  140. data/fixtures/{twitter-bootstrap-app/config.rb → import-app/source/test.html} +0 -0
  141. data/fixtures/import-app/static.html +1 -0
  142. data/fixtures/javascript-app/config.rb +0 -0
  143. data/fixtures/javascript-app/source/index.html +17 -0
  144. data/fixtures/minify-css-app/source/inline-css.php +8 -0
  145. data/fixtures/minify-css-app/source/stylesheets/base/_base.scss +15 -0
  146. data/fixtures/minify-css-app/source/stylesheets/base/_buttons.scss +35 -0
  147. data/fixtures/minify-css-app/source/stylesheets/base/_forms.scss +90 -0
  148. data/fixtures/minify-css-app/source/stylesheets/base/_grid-settings.scss +14 -0
  149. data/fixtures/minify-css-app/source/stylesheets/base/_lists.scss +31 -0
  150. data/fixtures/minify-css-app/source/stylesheets/base/_tables.scss +25 -0
  151. data/fixtures/minify-css-app/source/stylesheets/base/_typography.scss +49 -0
  152. data/fixtures/minify-css-app/source/stylesheets/base/_variables.scss +42 -0
  153. data/fixtures/minify-css-app/source/stylesheets/bourbon/_bourbon-deprecated-upcoming.scss +411 -0
  154. data/fixtures/minify-css-app/source/stylesheets/bourbon/_bourbon.scss +87 -0
  155. data/fixtures/minify-css-app/source/stylesheets/bourbon/addons/_border-color.scss +26 -0
  156. data/fixtures/minify-css-app/source/stylesheets/bourbon/addons/_border-radius.scss +48 -0
  157. data/fixtures/minify-css-app/source/stylesheets/bourbon/addons/_border-style.scss +25 -0
  158. data/fixtures/minify-css-app/source/stylesheets/bourbon/addons/_border-width.scss +25 -0
  159. data/fixtures/minify-css-app/source/stylesheets/bourbon/addons/_buttons.scss +64 -0
  160. data/fixtures/minify-css-app/source/stylesheets/bourbon/addons/_clearfix.scss +25 -0
  161. data/fixtures/minify-css-app/source/stylesheets/bourbon/addons/_ellipsis.scss +30 -0
  162. data/fixtures/minify-css-app/source/stylesheets/bourbon/addons/_font-stacks.scss +31 -0
  163. data/fixtures/minify-css-app/source/stylesheets/bourbon/addons/_hide-text.scss +27 -0
  164. data/fixtures/minify-css-app/source/stylesheets/bourbon/addons/_margin.scss +26 -0
  165. data/fixtures/minify-css-app/source/stylesheets/bourbon/addons/_padding.scss +26 -0
  166. data/fixtures/minify-css-app/source/stylesheets/bourbon/addons/_position.scss +48 -0
  167. data/fixtures/minify-css-app/source/stylesheets/bourbon/addons/_prefixer.scss +66 -0
  168. data/fixtures/minify-css-app/source/stylesheets/bourbon/addons/_retina-image.scss +25 -0
  169. data/fixtures/minify-css-app/source/stylesheets/bourbon/addons/_size.scss +51 -0
  170. data/fixtures/minify-css-app/source/stylesheets/bourbon/addons/_text-inputs.scss +113 -0
  171. data/fixtures/minify-css-app/source/stylesheets/bourbon/addons/_timing-functions.scss +34 -0
  172. data/fixtures/minify-css-app/source/stylesheets/bourbon/addons/_triangle.scss +63 -0
  173. data/fixtures/minify-css-app/source/stylesheets/bourbon/addons/_word-wrap.scss +29 -0
  174. data/fixtures/minify-css-app/source/stylesheets/bourbon/css3/_animation.scss +43 -0
  175. data/fixtures/minify-css-app/source/stylesheets/bourbon/css3/_appearance.scss +3 -0
  176. data/fixtures/minify-css-app/source/stylesheets/bourbon/css3/_backface-visibility.scss +3 -0
  177. data/fixtures/minify-css-app/source/stylesheets/bourbon/css3/_background-image.scss +42 -0
  178. data/fixtures/minify-css-app/source/stylesheets/bourbon/css3/_background.scss +55 -0
  179. data/fixtures/minify-css-app/source/stylesheets/bourbon/css3/_border-image.scss +59 -0
  180. data/fixtures/minify-css-app/source/stylesheets/bourbon/css3/_calc.scss +4 -0
  181. data/fixtures/minify-css-app/source/stylesheets/bourbon/css3/_columns.scss +47 -0
  182. data/fixtures/minify-css-app/source/stylesheets/bourbon/css3/_filter.scss +4 -0
  183. data/fixtures/minify-css-app/source/stylesheets/bourbon/css3/_flex-box.scss +287 -0
  184. data/fixtures/minify-css-app/source/stylesheets/bourbon/css3/_font-face.scss +24 -0
  185. data/fixtures/minify-css-app/source/stylesheets/bourbon/css3/_font-feature-settings.scss +4 -0
  186. data/fixtures/minify-css-app/source/stylesheets/bourbon/css3/_hidpi-media-query.scss +10 -0
  187. data/fixtures/minify-css-app/source/stylesheets/bourbon/css3/_hyphens.scss +4 -0
  188. data/fixtures/minify-css-app/source/stylesheets/bourbon/css3/_image-rendering.scss +14 -0
  189. data/fixtures/minify-css-app/source/stylesheets/bourbon/css3/_keyframes.scss +36 -0
  190. data/fixtures/minify-css-app/source/stylesheets/bourbon/css3/_linear-gradient.scss +38 -0
  191. data/fixtures/minify-css-app/source/stylesheets/bourbon/css3/_perspective.scss +8 -0
  192. data/fixtures/minify-css-app/source/stylesheets/bourbon/css3/_placeholder.scss +8 -0
  193. data/fixtures/minify-css-app/source/stylesheets/bourbon/css3/_radial-gradient.scss +39 -0
  194. data/fixtures/minify-css-app/source/stylesheets/bourbon/css3/_selection.scss +42 -0
  195. data/fixtures/minify-css-app/source/stylesheets/bourbon/css3/_text-decoration.scss +19 -0
  196. data/fixtures/minify-css-app/source/stylesheets/bourbon/css3/_transform.scss +15 -0
  197. data/fixtures/minify-css-app/source/stylesheets/bourbon/css3/_transition.scss +71 -0
  198. data/fixtures/minify-css-app/source/stylesheets/bourbon/css3/_user-select.scss +3 -0
  199. data/fixtures/minify-css-app/source/stylesheets/bourbon/functions/_assign-inputs.scss +11 -0
  200. data/fixtures/minify-css-app/source/stylesheets/bourbon/functions/_contains-falsy.scss +20 -0
  201. data/fixtures/minify-css-app/source/stylesheets/bourbon/functions/_contains.scss +26 -0
  202. data/fixtures/minify-css-app/source/stylesheets/bourbon/functions/_is-length.scss +11 -0
  203. data/fixtures/minify-css-app/source/stylesheets/bourbon/functions/_is-light.scss +21 -0
  204. data/fixtures/minify-css-app/source/stylesheets/bourbon/functions/_is-number.scss +11 -0
  205. data/fixtures/minify-css-app/source/stylesheets/bourbon/functions/_is-size.scss +13 -0
  206. data/fixtures/minify-css-app/source/stylesheets/bourbon/functions/_modular-scale.scss +69 -0
  207. data/fixtures/minify-css-app/source/stylesheets/bourbon/functions/_px-to-em.scss +13 -0
  208. data/fixtures/minify-css-app/source/stylesheets/bourbon/functions/_px-to-rem.scss +15 -0
  209. data/fixtures/minify-css-app/source/stylesheets/bourbon/functions/_shade.scss +24 -0
  210. data/fixtures/minify-css-app/source/stylesheets/bourbon/functions/_strip-units.scss +17 -0
  211. data/fixtures/minify-css-app/source/stylesheets/bourbon/functions/_tint.scss +24 -0
  212. data/fixtures/minify-css-app/source/stylesheets/bourbon/functions/_transition-property-name.scss +22 -0
  213. data/fixtures/minify-css-app/source/stylesheets/bourbon/functions/_unpack.scss +27 -0
  214. data/fixtures/minify-css-app/source/stylesheets/bourbon/helpers/_convert-units.scss +21 -0
  215. data/fixtures/minify-css-app/source/stylesheets/bourbon/helpers/_directional-values.scss +96 -0
  216. data/fixtures/minify-css-app/source/stylesheets/bourbon/helpers/_font-source-declaration.scss +43 -0
  217. data/fixtures/minify-css-app/source/stylesheets/bourbon/helpers/_gradient-positions-parser.scss +13 -0
  218. data/fixtures/minify-css-app/source/stylesheets/bourbon/helpers/_linear-angle-parser.scss +25 -0
  219. data/fixtures/minify-css-app/source/stylesheets/bourbon/helpers/_linear-gradient-parser.scss +41 -0
  220. data/fixtures/minify-css-app/source/stylesheets/bourbon/helpers/_linear-positions-parser.scss +61 -0
  221. data/fixtures/minify-css-app/source/stylesheets/bourbon/helpers/_linear-side-corner-parser.scss +31 -0
  222. data/fixtures/minify-css-app/source/stylesheets/bourbon/helpers/_radial-arg-parser.scss +69 -0
  223. data/fixtures/minify-css-app/source/stylesheets/bourbon/helpers/_radial-gradient-parser.scss +50 -0
  224. data/fixtures/minify-css-app/source/stylesheets/bourbon/helpers/_radial-positions-parser.scss +18 -0
  225. data/fixtures/minify-css-app/source/stylesheets/bourbon/helpers/_render-gradients.scss +26 -0
  226. data/fixtures/minify-css-app/source/stylesheets/bourbon/helpers/_shape-size-stripper.scss +10 -0
  227. data/fixtures/minify-css-app/source/stylesheets/bourbon/helpers/_str-to-num.scss +50 -0
  228. data/fixtures/minify-css-app/source/stylesheets/bourbon/settings/_asset-pipeline.scss +7 -0
  229. data/fixtures/minify-css-app/source/stylesheets/bourbon/settings/_prefixer.scss +9 -0
  230. data/fixtures/minify-css-app/source/stylesheets/bourbon/settings/_px-to-em.scss +1 -0
  231. data/fixtures/minify-css-app/source/stylesheets/site.xcss.sass +6 -0
  232. data/fixtures/minify-js-app/source/inline-js.php +22 -0
  233. data/fixtures/minify-js-app/source/javascripts/js_test.xjs +8 -0
  234. data/fixtures/partials-app/source/locals.html.erb +1 -1
  235. data/fixtures/preview-server-app/bin/dns_server.rb +33 -0
  236. data/fixtures/preview-server-app/config.rb +0 -0
  237. data/fixtures/preview-server-app/source/index.html.erb +1 -0
  238. data/fixtures/preview-server-app/source/layout.erb +9 -0
  239. data/fixtures/preview-server-app/source/layouts/custom.erb +8 -0
  240. data/fixtures/preview-server-app/source/real.html +1 -0
  241. data/fixtures/preview-server-app/source/real/index.html.erb +5 -0
  242. data/fixtures/preview-server-app/source/should_be_ignored.html +1 -0
  243. data/fixtures/preview-server-app/source/should_be_ignored2.html +1 -0
  244. data/fixtures/preview-server-app/source/should_be_ignored3.html +1 -0
  245. data/fixtures/preview-server-app/source/static.html +1 -0
  246. data/fixtures/preview-server-hook-app/config.rb +19 -0
  247. data/fixtures/preview-server-hook-app/source/index.html.erb +9 -0
  248. data/fixtures/related-files-app/config.rb +0 -0
  249. data/fixtures/related-files-app/source/index.html.erb +0 -0
  250. data/fixtures/related-files-app/source/partials/_test.erb +0 -0
  251. data/fixtures/related-files-app/source/partials/_test2.haml +0 -0
  252. data/fixtures/related-files-app/source/stylesheets/_include3.sass +0 -0
  253. data/fixtures/related-files-app/source/stylesheets/_include4.scss +0 -0
  254. data/fixtures/related-files-app/source/stylesheets/include1.css +0 -0
  255. data/fixtures/related-files-app/source/stylesheets/include2.css.scss +0 -0
  256. data/fixtures/related-files-app/source/stylesheets/site.css.scss +0 -0
  257. data/fixtures/relative-assets-app/source/absolute_image_relative_css.html.erb +9 -0
  258. data/fixtures/relative-assets-app/source/javascripts/app.js +3 -0
  259. data/fixtures/relative-assets-app/source/relative_image.html.erb +1 -0
  260. data/fixtures/relative-assets-app/source/relative_image_absolute_css.html.erb +9 -0
  261. data/lib/middleman-core/application.rb +44 -18
  262. data/lib/middleman-core/builder.rb +1 -27
  263. data/lib/middleman-core/callback_manager.rb +3 -3
  264. data/lib/middleman-core/configuration.rb +1 -1
  265. data/lib/middleman-core/core_extensions/collections/step_context.rb +1 -1
  266. data/lib/middleman-core/core_extensions/data.rb +2 -2
  267. data/lib/middleman-core/core_extensions/default_helpers.rb +37 -3
  268. data/lib/middleman-core/core_extensions/external_helpers.rb +2 -1
  269. data/lib/middleman-core/core_extensions/front_matter.rb +5 -2
  270. data/lib/middleman-core/core_extensions/i18n.rb +95 -29
  271. data/lib/middleman-core/core_extensions/routing.rb +2 -2
  272. data/lib/middleman-core/dns_resolver.rb +73 -0
  273. data/lib/middleman-core/dns_resolver/basic_network_resolver.rb +52 -0
  274. data/lib/middleman-core/dns_resolver/hosts_resolver.rb +63 -0
  275. data/lib/middleman-core/dns_resolver/local_link_resolver.rb +44 -0
  276. data/lib/middleman-core/dns_resolver/network_resolver.rb +42 -0
  277. data/lib/middleman-core/extension.rb +1 -1
  278. data/lib/middleman-core/extensions/asset_hash.rb +3 -3
  279. data/lib/middleman-core/extensions/automatic_image_sizes.rb +9 -1
  280. data/lib/middleman-core/extensions/minify_css.rb +49 -17
  281. data/lib/middleman-core/extensions/minify_javascript.rb +60 -30
  282. data/lib/middleman-core/extensions/relative_assets.rb +13 -0
  283. data/lib/middleman-core/file_renderer.rb +2 -2
  284. data/lib/middleman-core/meta_pages.rb +1 -1
  285. data/lib/middleman-core/preview_server.rb +109 -47
  286. data/lib/middleman-core/preview_server/checks.rb +81 -0
  287. data/lib/middleman-core/preview_server/information.rb +273 -0
  288. data/lib/middleman-core/preview_server/network_interface_inventory.rb +65 -0
  289. data/lib/middleman-core/preview_server/server_hostname.rb +39 -0
  290. data/lib/middleman-core/preview_server/server_information.rb +153 -0
  291. data/lib/middleman-core/preview_server/server_information_callback_proxy.rb +35 -0
  292. data/lib/middleman-core/preview_server/server_information_validator.rb +18 -0
  293. data/lib/middleman-core/preview_server/server_ip_address.rb +55 -0
  294. data/lib/middleman-core/preview_server/server_url.rb +63 -0
  295. data/lib/middleman-core/preview_server/tcp_port_prober.rb +29 -0
  296. data/lib/middleman-core/rack.rb +7 -1
  297. data/lib/middleman-core/renderers/haml.rb +1 -2
  298. data/lib/middleman-core/renderers/liquid.rb +1 -1
  299. data/lib/middleman-core/renderers/sass.rb +23 -36
  300. data/lib/middleman-core/renderers/sass_functions.rb +20 -22
  301. data/lib/middleman-core/sitemap/extensions/ignores.rb +5 -1
  302. data/lib/middleman-core/sitemap/extensions/import.rb +108 -0
  303. data/lib/middleman-core/sitemap/extensions/move_file.rb +68 -0
  304. data/lib/middleman-core/sitemap/extensions/proxies.rb +7 -2
  305. data/lib/middleman-core/sitemap/extensions/redirects.rb +1 -0
  306. data/lib/middleman-core/sitemap/resource.rb +3 -7
  307. data/lib/middleman-core/sitemap/store.rb +22 -19
  308. data/lib/middleman-core/sources.rb +2 -3
  309. data/lib/middleman-core/sources/source_watcher.rb +21 -37
  310. data/lib/middleman-core/step_definitions.rb +1 -0
  311. data/lib/middleman-core/step_definitions/builder_steps.rb +0 -47
  312. data/lib/middleman-core/step_definitions/commandline_steps.rb +88 -0
  313. data/lib/middleman-core/step_definitions/middleman_steps.rb +51 -6
  314. data/lib/middleman-core/step_definitions/server_steps.rb +21 -53
  315. data/lib/middleman-core/template_context.rb +14 -13
  316. data/lib/middleman-core/util.rb +127 -4
  317. data/lib/middleman-core/util/data.rb +87 -138
  318. data/lib/middleman-core/version.rb +1 -1
  319. data/middleman-core.gemspec +11 -11
  320. data/spec/middleman-core/dns_resolver_spec.rb +118 -0
  321. data/spec/middleman-core/preview_server/server_hostname_spec.rb +39 -0
  322. data/spec/middleman-core/preview_server/server_ip_address_spec.rb +43 -0
  323. data/spec/middleman-core/util_spec.rb +111 -3
  324. data/spec/spec_helper.rb +29 -1
  325. data/spec/support/given.rb +42 -0
  326. metadata +528 -221
  327. data/features/twitter-bootstrap-compile.feature +0 -6
  328. data/features/working_directory.feature +0 -33
  329. data/fixtures/collections-app/config.rb +0 -16
  330. data/fixtures/collections-app/source/index.html.erb +0 -26
  331. data/fixtures/twitter-bootstrap-app/source/twitter-bootstrap-ebc6935/.gitignore +0 -36
  332. data/fixtures/twitter-bootstrap-app/source/twitter-bootstrap-ebc6935/.travis.yml +0 -3
  333. data/fixtures/twitter-bootstrap-app/source/twitter-bootstrap-ebc6935/LICENSE +0 -176
  334. data/fixtures/twitter-bootstrap-app/source/twitter-bootstrap-ebc6935/README.md +0 -137
  335. data/fixtures/twitter-bootstrap-app/source/twitter-bootstrap-ebc6935/img/glyphicons-halflings-white.png +0 -0
  336. data/fixtures/twitter-bootstrap-app/source/twitter-bootstrap-ebc6935/img/glyphicons-halflings.png +0 -0
  337. data/fixtures/twitter-bootstrap-app/source/twitter-bootstrap-ebc6935/js/.jshintrc +0 -10
  338. data/fixtures/twitter-bootstrap-app/source/twitter-bootstrap-ebc6935/js/README.md +0 -112
  339. data/fixtures/twitter-bootstrap-app/source/twitter-bootstrap-ebc6935/js/bootstrap-alert.js +0 -90
  340. data/fixtures/twitter-bootstrap-app/source/twitter-bootstrap-ebc6935/js/bootstrap-button.js +0 -96
  341. data/fixtures/twitter-bootstrap-app/source/twitter-bootstrap-ebc6935/js/bootstrap-carousel.js +0 -169
  342. data/fixtures/twitter-bootstrap-app/source/twitter-bootstrap-ebc6935/js/bootstrap-collapse.js +0 -157
  343. data/fixtures/twitter-bootstrap-app/source/twitter-bootstrap-ebc6935/js/bootstrap-dropdown.js +0 -100
  344. data/fixtures/twitter-bootstrap-app/source/twitter-bootstrap-ebc6935/js/bootstrap-modal.js +0 -218
  345. data/fixtures/twitter-bootstrap-app/source/twitter-bootstrap-ebc6935/js/bootstrap-popover.js +0 -98
  346. data/fixtures/twitter-bootstrap-app/source/twitter-bootstrap-ebc6935/js/bootstrap-scrollspy.js +0 -151
  347. data/fixtures/twitter-bootstrap-app/source/twitter-bootstrap-ebc6935/js/bootstrap-tab.js +0 -135
  348. data/fixtures/twitter-bootstrap-app/source/twitter-bootstrap-ebc6935/js/bootstrap-tooltip.js +0 -275
  349. data/fixtures/twitter-bootstrap-app/source/twitter-bootstrap-ebc6935/js/bootstrap-transition.js +0 -61
  350. data/fixtures/twitter-bootstrap-app/source/twitter-bootstrap-ebc6935/js/bootstrap-typeahead.js +0 -285
  351. data/fixtures/twitter-bootstrap-app/source/twitter-bootstrap-ebc6935/js/tests/index.html +0 -54
  352. data/fixtures/twitter-bootstrap-app/source/twitter-bootstrap-ebc6935/js/tests/phantom.js +0 -63
  353. data/fixtures/twitter-bootstrap-app/source/twitter-bootstrap-ebc6935/js/tests/server.js +0 -14
  354. data/fixtures/twitter-bootstrap-app/source/twitter-bootstrap-ebc6935/js/tests/unit/bootstrap-alert.js +0 -56
  355. data/fixtures/twitter-bootstrap-app/source/twitter-bootstrap-ebc6935/js/tests/unit/bootstrap-button.js +0 -77
  356. data/fixtures/twitter-bootstrap-app/source/twitter-bootstrap-ebc6935/js/tests/unit/bootstrap-carousel.js +0 -28
  357. data/fixtures/twitter-bootstrap-app/source/twitter-bootstrap-ebc6935/js/tests/unit/bootstrap-collapse.js +0 -54
  358. data/fixtures/twitter-bootstrap-app/source/twitter-bootstrap-ebc6935/js/tests/unit/bootstrap-dropdown.js +0 -87
  359. data/fixtures/twitter-bootstrap-app/source/twitter-bootstrap-ebc6935/js/tests/unit/bootstrap-modal.js +0 -114
  360. data/fixtures/twitter-bootstrap-app/source/twitter-bootstrap-ebc6935/js/tests/unit/bootstrap-phantom.js +0 -21
  361. data/fixtures/twitter-bootstrap-app/source/twitter-bootstrap-ebc6935/js/tests/unit/bootstrap-popover.js +0 -93
  362. data/fixtures/twitter-bootstrap-app/source/twitter-bootstrap-ebc6935/js/tests/unit/bootstrap-scrollspy.js +0 -31
  363. data/fixtures/twitter-bootstrap-app/source/twitter-bootstrap-ebc6935/js/tests/unit/bootstrap-tab.js +0 -61
  364. data/fixtures/twitter-bootstrap-app/source/twitter-bootstrap-ebc6935/js/tests/unit/bootstrap-tooltip.js +0 -155
  365. data/fixtures/twitter-bootstrap-app/source/twitter-bootstrap-ebc6935/js/tests/unit/bootstrap-transition.js +0 -13
  366. data/fixtures/twitter-bootstrap-app/source/twitter-bootstrap-ebc6935/js/tests/unit/bootstrap-typeahead.js +0 -148
  367. data/fixtures/twitter-bootstrap-app/source/twitter-bootstrap-ebc6935/js/tests/vendor/jquery.js +0 -9252
  368. data/fixtures/twitter-bootstrap-app/source/twitter-bootstrap-ebc6935/js/tests/vendor/qunit.css +0 -232
  369. data/fixtures/twitter-bootstrap-app/source/twitter-bootstrap-ebc6935/js/tests/vendor/qunit.js +0 -1510
  370. data/fixtures/twitter-bootstrap-app/source/twitter-bootstrap-ebc6935/less/_accordion.less +0 -33
  371. data/fixtures/twitter-bootstrap-app/source/twitter-bootstrap-ebc6935/less/_alerts.less +0 -58
  372. data/fixtures/twitter-bootstrap-app/source/twitter-bootstrap-ebc6935/less/_breadcrumbs.less +0 -24
  373. data/fixtures/twitter-bootstrap-app/source/twitter-bootstrap-ebc6935/less/_button-groups.less +0 -191
  374. data/fixtures/twitter-bootstrap-app/source/twitter-bootstrap-ebc6935/less/_buttons.less +0 -191
  375. data/fixtures/twitter-bootstrap-app/source/twitter-bootstrap-ebc6935/less/_carousel.less +0 -121
  376. data/fixtures/twitter-bootstrap-app/source/twitter-bootstrap-ebc6935/less/_close.less +0 -29
  377. data/fixtures/twitter-bootstrap-app/source/twitter-bootstrap-ebc6935/less/_code.less +0 -57
  378. data/fixtures/twitter-bootstrap-app/source/twitter-bootstrap-ebc6935/less/_component-animations.less +0 -20
  379. data/fixtures/twitter-bootstrap-app/source/twitter-bootstrap-ebc6935/less/_dropdowns.less +0 -143
  380. data/fixtures/twitter-bootstrap-app/source/twitter-bootstrap-ebc6935/less/_forms.less +0 -583
  381. data/fixtures/twitter-bootstrap-app/source/twitter-bootstrap-ebc6935/less/_grid.less +0 -5
  382. data/fixtures/twitter-bootstrap-app/source/twitter-bootstrap-ebc6935/less/_hero-unit.less +0 -22
  383. data/fixtures/twitter-bootstrap-app/source/twitter-bootstrap-ebc6935/less/_labels-badges.less +0 -55
  384. data/fixtures/twitter-bootstrap-app/source/twitter-bootstrap-ebc6935/less/_layouts.less +0 -17
  385. data/fixtures/twitter-bootstrap-app/source/twitter-bootstrap-ebc6935/less/_mixins.less +0 -646
  386. data/fixtures/twitter-bootstrap-app/source/twitter-bootstrap-ebc6935/less/_modals.less +0 -90
  387. data/fixtures/twitter-bootstrap-app/source/twitter-bootstrap-ebc6935/less/_navbar.less +0 -358
  388. data/fixtures/twitter-bootstrap-app/source/twitter-bootstrap-ebc6935/less/_navs.less +0 -363
  389. data/fixtures/twitter-bootstrap-app/source/twitter-bootstrap-ebc6935/less/_pager.less +0 -36
  390. data/fixtures/twitter-bootstrap-app/source/twitter-bootstrap-ebc6935/less/_pagination.less +0 -56
  391. data/fixtures/twitter-bootstrap-app/source/twitter-bootstrap-ebc6935/less/_popovers.less +0 -49
  392. data/fixtures/twitter-bootstrap-app/source/twitter-bootstrap-ebc6935/less/_progress-bars.less +0 -117
  393. data/fixtures/twitter-bootstrap-app/source/twitter-bootstrap-ebc6935/less/_reset.less +0 -131
  394. data/fixtures/twitter-bootstrap-app/source/twitter-bootstrap-ebc6935/less/_responsive-1200px-min.less +0 -26
  395. data/fixtures/twitter-bootstrap-app/source/twitter-bootstrap-ebc6935/less/_responsive-767px-max.less +0 -149
  396. data/fixtures/twitter-bootstrap-app/source/twitter-bootstrap-ebc6935/less/_responsive-768px-979px.less +0 -17
  397. data/fixtures/twitter-bootstrap-app/source/twitter-bootstrap-ebc6935/less/_responsive-navbar.less +0 -153
  398. data/fixtures/twitter-bootstrap-app/source/twitter-bootstrap-ebc6935/less/_responsive-utilities.less +0 -41
  399. data/fixtures/twitter-bootstrap-app/source/twitter-bootstrap-ebc6935/less/_scaffolding.less +0 -29
  400. data/fixtures/twitter-bootstrap-app/source/twitter-bootstrap-ebc6935/less/_sprites.less +0 -191
  401. data/fixtures/twitter-bootstrap-app/source/twitter-bootstrap-ebc6935/less/_tables.less +0 -176
  402. data/fixtures/twitter-bootstrap-app/source/twitter-bootstrap-ebc6935/less/_thumbnails.less +0 -47
  403. data/fixtures/twitter-bootstrap-app/source/twitter-bootstrap-ebc6935/less/_tooltip.less +0 -35
  404. data/fixtures/twitter-bootstrap-app/source/twitter-bootstrap-ebc6935/less/_type.less +0 -232
  405. data/fixtures/twitter-bootstrap-app/source/twitter-bootstrap-ebc6935/less/_utilities.less +0 -23
  406. data/fixtures/twitter-bootstrap-app/source/twitter-bootstrap-ebc6935/less/_variables.less +0 -206
  407. data/fixtures/twitter-bootstrap-app/source/twitter-bootstrap-ebc6935/less/_wells.less +0 -27
  408. data/fixtures/twitter-bootstrap-app/source/twitter-bootstrap-ebc6935/less/bootstrap.less +0 -62
  409. data/fixtures/twitter-bootstrap-app/source/twitter-bootstrap-ebc6935/less/responsive.less +0 -48
  410. data/fixtures/twitter-bootstrap-app/source/twitter-bootstrap-ebc6935/less/tests/css-tests.css +0 -52
  411. data/fixtures/twitter-bootstrap-app/source/twitter-bootstrap-ebc6935/less/tests/css-tests.html +0 -917
  412. data/fixtures/twitter-bootstrap-app/source/twitter-bootstrap-ebc6935/less/tests/forms.html +0 -179
  413. data/fixtures/twitter-bootstrap-app/source/twitter-bootstrap-ebc6935/less/tests/navbar.html +0 -108
  414. data/fixtures/twitter-bootstrap-app/source/twitter-bootstrap-ebc6935/package.json +0 -25
  415. data/lib/middleman-core/cli/server.rb +0 -96
@@ -0,0 +1,42 @@
1
+ require 'middleman-core/dns_resolver/basic_network_resolver'
2
+
3
+ module Middleman
4
+ class DnsResolver
5
+ # Use network name server to resolve ips and names
6
+ class NetworkResolver < BasicNetworkResolver
7
+ def initialize(opts={})
8
+ super
9
+
10
+ @resolver = opts.fetch(:resolver, Resolv::DNS.new(nameserver_config))
11
+ self.timeouts = timeouts
12
+ end
13
+
14
+ private
15
+
16
+ # Hosts + Ports for MDNS resolver
17
+ #
18
+ # This looks for MM_MDNSRC in your environment. If you are going to use
19
+ # IPv6-addresses: Make sure you do not forget to add the port at the end.
20
+ #
21
+ # MM_MDNSRC=ip:port ip:port
22
+ #
23
+ # @return [Hash]
24
+ # Returns the configuration for the nameserver
25
+ #
26
+ # @example
27
+ # export MM_MDNSRC="224.0.0.251:5353 ff02::fb:5353"
28
+ #
29
+ def nameserver_config
30
+ return unless ENV.key?('MM_DNSRC') && ENV['MM_DNSRC']
31
+
32
+ address, port = ENV['MM_DNSRC'].split(/:/)
33
+
34
+ {
35
+ nameserver_port: [[address, port.to_i]]
36
+ }
37
+ rescue StandardError
38
+ {}
39
+ end
40
+ end
41
+ end
42
+ end
@@ -61,7 +61,7 @@ module Middleman
61
61
  # * `app.before_render {|body, path, locs, template_class| ... }` - Manipulate template sources before they are rendered.
62
62
  # * `app.after_render {|content, path, locs, template_class| ... }` - Manipulate output text after a template has been rendered. It is also common to install a Rack middleware to do this instead.
63
63
  # * `app.ready { ... }` - Run code once Middleman is ready to serve or build files (after `after_configuration`).
64
- # * `app.compass_config { |compass_config| ... }` - Manipulate the Compass configuration after it has been set up.
64
+
65
65
  #
66
66
  # @see http://middlemanapp.com/advanced/custom/ Middleman Custom Extensions Documentation
67
67
  class Extension
@@ -3,7 +3,7 @@ require 'middleman-core/util'
3
3
  require 'middleman-core/rack'
4
4
 
5
5
  class Middleman::Extensions::AssetHash < ::Middleman::Extension
6
- option :exts, %w(.jpg .jpeg .png .gif .webp .js .css .otf .woff .woff2 .eot .ttf .svg), 'List of extensions that get asset hashes appended to them.'
6
+ option :exts, %w(.jpg .jpeg .png .gif .webp .js .css .otf .woff .woff2 .eot .ttf .svg .svgz), 'List of extensions that get asset hashes appended to them.'
7
7
  option :ignore, [], 'Regexes of filenames to skip adding asset hashes to'
8
8
 
9
9
  def initialize(app, options_hash={}, &block)
@@ -17,10 +17,10 @@ class Middleman::Extensions::AssetHash < ::Middleman::Extension
17
17
  def after_configuration
18
18
  # Allow specifying regexes to ignore, plus always ignore apple touch icons
19
19
  @ignore = Array(options.ignore) + [/^apple-touch-icon/]
20
-
20
+
21
21
  app.use ::Middleman::Middleware::InlineURLRewriter,
22
22
  id: :asset_hash,
23
- url_extensions: options.exts,
23
+ url_extensions: options.exts.sort.reverse,
24
24
  source_extensions: %w(.htm .html .php .css .js),
25
25
  ignore: @ignore,
26
26
  middleman_app: app,
@@ -23,7 +23,15 @@ class Middleman::Extensions::AutomaticImageSizes < ::Middleman::Extension
23
23
 
24
24
  if file && file[:full_path].exist?
25
25
  begin
26
- width, height = ::FastImage.size(file[:full_path].to_s, raise_on_failure: true)
26
+ full_path = file[:full_path].to_s
27
+ width, height = ::FastImage.size(full_path, raise_on_failure: true)
28
+ # Check for @2x and @3x image
29
+ retina = full_path.match(/@(\d)x\.[a-zA-Z]{3,4}$/)
30
+ if retina
31
+ factor = retina[1].to_i
32
+ width /= factor
33
+ height /= factor
34
+ end
27
35
  params[:width] = width
28
36
  params[:height] = height
29
37
  rescue FastImage::UnknownImageType
@@ -8,12 +8,16 @@ class Middleman::Extensions::MinifyCss < ::Middleman::Extension
8
8
  require 'sass'
9
9
  SassCompressor
10
10
  }, 'Set the CSS compressor to use.'
11
+ option :content_types, %w(text/css), 'Content types of resources that contain CSS'
12
+ option :inline_content_types, %w(text/html text/php), 'Content types of resources that contain inline CSS'
11
13
 
12
14
  def ready
13
15
  # Setup Rack middleware to minify CSS
14
16
  app.use Rack, compressor: options[:compressor],
15
17
  ignore: Array(options[:ignore]) + [/\.min\./],
16
- inline: options[:inline]
18
+ inline: options[:inline],
19
+ content_types: options[:content_types],
20
+ inline_content_types: options[:inline_content_types]
17
21
  end
18
22
 
19
23
  class SassCompressor
@@ -45,6 +49,8 @@ class Middleman::Extensions::MinifyCss < ::Middleman::Extension
45
49
  @compressor = options.fetch(:compressor)
46
50
  @compressor = @compressor.to_proc if @compressor.respond_to? :to_proc
47
51
  @compressor = @compressor.call if @compressor.is_a? Proc
52
+ @content_types = options[:content_types]
53
+ @inline_content_types = options[:inline_content_types]
48
54
  end
49
55
 
50
56
  # Rack interface
@@ -53,19 +59,18 @@ class Middleman::Extensions::MinifyCss < ::Middleman::Extension
53
59
  def call(env)
54
60
  status, headers, response = @app.call(env)
55
61
 
56
- if inline_html_content?(env['PATH_INFO'])
57
- minified = ::Middleman::Util.extract_response_text(response)
58
- minified.gsub!(INLINE_CSS_REGEX) do
59
- $1 << @compressor.compress($2) << $3
60
- end
62
+ content_type = headers['Content-Type'].try(:slice, /^[^;]*/)
63
+ path = env['PATH_INFO']
61
64
 
65
+ minified = if @inline && minifiable_inline?(content_type)
66
+ minify_inline(::Middleman::Util.extract_response_text(response))
67
+ elsif minifiable?(content_type) && !ignore?(path)
68
+ minify(::Middleman::Util.extract_response_text(response))
69
+ end
70
+
71
+ if minified
62
72
  headers['Content-Length'] = ::Rack::Utils.bytesize(minified).to_s
63
73
  response = [minified]
64
- elsif standalone_css_content?(env['PATH_INFO'])
65
- minified_css = @compressor.compress(::Middleman::Util.extract_response_text(response))
66
-
67
- headers['Content-Length'] = ::Rack::Utils.bytesize(minified_css).to_s
68
- response = [minified_css]
69
74
  end
70
75
 
71
76
  [status, headers, response]
@@ -73,14 +78,41 @@ class Middleman::Extensions::MinifyCss < ::Middleman::Extension
73
78
 
74
79
  private
75
80
 
76
- Contract String => Bool
77
- def inline_html_content?(path)
78
- (path.end_with?('.html') || path.end_with?('.php')) && @inline
81
+ # Whether the path should be ignored
82
+ # @param [String] path
83
+ # @return [Boolean]
84
+ def ignore?(path)
85
+ @ignore.any? { |ignore| Middleman::Util.path_match(ignore, path) }
86
+ end
87
+
88
+ # Whether this type of content can be minified
89
+ # @param [String, nil] content_type
90
+ # @return [Boolean]
91
+ def minifiable?(content_type)
92
+ @content_types.include?(content_type)
79
93
  end
80
94
 
81
- Contract String => Bool
82
- def standalone_css_content?(path)
83
- path.end_with?('.css') && @ignore.none? { |ignore| Middleman::Util.path_match(ignore, path) }
95
+ # Whether this type of content contains inline content that can be minified
96
+ # @param [String, nil] content_type
97
+ # @return [Boolean]
98
+ def minifiable_inline?(content_type)
99
+ @inline_content_types.include?(content_type)
100
+ end
101
+
102
+ # Minify the content
103
+ # @param [String] content
104
+ # @return [String]
105
+ def minify(content)
106
+ @compressor.compress(content)
107
+ end
108
+
109
+ # Detect and minify inline content
110
+ # @param [String] content
111
+ # @return [String]
112
+ def minify_inline(content)
113
+ content.gsub(INLINE_CSS_REGEX) do
114
+ $1 + minify($2) + $3
115
+ end
84
116
  end
85
117
  end
86
118
  end
@@ -8,17 +8,22 @@ class Middleman::Extensions::MinifyJavascript < ::Middleman::Extension
8
8
  require 'uglifier'
9
9
  ::Uglifier.new
10
10
  }, 'Set the JS compressor to use.'
11
+ option :content_types, %w(application/javascript), 'Content types of resources that contain JS'
12
+ option :inline_content_types, %w(text/html text/php), 'Content types of resources that contain inline JS'
11
13
 
12
14
  def ready
13
- # Setup Rack middleware to minify CSS
15
+ # Setup Rack middleware to minify JS
14
16
  app.use Rack, compressor: options[:compressor],
15
17
  ignore: Array(options[:ignore]) + [/\.min\./],
16
- inline: options[:inline]
18
+ inline: options[:inline],
19
+ content_types: options[:content_types],
20
+ inline_content_types: options[:inline_content_types]
17
21
  end
18
22
 
19
23
  # Rack middleware to look for JS and compress it
20
24
  class Rack
21
25
  include Contracts
26
+ INLINE_JS_REGEX = /(<script[^>]*>\s*(?:\/\/(?:(?:<!--)|(?:<!\[CDATA\[))\n)?)(.*?)((?:(?:\n\s*)?\/\/(?:(?:-->)|(?:\]\]>)))?\s*<\/script>)/m
22
27
 
23
28
  # Init
24
29
  # @param [Class] app
@@ -36,6 +41,8 @@ class Middleman::Extensions::MinifyJavascript < ::Middleman::Extension
36
41
  @compressor = options.fetch(:compressor)
37
42
  @compressor = @compressor.to_proc if @compressor.respond_to? :to_proc
38
43
  @compressor = @compressor.call if @compressor.is_a? Proc
44
+ @content_types = options[:content_types]
45
+ @inline_content_types = options[:inline_content_types]
39
46
  end
40
47
 
41
48
  # Rack interface
@@ -44,25 +51,18 @@ class Middleman::Extensions::MinifyJavascript < ::Middleman::Extension
44
51
  def call(env)
45
52
  status, headers, response = @app.call(env)
46
53
 
47
- path = env['PATH_INFO']
54
+ type = headers['Content-Type'].try(:slice, /^[^;]*/)
55
+ @path = env['PATH_INFO']
48
56
 
49
- begin
50
- if @inline && (path.end_with?('.html') || path.end_with?('.php'))
51
- uncompressed_source = ::Middleman::Util.extract_response_text(response)
52
-
53
- minified = minify_inline_content(uncompressed_source)
54
-
55
- headers['Content-Length'] = ::Rack::Utils.bytesize(minified).to_s
56
- response = [minified]
57
- elsif path.end_with?('.js') && @ignore.none? { |ignore| Middleman::Util.path_match(ignore, path) }
58
- uncompressed_source = ::Middleman::Util.extract_response_text(response)
59
- minified = @compressor.compress(uncompressed_source)
57
+ minified = if @inline && minifiable_inline?(type)
58
+ minify_inline(::Middleman::Util.extract_response_text(response))
59
+ elsif minifiable?(type) && !ignore?(@path)
60
+ minify(::Middleman::Util.extract_response_text(response))
61
+ end
60
62
 
61
- headers['Content-Length'] = ::Rack::Utils.bytesize(minified).to_s
62
- response = [minified]
63
- end
64
- rescue ExecJS::ProgramError => e
65
- warn "WARNING: Couldn't compress JavaScript in #{path}: #{e.message}"
63
+ if minified
64
+ headers['Content-Length'] = ::Rack::Utils.bytesize(minified).to_s
65
+ response = [minified]
66
66
  end
67
67
 
68
68
  [status, headers, response]
@@ -70,20 +70,50 @@ class Middleman::Extensions::MinifyJavascript < ::Middleman::Extension
70
70
 
71
71
  private
72
72
 
73
- Contract String => String
74
- def minify_inline_content(uncompressed_source)
75
- uncompressed_source.gsub(/(<script[^>]*>\s*(?:\/\/(?:(?:<!--)|(?:<!\[CDATA\[))\n)?)(.*?)((?:(?:\n\s*)?\/\/(?:(?:-->)|(?:\]\]>)))?\s*<\/script>)/m) do |match|
73
+ # Whether the path should be ignored
74
+ # @param [String] path
75
+ # @return [Boolean]
76
+ def ignore?(path)
77
+ @ignore.any? { |ignore| Middleman::Util.path_match(ignore, path) }
78
+ end
79
+
80
+ # Whether this type of content can be minified
81
+ # @param [String, nil] content_type
82
+ # @return [Boolean]
83
+ def minifiable?(content_type)
84
+ @content_types.include?(content_type)
85
+ end
86
+
87
+ # Whether this type of content contains inline content that can be minified
88
+ # @param [String, nil] content_type
89
+ # @return [Boolean]
90
+ def minifiable_inline?(content_type)
91
+ @inline_content_types.include?(content_type)
92
+ end
93
+
94
+ # Minify the content
95
+ # @param [String] content
96
+ # @return [String]
97
+ def minify(content)
98
+ @compressor.compress(content)
99
+ rescue ExecJS::ProgramError => e
100
+ warn "WARNING: Couldn't compress JavaScript in #{@path}: #{e.message}"
101
+ content
102
+ end
103
+
104
+ # Detect and minify inline content
105
+ # @param [String] content
106
+ # @return [String]
107
+ def minify_inline(content)
108
+ content.gsub(INLINE_JS_REGEX) do |match|
76
109
  first = $1
77
- javascript = $2
110
+ inline_content = $2
78
111
  last = $3
79
112
 
80
- # Only compress script tags that contain JavaScript (as opposed
81
- # to something like jQuery templates, identified with a "text/html"
82
- # type.
83
- if first =~ /<script>/ || first.include?('text/javascript')
84
- minified_js = @compressor.compress(javascript)
85
-
86
- first << minified_js << last
113
+ # Only compress script tags that contain JavaScript (as opposed to
114
+ # something like jQuery templates, identified with a "text/html" type).
115
+ if first.include?('<script>') || first.include?('text/javascript')
116
+ first + minify(inline_content) + last
87
117
  else
88
118
  match
89
119
  end
@@ -22,6 +22,19 @@ class Middleman::Extensions::RelativeAssets < ::Middleman::Extension
22
22
  proc: method(:rewrite_url)
23
23
  end
24
24
 
25
+ helpers do
26
+ # asset_url override for relative assets
27
+ # @param [String] path
28
+ # @param [String] prefix
29
+ # @param [Hash] options Additional options.
30
+ # @return [String]
31
+ def asset_url(path, prefix='', options={})
32
+ options[:relative] = true unless options.key?(:relative)
33
+
34
+ super(path, prefix, options)
35
+ end
36
+ end
37
+
25
38
  Contract String, Or[String, Pathname], Any => Maybe[String]
26
39
  def rewrite_url(asset_path, dirpath, request_path)
27
40
  uri = ::Addressable::URI.parse(asset_path)
@@ -29,7 +29,7 @@ module Middleman
29
29
  # @param [Class] context
30
30
  # @return [String]
31
31
  Contract Hash, Hash, Any, Maybe[Proc] => String
32
- def render(locs={}, opts={}, context, &block)
32
+ def render(locs, opts, context, &block)
33
33
  path = @path.dup
34
34
 
35
35
  # Detect the remdering engine from the extension
@@ -97,7 +97,7 @@ module Middleman
97
97
  Contract String
98
98
  def template_data_for_file
99
99
  if @app.extensions[:front_matter]
100
- @app.extensions[:front_matter].template_data_for_file(@path) || ''
100
+ @app.extensions[:front_matter].template_data_for_file(@path) || File.read(@path)
101
101
  else
102
102
  file = @app.files.find(:source, @path)
103
103
  file.read if file
@@ -95,7 +95,7 @@ module Middleman
95
95
  # Render a template with the given name and locals
96
96
  def template(template_name, locals={})
97
97
  template_path = File.join(File.dirname(__FILE__), 'meta_pages', 'templates', template_name)
98
- content = Tilt.new(template_path).render(nil, locals)
98
+ content = Tilt.new(template_path).render(::Object.new, locals)
99
99
  response(content)
100
100
  end
101
101
 
@@ -1,32 +1,50 @@
1
1
  require 'webrick'
2
2
  require 'webrick/https'
3
3
  require 'openssl'
4
- require 'socket'
5
4
  require 'middleman-core/meta_pages'
6
5
  require 'middleman-core/logger'
7
6
  require 'middleman-core/rack'
7
+ require 'middleman-core/preview_server/server_information'
8
+ require 'middleman-core/preview_server/server_url'
9
+ require 'middleman-core/preview_server/server_information_callback_proxy'
8
10
 
9
11
  # rubocop:disable GlobalVars
10
12
  module Middleman
11
- module PreviewServer
13
+ class PreviewServer
12
14
  class << self
13
15
  extend Forwardable
14
16
 
15
- attr_reader :app, :host, :port, :ssl_certificate, :ssl_private_key
16
- def_delegator :app, :logger
17
-
18
- def https?
19
- @https
20
- end
17
+ attr_reader :app, :ssl_certificate, :ssl_private_key, :environment, :server_information
21
18
 
22
19
  # Start an instance of Middleman::Application
23
20
  # @return [void]
24
21
  def start(opts={})
22
+ # Do not buffer output, otherwise testing of output does not work
23
+ $stdout.sync = true
24
+ $stderr.sync = true
25
+
25
26
  @options = opts
27
+ @server_information = ServerInformation.new
28
+ @server_information.https = (@options[:https] == true)
29
+
30
+ # New app evaluates the middleman configuration. Since this can be
31
+ # invalid as well, we need to evaluate the configuration BEFORE
32
+ # checking for validity
33
+ the_app = initialize_new_app
34
+
35
+ # And now comes the check
36
+ unless server_information.valid?
37
+ $stderr.puts %(== Running Middleman failed: #{server_information.reason}. Please fix that and try again.)
38
+ exit 1
39
+ end
40
+
41
+ mount_instance(the_app)
26
42
 
27
- mount_instance(new_app)
28
- logger.info "== The Middleman is standing watch at #{uri}"
29
- logger.info "== Inspect your site configuration at #{uri + '__middleman'}"
43
+ app.logger.debug %(== Server information is provided by #{server_information.handler})
44
+ app.logger.debug %(== The Middleman is running in "#{environment}" environment)
45
+ app.logger.debug format('== The Middleman preview server is bound to %s', ServerUrl.new(hosts: server_information.listeners, port: server_information.port, https: server_information.https?).to_bind_addresses.join(', '))
46
+ app.logger.info format('== View your site at %s', ServerUrl.new(hosts: server_information.site_addresses, port: server_information.port, https: server_information.https?).to_urls.join(', '))
47
+ app.logger.info format('== Inspect your site configuration at %s', ServerUrl.new(hosts: server_information.site_addresses, port: server_information.port, https: server_information.https?).to_config_urls.join(', '))
30
48
 
31
49
  @initialized ||= false
32
50
  return if @initialized
@@ -38,6 +56,21 @@ module Middleman
38
56
  # reloading later on.
39
57
  ::Middleman::Profiling.report('server_start')
40
58
 
59
+ app.execute_callbacks(:before_server, [ServerInformationCallbackProxy.new(server_information)])
60
+
61
+ if @options[:daemon]
62
+ # To output the child PID, let's make preview server a daemon by hand
63
+ if child_pid = fork
64
+ app.logger.info "== Middleman preview server is running in background with PID #{child_pid}"
65
+ Process.detach child_pid
66
+ exit 0
67
+ else
68
+ $stdout.reopen('/dev/null', 'w')
69
+ $stderr.reopen('/dev/null', 'w')
70
+ $stdin.reopen('/dev/null', 'r')
71
+ end
72
+ end
73
+
41
74
  loop do
42
75
  @webrick.start
43
76
 
@@ -56,7 +89,7 @@ module Middleman
56
89
  # @return [void]
57
90
  def stop
58
91
  begin
59
- logger.info '== The Middleman is shutting down'
92
+ app.logger.info '== The Middleman is shutting down'
60
93
  rescue
61
94
  # if the user closed their terminal STDOUT/STDERR won't exist
62
95
  end
@@ -67,13 +100,13 @@ module Middleman
67
100
  # Simply stop, then start the server
68
101
  # @return [void]
69
102
  def reload
70
- logger.info '== The Middleman is reloading'
103
+ app.logger.info '== The Middleman is reloading'
71
104
 
72
105
  begin
73
- app = new_app
106
+ app = initialize_new_app
74
107
  rescue => e
75
- logger.error "Error reloading Middleman: #{e}\n#{e.backtrace.join("\n")}"
76
- logger.info '== The Middleman is still running the application from before the error'
108
+ $stderr.puts "Error reloading Middleman: #{e}\n#{e.backtrace.join("\n")}"
109
+ app.logger.info '== The Middleman is still running the application from before the error'
77
110
  return
78
111
  end
79
112
 
@@ -84,7 +117,7 @@ module Middleman
84
117
 
85
118
  mount_instance(app)
86
119
 
87
- logger.info '== The Middleman has reloaded'
120
+ app.logger.info '== The Middleman has reloaded'
88
121
  end
89
122
 
90
123
  # Stop the current instance, exit Webrick
@@ -96,7 +129,7 @@ module Middleman
96
129
 
97
130
  private
98
131
 
99
- def new_app
132
+ def initialize_new_app
100
133
  opts = @options.dup
101
134
 
102
135
  ::Middleman::Logger.singleton(
@@ -110,8 +143,10 @@ module Middleman
110
143
  config[:watcher_force_polling] = opts[:force_polling]
111
144
  config[:watcher_latency] = opts[:latency]
112
145
 
113
- config[:host] = opts[:host] if opts[:host]
114
146
  config[:port] = opts[:port] if opts[:port]
147
+ config[:bind_address] = opts[:bind_address]
148
+ config[:server_name] = opts[:server_name]
149
+ config[:https] = opts[:https] unless opts[:https].nil?
115
150
  config[:ssl_certificate] = opts[:ssl_certificate] if opts[:ssl_certificate]
116
151
  config[:ssl_private_key] = opts[:ssl_private_key] if opts[:ssl_private_key]
117
152
 
@@ -120,13 +155,13 @@ module Middleman
120
155
  %r{^config\.rb$},
121
156
  %r{^environments/[^\.](.*)\.rb$},
122
157
  %r{^lib/[^\.](.*)\.rb$},
123
- %r{^#{@app.config[:helpers_dir]}/[^\.](.*)\.rb$}
158
+ %r{^#{config[:helpers_dir]}/[^\.](.*)\.rb$}
124
159
  ]
125
160
 
126
161
  # config.rb
127
162
  watcher = files.watch :reload,
128
- path: root,
129
- only: match_against
163
+ path: root,
164
+ only: match_against
130
165
 
131
166
  # Hack around node_modules in root.
132
167
  watcher.listener.ignore(/^node_modules/)
@@ -139,9 +174,16 @@ module Middleman
139
174
  end
140
175
  end
141
176
 
142
- @host = app.config[:host]
143
- @port = app.config[:port]
144
- @https = app.config[:https]
177
+ # store configured port to make a check later on possible
178
+ configured_port = app.config[:port]
179
+
180
+ # Use configuration values to set `bind_address` etc. in
181
+ # `server_information`
182
+ server_information.use app.config
183
+
184
+ app.logger.warn format('== The Middleman uses a different port "%s" then the configured one "%s" because some other server is listening on that port.', server_information.port, configured_port) unless app.config[:port] == configured_port
185
+
186
+ @environment = app.config[:environment]
145
187
 
146
188
  @ssl_certificate = app.config[:ssl_certificate]
147
189
  @ssl_private_key = app.config[:ssl_private_key]
@@ -179,13 +221,14 @@ module Middleman
179
221
  # @return [void]
180
222
  def setup_webrick(is_logging)
181
223
  http_opts = {
182
- BindAddress: host,
183
- Port: port,
224
+ Port: server_information.port,
184
225
  AccessLog: [],
226
+ ServerName: server_information.server_name,
227
+ BindAddress: server_information.bind_address.to_s,
185
228
  DoNotReverseLookup: true
186
229
  }
187
230
 
188
- if https?
231
+ if server_information.https?
189
232
  http_opts[:SSLEnable] = true
190
233
 
191
234
  if ssl_certificate || ssl_private_key
@@ -198,6 +241,9 @@ module Middleman
198
241
  %w(CN localhost),
199
242
  %w(CN #{host})
200
243
  ].uniq
244
+ cert, key = create_self_signed_cert(1024, [['CN', server_information.server_name]], server_information.site_addresses, 'Middleman Preview Server')
245
+ http_opts[:SSLCertificate] = cert
246
+ http_opts[:SSLPrivateKey] = key
201
247
  end
202
248
  end
203
249
 
@@ -210,11 +256,44 @@ module Middleman
210
256
  begin
211
257
  ::WEBrick::HTTPServer.new(http_opts)
212
258
  rescue Errno::EADDRINUSE
213
- logger.error "== Port #{port} is unavailable. Either close the instance of Middleman already running on #{port} or start this Middleman on a new port with: --port=#{unused_tcp_port}"
214
- exit(1)
259
+ $stderr.puts %(== Port "#{http_opts[:Port]}" is in use. This should not have happened. Please start "middleman server" again.)
215
260
  end
216
261
  end
217
262
 
263
+ # Copy of https://github.com/nahi/ruby/blob/webrick_trunk/lib/webrick/ssl.rb#L39
264
+ # that uses a different serial number each time the cert is generated in order to
265
+ # avoid errors in Firefox. Also doesn't print out stuff to $stderr unnecessarily.
266
+ def create_self_signed_cert(bits, cn, aliases, comment)
267
+ rsa = OpenSSL::PKey::RSA.new(bits)
268
+ cert = OpenSSL::X509::Certificate.new
269
+ cert.version = 2
270
+ cert.serial = Time.now.to_i % (1 << 20)
271
+ name = OpenSSL::X509::Name.new(cn)
272
+ cert.subject = name
273
+ cert.issuer = name
274
+ cert.not_before = Time.now
275
+ cert.not_after = Time.now + (365 * 24 * 60 * 60)
276
+ cert.public_key = rsa.public_key
277
+
278
+ ef = OpenSSL::X509::ExtensionFactory.new(nil, cert)
279
+ ef.issuer_certificate = cert
280
+ cert.extensions = [
281
+ ef.create_extension('basicConstraints', 'CA:FALSE'),
282
+ ef.create_extension('keyUsage', 'keyEncipherment'),
283
+ ef.create_extension('subjectKeyIdentifier', 'hash'),
284
+ ef.create_extension('extendedKeyUsage', 'serverAuth'),
285
+ ef.create_extension('nsComment', comment)
286
+ ]
287
+ aki = ef.create_extension('authorityKeyIdentifier',
288
+ 'keyid:always,issuer:always')
289
+ cert.add_extension(aki)
290
+ cert.add_extension ef.create_extension('subjectAltName', aliases.map { |d| "DNS: #{d}" }.join(','))
291
+
292
+ cert.sign(rsa, OpenSSL::Digest::SHA1.new)
293
+
294
+ [cert, rsa]
295
+ end
296
+
218
297
  # Attach a new Middleman::Application instance
219
298
  # @param [Middleman::Application] app
220
299
  # @return [void]
@@ -236,23 +315,6 @@ module Middleman
236
315
 
237
316
  @app = nil
238
317
  end
239
-
240
- # Returns the URI the preview server will run on
241
- # @return [URI]
242
- def uri
243
- host = (@host == '0.0.0.0') ? 'localhost' : @host
244
- scheme = https? ? 'https' : 'http'
245
- URI("#{scheme}://#{host}:#{@port}")
246
- end
247
-
248
- # Returns unused TCP port
249
- # @return [Fixnum]
250
- def unused_tcp_port
251
- server = TCPServer.open(0)
252
- port = server.addr[1]
253
- server.close
254
- port
255
- end
256
318
  end
257
319
 
258
320
  class FilteredWebrickLog < ::WEBrick::Log