├── .gitignore ├── CHANGELOG.md ├── LICENSE-2.0.txt ├── README.asciidoc ├── dev-resources ├── META-INF │ └── assets │ │ ├── aviso-logo.png │ │ ├── coffeescript-source.coffee │ │ ├── colors.less │ │ ├── common.jade │ │ ├── f1.txt │ │ ├── invalid-coffeescript.coffee │ │ ├── invalid-less.less │ │ ├── jade-helper.jade │ │ ├── jade-source.jade │ │ ├── jade-uris-helper.jade │ │ ├── logo.less │ │ ├── sample.less │ │ ├── stack │ │ ├── barney.js │ │ ├── bedrock.stack │ │ ├── bootstrap.js │ │ ├── bootstrap.stack │ │ ├── compiled.stack │ │ ├── fred.js │ │ ├── jquery.js │ │ ├── local.less │ │ ├── meta.stack │ │ ├── missing-component.stack │ │ ├── stack.coffee │ │ └── style.stack │ │ └── sub │ │ ├── f2.txt │ │ ├── f3.txt │ │ ├── jade-include.jade │ │ └── samedir.jade ├── example.clj ├── expected │ ├── bedrock.js │ ├── bootstrap-webjars.css │ ├── coffeescript-source.js │ ├── coffeescript-source.map │ ├── compiled-stack.js │ ├── jade-include.html │ ├── jade-source.html │ ├── jade-uris-helper.html │ ├── logo.css │ ├── meta.js │ ├── minimized │ │ ├── bootstrap.css │ │ ├── bootstrap.js │ │ ├── fred.js │ │ ├── meta.js │ │ └── sample.css │ ├── sample.css │ ├── sample.map │ └── style.css ├── logback-test.xml └── user.clj ├── java ├── com │ └── yahoo │ │ └── platform │ │ └── yui │ │ └── compressor │ │ └── CssCompressor.java └── io │ └── aviso │ └── twixt │ └── shims │ └── CompilerShim.java ├── manual ├── _template.html ├── dexy.yaml └── en │ ├── caching.ad │ ├── configuration.ad │ ├── cssmin.ad │ ├── direct-uris.ad │ ├── exceptions.ad │ ├── getting-started.ad │ ├── index.ad │ ├── jade.ad │ ├── jsmin.ad │ ├── meta.edn │ ├── notes.ad │ ├── stacks.ad │ ├── uris.ad │ └── webjars.ad ├── project.clj ├── resources └── META-INF │ ├── assets │ └── twixt │ │ ├── exception.coffee │ │ └── exception.less │ └── twixt │ ├── coffee-script.js │ └── invoke-coffeescript.js ├── spec └── io │ └── aviso │ ├── twixt_spec.clj │ └── utils_spec.clj └── src └── io └── aviso ├── twixt.clj └── twixt ├── asset.clj ├── coffee_script.clj ├── compress.clj ├── css_minification.clj ├── css_rewrite.clj ├── exceptions.clj ├── export.clj ├── fs_cache.clj ├── jade.clj ├── js_minification.clj ├── less.clj ├── memory_cache.clj ├── rhino.clj ├── ring.clj ├── schemas.clj ├── stacks.clj ├── startup.clj └── utils.clj /.gitignore: -------------------------------------------------------------------------------- 1 | target/ 2 | doc 3 | .lein-failures 4 | .DS_Store 5 | twixt.sublime-* 6 | pom.* 7 | .* 8 | *.iml 9 | .dexy 10 | -------------------------------------------------------------------------------- /CHANGELOG.md: -------------------------------------------------------------------------------- 1 | ## 0.1.25 - 21 Mar 2017 2 | 3 | Update to Clojure 1.8.0 4 | Fix bad ns declaration in io.aviso.twixt.fs-cache 5 | 6 | ## 0.1.24 - 18 Feb 2016 7 | 8 | Redeployed on Clojars, POM problematic in previous deploy. 9 | 10 | ## 0.1.23 - 18 Feb 2016 11 | 12 | Bugfixes 13 | - Not passing the merged 'twixt-options' to 'wrap-with-exporter' in 'startup/wrap-with-twixt' 14 | - Wrong arg passed to 'ex-info' in 'js-minification/minimize-javascript-asset' 15 | - [Issue 26](https://github.com/AvisoNovate/twixt/issues/26) 16 | 17 | ## 0.1.22 - 30 Sep 2015 18 | 19 | Updated a bunch of dependencies to latest version, most important: clojure 1.7 and schema 1.0.1 . 20 | 21 | ## 0.1.21 - 21 Aug 2015 22 | 23 | Ensured that small amount of Java in the project is compiled for JDK 1.6. 24 | 25 | Removed a few reflection warnings. 26 | 27 | Upgraded various dependencies to latest. 28 | 29 | The twixt helper provided to Jade templates now has a uris() method, which accepts an array of asset path strings. 30 | 31 | ## 0.1.20 - 12 Aug 2015 32 | 33 | Added support for CSS Compression (using YUICompressor). 34 | This is normally enabled only in production mode. 35 | 36 | Revised the approach to caching significantly. 37 | File-system caching is now enabled for both production and development, but uses 38 | different sub-directories of the specified root cache folder. 39 | 40 | There have been some non-backwards compatible changes to several functions as a result. 41 | 42 | In the Twixt options map, the :cache-folder key has been renamed to :cache-dir, and moved 43 | under a new key, :cache. 44 | 45 | The new caching and locking approach helps ensure that multiple threads do no attempt 46 | to perform the same compilation steps in parallel. 47 | There is now a per-asset lock used to prevent duplicate conflicting work across 48 | multiple threads. 49 | 50 | ## 0.1.19 - 7 Aug 2015 51 | 52 | Fixed bug where compressed assets would cause a double exception (an exception, caused by 53 | an exception building the exception report). 54 | 55 | ## 0.1.18 - 31 Jul 2015 56 | 57 | Reverted Clojure compatibility to 1.6.0. 58 | 59 | ## 0.1.17 - 29 Jul 2015 60 | 61 | Updated dependencies, including Clojure to 1.7.0, and CoffeeScript compiler to 1.9.3. 62 | 63 | Clarified many APIs by introducing Prismatic Schema signatures. 64 | 65 | Added support for finding assets inside [WebJars](http://www.webjars.org/). 66 | 67 | Twixt can now export assets to specific locations (typically, under public/resources). 68 | Exports are live: monitored for changes and dynamically re-exported as needed. 69 | This is intended largely to allow Less stylesheets to live-update when using 70 | [Figwheel](https://github.com/bhauman/lein-figwheel). 71 | 72 | ## 0.1.16 - 10 Jun 2015 73 | 74 | Update dependencies to latest. 75 | 76 | ## 0.1.15 - 7 Nov 2014 77 | 78 | Added medley (0.5.3) as a dependency. 79 | 80 | ## 0.1.14 - Oct 24 2014 81 | 82 | * In-line source maps for compiled Less files (in development mode) 83 | * Update Less4J dependency to 0.8.3 84 | 85 | No closed issues 86 | 87 | ## 0.1.13 - 6 Jun 2014 88 | 89 | * Source maps for CoffeeScript 90 | * Source maps for Less (partial, pending improvements to Less4J) 91 | * Support for stacks: aggregated assets that combine into a single virtual asset 92 | * JavaScript Minification (via Google Closure) 93 | 94 | [Closed Issues](https://github.com/AvisoNovate/twixt/issues?q=milestone%3A0.1.13) 95 | 96 | ## 0.1.12 - 29 Apr 2014 97 | 98 | * Minor bug fixes; some relative paths computed incorrectly 99 | 100 | [Closed Issues](https://github.com/AvisoNovate/twixt/issues?q=milestone%3A0.1.12) 101 | 102 | ## 0.1.11 - 13 Mar 2014 103 | 104 | * Adds support for Jade helpers and variables 105 | 106 | [Closed Issues](https://github.com/AvisoNovate/twixt/issues?q=milestone%3A0.1.11) 107 | 108 | ## 0.1.10 - 7 Mar 2014 109 | 110 | * Jade `include` directive supported, with dependency tracking 111 | 112 | [Closed Issues](https://github.com/AvisoNovate/twixt/issues?q=milestone%3A0.1.10) 113 | 114 | ## 0.1.9 - 26 Feb 2014 115 | 116 | * Exception report displays map keys in sorted order 117 | * Exception report displays some system properties as lists (e.g., `java.class.path`) 118 | 119 | [Closed Issues](https://github.com/AvisoNovate/twixt/issues?q=milestone%3A0.1.9) 120 | -------------------------------------------------------------------------------- /LICENSE-2.0.txt: -------------------------------------------------------------------------------- 1 | 2 | Apache License 3 | Version 2.0, January 2004 4 | http://www.apache.org/licenses/ 5 | 6 | TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION 7 | 8 | 1. Definitions. 9 | 10 | "License" shall mean the terms and conditions for use, reproduction, 11 | and distribution as defined by Sections 1 through 9 of this document. 12 | 13 | "Licensor" shall mean the copyright owner or entity authorized by 14 | the copyright owner that is granting the License. 15 | 16 | "Legal Entity" shall mean the union of the acting entity and all 17 | other entities that control, are controlled by, or are under common 18 | control with that entity. For the purposes of this definition, 19 | "control" means (i) the power, direct or indirect, to cause the 20 | direction or management of such entity, whether by contract or 21 | otherwise, or (ii) ownership of fifty percent (50%) or more of the 22 | outstanding shares, or (iii) beneficial ownership of such entity. 23 | 24 | "You" (or "Your") shall mean an individual or Legal Entity 25 | exercising permissions granted by this License. 26 | 27 | "Source" form shall mean the preferred form for making modifications, 28 | including but not limited to software source code, documentation 29 | source, and configuration files. 30 | 31 | "Object" form shall mean any form resulting from mechanical 32 | transformation or translation of a Source form, including but 33 | not limited to compiled object code, generated documentation, 34 | and conversions to other media types. 35 | 36 | "Work" shall mean the work of authorship, whether in Source or 37 | Object form, made available under the License, as indicated by a 38 | copyright notice that is included in or attached to the work 39 | (an example is provided in the Appendix below). 40 | 41 | "Derivative Works" shall mean any work, whether in Source or Object 42 | form, that is based on (or derived from) the Work and for which the 43 | editorial revisions, annotations, elaborations, or other modifications 44 | represent, as a whole, an original work of authorship. For the purposes 45 | of this License, Derivative Works shall not include works that remain 46 | separable from, or merely link (or bind by name) to the interfaces of, 47 | the Work and Derivative Works thereof. 48 | 49 | "Contribution" shall mean any work of authorship, including 50 | the original version of the Work and any modifications or additions 51 | to that Work or Derivative Works thereof, that is intentionally 52 | submitted to Licensor for inclusion in the Work by the copyright owner 53 | or by an individual or Legal Entity authorized to submit on behalf of 54 | the copyright owner. For the purposes of this definition, "submitted" 55 | means any form of electronic, verbal, or written communication sent 56 | to the Licensor or its representatives, including but not limited to 57 | communication on electronic mailing lists, source code control systems, 58 | and issue tracking systems that are managed by, or on behalf of, the 59 | Licensor for the purpose of discussing and improving the Work, but 60 | excluding communication that is conspicuously marked or otherwise 61 | designated in writing by the copyright owner as "Not a Contribution." 62 | 63 | "Contributor" shall mean Licensor and any individual or Legal Entity 64 | on behalf of whom a Contribution has been received by Licensor and 65 | subsequently incorporated within the Work. 66 | 67 | 2. Grant of Copyright License. Subject to the terms and conditions of 68 | this License, each Contributor hereby grants to You a perpetual, 69 | worldwide, non-exclusive, no-charge, royalty-free, irrevocable 70 | copyright license to reproduce, prepare Derivative Works of, 71 | publicly display, publicly perform, sublicense, and distribute the 72 | Work and such Derivative Works in Source or Object form. 73 | 74 | 3. Grant of Patent License. Subject to the terms and conditions of 75 | this License, each Contributor hereby grants to You a perpetual, 76 | worldwide, non-exclusive, no-charge, royalty-free, irrevocable 77 | (except as stated in this section) patent license to make, have made, 78 | use, offer to sell, sell, import, and otherwise transfer the Work, 79 | where such license applies only to those patent claims licensable 80 | by such Contributor that are necessarily infringed by their 81 | Contribution(s) alone or by combination of their Contribution(s) 82 | with the Work to which such Contribution(s) was submitted. If You 83 | institute patent litigation against any entity (including a 84 | cross-claim or counterclaim in a lawsuit) alleging that the Work 85 | or a Contribution incorporated within the Work constitutes direct 86 | or contributory patent infringement, then any patent licenses 87 | granted to You under this License for that Work shall terminate 88 | as of the date such litigation is filed. 89 | 90 | 4. Redistribution. You may reproduce and distribute copies of the 91 | Work or Derivative Works thereof in any medium, with or without 92 | modifications, and in Source or Object form, provided that You 93 | meet the following conditions: 94 | 95 | (a) You must give any other recipients of the Work or 96 | Derivative Works a copy of this License; and 97 | 98 | (b) You must cause any modified files to carry prominent notices 99 | stating that You changed the files; and 100 | 101 | (c) You must retain, in the Source form of any Derivative Works 102 | that You distribute, all copyright, patent, trademark, and 103 | attribution notices from the Source form of the Work, 104 | excluding those notices that do not pertain to any part of 105 | the Derivative Works; and 106 | 107 | (d) If the Work includes a "NOTICE" text file as part of its 108 | distribution, then any Derivative Works that You distribute must 109 | include a readable copy of the attribution notices contained 110 | within such NOTICE file, excluding those notices that do not 111 | pertain to any part of the Derivative Works, in at least one 112 | of the following places: within a NOTICE text file distributed 113 | as part of the Derivative Works; within the Source form or 114 | documentation, if provided along with the Derivative Works; or, 115 | within a display generated by the Derivative Works, if and 116 | wherever such third-party notices normally appear. The contents 117 | of the NOTICE file are for informational purposes only and 118 | do not modify the License. You may add Your own attribution 119 | notices within Derivative Works that You distribute, alongside 120 | or as an addendum to the NOTICE text from the Work, provided 121 | that such additional attribution notices cannot be construed 122 | as modifying the License. 123 | 124 | You may add Your own copyright statement to Your modifications and 125 | may provide additional or different license terms and conditions 126 | for use, reproduction, or distribution of Your modifications, or 127 | for any such Derivative Works as a whole, provided Your use, 128 | reproduction, and distribution of the Work otherwise complies with 129 | the conditions stated in this License. 130 | 131 | 5. Submission of Contributions. Unless You explicitly state otherwise, 132 | any Contribution intentionally submitted for inclusion in the Work 133 | by You to the Licensor shall be under the terms and conditions of 134 | this License, without any additional terms or conditions. 135 | Notwithstanding the above, nothing herein shall supersede or modify 136 | the terms of any separate license agreement you may have executed 137 | with Licensor regarding such Contributions. 138 | 139 | 6. Trademarks. This License does not grant permission to use the trade 140 | names, trademarks, service marks, or product names of the Licensor, 141 | except as required for reasonable and customary use in describing the 142 | origin of the Work and reproducing the content of the NOTICE file. 143 | 144 | 7. Disclaimer of Warranty. Unless required by applicable law or 145 | agreed to in writing, Licensor provides the Work (and each 146 | Contributor provides its Contributions) on an "AS IS" BASIS, 147 | WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or 148 | implied, including, without limitation, any warranties or conditions 149 | of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A 150 | PARTICULAR PURPOSE. You are solely responsible for determining the 151 | appropriateness of using or redistributing the Work and assume any 152 | risks associated with Your exercise of permissions under this License. 153 | 154 | 8. Limitation of Liability. In no event and under no legal theory, 155 | whether in tort (including negligence), contract, or otherwise, 156 | unless required by applicable law (such as deliberate and grossly 157 | negligent acts) or agreed to in writing, shall any Contributor be 158 | liable to You for damages, including any direct, indirect, special, 159 | incidental, or consequential damages of any character arising as a 160 | result of this License or out of the use or inability to use the 161 | Work (including but not limited to damages for loss of goodwill, 162 | work stoppage, computer failure or malfunction, or any and all 163 | other commercial damages or losses), even if such Contributor 164 | has been advised of the possibility of such damages. 165 | 166 | 9. Accepting Warranty or Additional Liability. While redistributing 167 | the Work or Derivative Works thereof, You may choose to offer, 168 | and charge a fee for, acceptance of support, warranty, indemnity, 169 | or other liability obligations and/or rights consistent with this 170 | License. However, in accepting such obligations, You may act only 171 | on Your own behalf and on Your sole responsibility, not on behalf 172 | of any other Contributor, and only if You agree to indemnify, 173 | defend, and hold each Contributor harmless for any liability 174 | incurred by, or claims asserted against, such Contributor by reason 175 | of your accepting any such warranty or additional liability. 176 | 177 | END OF TERMS AND CONDITIONS 178 | 179 | APPENDIX: How to apply the Apache License to your work. 180 | 181 | To apply the Apache License to your work, attach the following 182 | boilerplate notice, with the fields enclosed by brackets "[]" 183 | replaced with your own identifying information. (Don't include 184 | the brackets!) The text should be enclosed in the appropriate 185 | comment syntax for the file format. We also recommend that a 186 | file or class name and description of purpose be included on the 187 | same "printed page" as the copyright notice for easier 188 | identification within third-party archives. 189 | 190 | Copyright [yyyy] [name of copyright owner] 191 | 192 | Licensed under the Apache License, Version 2.0 (the "License"); 193 | you may not use this file except in compliance with the License. 194 | You may obtain a copy of the License at 195 | 196 | http://www.apache.org/licenses/LICENSE-2.0 197 | 198 | Unless required by applicable law or agreed to in writing, software 199 | distributed under the License is distributed on an "AS IS" BASIS, 200 | WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 201 | See the License for the specific language governing permissions and 202 | limitations under the License. 203 | -------------------------------------------------------------------------------- /README.asciidoc: -------------------------------------------------------------------------------- 1 | == Twixt - Awesome asset management for Clojure web applications 2 | 3 | image:http://clojars.org/io.aviso/twixt/latest-version.svg[Clojars Project, link="http://clojars.org/io.aviso/twixt"] 4 | 5 | image:https://drone.io/github.com/AvisoNovate/twixt/status.png[Build Status, link="https://drone.io/github.com/AvisoNovate/twixt"] 6 | 7 | Twixt is an extensible asset pipeline for use in Clojure web applications. 8 | It is designed to complement an application built using Ring and related libraries, such as Compojure. 9 | Twixt provides content transformation (such as Less to CSS), support for efficient immutable resources, 10 | and best-of-breed exception reporting. 11 | 12 | Twixt is available under the terms of the Apache Software License 2.0. 13 | 14 | link:https://portal.aviso.io/#/docs/open-source[Full Documentation] 15 | -------------------------------------------------------------------------------- /dev-resources/META-INF/assets/aviso-logo.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/AvisoNovate/twixt/21fff9321cd760459346604263bea8f747cfb1e2/dev-resources/META-INF/assets/aviso-logo.png -------------------------------------------------------------------------------- /dev-resources/META-INF/assets/coffeescript-source.coffee: -------------------------------------------------------------------------------- 1 | println "CoffeeScript is fun!" -------------------------------------------------------------------------------- /dev-resources/META-INF/assets/colors.less: -------------------------------------------------------------------------------- 1 | @body-color: #898989; 2 | -------------------------------------------------------------------------------- /dev-resources/META-INF/assets/common.jade: -------------------------------------------------------------------------------- 1 | p Content from common.jade -------------------------------------------------------------------------------- /dev-resources/META-INF/assets/f1.txt: -------------------------------------------------------------------------------- 1 | file 1 2 | -------------------------------------------------------------------------------- /dev-resources/META-INF/assets/invalid-coffeescript.coffee: -------------------------------------------------------------------------------- 1 | define ["dep1", "dep2"], 2 | (dep1, dep2) -> 3 | 4 | # Error is the missing comma after the string: 5 | dep1.doSomething "atrocious" 6 | argument: dep2 7 | -------------------------------------------------------------------------------- /dev-resources/META-INF/assets/invalid-less.less: -------------------------------------------------------------------------------- 1 | body { 2 | 3 | p { 4 | color: red; 5 | 6 | // Missing close brace: -------------------------------------------------------------------------------- /dev-resources/META-INF/assets/jade-helper.jade: -------------------------------------------------------------------------------- 1 | img(src=twixt.uri("aviso-logo.png"), title=logoTitle) -------------------------------------------------------------------------------- /dev-resources/META-INF/assets/jade-source.jade: -------------------------------------------------------------------------------- 1 | .sidebar-placeholder 2 | h1 Sidebar Empty 3 | p.lead 4 | | The sidebar is used when selecting certain objects or operations 5 | | in the main user interface. It will open when needed, 6 | | and can be closed using the 7 | button: i.icon-chevron-left 8 | | button. 9 | -------------------------------------------------------------------------------- /dev-resources/META-INF/assets/jade-uris-helper.jade: -------------------------------------------------------------------------------- 1 | ul: for s in twixt.uris(["stack/compiled.stack", "/colors.less"]) 2 | li= s -------------------------------------------------------------------------------- /dev-resources/META-INF/assets/logo.less: -------------------------------------------------------------------------------- 1 | div.logo { 2 | background: data-uri("aviso-logo.png"); 3 | } -------------------------------------------------------------------------------- /dev-resources/META-INF/assets/sample.less: -------------------------------------------------------------------------------- 1 | @import "colors.less"; 2 | 3 | body { 4 | color: @body-color; 5 | } -------------------------------------------------------------------------------- /dev-resources/META-INF/assets/stack/barney.js: -------------------------------------------------------------------------------- 1 | var barney = "Barney Rubble"; 2 | -------------------------------------------------------------------------------- /dev-resources/META-INF/assets/stack/bedrock.stack: -------------------------------------------------------------------------------- 1 | {:content-type "text/javascript" 2 | :components ["fred.js" "barney.js"]} -------------------------------------------------------------------------------- /dev-resources/META-INF/assets/stack/bootstrap.stack: -------------------------------------------------------------------------------- 1 | {:content-type "text/javascript" 2 | :components ["jquery.js", "bootstrap.js"]} -------------------------------------------------------------------------------- /dev-resources/META-INF/assets/stack/compiled.stack: -------------------------------------------------------------------------------- 1 | {:content-type "text/javascript" 2 | :components ["../coffeescript-source.coffee" "stack.coffee"]} -------------------------------------------------------------------------------- /dev-resources/META-INF/assets/stack/fred.js: -------------------------------------------------------------------------------- 1 | var fred = "Fred Flintstone"; 2 | -------------------------------------------------------------------------------- /dev-resources/META-INF/assets/stack/local.less: -------------------------------------------------------------------------------- 1 | div.logo { 2 | background: url("../aviso-logo.png"); 3 | } -------------------------------------------------------------------------------- /dev-resources/META-INF/assets/stack/meta.stack: -------------------------------------------------------------------------------- 1 | {:content-type "text/javascript" 2 | :components ["bedrock.stack" "compiled.stack"]} -------------------------------------------------------------------------------- /dev-resources/META-INF/assets/stack/missing-component.stack: -------------------------------------------------------------------------------- 1 | {:content-type "text/javascript" 2 | :components ["does-not-exist.coffee"]} -------------------------------------------------------------------------------- /dev-resources/META-INF/assets/stack/stack.coffee: -------------------------------------------------------------------------------- 1 | showDialog = (text) -> window.alert text 2 | 3 | $("#logo").click showDialog 4 | -------------------------------------------------------------------------------- /dev-resources/META-INF/assets/stack/style.stack: -------------------------------------------------------------------------------- 1 | {:content-type "text/css" 2 | :components ["local.less" "../sample.less"]} -------------------------------------------------------------------------------- /dev-resources/META-INF/assets/sub/f2.txt: -------------------------------------------------------------------------------- 1 | file 2 2 | -------------------------------------------------------------------------------- /dev-resources/META-INF/assets/sub/f3.txt: -------------------------------------------------------------------------------- 1 | file 3 2 | -------------------------------------------------------------------------------- /dev-resources/META-INF/assets/sub/jade-include.jade: -------------------------------------------------------------------------------- 1 | include samedir.jade 2 | include ../common 3 | p Content from "sub/jade-include.jade" 4 | -------------------------------------------------------------------------------- /dev-resources/META-INF/assets/sub/samedir.jade: -------------------------------------------------------------------------------- 1 | p Content from "sub/samedir.jade" 2 | -------------------------------------------------------------------------------- /dev-resources/example.clj: -------------------------------------------------------------------------------- 1 | (ns example 2 | (:use ring.adapter.jetty 3 | io.aviso.twixt.startup) 4 | (:require [io.aviso.tracker :as t] 5 | [io.aviso.twixt :refer [get-asset-uris default-options]])) 6 | 7 | (defn handler 8 | [request] 9 | (t/track "Invoking handler (that throws exceptions)" 10 | (if (= (:uri request) "/fail") 11 | ;; This will fail at some depth: 12 | (doall 13 | (get-asset-uris (:twixt request) "invalid-coffeescript.coffee"))))) 14 | 15 | (defn app 16 | [] 17 | (wrap-with-twixt handler (assoc-in default-options [:cache :cache-dir] "target/twixt-cache") false)) 18 | 19 | (defn launch 20 | [] 21 | (run-jetty (app) {:port 8888 :join? false})) 22 | 23 | (defn shutdown 24 | [server] 25 | (.stop server)) -------------------------------------------------------------------------------- /dev-resources/expected/bedrock.js: -------------------------------------------------------------------------------- 1 | var fred = "Fred Flintstone"; 2 | var barney = "Barney Rubble"; 3 | -------------------------------------------------------------------------------- /dev-resources/expected/coffeescript-source.js: -------------------------------------------------------------------------------- 1 | // Generated by CoffeeScript 1.9.3 2 | (function() { 3 | println("CoffeeScript is fun!"); 4 | 5 | }).call(this); 6 | 7 | //# sourceMappingURL=coffeescript-source.coffee@source.map -------------------------------------------------------------------------------- /dev-resources/expected/coffeescript-source.map: -------------------------------------------------------------------------------- 1 | { 2 | "version": 3, 3 | "file": "", 4 | "sourceRoot": "", 5 | "sources": [ 6 | "coffeescript-source.coffee" 7 | ], 8 | "names": [], 9 | "mappings": ";AAAA;EAAA,OAAA,CAAQ,sBAAR;AAAA", 10 | "sourcesContent": [ 11 | "println \"CoffeeScript is fun!\"" 12 | ] 13 | } -------------------------------------------------------------------------------- /dev-resources/expected/compiled-stack.js: -------------------------------------------------------------------------------- 1 | // Generated by CoffeeScript 1.9.3 2 | (function() { 3 | println("CoffeeScript is fun!"); 4 | 5 | }).call(this); 6 | 7 | //# sourceMappingURL=coffeescript-source.coffee@source.map 8 | // Generated by CoffeeScript 1.9.3 9 | (function() { 10 | var showDialog; 11 | 12 | showDialog = function(text) { 13 | return window.alert(text); 14 | }; 15 | 16 | $("#logo").click(showDialog); 17 | 18 | }).call(this); 19 | 20 | //# sourceMappingURL=stack.coffee@source.map 21 | -------------------------------------------------------------------------------- /dev-resources/expected/jade-include.html: -------------------------------------------------------------------------------- 1 |

Content from "sub/samedir.jade"

2 |

Content from common.jade

3 |

Content from "sub/jade-include.jade"

-------------------------------------------------------------------------------- /dev-resources/expected/jade-source.html: -------------------------------------------------------------------------------- 1 | -------------------------------------------------------------------------------- /dev-resources/expected/jade-uris-helper.html: -------------------------------------------------------------------------------- 1 | -------------------------------------------------------------------------------- /dev-resources/expected/logo.css: -------------------------------------------------------------------------------- 1 | div.logo { 2 | background: url(""); 3 | } 4 | 5 | /*# sourceMappingURL=logo.less@source.map */ -------------------------------------------------------------------------------- /dev-resources/expected/meta.js: -------------------------------------------------------------------------------- 1 | var fred = "Fred Flintstone"; 2 | var barney = "Barney Rubble"; 3 | // Generated by CoffeeScript 1.9.3 4 | (function() { 5 | println("CoffeeScript is fun!"); 6 | 7 | }).call(this); 8 | 9 | //# sourceMappingURL=coffeescript-source.coffee@source.map 10 | // Generated by CoffeeScript 1.9.3 11 | (function() { 12 | var showDialog; 13 | 14 | showDialog = function(text) { 15 | return window.alert(text); 16 | }; 17 | 18 | $("#logo").click(showDialog); 19 | 20 | }).call(this); 21 | 22 | //# sourceMappingURL=stack.coffee@source.map -------------------------------------------------------------------------------- /dev-resources/expected/minimized/fred.js: -------------------------------------------------------------------------------- 1 | var fred="Fred Flintstone"; 2 | -------------------------------------------------------------------------------- /dev-resources/expected/minimized/meta.js: -------------------------------------------------------------------------------- 1 | var fred="Fred Flintstone",barney="Barney Rubble";(function(){println("CoffeeScript is fun!")}).call(this);(function(){$("#logo").click(function(a){return window.alert(a)})}).call(this); 2 | -------------------------------------------------------------------------------- /dev-resources/expected/minimized/sample.css: -------------------------------------------------------------------------------- 1 | body{color:#898989} 2 | -------------------------------------------------------------------------------- /dev-resources/expected/sample.css: -------------------------------------------------------------------------------- 1 | body { 2 | color: #898989; 3 | } 4 | 5 | /*# sourceMappingURL=sample.less@source.map */ 6 | -------------------------------------------------------------------------------- /dev-resources/expected/sample.map: -------------------------------------------------------------------------------- 1 | { 2 | "version":3, 3 | "file":"sample.css", 4 | "lineCount":1, 5 | "mappings":"ACEAA;", 6 | "sources":["colors.less","sample.less"], 7 | "sourcesContent":["@body-color: #898989;\n","@import \"colors.less\";\n\nbody {\n color: @body-color;\n}"], 8 | "names":["body"] 9 | } -------------------------------------------------------------------------------- /dev-resources/expected/style.css: -------------------------------------------------------------------------------- 1 | div.logo { 2 | background: url("/assets/8ee745bf/aviso-logo.png"); 3 | } 4 | 5 | /*# sourceMappingURL=local.less@source.map */ 6 | body { 7 | color: #898989; 8 | } 9 | 10 | /*# sourceMappingURL=sample.less@source.map */ -------------------------------------------------------------------------------- /dev-resources/logback-test.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | %date{HH:mm:ss.SSS} %-5level [%thread] %logger{} - %msg%n 7 | 8 | 9 | 10 | 11 | 12 | 13 | 14 | 15 | 16 | 17 | 18 | 19 | 20 | 21 | 22 | -------------------------------------------------------------------------------- /dev-resources/user.clj: -------------------------------------------------------------------------------- 1 | (ns user 2 | (:use speclj.config 3 | clojure.pprint) 4 | (:require [schema.core :as s])) 5 | 6 | (alter-var-root #'default-config assoc :color true :reporters ["documentation"]) 7 | 8 | (s/set-fn-validation! true) -------------------------------------------------------------------------------- /java/com/yahoo/platform/yui/compressor/CssCompressor.java: -------------------------------------------------------------------------------- 1 | /* 2 | * YUI Compressor 3 | * http://developer.yahoo.com/yui/compressor/ 4 | * Author: Julien Lecomte - http://www.julienlecomte.net/ 5 | * Author: Isaac Schlueter - http://foohack.com/ 6 | * Author: Stoyan Stefanov - http://phpied.com/ 7 | * Contributor: Dan Beam - http://danbeam.org/ 8 | * Copyright (c) 2013 Yahoo! Inc. All rights reserved. 9 | * The copyrights embodied in the content of this file are licensed 10 | * by Yahoo! Inc. under the BSD (revised) open source license. 11 | */ 12 | package com.yahoo.platform.yui.compressor; 13 | 14 | import java.io.IOException; 15 | import java.io.Reader; 16 | import java.io.Writer; 17 | import java.util.regex.Pattern; 18 | import java.util.regex.Matcher; 19 | import java.util.ArrayList; 20 | 21 | public class CssCompressor { 22 | 23 | private StringBuffer srcsb = new StringBuffer(); 24 | 25 | public CssCompressor(Reader in) throws IOException { 26 | // Read the stream... 27 | int c; 28 | while ((c = in.read()) != -1) { 29 | srcsb.append((char) c); 30 | } 31 | } 32 | 33 | /** 34 | * @param css - full css string 35 | * @param preservedToken - token to preserve 36 | * @param tokenRegex - regex to find token 37 | * @param removeWhiteSpace - remove any white space in the token 38 | * @param preservedTokens - array of token values 39 | * @return 40 | */ 41 | protected String preserveToken(String css, String preservedToken, 42 | String tokenRegex, boolean removeWhiteSpace, ArrayList preservedTokens) { 43 | 44 | int maxIndex = css.length() - 1; 45 | int appendIndex = 0; 46 | 47 | StringBuffer sb = new StringBuffer(); 48 | 49 | Pattern p = Pattern.compile(tokenRegex); 50 | Matcher m = p.matcher(css); 51 | 52 | while (m.find()) { 53 | int startIndex = m.start() + (preservedToken.length() + 1); 54 | String terminator = m.group(1); 55 | 56 | // skip this, if CSS was already copied to "sb" upto this position 57 | if (m.start() < appendIndex) { 58 | continue; 59 | } 60 | 61 | if (terminator.length() == 0) { 62 | terminator = ")"; 63 | } 64 | 65 | boolean foundTerminator = false; 66 | 67 | int endIndex = m.end() - 1; 68 | while(foundTerminator == false && endIndex+1 <= maxIndex) { 69 | endIndex = css.indexOf(terminator, endIndex+1); 70 | 71 | if (endIndex <= 0) { 72 | break; 73 | } else if ((endIndex > 0) && (css.charAt(endIndex-1) != '\\')) { 74 | foundTerminator = true; 75 | if (!")".equals(terminator)) { 76 | endIndex = css.indexOf(")", endIndex); 77 | } 78 | } 79 | } 80 | 81 | // Enough searching, start moving stuff over to the buffer 82 | sb.append(css.substring(appendIndex, m.start())); 83 | 84 | if (foundTerminator) { 85 | String token = css.substring(startIndex, endIndex); 86 | if(removeWhiteSpace) 87 | token = token.replaceAll("\\s+", ""); 88 | preservedTokens.add(token); 89 | 90 | String preserver = preservedToken + "(___YUICSSMIN_PRESERVED_TOKEN_" + (preservedTokens.size() - 1) + "___)"; 91 | sb.append(preserver); 92 | 93 | appendIndex = endIndex + 1; 94 | } else { 95 | // No end terminator found, re-add the whole match. Should we throw/warn here? 96 | sb.append(css.substring(m.start(), m.end())); 97 | appendIndex = m.end(); 98 | } 99 | } 100 | 101 | sb.append(css.substring(appendIndex)); 102 | 103 | return sb.toString(); 104 | } 105 | 106 | public void compress(Writer out, int linebreakpos) 107 | throws IOException { 108 | 109 | Pattern p; 110 | Matcher m; 111 | String css = srcsb.toString(); 112 | 113 | int startIndex = 0; 114 | int endIndex = 0; 115 | int i = 0; 116 | int max = 0; 117 | ArrayList preservedTokens = new ArrayList(0); 118 | ArrayList comments = new ArrayList(0); 119 | String token; 120 | int totallen = css.length(); 121 | String placeholder; 122 | 123 | 124 | StringBuffer sb = new StringBuffer(css); 125 | 126 | // collect all comment blocks... 127 | while ((startIndex = sb.indexOf("/*", startIndex)) >= 0) { 128 | endIndex = sb.indexOf("*/", startIndex + 2); 129 | if (endIndex < 0) { 130 | endIndex = totallen; 131 | } 132 | 133 | token = sb.substring(startIndex + 2, endIndex); 134 | comments.add(token); 135 | sb.replace(startIndex + 2, endIndex, "___YUICSSMIN_PRESERVE_CANDIDATE_COMMENT_" + (comments.size() - 1) + "___"); 136 | startIndex += 2; 137 | } 138 | css = sb.toString(); 139 | 140 | 141 | css = this.preserveToken(css, "url", "(?i)url\\(\\s*([\"']?)data\\:", true, preservedTokens); 142 | css = this.preserveToken(css, "calc", "(?i)calc\\(\\s*([\"']?)", false, preservedTokens); 143 | css = this.preserveToken(css, "progid:DXImageTransform.Microsoft.Matrix", "(?i)progid:DXImageTransform.Microsoft.Matrix\\s*([\"']?)", false, preservedTokens); 144 | 145 | 146 | // preserve strings so their content doesn't get accidentally minified 147 | sb = new StringBuffer(); 148 | p = Pattern.compile("(\"([^\\\\\"]|\\\\.|\\\\)*\")|(\'([^\\\\\']|\\\\.|\\\\)*\')"); 149 | m = p.matcher(css); 150 | while (m.find()) { 151 | token = m.group(); 152 | char quote = token.charAt(0); 153 | token = token.substring(1, token.length() - 1); 154 | 155 | // maybe the string contains a comment-like substring? 156 | // one, maybe more? put'em back then 157 | if (token.indexOf("___YUICSSMIN_PRESERVE_CANDIDATE_COMMENT_") >= 0) { 158 | for (i = 0, max = comments.size(); i < max; i += 1) { 159 | token = token.replace("___YUICSSMIN_PRESERVE_CANDIDATE_COMMENT_" + i + "___", comments.get(i).toString()); 160 | } 161 | } 162 | 163 | // minify alpha opacity in filter strings 164 | token = token.replaceAll("(?i)progid:DXImageTransform.Microsoft.Alpha\\(Opacity=", "alpha(opacity="); 165 | 166 | preservedTokens.add(token); 167 | String preserver = quote + "___YUICSSMIN_PRESERVED_TOKEN_" + (preservedTokens.size() - 1) + "___" + quote; 168 | m.appendReplacement(sb, preserver); 169 | } 170 | m.appendTail(sb); 171 | css = sb.toString(); 172 | 173 | 174 | // strings are safe, now wrestle the comments 175 | for (i = 0, max = comments.size(); i < max; i += 1) { 176 | 177 | token = comments.get(i).toString(); 178 | placeholder = "___YUICSSMIN_PRESERVE_CANDIDATE_COMMENT_" + i + "___"; 179 | 180 | // ! in the first position of the comment means preserve 181 | // so push to the preserved tokens while stripping the ! 182 | if (token.startsWith("!")) { 183 | preservedTokens.add(token); 184 | css = css.replace(placeholder, "___YUICSSMIN_PRESERVED_TOKEN_" + (preservedTokens.size() - 1) + "___"); 185 | continue; 186 | } 187 | 188 | // \ in the last position looks like hack for Mac/IE5 189 | // shorten that to /*\*/ and the next one to /**/ 190 | if (token.endsWith("\\")) { 191 | preservedTokens.add("\\"); 192 | css = css.replace(placeholder, "___YUICSSMIN_PRESERVED_TOKEN_" + (preservedTokens.size() - 1) + "___"); 193 | i = i + 1; // attn: advancing the loop 194 | preservedTokens.add(""); 195 | css = css.replace("___YUICSSMIN_PRESERVE_CANDIDATE_COMMENT_" + i + "___", "___YUICSSMIN_PRESERVED_TOKEN_" + (preservedTokens.size() - 1) + "___"); 196 | continue; 197 | } 198 | 199 | // keep empty comments after child selectors (IE7 hack) 200 | // e.g. html >/**/ body 201 | if (token.length() == 0) { 202 | startIndex = css.indexOf(placeholder); 203 | if (startIndex > 2) { 204 | if (css.charAt(startIndex - 3) == '>') { 205 | preservedTokens.add(""); 206 | css = css.replace(placeholder, "___YUICSSMIN_PRESERVED_TOKEN_" + (preservedTokens.size() - 1) + "___"); 207 | } 208 | } 209 | } 210 | 211 | // in all other cases kill the comment 212 | css = css.replace("/*" + placeholder + "*/", ""); 213 | } 214 | 215 | // preserve \9 IE hack 216 | final String backslash9 = "\\9"; 217 | while (css.indexOf(backslash9) > -1) { 218 | preservedTokens.add(backslash9); 219 | css = css.replace(backslash9, "___YUICSSMIN_PRESERVED_TOKEN_" + (preservedTokens.size() - 1) + "___"); 220 | } 221 | 222 | // Normalize all whitespace strings to single spaces. Easier to work with that way. 223 | css = css.replaceAll("\\s+", " "); 224 | 225 | // Remove the spaces before the things that should not have spaces before them. 226 | // But, be careful not to turn "p :link {...}" into "p:link{...}" 227 | // Swap out any pseudo-class colons with the token, and then swap back. 228 | sb = new StringBuffer(); 229 | p = Pattern.compile("(^|\\})(([^\\{:])+:)+([^\\{]*\\{)"); 230 | m = p.matcher(css); 231 | while (m.find()) { 232 | String s = m.group(); 233 | s = s.replaceAll(":", "___YUICSSMIN_PSEUDOCLASSCOLON___"); 234 | s = s.replaceAll( "\\\\", "\\\\\\\\" ).replaceAll( "\\$", "\\\\\\$" ); 235 | m.appendReplacement(sb, s); 236 | } 237 | m.appendTail(sb); 238 | css = sb.toString(); 239 | // Remove spaces before the things that should not have spaces before them. 240 | css = css.replaceAll("\\s+([!{};:>+\\(\\)\\],])", "$1"); 241 | // Restore spaces for !important 242 | css = css.replaceAll("!important", " !important"); 243 | // bring back the colon 244 | css = css.replaceAll("___YUICSSMIN_PSEUDOCLASSCOLON___", ":"); 245 | 246 | // retain space for special IE6 cases 247 | sb = new StringBuffer(); 248 | p = Pattern.compile("(?i):first\\-(line|letter)(\\{|,)"); 249 | m = p.matcher(css); 250 | while (m.find()) { 251 | m.appendReplacement(sb, ":first-" + m.group(1).toLowerCase() + " " + m.group(2)); 252 | } 253 | m.appendTail(sb); 254 | css = sb.toString(); 255 | 256 | // no space after the end of a preserved comment 257 | css = css.replaceAll("\\*/ ", "*/"); 258 | 259 | // If there are multiple @charset directives, push them to the top of the file. 260 | sb = new StringBuffer(); 261 | p = Pattern.compile("(?i)^(.*)(@charset)( \"[^\"]*\";)"); 262 | m = p.matcher(css); 263 | while (m.find()) { 264 | String s = m.group(1).replaceAll("\\\\", "\\\\\\\\").replaceAll("\\$", "\\\\\\$"); 265 | m.appendReplacement(sb, m.group(2).toLowerCase() + m.group(3) + s); 266 | } 267 | m.appendTail(sb); 268 | css = sb.toString(); 269 | 270 | // When all @charset are at the top, remove the second and after (as they are completely ignored). 271 | sb = new StringBuffer(); 272 | p = Pattern.compile("(?i)^((\\s*)(@charset)( [^;]+;\\s*))+"); 273 | m = p.matcher(css); 274 | while (m.find()) { 275 | m.appendReplacement(sb, m.group(2) + m.group(3).toLowerCase() + m.group(4)); 276 | } 277 | m.appendTail(sb); 278 | css = sb.toString(); 279 | 280 | // lowercase some popular @directives (@charset is done right above) 281 | sb = new StringBuffer(); 282 | p = Pattern.compile("(?i)@(font-face|import|(?:-(?:atsc|khtml|moz|ms|o|wap|webkit)-)?keyframe|media|page|namespace)"); 283 | m = p.matcher(css); 284 | while (m.find()) { 285 | m.appendReplacement(sb, '@' + m.group(1).toLowerCase()); 286 | } 287 | m.appendTail(sb); 288 | css = sb.toString(); 289 | 290 | // lowercase some more common pseudo-elements 291 | sb = new StringBuffer(); 292 | p = Pattern.compile("(?i):(active|after|before|checked|disabled|empty|enabled|first-(?:child|of-type)|focus|hover|last-(?:child|of-type)|link|only-(?:child|of-type)|root|:selection|target|visited)"); 293 | m = p.matcher(css); 294 | while (m.find()) { 295 | m.appendReplacement(sb, ':' + m.group(1).toLowerCase()); 296 | } 297 | m.appendTail(sb); 298 | css = sb.toString(); 299 | 300 | // lowercase some more common functions 301 | sb = new StringBuffer(); 302 | p = Pattern.compile("(?i):(lang|not|nth-child|nth-last-child|nth-last-of-type|nth-of-type|(?:-(?:moz|webkit)-)?any)\\("); 303 | m = p.matcher(css); 304 | while (m.find()) { 305 | m.appendReplacement(sb, ':' + m.group(1).toLowerCase() + '('); 306 | } 307 | m.appendTail(sb); 308 | css = sb.toString(); 309 | 310 | // lower case some common function that can be values 311 | // NOTE: rgb() isn't useful as we replace with #hex later, as well as and() is already done for us right after this 312 | sb = new StringBuffer(); 313 | p = Pattern.compile("(?i)([:,\\( ]\\s*)(attr|color-stop|from|rgba|to|url|(?:-(?:atsc|khtml|moz|ms|o|wap|webkit)-)?(?:calc|max|min|(?:repeating-)?(?:linear|radial)-gradient)|-webkit-gradient)"); 314 | m = p.matcher(css); 315 | while (m.find()) { 316 | m.appendReplacement(sb, m.group(1) + m.group(2).toLowerCase()); 317 | } 318 | m.appendTail(sb); 319 | css = sb.toString(); 320 | 321 | // Put the space back in some cases, to support stuff like 322 | // @media screen and (-webkit-min-device-pixel-ratio:0){ 323 | css = css.replaceAll("(?i)\\band\\(", "and ("); 324 | 325 | // Remove the spaces after the things that should not have spaces after them. 326 | css = css.replaceAll("([!{}:;>+\\(\\[,])\\s+", "$1"); 327 | 328 | // remove unnecessary semicolons 329 | css = css.replaceAll(";+}", "}"); 330 | 331 | // Replace 0(px,em,%) with 0. 332 | css = css.replaceAll("(?i)(^|[^.0-9])(?:0?\\.)?0(?:px|em|%|in|cm|mm|pc|pt|ex|deg|g?rad|m?s|k?hz)", "$10"); 333 | // Replace x.0(px,em,%) with x(px,em,%). 334 | css = css.replaceAll("([0-9])\\.0(px|em|%|in|cm|mm|pc|pt|ex|deg|g?rad|m?s|k?hz| |;)", "$1$2"); 335 | 336 | // Replace 0 0 0 0; with 0. 337 | css = css.replaceAll(":0 0 0 0(;|})", ":0$1"); 338 | css = css.replaceAll(":0 0 0(;|})", ":0$1"); 339 | css = css.replaceAll(":0 0(;|})", ":0$1"); 340 | 341 | 342 | // Replace background-position:0; with background-position:0 0; 343 | // same for transform-origin 344 | sb = new StringBuffer(); 345 | p = Pattern.compile("(?i)(background-position|webkit-mask-position|transform-origin|webkit-transform-origin|moz-transform-origin|o-transform-origin|ms-transform-origin):0(;|})"); 346 | m = p.matcher(css); 347 | while (m.find()) { 348 | m.appendReplacement(sb, m.group(1).toLowerCase() + ":0 0" + m.group(2)); 349 | } 350 | m.appendTail(sb); 351 | css = sb.toString(); 352 | 353 | // Replace 0.6 to .6, but only when preceded by : or a white-space 354 | css = css.replaceAll("(:|\\s)0+\\.(\\d+)", "$1.$2"); 355 | 356 | // Shorten colors from rgb(51,102,153) to #336699 357 | // This makes it more likely that it'll get further compressed in the next step. 358 | p = Pattern.compile("rgb\\s*\\(\\s*([0-9,\\s]+)\\s*\\)"); 359 | m = p.matcher(css); 360 | sb = new StringBuffer(); 361 | while (m.find()) { 362 | String[] rgbcolors = m.group(1).split(","); 363 | StringBuffer hexcolor = new StringBuffer("#"); 364 | for (i = 0; i < rgbcolors.length; i++) { 365 | int val = Integer.parseInt(rgbcolors[i]); 366 | if (val < 16) { 367 | hexcolor.append("0"); 368 | } 369 | 370 | // If someone passes an RGB value that's too big to express in two characters, round down. 371 | // Probably should throw out a warning here, but generating valid CSS is a bigger concern. 372 | if (val > 255) { 373 | val = 255; 374 | } 375 | hexcolor.append(Integer.toHexString(val)); 376 | } 377 | m.appendReplacement(sb, hexcolor.toString()); 378 | } 379 | m.appendTail(sb); 380 | css = sb.toString(); 381 | 382 | // Shorten colors from #AABBCC to #ABC. Note that we want to make sure 383 | // the color is not preceded by either ", " or =. Indeed, the property 384 | // filter: chroma(color="#FFFFFF"); 385 | // would become 386 | // filter: chroma(color="#FFF"); 387 | // which makes the filter break in IE. 388 | // We also want to make sure we're only compressing #AABBCC patterns inside { }, not id selectors ( #FAABAC {} ) 389 | // We also want to avoid compressing invalid values (e.g. #AABBCCD to #ABCD) 390 | p = Pattern.compile("(\\=\\s*?[\"']?)?" + "#([0-9a-fA-F])([0-9a-fA-F])([0-9a-fA-F])([0-9a-fA-F])([0-9a-fA-F])([0-9a-fA-F])" + "(:?\\}|[^0-9a-fA-F{][^{]*?\\})"); 391 | 392 | m = p.matcher(css); 393 | sb = new StringBuffer(); 394 | int index = 0; 395 | 396 | while (m.find(index)) { 397 | 398 | sb.append(css.substring(index, m.start())); 399 | 400 | boolean isFilter = (m.group(1) != null && !"".equals(m.group(1))); 401 | 402 | if (isFilter) { 403 | // Restore, as is. Compression will break filters 404 | sb.append(m.group(1) + "#" + m.group(2) + m.group(3) + m.group(4) + m.group(5) + m.group(6) + m.group(7)); 405 | } else { 406 | if( m.group(2).equalsIgnoreCase(m.group(3)) && 407 | m.group(4).equalsIgnoreCase(m.group(5)) && 408 | m.group(6).equalsIgnoreCase(m.group(7))) { 409 | 410 | // #AABBCC pattern 411 | sb.append("#" + (m.group(3) + m.group(5) + m.group(7)).toLowerCase()); 412 | 413 | } else { 414 | 415 | // Non-compressible color, restore, but lower case. 416 | sb.append("#" + (m.group(2) + m.group(3) + m.group(4) + m.group(5) + m.group(6) + m.group(7)).toLowerCase()); 417 | } 418 | } 419 | 420 | index = m.end(7); 421 | } 422 | 423 | sb.append(css.substring(index)); 424 | css = sb.toString(); 425 | 426 | // Replace #f00 -> red 427 | css = css.replaceAll("(:|\\s)(#f00)(;|})", "$1red$3"); 428 | // Replace other short color keywords 429 | css = css.replaceAll("(:|\\s)(#000080)(;|})", "$1navy$3"); 430 | css = css.replaceAll("(:|\\s)(#808080)(;|})", "$1gray$3"); 431 | css = css.replaceAll("(:|\\s)(#808000)(;|})", "$1olive$3"); 432 | css = css.replaceAll("(:|\\s)(#800080)(;|})", "$1purple$3"); 433 | css = css.replaceAll("(:|\\s)(#c0c0c0)(;|})", "$1silver$3"); 434 | css = css.replaceAll("(:|\\s)(#008080)(;|})", "$1teal$3"); 435 | css = css.replaceAll("(:|\\s)(#ffa500)(;|})", "$1orange$3"); 436 | css = css.replaceAll("(:|\\s)(#800000)(;|})", "$1maroon$3"); 437 | 438 | // border: none -> border:0 439 | sb = new StringBuffer(); 440 | p = Pattern.compile("(?i)(border|border-top|border-right|border-bottom|border-left|outline|background):none(;|})"); 441 | m = p.matcher(css); 442 | while (m.find()) { 443 | m.appendReplacement(sb, m.group(1).toLowerCase() + ":0" + m.group(2)); 444 | } 445 | m.appendTail(sb); 446 | css = sb.toString(); 447 | 448 | // shorter opacity IE filter 449 | css = css.replaceAll("(?i)progid:DXImageTransform.Microsoft.Alpha\\(Opacity=", "alpha(opacity="); 450 | 451 | // Find a fraction that is used for Opera's -o-device-pixel-ratio query 452 | // Add token to add the "\" back in later 453 | css = css.replaceAll("\\(([\\-A-Za-z]+):([0-9]+)\\/([0-9]+)\\)", "($1:$2___YUI_QUERY_FRACTION___$3)"); 454 | 455 | // Remove empty rules. 456 | css = css.replaceAll("[^\\}\\{/;]+\\{\\}", ""); 457 | 458 | // Add "\" back to fix Opera -o-device-pixel-ratio query 459 | css = css.replaceAll("___YUI_QUERY_FRACTION___", "/"); 460 | 461 | // TODO: Should this be after we re-insert tokens. These could alter the break points. However then 462 | // we'd need to make sure we don't break in the middle of a string etc. 463 | if (linebreakpos >= 0) { 464 | // Some source control tools don't like it when files containing lines longer 465 | // than, say 8000 characters, are checked in. The linebreak option is used in 466 | // that case to split long lines after a specific column. 467 | i = 0; 468 | int linestartpos = 0; 469 | sb = new StringBuffer(css); 470 | while (i < sb.length()) { 471 | char c = sb.charAt(i++); 472 | if (c == '}' && i - linestartpos > linebreakpos) { 473 | sb.insert(i, '\n'); 474 | linestartpos = i; 475 | } 476 | } 477 | 478 | css = sb.toString(); 479 | } 480 | 481 | // Replace multiple semi-colons in a row by a single one 482 | // See SF bug #1980989 483 | css = css.replaceAll(";;+", ";"); 484 | 485 | // restore preserved comments and strings 486 | for(i = 0, max = preservedTokens.size(); i < max; i++) { 487 | css = css.replace("___YUICSSMIN_PRESERVED_TOKEN_" + i + "___", preservedTokens.get(i).toString()); 488 | } 489 | 490 | // Trim the final string (for any leading or trailing white spaces) 491 | css = css.trim(); 492 | 493 | // Write the output... 494 | out.write(css); 495 | } 496 | } -------------------------------------------------------------------------------- /java/io/aviso/twixt/shims/CompilerShim.java: -------------------------------------------------------------------------------- 1 | package io.aviso.twixt.shims; 2 | 3 | import com.google.javascript.jscomp.*; 4 | import com.google.javascript.jscomp.Compiler; 5 | 6 | import java.io.IOException; 7 | import java.io.InputStream; 8 | import java.nio.charset.StandardCharsets; 9 | import java.util.ArrayList; 10 | import java.util.Arrays; 11 | import java.util.Collections; 12 | import java.util.List; 13 | 14 | 15 | /** 16 | * A shim for creating and executing the Google Closure JavaScript compiler. 17 | */ 18 | public class CompilerShim { 19 | 20 | public static List runCompiler(CompilerOptions options, 21 | CompilationLevel level, 22 | String filePath, 23 | InputStream source) throws IOException { 24 | Compiler compiler = new Compiler(); 25 | 26 | compiler.disableThreads(); 27 | 28 | level.setOptionsForCompilationLevel(options); 29 | 30 | SourceFile sourceFile = SourceFile.fromInputStream(filePath, source, StandardCharsets.UTF_8); 31 | 32 | List externs = Collections.emptyList(); 33 | List sources = Arrays.asList(sourceFile); 34 | 35 | Result result = compiler.compile(externs, sources, options); 36 | 37 | List resultTuple = new ArrayList(); 38 | 39 | if (result.success) { 40 | resultTuple.add(null); 41 | resultTuple.add(compiler.toSource()); 42 | } else { 43 | resultTuple.add(result.errors); 44 | } 45 | 46 | return resultTuple; 47 | } 48 | } 49 | -------------------------------------------------------------------------------- /manual/_template.html: -------------------------------------------------------------------------------- 1 | title: {{ page_title }} 2 | --- 3 | {{ content }} 4 | -------------------------------------------------------------------------------- /manual/dexy.yaml: -------------------------------------------------------------------------------- 1 | docs: 2 | - "*/*.ad|jinja|yamlargs|asciidoctor|soups": 3 | - asciidoctor: { ext : .html , args: '-s' } 4 | - apply-ws-to-content: True 5 | - code 6 | 7 | meta: 8 | - "*/meta.edn": 9 | - output: True 10 | code: 11 | - "*/*.clj" -------------------------------------------------------------------------------- /manual/en/caching.ad: -------------------------------------------------------------------------------- 1 | title: Caching 2 | --- 3 | 4 | It is desirable to have Twixt be able to serve-up files quickly, especially in production. 5 | However, that is counter-balanced by the need to ensure the *correct* content is served. 6 | 7 | The Twixt options specifies the location of the file system cache used to store compiled assets. 8 | 9 | The location of this is specified in the key `[:cache :cache-dir]` in the Twixt options. 10 | 11 | The cache-dir is the root level directory; actually caching occurs in sub-folders for "development" 12 | or "production". 13 | 14 | The cache folder is not cleared on startup; this allow compilation and other transformations 15 | from previous executions 16 | 17 | Generally, when a request for an asset is received, and the asset is in the cache, then 18 | the response is served from the cache. 19 | 20 | Periodically (as defined by the `[:cache :cache-interval-ms]` options key), a request for an asset 21 | will perform a check against the dependencies of the asset; if any of the underlying resources have changed, then the 22 | cache entry for the asset will be discarded and rebuilt. 23 | 24 | In production, you may want to set the cache interval to be much higher than in development. 25 | 26 | On concern is that a change in configuration can also invalidate caches. 27 | For example, if you upgrade to a newer version of Twixt, or make changes to the configuration 28 | of Twixt between executions, then the cached versions of the assets may not match what 29 | Twixt would generate. 30 | 31 | You should take this into account while developing and especially when deploying: it may be necessary 32 | to manually delete the file system cache as part of deploying an application upgrade. 33 | 34 | -------------------------------------------------------------------------------- /manual/en/configuration.ad: -------------------------------------------------------------------------------- 1 | title: Configuration 2 | --- 3 | 4 | Twixt's configuration is used to determine where to locate asset resources on the classpath, 5 | and what folder to serve them under. It also maps file name extensions to MIME types, and 6 | configures the file system cache. 7 | 8 | The default options: 9 | 10 | [source,clojure] 11 | ---- 12 | (def default-options 13 | {:path-prefix "/assets/" 14 | :content-types mime/default-mime-types 15 | :resolver-factories [...] 16 | :twixt-template {} 17 | :content-transformers {} 18 | :js-optimizations :default 19 | :compressable #{"text/*" "application/edn" "application/json"} 20 | :cache {:cache-dir (System/getProperty "twixt.cache-dir" (System/getProperty "java.io.tmpdir")) 21 | :check-interval-ms 1000} 22 | :exports {:interval-ms 5000 23 | :output-dir "resources/public" 24 | :output-uri "" 25 | :assets []}} 26 | ---- 27 | 28 | You can override :path-prefix to change the root URL for assets; / is an acceptable value. 29 | 30 | The :content-types key maps file extensions to MIME types. 31 | 32 | The :content-transformers key is a map of content type keys (as strings, such as "text/coffeescript") to a 33 | transformation function; The CoffeeScript, Jade, and Less compilers operate by adding entries to :content-types and :content-transformers. 34 | 35 | The :compressable key is a set used to identify which content types are compressable; note the use of the /* suffix to indicate 36 | that all text content types are compressable. Anything not explicitly compressable is considered non-compressable. 37 | 38 | The :twixt-template key is a map that provides default values for the :twixt request key. 39 | This is often used to provide information specific to particular content transformers. 40 | 41 | The :js-optimisations key is a keyword that specifies which Google Closure compilation level should be used to minimize 42 | JavaScript files. 43 | This may be one of :default, :none, :simple, :whitespace, :advanced. 44 | 45 | The :default option corresponds to :none in development mode, and :simple in production mode. 46 | 47 | The :resolver-factories key is a list of functions that create Asset resolvers. 48 | Each factory function is passed the Twixt configuration and returns an Asset handler. 49 | The two built-in factories search for standard resources (under META-INF/assets) and WebJars resources (under META-INF/resources/webjars). 50 | 51 | The :cache key defines where the file system cache (of compiled assets) is stored, and 52 | a what interval should invalidation checks occur (invalidation checks detect when any of the files from which an asset 53 | is obtained and compiled have changed). 54 | 55 | :exports is used to configure static exports. A preset list of exports are located by their asset name and output under 56 | the configured output directory. 57 | 58 | Each value in the :assets collection can be a string, or a tuple of asset name and output alias. 59 | This allows an asset to be exported and renamed at the same time. 60 | 61 | For example, to make the asset css/site.less visible as css/site.css, you would supply a tuple: 62 | 63 | [source,clojure] 64 | ---- 65 | (-> default-options 66 | (update-in [:exports :assets] conj ["css/site.less" "css/site.css"])) 67 | ---- 68 | 69 | This would locate the resource file META-INF/assets/css/site.less, compile it from Less to CSS, and copy it 70 | to resources/public/css/site.css. 71 | A call to the io.aviso.twixt/get-asset-uri function would return the URI "/css/site.css", 72 | which is exactly what the client will need to read the downloaded file. 73 | 74 | Assets are scanned for changes at intervals, and exports occur automatically. 75 | The thread which performs asset exports is started as needed, and will shutdown 76 | once the Ring handler pipeline is garbage collected; you should generally include a call to (System/gc) to 77 | ensure this happens promptly. 78 | 79 | The output-uri is the URI corresponding to the output-dir. 80 | 81 | Twixt does *not* provide a resource handler (to pick up files exported to the output directory) on its own; this is something 82 | you must supply when setting up your Ring pipeline. 83 | 84 | 85 | -------------------------------------------------------------------------------- /manual/en/cssmin.ad: -------------------------------------------------------------------------------- 1 | title: CSS Minification 2 | --- 3 | 4 | CSS minification is enabled by default in production mode, but is normally disabled in development mode. 5 | 6 | The :minimize-css key in the link:configuration.html[Twixt options] overrides the default. 7 | 8 | -------------------------------------------------------------------------------- /manual/en/direct-uris.ad: -------------------------------------------------------------------------------- 1 | title: Direct URIs 2 | --- 3 | 4 | Sometimes it is not possible to determine the full asset URI ahead of time; 5 | a common example would be a client-side 6 | framework, such as http://angularjs.org[AngularJS] that wants to load HTML templates dynamically, at runtime. It will know 7 | the path to the asset, but will not know the checksum. 8 | 9 | In this case, an *optional* Ring middleware can be used: +wrap-with-asset-redirector+. 10 | 11 | This middleware identifies requests that match existing assets and responds with a 302 redirect to the proper asset URL. 12 | For example, the asset stored as +META-INF/assets/blueant/cla.html+ can be accessed as +/blueant/cla.html+, and will be sent a redirect 13 | to +/assets/123abc/blueant/cla.html+. 14 | -------------------------------------------------------------------------------- /manual/en/exceptions.ad: -------------------------------------------------------------------------------- 1 | title: Exception Reporting 2 | --- 3 | 4 | Twixt features a very readable HTML exception report page, which displays: 5 | 6 | * The entire stack of nested exceptions, top to bottom 7 | * The stack trace for only the root exception 8 | * Demangled namespace and function names for Clojure stack frames: `io.aviso.twixt/new-twixt/reify/middleware/fn` instead of 9 | `io.aviso.twixt$new_twixt$reify__954$middleware__959$fn__960.invoke()`. 10 | * The contents of the Ring request map 11 | * All JVM system properties 12 | 13 | Twixt provides middleware that wraps the request in a `try` block; exceptions are caught and converted to the HTML exception report, which is passed down to the client. 14 | Twixt does not attempt to perform content negotiation: it always sends a `text/html` stream. 15 | -------------------------------------------------------------------------------- /manual/en/getting-started.ad: -------------------------------------------------------------------------------- 1 | title: Getting Started 2 | --- 3 | 4 | Twixt serves resources located on the classpath, in the +META-INF/assets/+ folder. 5 | The contents of this folder is accessible to clients, by default, via the URL +/assets/+. 6 | 7 | By design, assets are segregated from the rest of your code. 8 | This prevents a malicious client from directly accessing your code or configuration files. 9 | Anything outside the +META-INF/assets/+ folder is inaccessible via Twixt. 10 | 11 | Twixt maps file extensions to MIME types; it will then transform certain MIME types; for example +.coffee+ files are compiled to JavaScript. 12 | 13 | [source,clojure] 14 | ---- 15 | (ns example.app 16 | (:use compojure.core 17 | ring.adapter.jetty 18 | [io.aviso.twixt.startup :as startup])) 19 | 20 | ;;; Use Compojure to map routes to handlers 21 | (defroutes app ...) 22 | 23 | ;;; Create twixt wrappers and handler to handle /asset/ URLs; 24 | ;;; true -> in development mode 25 | (def app-handler 26 | (startup/wrap-with-twixt app true) 27 | 28 | ;;; Use the Ring Jetty adapter to serve your application 29 | (run-jetty app-handler) 30 | ---- 31 | 32 | NOTE: Twixt changes its behavior in a number of ways in development mode (as opposed to the normal 33 | production mode). 34 | For example, Less compilation will pretty-print the generated HTML markup in development 35 | mode. 36 | In production mode, the markup omits all unnecessary white space. 37 | 38 | The Twixt middleware intercepts requests for the +/assets/+ URI that map to actual files; non-matching requests, or 39 | requests for assets that do not exist, are delegated down to the wrapped handlers. 40 | 41 | In development mode, Twixt will write compiled files to the file system (you can configure where if you like). 42 | On a restart of the application, it will use those cached files if the source files have not changed. This is important, 43 | as compilation of some resources, such as CoffeeScript, can take several seconds (due to the speed, or lack thereof, of 44 | the Rhino JavaScript engine). 45 | 46 | The Twixt API includes alternate functions for constructing both the Ring middleware, and Twixt's own 47 | asset pipeline; this allows you to add new features, or exclude unwanted features. Please reference the 48 | code to see how to configure Twixt options, assemble the Twixt asset pipeline, and finally, provide the necessary 49 | Ring middleware. -------------------------------------------------------------------------------- /manual/en/index.ad: -------------------------------------------------------------------------------- 1 | title: About 2 | --- 3 | 4 | == Awesome asset management for Clojure web applications 5 | 6 | %={:type :github :user "AvisoNovate" :repo "twixt"}% 7 | 8 | image:http://clojars.org/io.aviso/twixt/latest-version.svg[Clojars Project, link="http://clojars.org/io.aviso/twixt"] 9 | 10 | image:https://drone.io/github.com/AvisoNovate/twixt/status.png[Build Status, link="https://drone.io/github.com/AvisoNovate/twixt"] 11 | 12 | Twixt is an extensible asset pipeline for use in Clojure web applications. 13 | It is designed to complement an application built using Ring and related libraries, such as Compojure. 14 | Twixt provides content transformation (such as Less to CSS), support for efficient immutable resources, 15 | and best-of-breed exception reporting. 16 | 17 | Twixt draws inspiration from http://tapestry.apache.org[Apache Tapestry] and https://github.com/edgecase/dieter[Dieter]. 18 | 19 | Twixt currently supports: 20 | 21 | * link:exceptions.html[Best-of-breed exception report] 22 | * link:http://coffeescript.org/[CoffeeScript] to JavaScript compilation (using Rhino) - including source maps 23 | * JavaScript Minimization (using https://developers.google.com/closure/compiler/[Google Closure]) 24 | * Jade to HTML compilation (using https://github.com/neuland/jade4j[jade4j]) 25 | * Less to CSS compilation (using https://github.com/SomMeri/less4j[less4j]) 26 | * link:stacks.html[JavaScript and CSS aggregation] 27 | * File system caching of compiled content 28 | * Automatic GZip compression 29 | * Automatic CSS Minification (using http://yui.github.io/yuicompressor/[YUICompressor]) 30 | * Works with link:webjars.html[WebJars] 31 | 32 | == Stability 33 | 34 | *Alpha*: Many features are not yet implemented and the code is likely to change in many ways going forward ... but still very useful! 35 | -------------------------------------------------------------------------------- /manual/en/jade.ad: -------------------------------------------------------------------------------- 1 | title: Jade 2 | --- 3 | 4 | link:http://jade-lang.com/[Jade] is a wonderful template engine ... as long as you 5 | are comfortable with significant indentation. 6 | 7 | Like many such tools, the real power comes from being able to extend Jade in various ways. 8 | 9 | == twixt helper 10 | 11 | Twixt places a helper object, +twixt+, into scope for your templates. +twixt+ supplies a single method, +uri+. 12 | You can pass the +uri+ method a relative path, or an absolute path (starting with a slash). 13 | 14 | ---- 15 | img(src=twixt.uri("logo.png")) 16 | ---- 17 | 18 | WARNING: When the path is relative, it is evaluated relative to the main Jade asset 19 | (and explicitly not relative to any +include+ -ed Jade sources). 20 | 21 | This will output a fully qualified asset URI: 22 | 23 | ---- 24 | 25 | ---- 26 | 27 | The +uris+ method accepts an array of paths; an returns an array of individual asset URIs. 28 | This is useful when the URI references a stack (which will be a single asset in production, 29 | but multiple assets in development). 30 | 31 | --- 32 | for script in twixt.uris(["js/app.stack"]) 33 | script(src=script) 34 | ---- 35 | 36 | == Defining your own helpers 37 | 38 | It is possible to define your own helper objects. 39 | 40 | Helper objects are defined inside the Twixt context under keys +:jade+ +:helpers+. 41 | This is a map of _string_ keys to creator functions. 42 | 43 | Each creator function is passed the main Jade asset, and the Twixt context. 44 | It uses this to initialize and return a helper object. 45 | A new set of helper objects is created for each individual Jade compilation. 46 | 47 | Generally, you will want to define a protocol, then use +reify+. For example, this is the implementation of the +twixt+ helper: 48 | 49 | ---- 50 | (defprotocol TwixtHelper 51 | "A Jade4J helper object that is used to allow a template to resolve asset URIs." 52 | (uri 53 | [this path] 54 | "Used to obtain the URI for a given asset, identified by its asset path. 55 | The path may be relative to the currently compiling asset, or may be absolute (with a leading slash). 56 | 57 | Throws an exception if the asset it not found.") 58 | (uris 59 | [this paths] 60 | "Used to obtain multiple URIs for any number of assets, each identified by a path. 61 | 62 | paths is a seq of asset path strings (eitehr relative to the current asset, or absolute). 63 | 64 | *Added in 0.1.21*")) 65 | 66 | (defn- create-twixt-helper 67 | [asset context] 68 | (reify TwixtHelper 69 | (uri [_ path] 70 | (twixt/get-asset-uri context (complete-path asset path))) 71 | (uris [_ paths] 72 | (->> paths 73 | (map (partial complete-path asset)) 74 | (mapcat (partial twixt/get-asset-uris context)))))) 75 | ---- 76 | 77 | NOTE: Any asset URI will cause the asset in question to be added as a dependency of the main Jade template. This means 78 | that changing the referenced asset will cause the Jade template to be re-compiled. This makes sense: changing an image 79 | file will change the URI for the image file (due to content snapshotting), which means that the Jade output should also change. 80 | 81 | Creator functions can be added to the Twixt context using Ring middleware: 82 | 83 | ---- 84 | (handler (assoc-in request [:twixt :jade :helpers "adrotation"] 85 | create-ad-rotation-helper)) 86 | ---- 87 | 88 | However, more frequently, you will just add to the Twixt options in your application's startup code: 89 | 90 | ---- 91 | (assoc-in twixt/default-options [:twixt-template :jade :helpers "adrotation"] 92 | create-ad-rotation-helper)) 93 | ---- 94 | 95 | This +:twixt-template+ key is used to create the +:twixt+ Ring request key. 96 | 97 | == Defining your own variables 98 | 99 | Variables are much the same as helpers, with two differences: 100 | 101 | * The key is +:variables+ (under +:jade+, in the Twixt context) 102 | * The value is the exact object to expose to the template 103 | 104 | You can expose Clojure functions as variables if you wish; the Jade template should use +func.invoke()+ to call the function. 105 | 106 | == Helper / Variable pitfalls 107 | 108 | The main issue with helpers and variables relates to cache invalidation. 109 | Twixt bases cache invalidation entirely on the contents of the underlying files. 110 | There is no way for Twixt to know to invalidate the cache just because the implementation 111 | of a helper has changed, even if that means different markup is being rendered. This 112 | is one of the primary reasons that link:caching.html[disk cache is disabled in production]. 113 | 114 | There is currently an ambiguity that comes into play when the referenced asset is a compressable file type (e.g., not an image 115 | file). This can cause the Jade compiler to generate a compressed URI that, for a different request and client, will not be useful. 116 | 117 | -------------------------------------------------------------------------------- /manual/en/jsmin.ad: -------------------------------------------------------------------------------- 1 | title: JavaScript Minimization 2 | --- 3 | 4 | JavaScript minimization is enabled by default only in production mode. 5 | 6 | It is set up for _simple optimizations_; this removes whitespace and may shorten variable and function names. 7 | 8 | The optimization level is configurable using the +:js-optimisations+ key as described in %={:type :link :page "configuration"}. 9 | 10 | It is recommended that you make use of JavaScript stacks; this allows the compiler to only be executed once 11 | on a larger sample of JavaScript. -------------------------------------------------------------------------------- /manual/en/meta.edn: -------------------------------------------------------------------------------- 1 | {:name "Twixt Manual" 2 | :description "Awesome asset management for Clojure web applications." 3 | :github "AvisoNovate/twixt" 4 | :api "http://howardlewisship.com/io.aviso/twixt/" 5 | :pages ["index" 6 | "getting-started" 7 | "uris" 8 | "exceptions" 9 | "jade" 10 | "stacks" 11 | "caching" 12 | "jsmin" 13 | "cssmin" 14 | "configuration" 15 | "direct-uris" 16 | "webjars" 17 | "notes"]} 18 | -------------------------------------------------------------------------------- /manual/en/notes.ad: -------------------------------------------------------------------------------- 1 | title: Notes 2 | --- 3 | 4 | == Future Plans 5 | 6 | The goal is to achieve at least parity with Apache Tapestry, plus some additional features specific to Clojure. This means: 7 | 8 | * E-Tags support 9 | * ClojureScript compilation 10 | * CSS Minification 11 | * RequireJS support and AMD modules 12 | * Break out the the Less, Jade, CoffeeScript, exception reporting support, etc. into a-la-carte modules 13 | * "Warm up" the cache at startup (in production) 14 | 15 | 16 | == A note about feedback 17 | 18 | http://tapestryjava.blogspot.com/2013/05/once-more-feedback-please.html[Feedback] is very important to me; I often find 19 | Clojure just a bit frustrating, because if there is an error in your code, it can be a bit of a challenge to track the problem 20 | backwards from the failure to the offending code. Part of this is inherent in functional programming, part of it is related to lazy evaluation, 21 | and part is the trade-off between a typed and untyped language. 22 | 23 | In any case, it is very important to me that when thing go wrong, you are provided with a detailed description of the failure. 24 | Twixt has a mechanism for tracking the operations it is attempting, to give you insight into what exactly failed if there 25 | is an error. For example, (from the test suite): 26 | 27 | ---- 28 | ERROR [ qtp2166970-29] io.aviso.twixt.coffee-script An exception has occurred: 29 | ERROR [ qtp2166970-29] io.aviso.twixt.coffee-script [ 1] - Invoking handler (that throws exceptions) 30 | ERROR [ qtp2166970-29] io.aviso.twixt.coffee-script [ 2] - Accessing asset `invalid-coffeescript.coffee' 31 | ERROR [ qtp2166970-29] io.aviso.twixt.coffee-script [ 3] - Compiling `META-INF/assets/invalid-coffeescript.coffee' to JavaScript 32 | ERROR [ qtp2166970-29] io.aviso.twixt.coffee-script META-INF/assets/invalid-coffeescript.coffee:6:1: error: unexpected INDENT 33 | argument: dep2 34 | ^^^^^^ 35 | java.lang.RuntimeException: META-INF/assets/invalid-coffeescript.coffee:6:1: error: unexpected INDENT 36 | argument: dep2 37 | ^^^^^^ 38 | .... 39 | ---- 40 | 41 | In other words, when there's a failure, Twixt can tell you the steps that led up the failure, which is 90% of solving the problem in the first place. 42 | 43 | Twixt's exception report captures all of this and presents it as readable HTML. 44 | The exception report page also does a decent job of de-mangling Java class names to Clojure namespaces and function names. 45 | 46 | == How does Twixt differ from Dieter? 47 | 48 | On the application I was building, I had a requirement to deploy as a JAR; Dieter expects all the assets to be on the filesystem; I spent some time attempting to hack the Dieter code to allow resources on the classpath as well. 49 | When that proved unsuccessful, I decided to build out something a bit more ambitious, that would support the features that have accumulated in Tapestry over the last few years. 50 | 51 | Twixt also embraces http://www.infoq.com/presentations/Clojure-Large-scale-patterns-techniques[system as transient state], meaning nothing is stored statically. 52 | 53 | Twixt will grow further apart from Dieter as the more advanced pieces are put into place. -------------------------------------------------------------------------------- /manual/en/stacks.ad: -------------------------------------------------------------------------------- 1 | title: Stacks 2 | --- 3 | 4 | In development, you often want to have many small source files downloaded individually to the browser. 5 | This is simpler to debug, and faster ... a change to one file will be a small recompile of just that file. 6 | 7 | In production, it's a different story; you want the client to make as few requests to the server as possible. 8 | 9 | This can be accomplished using _stacks_. 10 | 11 | Stacks allow you to group together related files of the same type into a single asset. Commonly, this is used 12 | to aggregate JavaScript or CSS. 13 | 14 | In development mode, you will see the individual files of the stack; 15 | in production mode, the stack is represented by a single URI which maps to the aggregated content of all the 16 | files in the stack. 17 | 18 | A stack file is written in https://github.com/edn-format/edn[EDN]. 19 | Each stack file contains a +:content-type+ key, and a +:components+ key. 20 | Stack files have a +.stack+ extension. 21 | 22 | [source,clojure] 23 | ---- 24 | {:content-type "text/css" 25 | :components "bootstrap3/bootstrap.css" 26 | "app.less" 27 | "ie-fixes.less"} 28 | ---- 29 | 30 | When using stacks, you will want a slight tweak to your page template: 31 | 32 | [source,clojure] 33 | ---- 34 | (defhtml index 35 | [{context :twixt :as request}] 36 | (html 37 | (doctype :html5 38 | [:html 39 | [:head 40 | [:title "My Clojure App"] 41 | (apply include-css (get-asset-uris context "css/app-css.stack")) 42 | ... 43 | ---- 44 | 45 | Since +get-asset-uris+ will return a collection of URIs (unlike +get-asset-uri+ which always returns just one), 46 | we must change +include-css+ to +apply include-css+. 47 | 48 | This template will work in development (+get-asset-uris+ returning several URIs) and in production (just a 49 | single URI). 50 | 51 | It is possible for stacks to include other stacks as components. 52 | 53 | Stack components are _included_ not _imported_; if a component asset is listed more than once, its content will 54 | be aggregated more than once. -------------------------------------------------------------------------------- /manual/en/uris.ad: -------------------------------------------------------------------------------- 1 | title: URIs 2 | --- 3 | 4 | At its core, Twixt is a set of Ring middleware that maps certain URI patterns to matching files on the classpath, 5 | and does some transformations along the way. 6 | 7 | Currently, the mapping is very straightforward: the path +/assets/123abc/css/style.less+ is mapped to resource 8 | +META-INF/assets/css/style.less+ which is read into memory and transformed from Less to CSS. 9 | Embedded in the middle of the URL is the content checksum for the file (+123abc+). 10 | 11 | NOTE: Placing this information directly into the URI is called _fingerprinting_. 12 | 13 | In your application, assets will change during development, or between production deployments. The URIs provided to 14 | the client agent (the web browser) is a _resource_; on the web, resources are immutable, even though 15 | in your workspace, files change all the time. 16 | The checksum in the URI is based on the actual content of the file; 17 | whenever the underlying content changes, then a new checksum, new URI, and therefore, new resource will be referenced by the URI. 18 | 19 | Twixt sets headers to indicate a far-future expiration date for the resource; 20 | the upshot of which is that, once the resource for an asset is downloaded to the client browser, the browser will not ask for it again. 21 | This is great for overall performance, since it reduces the amount of network I/O 22 | the client will encounter, especially on subsequent visits to the same site. 23 | 24 | It also reduces the load on the server in two ways: 25 | 26 | * Fewer requests will need to be sent to the server as client will use their local cache. 27 | * Content can be cached by intermediary servers. 28 | 29 | The checksum has a optional "z" prefix; this indicates a GZip compressed resource. 30 | 31 | On a request that renders markup, Twixt will detect if the client supports GZip compression. 32 | If so, then for assets where compression makes sense (such as JavaScript, and 33 | specifically excluding image formats) then Twixt will generate an alternate URI that 34 | indicates access to the compressed asset. 35 | 36 | Because of this, when referencing assets inside your templates, you must pass paths (relative to +META-INF/assets+) 37 | through Twixt to get URIs that will work in the browser: 38 | 39 | [source,clojure] 40 | ---- 41 | (defhtml index 42 | [{context :twixt :as request}] 43 | (html 44 | (doctype :html5 45 | [:html 46 | [:head 47 | [:title "My Clojure App"] 48 | (include-css (get-asset-uri context "css/style.less")) 49 | ... 50 | ---- 51 | 52 | The parameter to +defhtml+ is the Ring request map; Twixt has injected middleware that provides the Twixt context under 53 | the +:twixt+ key. 54 | 55 | +get-asset-uri+ is defined in the +io.aviso.twixt+ namespace. 56 | 57 | Twixt must do all necessary compilations and other transformations, to arrive at final content for which a checksum 58 | can be computed. 59 | Although this can slow the initial HTML render, it is also good because any exceptions, such as compilation errors, will occur immediately, 60 | rather than when the asset's content is later retrieved by the client. 61 | 62 | When a client requests an asset that exists, but supplies an incorrect checksum, 63 | Twixt will respond with a 301 redirect HTTP response, 64 | directing the client to the correct resource (with the correct checksum). 65 | This is an extremely unlikely scenario that would involve a running client in a race with a redeployed application. -------------------------------------------------------------------------------- /manual/en/webjars.ad: -------------------------------------------------------------------------------- 1 | title: WebJars 2 | --- 3 | 4 | link:http://www.webjars.org/[Web Jars] is any easy way to include any number of pre-packaged sets of resources 5 | as a JAR file. 6 | Why mess around with getting a distribution, extracting it, and possibly checking those files into your project's 7 | source code control? 8 | Just add the WebJar to the classpath (inside your project.clj): 9 | 10 | [source,clojure] 11 | ---- 12 | [org.webjars/d3js "3.5.5-1"] 13 | ---- 14 | 15 | And you can now reference the files within that WebJar: 16 | 17 | [source,clojure] 18 | ---- 19 | (find-asset-uri context "d3js/d3.js) 20 | ---- 21 | 22 | That's it ... you don't even need to use the version number in the call to find-asset-uri; the version 23 | number needed internally to build the path to the referenced file (in this example, /META-INF/resources/webjars/d3j3/3.5.5-1/d3.js) 24 | is determined automatically. 25 | 26 | The referenced file will be read as normal and processed just like any file under META-INF/assets. 27 | 28 | Twixt uses dependencies on Bootstrap (3.3.5) and jQuery (1.11.1), which are used in the link:exceptions.html[exception report page]. -------------------------------------------------------------------------------- /project.clj: -------------------------------------------------------------------------------- 1 | (defproject io.aviso/twixt "0.1.25" 2 | :description "An extensible asset pipeline for Clojure web applications" 3 | :url "https://github.com/AvisoNovate/twixt" 4 | :license {:name "Apache Sofware Licencse 2.0" 5 | :url "http://www.apache.org/licenses/LICENSE-2.0.html"} 6 | :dependencies [[org.clojure/clojure "1.8.0"] 7 | [io.aviso/pretty "0.1.19"] 8 | [io.aviso/tracker "0.1.7"] 9 | [medley "0.7.0"] 10 | [ring/ring-core "1.4.0"] 11 | [com.google.javascript/closure-compiler "v20150729"] 12 | [org.mozilla/rhino "1.7.7"] 13 | [com.github.sommeri/less4j "1.14.0"] 14 | [de.neuland-bfi/jade4j "0.4.3"] 15 | [prismatic/schema "1.0.1"] 16 | [hiccup "1.0.5"] 17 | [org.webjars/bootstrap "3.3.5"] 18 | [org.webjars/webjars-locator-core "0.28"]] 19 | ;; We keep a local copy of YUICompressor's CSSCompressor; we don't want the rest of the dependency 20 | ;; since it does evil things to patch Rhino to support JavaScript compression. 21 | :java-source-paths ["java"] 22 | :javac-options ["-target" "1.6" "-source" "1.6"] 23 | :test-paths ["spec"] 24 | :plugins [[speclj "3.2.0"] 25 | [lein-shell "0.4.0"]] 26 | :shell {:commands {"scp" {:dir "doc"}}} 27 | :aliases {"deploy-doc" ["shell" 28 | "scp" "-r" "." "hlship_howardlewisship@ssh.phx.nearlyfreespeech.net:io.aviso/twixt"] 29 | "release" ["do" 30 | "clean," 31 | "spec,", 32 | "doc," 33 | "deploy-doc," 34 | "deploy" "clojars"]} 35 | :codox {:src-dir-uri "https://github.com/AvisoNovate/twixt/blob/master/" 36 | :src-linenum-anchor-prefix "L" 37 | :defaults {:doc/format :markdown}} 38 | :profiles {:dev {:dependencies [[ch.qos.logback/logback-classic "1.1.3"] 39 | [speclj "3.3.1"] 40 | [ring/ring-jetty-adapter "1.4.0"]] 41 | :jvm-opts ["-Xmx1g"]} 42 | :1.6 {:dependencies [[org.clojure/clojure "1.6.0"] 43 | [speclj "3.2.0"] 44 | [medley "0.6.0"]]}}) 45 | -------------------------------------------------------------------------------- /resources/META-INF/assets/twixt/exception.coffee: -------------------------------------------------------------------------------- 1 | # Adds a simple event handler that toggles body.hide-filtered when the toggle filter button is clicked. 2 | 3 | $ -> 4 | $("body").on "click", "[data-action=toggle-filter]", -> 5 | $("body").toggleClass "hide-filtered" -------------------------------------------------------------------------------- /resources/META-INF/assets/twixt/exception.less: -------------------------------------------------------------------------------- 1 | // CSS used on the exception report view; expects Bootstrap CSS to be present as well. 2 | 3 | body { 4 | margin-top: 5px; 5 | } 6 | 7 | dl:not(.dl-horizontal) dd { 8 | margin-left: 25px; 9 | } 10 | 11 | td.function-name, td.source-location { 12 | text-align: right; 13 | } 14 | 15 | .package-name { 16 | font-size: x-small; 17 | } 18 | 19 | .filtered { 20 | color: rgb(136, 136, 136); 21 | } 22 | 23 | .hide-filtered .filtered { 24 | display: none; 25 | } 26 | 27 | .spacing-below { 28 | margin-bottom: 5px; 29 | } -------------------------------------------------------------------------------- /resources/META-INF/twixt/invoke-coffeescript.js: -------------------------------------------------------------------------------- 1 | // Compiles CoffeeScript source to JavaScript. 2 | // 3 | // input - string containing contents of the file 4 | // filepath - name of file, used to report errors 5 | // filename - last term in filepath 6 | // 7 | // Returns { output: , sourceMap: source-map-as-string } or { exception: } 8 | function compileCoffeeScriptSource(input, filepath, filename) { 9 | try { 10 | var result = CoffeeScript.compile(input, { 11 | header: true, 12 | filename: filepath, 13 | sourceFiles: [filename], 14 | sourceMap: true, 15 | inline: true}); 16 | 17 | return { 18 | output: result.js + "\n//# sourceMappingURL=" + filename + "@source.map\n" , 19 | sourceMap: result.v3SourceMap 20 | }; 21 | } 22 | catch (err) { 23 | return { exception: err.toString() }; 24 | } 25 | } 26 | -------------------------------------------------------------------------------- /spec/io/aviso/twixt_spec.clj: -------------------------------------------------------------------------------- 1 | (ns io.aviso.twixt-spec 2 | (:use speclj.core 3 | clojure.pprint 4 | clojure.template 5 | [io.aviso.twixt :exclude [find-asset]] 6 | io.aviso.twixt.utils) 7 | (:require [clojure.java.io :as io] 8 | [ring.middleware.resource :as resource] 9 | [io.aviso.twixt 10 | [coffee-script :as cs] 11 | [jade :as jade] 12 | [less :as less] 13 | [ring :as ring] 14 | [startup :as startup] 15 | [stacks :as stacks]] 16 | [clojure.string :as str] 17 | [clojure.tools.logging :as l] 18 | [io.aviso.twixt :as twixt])) 19 | 20 | (defn read-as-trimmed-string 21 | [content] 22 | (-> content 23 | read-content 24 | String. 25 | .trim)) 26 | 27 | (defn read-asset-content 28 | [asset] 29 | (assert asset "Can't read content from nil asset.") 30 | (-> 31 | asset 32 | :content 33 | read-as-trimmed-string)) 34 | 35 | (defn read-attachment-content 36 | [asset attachment-name] 37 | (assert asset "Can't read content from nil asset.") 38 | (-> 39 | asset 40 | (get-in [:attachments attachment-name]) 41 | :content 42 | read-as-trimmed-string)) 43 | 44 | (defn read-resource-content 45 | [path] 46 | (-> 47 | path 48 | io/resource 49 | read-as-trimmed-string)) 50 | 51 | (defmacro should-have-content 52 | [expected actual] 53 | `(let [expected# ~expected 54 | actual# ~actual] 55 | (when (not (= expected# actual#)) 56 | (-fail (format "Expected content did not match actual content:%n----%n%s%n----" 57 | actual#))))) 58 | 59 | (defn have-same-content 60 | [expected-content-resource asset] 61 | (let [expected (read-resource-content expected-content-resource) 62 | actual (read-asset-content asset)] 63 | (if (= expected actual) 64 | true 65 | (printf "Content of `%s' did not match `%s'\n----\n%s\n----\n" 66 | (:resource-path asset) 67 | expected-content-resource 68 | actual)))) 69 | 70 | (defn- sorted-dependencies 71 | [asset] 72 | (->> asset :dependencies vals (map :asset-path) sort)) 73 | 74 | (defn- remove-hash-from-uri 75 | [^String asset-uri] 76 | (str/join "/" (-> 77 | asset-uri 78 | (.split "/") 79 | vec 80 | (assoc 2 "[hash]")))) 81 | 82 | ;; Note: updating the CoffeeScript compiler will often change the outputs, including checksums, not least because 83 | ;; the compiler injects a comment with the compiler version. 84 | 85 | (def compiled-coffeescript-checksum "54753589") 86 | 87 | (describe "io.aviso.twixt" 88 | 89 | (with-all cache-dir (format "%s/%x" (System/getProperty "java.io.tmpdir") (System/nanoTime))) 90 | 91 | (defn with-sub-cache-dir [twixt-options subdir] 92 | (update-in twixt-options [:cache :cache-dir] str "/" subdir)) 93 | 94 | (defn get-asset-with-options [twixt-options asset-path] 95 | (let [pipeline (default-asset-pipeline twixt-options)] 96 | (pipeline asset-path (assoc twixt-options :asset-pipeline pipeline)))) 97 | 98 | (with-all options (-> 99 | default-options 100 | (assoc :development-mode true 101 | :js-optimizations :none) 102 | (assoc-in [:cache :cache-dir] @cache-dir) 103 | ;; This is usually done by the startup namespace: 104 | cs/register-coffee-script 105 | jade/register-jade 106 | less/register-less 107 | stacks/register-stacks)) 108 | 109 | (with-all pipeline (default-asset-pipeline @options)) 110 | 111 | (with-all twixt-context (assoc @options :asset-pipeline @pipeline)) 112 | 113 | (defn find-asset 114 | [asset-path] 115 | (@pipeline asset-path @twixt-context)) 116 | 117 | (context "asset pipeline" 118 | 119 | (it "returns nil when an asset is not found" 120 | 121 | (should-be-nil (@pipeline "does/not/exist.gif" @options)))) 122 | 123 | (context "get-asset-uri" 124 | 125 | (it "throws IllegalArgumentException if an asset does not exist" 126 | (should-throw Exception 127 | "Asset path `does/not/exist.png' does not map to an available resource." 128 | (get-asset-uri @twixt-context "does/not/exist.png"))) 129 | 130 | (it "returns the correct asset URI" 131 | (should= (str "/assets/" compiled-coffeescript-checksum "/coffeescript-source.coffee") 132 | (find-asset-uri @twixt-context "coffeescript-source.coffee"))) 133 | 134 | 135 | (it "can find WebJars assets" 136 | (should 137 | (have-same-content "META-INF/resources/webjars/bootstrap/3.3.5/js/alert.js" 138 | (find-asset "bootstrap/js/alert.js")))) 139 | 140 | (it "can process compilation of WebJars assets" 141 | (should 142 | (have-same-content "expected/bootstrap-webjars.css" 143 | (find-asset "bootstrap/less/bootstrap.less"))))) 144 | 145 | (context "CoffeeScript compilation" 146 | 147 | (with-all asset (@pipeline "coffeescript-source.coffee" @twixt-context)) 148 | 149 | (it "can read and transform a source file" 150 | 151 | (should-not-be-nil @asset) 152 | (should-not-be-nil (:modified-at @asset)) 153 | 154 | (do-template [key expected] 155 | (should= expected (key @asset) 156 | {:resource-path "META-INF/assets/coffeescript-source.coffee" 157 | :asset-path "coffeescript-source.coffee" 158 | :content-type "text/javascript" 159 | :compiled true 160 | :size 160 161 | :checksum compiled-coffeescript-checksum}))) 162 | (it "has the correct compiled content" 163 | (should (have-same-content "expected/coffeescript-source.js" @asset))) 164 | 165 | (it "has the correct source.map attachment" 166 | (should-have-content 167 | (read-resource-content "expected/coffeescript-source.map") 168 | (read-attachment-content @asset "source.map"))) 169 | 170 | (it "has the expected dependencies" 171 | (should= (sorted-dependencies @asset) 172 | ["coffeescript-source.coffee"])) 173 | 174 | (it "throws an exception if the source is not valid" 175 | (try 176 | (@pipeline "invalid-coffeescript.coffee" @twixt-context) 177 | (should-fail) 178 | (catch Exception e 179 | (should (-> e .getMessage (.startsWith "META-INF/assets/invalid-coffeescript.coffee:6:1: error: unexpected indentation"))))))) 180 | 181 | (context "Jade compilation" 182 | 183 | (context "simple Jade source" 184 | (with-all asset (@pipeline "jade-source.jade" @twixt-context)) 185 | 186 | 187 | (it "can read and transform a source file" 188 | 189 | (should= "text/html" (:content-type @asset)) 190 | (should (:compiled @asset))) 191 | 192 | (it "has the correct compiled content" 193 | (should (have-same-content "expected/jade-source.html" @asset)))) 194 | 195 | (context "using Jade includes" 196 | 197 | (with-all asset (@pipeline "sub/jade-include.jade" @twixt-context)) 198 | 199 | (it "has the correct compiled content" 200 | (should (have-same-content "expected/jade-include.html" @asset))) 201 | 202 | (it "has the expected dependencies" 203 | (should= ["common.jade" "sub/jade-include.jade" "sub/samedir.jade"] 204 | (sorted-dependencies @asset)))) 205 | 206 | (context "using Jade helpers" 207 | 208 | (with-all context' (-> 209 | @twixt-context 210 | ;; The merge is normally part of the Ring code. 211 | (merge (:twixt-template @options)) 212 | ;; This could be done by Ring middleware, or by 213 | ;; modifying the :twixt-template as well. 214 | (assoc-in [:jade :variables "logoTitle"] "Our Logo"))) 215 | 216 | 217 | (with-all asset (@pipeline "jade-helper.jade" @context')) 218 | 219 | (it "has the correct compiled content ignoring attr order" 220 | (should (or (= "" (read-asset-content @asset)) 221 | (= "" (read-asset-content @asset))))) 222 | 223 | (it "includes a dependency on the asset accessed by twixt.uri()" 224 | (should= ["aviso-logo.png" "jade-helper.jade"] 225 | (sorted-dependencies @asset))) 226 | 227 | 228 | (it "supports multiple assets via twixt.uris()" 229 | (should (have-same-content "expected/jade-uris-helper.html" 230 | (@pipeline "jade-uris-helper.jade" @context')))))) 231 | 232 | (context "Less compilation" 233 | 234 | (context "basic compilation" 235 | (with-all asset (@pipeline "sample.less" @twixt-context)) 236 | 237 | (it "can read and transform a source file" 238 | (should= "text/css" (:content-type @asset)) 239 | (should (:compiled @asset))) 240 | 241 | (it "has the correct compiled content" 242 | (should (have-same-content "expected/sample.css" @asset))) 243 | 244 | (it "has an attached source.map" 245 | (should-have-content (read-resource-content "expected/sample.map") 246 | (read-attachment-content @asset "source.map"))) 247 | 248 | (it "includes dependencies for @import-ed files" 249 | (should= ["colors.less" "sample.less"] 250 | (sorted-dependencies @asset))) 251 | 252 | (it "throws an exception for compilation failures" 253 | (try 254 | (@pipeline "invalid-less.less" @twixt-context) 255 | (should-fail) 256 | (catch Exception e 257 | #_(-> e .getMessage println) 258 | (should (-> e .getMessage (.contains "META-INF/assets/invalid-less.less:3:5: no viable alternative at input 'p'"))))))) 259 | 260 | (context "using data-uri and getData method" 261 | 262 | (with-all asset (@pipeline "logo.less" @twixt-context)) 263 | 264 | (it "has the correct compiled content" 265 | (should (have-same-content "expected/logo.css" @asset))))) 266 | 267 | (context "stack support" 268 | 269 | (context "simple stack" 270 | 271 | (with-all asset (@pipeline "stack/bedrock.stack" @twixt-context)) 272 | 273 | (it "has the content type from the stack file" 274 | (should= "text/javascript" (:content-type @asset))) 275 | 276 | (it "is marked as compiled" 277 | (should (:compiled @asset))) 278 | 279 | (it "identifies the correct aggregated asset paths" 280 | (should= ["stack/fred.js" "stack/barney.js"] 281 | (-> @asset :aggregate-asset-paths))) 282 | 283 | (it "includes dependencies on every file in the stack" 284 | (should= ["stack/barney.js" "stack/bedrock.stack" "stack/fred.js"] 285 | (sorted-dependencies @asset))) 286 | 287 | (it "has the correct aggregated content" 288 | (should (have-same-content "expected/bedrock.js" @asset)))) 289 | 290 | (context "stack with compilation" 291 | 292 | (with-all asset (@pipeline "stack/compiled.stack" @twixt-context)) 293 | 294 | (it "identifies the correct aggregated asset path" 295 | (should= ["coffeescript-source.coffee" "stack/stack.coffee"] 296 | (-> @asset :aggregate-asset-paths))) 297 | 298 | (it "includes dependencies on every file in the stack" 299 | (should= ["coffeescript-source.coffee" "stack/compiled.stack" "stack/stack.coffee"] 300 | (sorted-dependencies @asset))) 301 | 302 | ;; Note: the compiled output includes the sourceMappingURL comments. 303 | ;; It is possible that will not work well in the client and may need to be filtered out. 304 | (it "has the correct aggregated content" 305 | (should (have-same-content "expected/compiled-stack.js" @asset)))) 306 | 307 | (context "stack of stacks" 308 | (with-all asset (@pipeline "stack/meta.stack" @twixt-context)) 309 | 310 | (it "identifies the correct aggregated asset path" 311 | (should= ["stack/fred.js" "stack/barney.js" "coffeescript-source.coffee" "stack/stack.coffee"] 312 | (-> @asset :aggregate-asset-paths))) 313 | 314 | (it "includes dependencies on every file in the stack" 315 | (should= ["coffeescript-source.coffee" "stack/barney.js" "stack/bedrock.stack" "stack/compiled.stack" "stack/fred.js" "stack/meta.stack" "stack/stack.coffee"] 316 | (sorted-dependencies @asset))) 317 | 318 | 319 | (it "has the correct aggregated content" 320 | (should (have-same-content "expected/meta.js" @asset)))) 321 | 322 | (context "stack with missing component" 323 | 324 | (it "throws a reasonable exception when a component is missing" 325 | (try 326 | (get-asset-uri @twixt-context "stack/missing-component.stack") 327 | (should-fail) 328 | (catch Exception e 329 | (should= "Could not locate resource `stack/does-not-exist.coffee' (a component of `stack/missing-component.stack')." 330 | (.getMessage e)))))) 331 | 332 | (context "CSS stack" 333 | 334 | (with-all asset (@pipeline "stack/style.stack" @twixt-context)) 335 | 336 | (it "has the content type from the stack file" 337 | (should= "text/css" (:content-type @asset))) 338 | 339 | (it "identifies the correct aggregated asset paths" 340 | (should= ["stack/local.less" "sample.less"] 341 | (-> @asset :aggregate-asset-paths))) 342 | 343 | (it "identifies the correct dependencies" 344 | ;; Remember that a change to aviso-logo.png will change the checksum in the 345 | ;; compiled and CSS-rewritten stack/local.css, so it must be a dependency 346 | ;; of the final asset. 347 | (should= ["aviso-logo.png" "colors.less" "sample.less" "stack/local.less" "stack/style.stack"] 348 | (sorted-dependencies @asset))) 349 | 350 | ;; Again, need to think about stripping out the sourceMappingURL lines. 351 | (it "contains the correct aggregated content" 352 | (should (have-same-content "expected/style.css" @asset)))) 353 | 354 | (context "get-asset-uri" 355 | (it "always returns the stack asset URI" 356 | (should= "/assets/[hash]/stack/bedrock.stack" 357 | (->> (get-asset-uri @twixt-context "stack/bedrock.stack") 358 | remove-hash-from-uri)))) 359 | 360 | (context "get-asset-uris" 361 | 362 | (it "returns a list of component asset URIs in place of a stack" 363 | 364 | (should= ["/assets/[hash]/stack/fred.js" "/assets/[hash]/stack/barney.js"] 365 | (->> (get-asset-uris @twixt-context "stack/bedrock.stack") 366 | (map remove-hash-from-uri)))) 367 | 368 | (it "returns just the stack asset URI in production mode" 369 | (should= ["/assets/[hash]/stack/bedrock.stack"] 370 | (->> (get-asset-uris (assoc @twixt-context :development-mode false) "stack/bedrock.stack") 371 | (map remove-hash-from-uri)))))) 372 | 373 | (context "JavaScript Minimization" 374 | (context "is enabled by default in production mode" 375 | 376 | (with-all prod-options (-> @options 377 | (assoc :development-mode false 378 | :js-optimizations :default) 379 | (with-sub-cache-dir "js-min-1"))) 380 | 381 | (with-all prod-pipeline (default-asset-pipeline @prod-options)) 382 | 383 | (with-all prod-twixt-context (assoc @prod-options :asset-pipeline @prod-pipeline)) 384 | 385 | (with-all asset (@prod-pipeline "stack/meta.stack" @prod-twixt-context)) 386 | 387 | (it "contains the correct minimized content" 388 | (should (have-same-content "expected/minimized/meta.js" @asset))) 389 | 390 | (it "can handle much larger files" 391 | (should (have-same-content "expected/minimized/bootstrap.js" 392 | (@prod-pipeline "stack/bootstrap.stack" @prod-twixt-context))))) 393 | 394 | (context "can be disabled in production mode" 395 | 396 | (with-all prod-options (-> @options 397 | (assoc :development-mode false 398 | :js-optimizations :none) 399 | (with-sub-cache-dir "js-min-2"))) 400 | 401 | (with-all prod-pipeline (default-asset-pipeline @prod-options)) 402 | 403 | (with-all prod-twixt-context (assoc @prod-options :asset-pipeline @prod-pipeline)) 404 | 405 | (with-all asset (@prod-pipeline "stack/meta.stack" @prod-twixt-context)) 406 | 407 | (it "contains the correct unoptimized content" 408 | (should (have-same-content "expected/meta.js" @asset)))) 409 | 410 | (context "can be enabled in development mode" 411 | 412 | (with-all dev-options (-> @options 413 | (assoc :js-optimizations :simple) 414 | (with-sub-cache-dir "js-min-3"))) 415 | 416 | (with-all dev-pipeline (default-asset-pipeline @dev-options)) 417 | 418 | (with-all dev-twixt-context (assoc @dev-options :asset-pipeline @dev-pipeline)) 419 | 420 | (with-all asset (@dev-pipeline "stack/fred.js" @dev-twixt-context)) 421 | 422 | (it "contains the correct unoptimized content" 423 | (should (have-same-content "expected/minimized/fred.js" @asset))))) 424 | 425 | (context "CSS minification" 426 | (it "can be expicitly enabled" 427 | (let [asset (-> @options 428 | (assoc :minimize-css true) 429 | (with-sub-cache-dir "css-min-1") 430 | (get-asset-with-options "sample.less"))] 431 | (should 432 | (have-same-content "expected/minimized/sample.css" asset)))) 433 | 434 | (it "is enabled by default in production mode" 435 | (let [asset (-> @options 436 | (assoc :development-mode false) 437 | (with-sub-cache-dir "css-min-2") 438 | ;; This is not a terrific idea in that a change to the Bootstrap dependency 439 | ;; will invalidate this test. On the other hand, it ensures (on the side) 440 | ;; that YUICompressor can handle compiling all of Bootstrap. 441 | (get-asset-with-options "bootstrap/less/bootstrap.less"))] 442 | (should 443 | (have-same-content "expected/minimized/bootstrap.css" asset))))) 444 | 445 | (context "asset redirector" 446 | (with-all wrapped (ring/wrap-with-asset-redirector (constantly nil))) 447 | 448 | (with-all request {:uri "/sample.less" :twixt @twixt-context}) 449 | 450 | (with-all response (@wrapped @request)) 451 | 452 | (it "sends a 302 response" 453 | 454 | (should= 302 (:status @response))) 455 | 456 | (it "sends an empty body" 457 | (should= "" (:body @response))) 458 | 459 | (it "sends a proper Location header" 460 | (should (re-matches #"/assets/.*/sample.less" (get-in @response [:headers "Location"])))) 461 | 462 | (it "returns nil for non-matching paths" 463 | (do-template [path] 464 | (should-be-nil (@wrapped {:uri path :twixt @twixt-context})) 465 | "/" 466 | "/a-folder" 467 | "/another/folder/"))) 468 | 469 | (context "exporter" 470 | 471 | (with-all ring-handler (startup/wrap-with-twixt 472 | (-> (fn [request] 473 | (-> request 474 | :twixt 475 | (get-asset-uri "sub/jade-include.jade"))) 476 | (resource/wrap-resource "target/exported")) 477 | (-> twixt/default-options 478 | (assoc-in [:exports :output-dir] "target/exported") 479 | (update-in [:exports :assets] into ["sub/jade-include.jade"])) 480 | true)) 481 | 482 | 483 | (it "can export a file" 484 | (let [response (@ring-handler {:request-method :get 485 | :uri "/sub/jade-include.jade"})] 486 | 487 | 488 | (should-have-content (read-resource-content "expected/jade-include.html") 489 | (-> "target/exported/sub/jade-include.jade" 490 | io/file 491 | read-as-trimmed-string)))) 492 | 493 | (it "exposes the exported alias as the asset URI" 494 | (should= "/sub/jade-include.jade" 495 | (@ring-handler {:request-method :get 496 | :uri "any-match-ok"})))) 497 | 498 | ;; Slightly bogus; this lets the mass of exceptions written out by the executing tests have a chance to finish 499 | ;; before speclj outputs the report; without it, you often get a jumble of console output (including formatted exceptions) 500 | ;; and the report as well. Perhaps another solution is to get speclj to pipe its output through clojure.tools.logging? 501 | (it "needs to slow down to let the console catch up" 502 | (Thread/sleep 1000)) 503 | 504 | (after-all (System/gc))) 505 | 506 | 507 | 508 | (run-specs) -------------------------------------------------------------------------------- /spec/io/aviso/utils_spec.clj: -------------------------------------------------------------------------------- 1 | (ns io.aviso.utils-spec 2 | (:use 3 | clojure.template 4 | io.aviso.twixt.utils 5 | speclj.core) 6 | (:require 7 | [clojure.java.io :as io])) 8 | 9 | 10 | (defn- get-modified-at 11 | [resource-path] 12 | (-> resource-path io/resource modified-at .getTime)) 13 | 14 | (describe "io.aviso.twixt.utils" 15 | 16 | (context "compute-relative-path" 17 | 18 | (it "can compute relative paths" 19 | 20 | (do-template [start relative expected] 21 | 22 | (should= expected 23 | (compute-relative-path start relative)) 24 | 25 | "foo/bar.gif" "baz.png" "foo/baz.png" 26 | 27 | "foo/bar.gif" "./baz.png" "foo/baz.png" 28 | 29 | "foo/bar.gif" "../zip.zap" "zip.zap" 30 | 31 | "foo/bar/baz/biff.gif" "../gnip/zip.zap" "foo/bar/gnip/zip.zap" 32 | 33 | "foo/bar/gif" "../frozz/pugh.pdf" "foo/frozz/pugh.pdf")) 34 | 35 | 36 | (it "throws IllegalArgumentException if ../ too many times" 37 | (should-throw IllegalArgumentException 38 | (compute-relative-path "foo/bar.png" "../../too-high.pdf")))) 39 | 40 | (context "access to time modified" 41 | 42 | ;; This is to verify some underpinnings 43 | 44 | (it "can access time modified for a local file" 45 | 46 | ;; This will be on the classpath, but on the filesystem since it is part of the current project. 47 | 48 | (should (-> "META-INF/assets/colors.less" 49 | get-modified-at 50 | pos?))) 51 | 52 | (it "can access time modified for a resource in a JAR" 53 | (should (-> "META-INF/leiningen/io.aviso/pretty/LICENSE" 54 | get-modified-at 55 | pos?))))) 56 | 57 | 58 | (run-specs) -------------------------------------------------------------------------------- /src/io/aviso/twixt.clj: -------------------------------------------------------------------------------- 1 | (ns io.aviso.twixt 2 | "Pipeline for accessing and transforming assets for streaming. 3 | 4 | Twixt integrates with Ring, as a set of Ring request filters. 5 | 6 | Twixt plugs into the Ring pipeline, but most of its work is done in terms of the asset pipeline. 7 | 8 | An asset pipeline handler is passed two values: an asset path and a Twixt context, and returns an asset. 9 | 10 | The asset path is a string, identifying the location of the asset on the classpath, beneath 11 | `META-INF/assets`. 12 | 13 | The Twixt context provides additional information that may be needed 14 | when resolving the asset. 15 | 16 | The asset itself is a map with a specific set of keys. 17 | 18 | As with Ring, there's the concept of middleware; functions that accept an asset handler (plus optional 19 | additional parameters) and return a new asset handler." 20 | (:require [clojure.java.io :as io] 21 | [io.aviso.toolchest.macros :refer [cond-let]] 22 | [io.aviso.tracker :as t] 23 | [io.aviso.twixt 24 | [asset :as asset] 25 | [compress :as compress] 26 | [css-rewrite :as rewrite] 27 | [css-minification :as cssmin] 28 | [fs-cache :as fs] 29 | [memory-cache :as mem] 30 | [js-minification :as js] 31 | [schemas :refer [Asset AssetHandler AssetPath AssetURI TwixtContext ResourcePath]] 32 | [utils :as utils]] 33 | [ring.util.mime-type :as mime] 34 | [schema.core :as s] 35 | [clojure.string :as str]) 36 | (:import [java.net URL] 37 | org.webjars.WebJarAssetLocator 38 | [java.io File])) 39 | 40 | ;;; Lots of stuff from Tapestry 5.4 is not yet implemented 41 | ;;; - multiple domains (the context, the file system, etc.) 42 | ;;; - CSS minification 43 | ;;; - AMD/RequireJS modules 44 | 45 | (defn- extract-file-extension [^String path] 46 | (let [dotx (.lastIndexOf path ".")] 47 | (.substring path (inc dotx)))) 48 | 49 | (defn- extract-content-type 50 | "Uses the resource-path's file extension to identify the content type." 51 | [content-types resource-path] 52 | (get content-types (extract-file-extension resource-path) "application/octet-stream")) 53 | 54 | (s/defn new-asset :- Asset 55 | "Create a new Asset. 56 | 57 | content-types 58 | : The content-types map from the Twixt options. 59 | 60 | asset-path 61 | : The path to the asset from the asset Root. This may be used in some tracking/debugging output. 62 | 63 | resource-path 64 | : The path to the asset on the classpath, used to locate the Asset's raw content. 65 | 66 | url 67 | : A URL used to access the raw content for the asset." 68 | {:size "0.1.17"} 69 | [content-types :- {s/Str s/Str} 70 | asset-path :- AssetPath 71 | resource-path :- ResourcePath 72 | url :- URL] 73 | (let [^bytes content-bytes (utils/read-content url) 74 | checksum (utils/compute-checksum content-bytes) 75 | modified-at (utils/modified-at url)] 76 | {:asset-path asset-path 77 | :resource-path resource-path 78 | :modified-at modified-at 79 | :content-type (extract-content-type content-types resource-path) 80 | :content content-bytes 81 | :size (alength content-bytes) 82 | :checksum checksum 83 | :dependencies {resource-path {:asset-path asset-path 84 | :checksum checksum 85 | :modified-at modified-at}}})) 86 | 87 | (s/defn make-asset-resolver :- AssetHandler 88 | "Factory for the standard Asset resolver function which converts a path into an Asset. 89 | 90 | The factory function is passed the Twixt options, which defines content types mappings. 91 | 92 | The resolver function is passed an asset path and a pipeline context (which is ignored); 93 | The asset path is converted to a classpath resource via the configuration; 94 | if the resource exists, it is converted to an Asset. 95 | 96 | If the Asset does not exist, the resolver returns nil. 97 | 98 | An Asset has the minimum following keys: 99 | 100 | :content 101 | : content of the asset in a form that is compatible with clojure.java.io 102 | 103 | :asset-path 104 | : path of the asset under the root folder /META-INF/assets/ 105 | 106 | :resource-path 107 | : full path of the underlying resource 108 | 109 | :content-type 110 | : MIME type of the content, as determined from the path's extension 111 | 112 | :size 113 | : size of the asset in bytes 114 | 115 | :checksum 116 | : Adler32 checksum of the content 117 | 118 | :modified-at 119 | : instant at which the file was last modified (not always trustworthy for files packaged in JARs) 120 | 121 | :compiled 122 | : _optional_ - true for assets that represent some form of compilation (or aggregation) and should be cached 123 | 124 | :aggregate-asset-paths 125 | : _optional_ - seq of asset paths from which a stack asset was constructed 126 | 127 | :dependencies 128 | : _optional_ - used to track underlying dependencies; a map of resource path to details about that resource 129 | (keys :asset-path, :checksum, and :modified-at) 130 | 131 | :attachments 132 | : _optional_ - map of string name to attachment (with keys :content, :size, and :content-type)" 133 | [twixt-options] 134 | (let [{:keys [content-types]} twixt-options] 135 | (fn [asset-path _context] 136 | (let [resource-path (str "META-INF/assets/" asset-path)] 137 | (if-let [url (io/resource resource-path)] 138 | (new-asset content-types asset-path resource-path url)))))) 139 | 140 | (s/defn make-webjars-asset-resolver :- AssetHandler 141 | "As with [[make-asset-resolver]], but finds assets inside WebJars. 142 | 143 | The path must start with a WebJar libary name, such as \"bootstrap\". 144 | The version number is injected into the resource path." 145 | {:since "0.1.17"} 146 | [twixt-options] 147 | (let [webjar-versions (into {} (-> (WebJarAssetLocator.) .getWebJars)) 148 | content-types (:content-types twixt-options)] 149 | (fn [asset-path _context] 150 | (cond-let 151 | [[webjar-name webjar-asset-path] (str/split asset-path #"/" 2) 152 | webjar-version (get webjar-versions webjar-name)] 153 | 154 | (nil? webjar-version) 155 | nil 156 | 157 | [resource-path (str "META-INF/resources/webjars/" webjar-name "/" webjar-version "/" webjar-asset-path) 158 | url (io/resource resource-path)] 159 | 160 | (some? url) 161 | (new-asset content-types asset-path resource-path url))))) 162 | 163 | 164 | (def default-options 165 | "Provides the default options when using Twixt; these rarely need to be changed except, perhaps, for :path-prefix 166 | or :cache-folder, or by plugins." 167 | {:path-prefix "/assets/" 168 | :content-types mime/default-mime-types 169 | :resolver-factories [make-asset-resolver make-webjars-asset-resolver] 170 | ;; Content transformer, e.g., compilers (such as CoffeeScript to JavaScript). Key is a content type, 171 | ;; value is a function passed an asset and Twixt context, and returns a new asset. 172 | :content-transformers {} 173 | ;; Identify which content types are compressable; all other content types are assumed to not be compressable. 174 | :compressable #{"text/*" "application/edn" "application/json"} 175 | :js-optimizations :default 176 | :cache {:cache-dir (System/getProperty "twixt.cache-dir" (System/getProperty "java.io.tmpdir")) 177 | :check-interval-ms 1000} 178 | :exports {:interval-ms 5000 179 | :output-dir "resources/public" 180 | :output-uri "" 181 | :assets []}}) 182 | 183 | 184 | (s/defn find-asset :- (s/maybe Asset) 185 | "Looks for a particular asset by asset path, returning the asset (if found)." 186 | {:added "0.1.17"} 187 | [asset-path :- AssetPath 188 | context :- TwixtContext] 189 | ((:asset-pipeline context) asset-path context)) 190 | 191 | (s/defn asset->uri :- s/Str 192 | "Converts an Asset to a URI that can be provided to a client." 193 | {:added "0.1.17"} 194 | [context :- TwixtContext 195 | asset :- Asset] 196 | (if-let [alias-uri (get (:asset-aliases context) (:asset-path asset))] 197 | alias-uri 198 | (asset/asset->request-path (:path-prefix context) asset))) 199 | 200 | (defn- get-single-asset 201 | [asset-pipeline context asset-path] 202 | {:pre [(some? asset-pipeline) 203 | (some? context) 204 | (some? asset-path)]} 205 | (or (asset-pipeline asset-path context) 206 | (throw (ex-info (format "Asset path `%s' does not map to an available resource." asset-path) 207 | context)))) 208 | 209 | (s/defn get-asset-uris :- [AssetURI] 210 | "Converts a number of asset paths into client URIs. 211 | Each path must exist. 212 | 213 | An asset path does not start with a leading slash. 214 | The default asset resolver locates each asset on the classpath under `META-INF/assets/`. 215 | 216 | Unlike [[get-asset-uri]], this function will (in _development mode_) expand stack assets 217 | into the asset URIs for all component assets. 218 | 219 | context 220 | : the :twixt key, extracted from the Ring request map 221 | 222 | paths 223 | : asset paths to convert to URIs" 224 | [{:keys [asset-pipeline development-mode] :as context} :- TwixtContext 225 | & paths :- [AssetPath]] 226 | (loop [asset-uris [] 227 | [path & more-paths] paths] 228 | (if-not (some? path) 229 | asset-uris 230 | (let [asset (get-single-asset asset-pipeline context path) 231 | aggregate-asset-paths (-> asset :aggregate-asset-paths seq)] 232 | (if (and development-mode aggregate-asset-paths) 233 | (recur (into asset-uris (apply get-asset-uris context aggregate-asset-paths)) 234 | more-paths) 235 | (recur (conj asset-uris (asset->uri context asset)) 236 | more-paths)))))) 237 | 238 | (s/defn get-asset-uri :- AssetURI 239 | "Converts a single asset paths into a client URI. 240 | Throws an exception if the path does not exist. 241 | 242 | The default asset resolver locates each asset on the classpath under `META-INF/assets/`. 243 | 244 | This works much the same as [[get-asset-uris]] except that stack asset will 245 | be a URI to the stack (whose content is the aggregation of the components of the stack) 246 | rather than the list of component asset URIs. 247 | This matches the behavior of `get-asset-uris` in _production_, but you should use 248 | `get-asset-uris` in development, since in development, you want the individual 249 | component assets, rather than the aggregated whole. 250 | 251 | context 252 | : the :twixt key, extracted from the Ring request map 253 | 254 | asset-path 255 | : path to the asset; asset paths do __not__ start with a leading slash" 256 | [context :- TwixtContext 257 | asset-path :- AssetPath] 258 | (let [{:keys [asset-pipeline]} context] 259 | (->> 260 | asset-path 261 | (get-single-asset asset-pipeline context) 262 | (asset->uri context)))) 263 | 264 | (s/defn find-asset-uri :- (s/maybe AssetURI) 265 | "Returns the URI for an asset, if it exists. 266 | If not, returns nil." 267 | [{:keys [asset-pipeline] :as context} 268 | asset-path :- AssetPath] 269 | (if-let [asset (asset-pipeline asset-path context)] 270 | (asset->uri context asset))) 271 | 272 | (s/defn wrap-pipeline-with-tracing :- AssetHandler 273 | "The first middleware in the asset pipeline, used to trace the construction of the asset." 274 | [asset-handler :- AssetHandler] 275 | (fn [asset-path context] 276 | (t/track 277 | #(format "Accessing asset `%s'" asset-path) 278 | (asset-handler asset-path context)))) 279 | 280 | (s/defn wrap-pipeline-with-per-content-type-transformation :- AssetHandler 281 | [asset-handler :- AssetHandler 282 | {:keys [content-transformers]}] 283 | (fn [asset-path context] 284 | (let [asset (asset-handler asset-path context) 285 | content-type (:content-type asset) 286 | transformer (get content-transformers content-type)] 287 | (if transformer 288 | (transformer asset context) 289 | asset)))) 290 | 291 | (s/defn default-wrap-pipeline-with-content-transformation :- AssetHandler 292 | "Used when constructing the asset pipeline, wraps a handler (normally, the asset resolver) 293 | with additional pipeline handlers based on 294 | the `:content-transformers` key of the Twixt options, plus JavaScript minification and CSS URL Rewriting 295 | and CSS Minification. 296 | 297 | The JavaScript and/or CSS minification may not be enabled in development mode." 298 | [asset-handler :- AssetHandler 299 | twixt-options] 300 | (-> 301 | asset-handler 302 | (wrap-pipeline-with-per-content-type-transformation twixt-options) 303 | (js/wrap-with-javascript-minimizations twixt-options) 304 | rewrite/wrap-with-css-rewriting 305 | (cssmin/wrap-with-css-minification twixt-options))) 306 | 307 | (s/defn default-wrap-pipeline-with-caching :- AssetHandler 308 | "Used when constructing the asset pipeline to wrap the handler with file system caching 309 | of compiled assets, and in-memory caching of all assets. 310 | 311 | This is invoked before adding support for compression." 312 | [asset-handler :- AssetHandler 313 | cache-dir :- File 314 | check-interval-ms :- s/Num] 315 | (-> 316 | asset-handler 317 | ;; The file system cache should come after anything downstream that might compile. 318 | (fs/wrap-with-filesystem-cache cache-dir) 319 | ;; This in-memory cache prevents constant checks agains the file system cache, or against 320 | ;; the asset resolvers. 321 | (mem/wrap-with-memory-cache check-interval-ms))) 322 | 323 | (s/defn wrap-pipeline-with-asset-resolver :- AssetHandler 324 | "Wraps the asset handler so that the :asset-resolver key is set to the asset resolver; the asset resolver 325 | is a way to bypass intermediate steps and gain access to the asset in its completely untransformed format." 326 | [asset-handler :- AssetHandler 327 | asset-resolver] 328 | (fn [asset-path context] 329 | (asset-handler asset-path (assoc context :asset-resolver asset-resolver)))) 330 | 331 | (defn- merge-handlers [handlers] 332 | (fn [asset-path context] 333 | (loop [[h & more-handlers] handlers] 334 | (cond-let 335 | (nil? h) 336 | nil 337 | 338 | [result (h asset-path context)] 339 | 340 | (some? result) 341 | result 342 | 343 | :else 344 | (recur more-handlers))))) 345 | 346 | (s/defn default-asset-pipeline :- AssetHandler 347 | "Sets up the default pipeline. 348 | 349 | The asset pipeline starts with a resolver, which is then intercepted using asset pipeline middleware. 350 | As with Ring, middleware is a function that accepts an asset-handler and returns an asset-handler. The asset-handler 351 | is passed an asset path and a context. The initial context is the value of the `:twixt` key from the 352 | Ring request map. 353 | 354 | In production mode, JavaScript will be minimized. 355 | 356 | The context will contain an :asset-pipeline key whose value is the asset pipeline in use. 357 | The context will contain a :path-prefix key, extracted from the twixt options. 358 | The context may also be passed to [[get-asset-uri]] (and related functions). 359 | 360 | In some cases, middlware may modify the context before passing it forward to the next asset-handler, typically 361 | by adding additional keys." 362 | [twixt-options] 363 | (let [{:keys [development-mode resolver-factories]} twixt-options 364 | asset-resolvers (for [factory resolver-factories] 365 | (factory twixt-options)) 366 | asset-resolver (merge-handlers asset-resolvers) 367 | ^String cache-path (-> twixt-options :cache :cache-dir) 368 | cache-dir (File. cache-path 369 | (if development-mode "dev" "prod")) 370 | check-interval-ms (-> twixt-options :cache :check-interval-ms)] 371 | (-> 372 | asset-resolver 373 | (default-wrap-pipeline-with-content-transformation twixt-options) 374 | (default-wrap-pipeline-with-caching cache-dir check-interval-ms) 375 | (compress/wrap-with-gzip-compression twixt-options) 376 | (wrap-pipeline-with-asset-resolver asset-resolver) 377 | wrap-pipeline-with-tracing))) 378 | 379 | -------------------------------------------------------------------------------- /src/io/aviso/twixt/asset.clj: -------------------------------------------------------------------------------- 1 | (ns io.aviso.twixt.asset 2 | "Utilities for dealing with a Twixt asset map.") 3 | 4 | (defn asset->request-path 5 | "Computes the complete asset path that can be referenced by a client in order to obtain 6 | the asset content. This includes the path prefix, the checksum, and the asset path itself. Compressed assets 7 | have their checksum prefixed with 'z'." 8 | [path-prefix asset] 9 | (str path-prefix 10 | (if (:compressed asset) "z" "") 11 | (:checksum asset) 12 | "/" 13 | (:asset-path asset))) 14 | -------------------------------------------------------------------------------- /src/io/aviso/twixt/coffee_script.clj: -------------------------------------------------------------------------------- 1 | (ns io.aviso.twixt.coffee-script 2 | "Provides asset pipeline middleware to perform CoffeeScript to JavaScript compilation. 3 | 4 | CoffeeScript assets include a source map as an attachment. 5 | The source map includes the source of the CoffeeScript file. 6 | 7 | The compiled JavaScript includes the directive for the browser to load the source map." 8 | (:import 9 | [java.util Map]) 10 | (:require 11 | [io.aviso.tracker :as t] 12 | [io.aviso.twixt 13 | [rhino :as rhino] 14 | [utils :as utils]])) 15 | 16 | (defn- ^String extract-value [^Map object key] 17 | (str (.get object key))) 18 | 19 | (defn- coffee-script-compiler [asset context] 20 | (let [file-path (:resource-path asset) 21 | file-name (utils/path->name file-path)] 22 | (t/timer 23 | #(format "Compiled `%s' to JavaScript in %.2f ms" file-path %) 24 | (t/track 25 | #(format "Compiling `%s' to JavaScript" file-path) 26 | (let [^Map result 27 | (rhino/invoke-javascript ["META-INF/twixt/coffee-script.js" "META-INF/twixt/invoke-coffeescript.js"] 28 | "compileCoffeeScriptSource" 29 | (-> asset :content utils/as-string) 30 | file-path 31 | file-name)] 32 | 33 | ;; The script returns an object with key "exception" or key "output": 34 | (when (.containsKey result "exception") 35 | (throw (RuntimeException. (extract-value result "exception")))) 36 | (-> asset 37 | (utils/create-compiled-asset "text/javascript" (extract-value result "output") nil) 38 | (utils/add-attachment "source.map" "application/json" (-> result (extract-value "sourceMap") utils/as-bytes)))))))) 39 | 40 | (defn register-coffee-script 41 | "Updates the Twixt options with support for compiling CoffeeScript into JavaScript." 42 | [options] 43 | (-> options 44 | (assoc-in [:content-types "coffee"] "text/coffeescript") 45 | (assoc-in [:content-transformers "text/coffeescript"] coffee-script-compiler))) -------------------------------------------------------------------------------- /src/io/aviso/twixt/compress.clj: -------------------------------------------------------------------------------- 1 | (ns io.aviso.twixt.compress 2 | "Asset pipeline middleware for handling compressable assets." 3 | (:require [clojure.string :as str] 4 | [clojure.java.io :as io] 5 | [io.aviso.twixt 6 | [memory-cache :as mem] 7 | [utils :as utils] 8 | [schemas :refer [AssetHandler]]] 9 | [schema.core :as s]) 10 | (:import [java.io ByteArrayOutputStream] 11 | [java.util.zip GZIPOutputStream])) 12 | 13 | (defn- make-wildcard [^String mime-type] 14 | (-> 15 | mime-type 16 | (.split "/") 17 | first 18 | (str "/*"))) 19 | 20 | (defn- is-compressable-mime-type? 21 | [compressable-types mime-type] 22 | (or 23 | (compressable-types mime-type) 24 | (compressable-types (make-wildcard mime-type)))) 25 | 26 | (defn- is-compressable? 27 | [compressable-types asset] 28 | (is-compressable-mime-type? compressable-types (:content-type asset))) 29 | 30 | (defn- compress-asset 31 | [asset] 32 | (let [bos (ByteArrayOutputStream. (:size asset))] 33 | (with-open [gz (GZIPOutputStream. bos) 34 | is (io/input-stream (:content asset))] 35 | (io/copy is gz)) 36 | (-> asset 37 | (utils/replace-asset-content (:content-type asset) (.toByteArray bos)) 38 | (assoc :compressed true)))) 39 | 40 | (defn- is-gzip-supported? 41 | [request] 42 | (if-let [encodings (-> request :headers (get "accept-encoding"))] 43 | (some #(.equalsIgnoreCase ^String % "gzip") 44 | (str/split encodings #",")))) 45 | 46 | (s/defn wrap-with-gzip-compression :- AssetHandler 47 | "Adds an in-memory cache that includes invalidation checks. 48 | 49 | Although compression is expensive enough that it is not desirable to 50 | perform compression on every request, it is also overkill to 51 | commit the compressed asset to a file system cache. 52 | 53 | The cache is only consulted if the :gzip-enabled options key is true." 54 | {:added "0.1.20"} 55 | [asset-handler :- AssetHandler 56 | twixt-options] 57 | (let [cache (mem/wrap-with-transforming-cache asset-handler 58 | (partial is-compressable? (:compressable twixt-options)) 59 | compress-asset)] 60 | (fn [asset-path options] 61 | (let [delegate (if (:gzip-enabled options) cache asset-handler)] 62 | (delegate asset-path options))))) 63 | 64 | (defn wrap-with-compression-analyzer 65 | "Ring middleware that analyzes the incoming request to determine if the 66 | client can accept compressed streams." 67 | [handler] 68 | (fn [request] 69 | (handler (assoc-in request [:twixt :gzip-enabled] (is-gzip-supported? request))))) -------------------------------------------------------------------------------- /src/io/aviso/twixt/css_minification.clj: -------------------------------------------------------------------------------- 1 | (ns io.aviso.twixt.css-minification 2 | "CSS Miniification based on YUICompressor." 3 | {:added "0.1.20"} 4 | (:require [schema.core :as s] 5 | [io.aviso.twixt.schemas :refer [AssetHandler]] 6 | [io.aviso.tracker :as t] 7 | [clojure.java.io :as io] 8 | [io.aviso.twixt.utils :as utils]) 9 | (:import [java.io StringWriter] 10 | [com.yahoo.platform.yui.compressor CssCompressor])) 11 | 12 | 13 | (defn- run-minimizer 14 | [{:keys [resource-path content] :as asset}] 15 | (t/timer 16 | #(format "Minimized `%s' (%,d bytes) in %.2f ms" 17 | resource-path (:size asset) %) 18 | (t/track 19 | #(format "Minimizing `%s' using YUICompressor." resource-path) 20 | (let [compressed (with-open [writer (StringWriter. 1000) 21 | reader (io/reader content)] 22 | (doto (CssCompressor. reader) 23 | (.compress writer -1)) 24 | 25 | (.flush writer) 26 | (-> writer .getBuffer str))] 27 | (utils/create-compiled-asset asset "text/css" compressed nil))))) 28 | 29 | (s/defn wrap-with-css-minification :- AssetHandler 30 | "Enabled CSS minification based on the :compress-css key of the Twixt options. 31 | If the key is not present, the default is to enable compression only in production mode. 32 | 33 | When not enabled, returns the provided handler unchanged." 34 | [handler :- AssetHandler {:keys [development-mode] :as twixt-options}] 35 | (if (get twixt-options :minimize-css (not development-mode)) 36 | (fn [asset-path {:keys [for-aggregation] :as context}] 37 | (let [{:keys [content-type] :as asset} (handler asset-path context)] 38 | (if (and (= "text/css" content-type) 39 | (not for-aggregation)) 40 | (run-minimizer asset) 41 | asset))) 42 | handler)) -------------------------------------------------------------------------------- /src/io/aviso/twixt/css_rewrite.clj: -------------------------------------------------------------------------------- 1 | (ns io.aviso.twixt.css-rewrite 2 | "Defines an asset pipeline filter used to rewrite URIs in CSS files. 3 | Relative URLs in CSS files will become invalid due to the inclusion of the individual asset's checksum in the 4 | asset request path; instead, the relative URIs are expanded to complete URIs that include the individual asset's checksum." 5 | (:require [clojure.string :as str] 6 | [io.aviso.tracker :as t] 7 | [io.aviso.twixt 8 | [asset :as asset] 9 | [utils :as utils] 10 | [schemas :refer [AssetHandler]]] 11 | [schema.core :as s])) 12 | 13 | (def ^:private url-pattern #"url\(\s*(['\"]?)(.+?)([\#\?].*?)?\1\s*\)") 14 | (def ^:private complete-url-pattern #"^[#/]|(\p{Alpha}\w*:)") 15 | 16 | (defn- rewrite-relative-url 17 | [asset-path context dependencies ^String relative-url] 18 | (if (.startsWith relative-url "data:") 19 | relative-url 20 | (t/track 21 | #(format "Rewriting relative URL `%s'" relative-url) 22 | (let [referenced-path (utils/compute-relative-path asset-path relative-url) 23 | asset ((:asset-pipeline context) referenced-path context)] 24 | (if-not asset 25 | (throw (ex-info 26 | (format "Unable to locate asset `%s'." referenced-path) 27 | {:source-asset asset-path 28 | :relative-url relative-url}))) 29 | 30 | (swap! dependencies utils/add-asset-as-dependency asset) 31 | (asset/asset->request-path (:path-prefix context) asset))))) 32 | 33 | (defn- css-url-match-handler [asset-path context dependencies match] 34 | (let [url (nth match 2) 35 | parameters (nth match 3)] 36 | (str "url(\"" 37 | (if (re-matches complete-url-pattern url) 38 | (str url parameters) 39 | (str (rewrite-relative-url asset-path context dependencies url) 40 | parameters)) 41 | "\")"))) 42 | 43 | (defn- rewrite-css 44 | [asset context] 45 | (t/track 46 | #(format "Rewriting URLs in `%s'" (:asset-path asset)) 47 | (let [content (utils/as-string (:content asset)) 48 | ;; Using an atom this way is clumsy. 49 | dependencies (atom (:dependencies asset)) 50 | content' (str/replace content 51 | url-pattern 52 | (partial css-url-match-handler (:asset-path asset) context dependencies))] 53 | (-> 54 | asset 55 | (utils/replace-asset-content "text/css" (utils/as-bytes content')) 56 | (assoc :dependencies @dependencies))))) 57 | 58 | (s/defn wrap-with-css-rewriting :- AssetHandler 59 | "Wraps the asset handler with the CSS URI rewriting logic needed for the client to be able to properly request the referenced assets. 60 | 61 | Rewriting occurs for individual CSS assets (including those created by compiling a Less source). It does not occur for 62 | aggregated CSS assets (since the individual assets will already have had URIs rewritten)." 63 | [handler :- AssetHandler] 64 | (fn [asset-path context] 65 | (let [asset (handler asset-path context)] 66 | (if (and 67 | (= "text/css" (:content-type asset)) 68 | (-> asset :aggregate-asset-paths empty?)) 69 | (rewrite-css asset context) 70 | asset)))) 71 | -------------------------------------------------------------------------------- /src/io/aviso/twixt/exceptions.clj: -------------------------------------------------------------------------------- 1 | (ns io.aviso.twixt.exceptions 2 | "Support for generating pretty and useful HTML reports when server-side exceptions occur." 3 | (:use hiccup.core 4 | hiccup.page 5 | ring.util.response) 6 | (:require [clojure.string :as s] 7 | [io.aviso 8 | [exception :as exception] 9 | [twixt :as t]]) 10 | (:import [clojure.lang APersistentMap Sequential PersistentHashMap] 11 | [java.util Map] 12 | [java.util.regex Pattern])) 13 | 14 | (def ^:private path-separator (System/getProperty "path.separator")) 15 | 16 | (defn- exception-message 17 | [^Throwable exception] 18 | (or (.getMessage exception) 19 | (-> exception .getClass .getName))) 20 | 21 | (defprotocol MarkupGeneration 22 | "Used to convert arbitrary values into markup strings. 23 | 24 | Extended onto nil, `String`, `APersistentMap`, `Sequential` and `Object`." 25 | (to-markup 26 | [value] 27 | "Returns HTML markup representing the value.")) 28 | 29 | (extend-type nil 30 | MarkupGeneration 31 | (to-markup [_] "nil")) 32 | h 33 | (extend-type String 34 | MarkupGeneration 35 | (to-markup [value] (h value))) 36 | 37 | (extend-type Object 38 | MarkupGeneration 39 | (to-markup [value] (-> value .toString h))) 40 | 41 | (extend-type APersistentMap 42 | MarkupGeneration 43 | (to-markup 44 | [m] 45 | (html 46 | (if (empty? m) 47 | [:em "empty map"] 48 | [:dl 49 | (apply concat 50 | (for [[k v] (sort-by str m)] 51 | [[:dt (to-markup k)] [:dd (to-markup v)]] 52 | )) 53 | ])))) 54 | 55 | (defn- seq->markup [coll] 56 | ;; Since *print-length* is normally nil, we provide an actual limit 57 | (let [limit (or *print-length* 10) 58 | values (take (inc limit) coll) 59 | trimmed (if (<= (count values) limit) 60 | values 61 | (concat 62 | (take limit values) 63 | ["..."]))] 64 | (for [v trimmed] 65 | [:li (to-markup v)]))) 66 | 67 | (extend-type Sequential 68 | MarkupGeneration 69 | (to-markup 70 | [coll] 71 | (html 72 | (if (empty? coll) 73 | [:em "none"] 74 | [:ul (seq->markup coll)])))) 75 | 76 | (extend-type Map 77 | MarkupGeneration 78 | (to-markup 79 | [^Map m] 80 | (to-markup (PersistentHashMap/create m)))) 81 | 82 | (defn- element->clojure-name [element] 83 | (let [names (:names element)] 84 | (list 85 | (s/join "/" (drop-last names)) 86 | "/" 87 | [:strong (last names)]))) 88 | 89 | (defn- element->java-name 90 | [{:keys [package class simple-class method]}] 91 | (list 92 | (if package 93 | (list 94 | [:span.package-name package] 95 | "." 96 | simple-class) 97 | class) 98 | " — " 99 | method)) 100 | 101 | (defn- apply-stack-frame-filter 102 | [filter frames] 103 | (loop [result [] 104 | [frame & more-frames] frames] 105 | (if (nil? frame) 106 | result 107 | (case (filter frame) 108 | :show 109 | (recur (conj result frame) more-frames) 110 | 111 | ;; We could perhaps do this differntly, but unlike console output, we treat these the same. 112 | (:hide :omit) 113 | (recur (conj result (assoc frame :filtered true)) more-frames) 114 | 115 | :terminate 116 | result)))) 117 | 118 | (defn- stack-frame->row-markup 119 | [{:keys [file line names] 120 | :as frame}] 121 | (let [clojure? (-> names empty? not) 122 | java-name (element->java-name frame)] 123 | [:tr (if (:filtered frame) {:class :filtered} {}) 124 | [:td.function-name 125 | (if clojure? 126 | (list 127 | (element->clojure-name frame) 128 | [:div.filtered java-name]) 129 | java-name) 130 | ] 131 | [:td.source-location file (if line ":")] 132 | [:td (or line "")] 133 | ])) 134 | 135 | (defn- single-exception->markup 136 | "Given an analyzed exception, generate markup for it." 137 | [twixt 138 | {^Throwable e :exception 139 | :keys [class-name message properties root]}] 140 | (html 141 | [:div.panel.panel-default 142 | [:div.panel-heading 143 | [:h3.panel-title class-name] 144 | ] 145 | [:div.panel-body 146 | [:h4 message] 147 | ;; TODO: sorting 148 | (when-not (empty? properties) 149 | [:dl 150 | (apply concat 151 | (for [[k v] (sort-by str properties)] 152 | [[:dt (-> k h)] [:dd (to-markup v)]])) 153 | ])] 154 | (if root 155 | (list 156 | [:div.btn-toolbar.spacing-below 157 | [:button.btn.btn-default.btn-sm {:data-action :toggle-filter} "Toggle Stack Frame Filter"]] 158 | [:table.table.table-hover.table-condensed.table-striped 159 | (->> 160 | e 161 | exception/expand-stack-trace 162 | (apply-stack-frame-filter (:stack-frame-filter twixt)) 163 | (map stack-frame->row-markup)) 164 | ])) 165 | ])) 166 | 167 | 168 | (defn exception->markup 169 | "Returns the markup (as a string) for a root exception and its stack of causes, 170 | including a stack trace for the deepest exception." 171 | [twixt root-exception] 172 | (->> 173 | root-exception 174 | exception/analyze-exception 175 | (map (partial single-exception->markup twixt)) 176 | (apply str))) 177 | 178 | (defn- is-path? 179 | [^String k ^String v] 180 | (and (.endsWith k ".path") 181 | (.contains v path-separator))) 182 | 183 | (defn- split-string [^String s ^String sep] 184 | (s/split s (-> sep Pattern/quote Pattern/compile))) 185 | 186 | (defn- path->markup [^String path] 187 | `[:ul 188 | ~@(for [v (.split path path-separator)] 189 | [:li v])]) 190 | 191 | (defn- sysproperties->markup 192 | [] 193 | (let [props (System/getProperties)] 194 | [:dl 195 | (apply concat 196 | (for [k (-> props keys sort) 197 | :let [v (.get props k)]] 198 | [[:dt k] 199 | [:dd (if (is-path? k v) (path->markup v) v)]]))])) 200 | 201 | (defn build-report 202 | "Builds an HTML exception report (as a string). 203 | 204 | request 205 | : Ring request map, which must contain the `:twixt` key 206 | 207 | exception 208 | : instance of Throwable to report" 209 | [request exception] 210 | (let [twixt (:twixt request)] 211 | (html5 212 | [:head 213 | [:title "Exception"] 214 | (apply include-css (t/get-asset-uris twixt 215 | "bootstrap/css/bootstrap.css" 216 | "twixt/exception.less"))] 217 | [:body.hide-filtered 218 | [:div.container 219 | [:div.panel.panel-danger 220 | [:div.panel-heading 221 | [:h3.panel-title "An unexpected exception has occurred."]] 222 | [:div.panel-body 223 | [:h3 (h (exception-message exception))] 224 | ] 225 | ] 226 | (exception->markup twixt exception) 227 | 228 | [:h3 "Request"] 229 | 230 | (to-markup request) 231 | 232 | [:h3 "System Properties"] 233 | (sysproperties->markup) 234 | ] 235 | (apply include-js (t/get-asset-uris twixt 236 | "jquery/jquery.js" 237 | "twixt/exception.coffee")) 238 | ]))) 239 | 240 | (defn wrap-with-exception-reporting 241 | "Wraps the handler to report any uncaught exceptions as an HTML exception report. 242 | This wrapper should wrap around other handlers (including the Twixt handler itself), but be nested within 243 | the twixt-setup handler (which provides the `:twixt` request map key)." 244 | [handler] 245 | (fn [request] 246 | (try 247 | (handler request) 248 | (catch Throwable t 249 | (-> 250 | (build-report request t) 251 | response 252 | (content-type "text/html") 253 | (status 500)))))) 254 | 255 | 256 | (defn default-stack-frame-filter 257 | "The default stack frame filter function, used by the HTML excepton report to identify frames that can be hidden 258 | by default. 259 | 260 | This implementation extends the standard frame filter (`io.aviso.repl/standard-frame-filter`), 261 | to also omit frames with no line number. 262 | 263 | The HTML exception report treats `:omit` and `:hide` identically; frames marked as either are 264 | initially hidden, but can be revealed in the client." 265 | [frame] 266 | (cond 267 | (nil? (:line frame)) 268 | :omit 269 | 270 | :else 271 | (exception/*default-frame-filter* frame))) 272 | 273 | (defn register-exception-reporting 274 | "Must be invoked to configure the Twixt options with the default `:stack-frame-filter` key, [[default-stack-frame-filter]]." 275 | [options] 276 | (assoc options :stack-frame-filter default-stack-frame-filter)) 277 | 278 | -------------------------------------------------------------------------------- /src/io/aviso/twixt/export.clj: -------------------------------------------------------------------------------- 1 | (ns io.aviso.twixt.export 2 | "Support for exporting assets to the file system. 3 | 4 | Ideally, exporting would be fully asynchronous; however that would require a dedicated thread, and 5 | that is a problem as there is no lifecycle for 6 | a Ring request handler to know when the Ring server is itself shutdown." 7 | {:added "0.1.17"} 8 | (:require [schema.core :as s] 9 | [io.aviso.twixt.schemas :refer [AssetPath ExportsConfiguration]] 10 | [io.aviso.tracker :as t] 11 | [io.aviso.toolchest.macros :refer [cond-let]] 12 | [io.aviso.twixt :as twixt] 13 | [clojure.java.io :as io] 14 | [clojure.tools.logging :as l])) 15 | 16 | (deftype ^:private Token [running? 17 | ^Thread thread] 18 | 19 | Object 20 | (finalize [_] 21 | (reset! running? false) 22 | (.interrupt thread))) 23 | 24 | (defn- export-asset 25 | [output-dir {:keys [asset-path content]} output-alias] 26 | (t/track 27 | (if (= asset-path output-alias) 28 | (format "Exporting `%s'." asset-path) 29 | (format "Exporting `%s' (as `%s')." asset-path output-alias)) 30 | 31 | (let [output-file (io/file output-dir output-alias)] 32 | (when-not (.exists output-file) 33 | (-> output-file .getParentFile .mkdirs)) 34 | 35 | (io/copy content output-file)))) 36 | 37 | (defn- check-exports 38 | [context output-dir assets checksums] 39 | (t/track 40 | "Checking exported assets for changes." 41 | (let [context' (assoc context :gzip-enabled false)] 42 | (doseq [[asset-path output-alias] assets] 43 | (try 44 | (t/track 45 | (format "Checking `%s' for changes." asset-path) 46 | (cond-let 47 | [asset (twixt/find-asset asset-path context')] 48 | 49 | (nil? asset) 50 | nil 51 | 52 | [asset-checksum (:checksum asset)] 53 | 54 | (= asset-checksum (get @checksums asset-path)) 55 | nil 56 | 57 | :else 58 | (do 59 | (export-asset output-dir asset output-alias) 60 | (swap! checksums assoc asset-path asset-checksum))) 61 | (catch Throwable _ 62 | ;; Reported by the tracker and ignored. 63 | ))))))) 64 | 65 | 66 | (defn- start-exporter-thread 67 | [context {:keys [interval-ms output-dir assets]}] 68 | (let [checksums (atom {}) 69 | ;; We want the assets to all be the same shape: asset path and output path. 70 | assets' (map #(if (string? %) 71 | [% %] 72 | %) 73 | assets) 74 | running? (atom true) 75 | first-pass (promise) 76 | export-body (fn [] 77 | (l/info "Twixt asset export thread started.") 78 | (while @running? 79 | (try 80 | (check-exports context output-dir assets' checksums) 81 | (deliver first-pass true) 82 | (Thread/sleep interval-ms) 83 | 84 | (catch Throwable _)) 85 | ;; Real errors are reported inside check-exports, and InterruptedException is ignored. 86 | ) 87 | (l/info "Shutting down Twixt asset export thread.")) 88 | export-thread (doto (Thread. ^Runnable export-body) 89 | (.setName "Twixt Export") 90 | (.setDaemon true) 91 | .start)] 92 | ;; Wait for completion of first export pass: 93 | 94 | @first-pass 95 | 96 | ;; The token is returned to the call, which keeps it in an atom. At shutdown time, 97 | ;; the entire Ring request pipeline will be GC'ed, at which point the token will become 98 | ;; weakly referenced. 99 | (Token. running? export-thread))) 100 | 101 | 102 | (defn- build-asset-aliases 103 | [assets output-uri] 104 | (reduce (fn [result asset] 105 | (if (string? asset) 106 | (assoc result asset (str output-uri "/" asset)) 107 | (let [[asset-path output-alias] asset] 108 | (assoc result asset-path (str output-uri "/" output-alias))))) 109 | {} 110 | assets)) 111 | 112 | (s/defn wrap-with-exporter 113 | "Wraps a Ring handler so that, periodically, assets identified in the configuration 114 | are checked for changes and copied out to the file system (as needed). 115 | 116 | The first checks and exports occur before delegating to the wrapped handler; this will allow 117 | a standard resource handler (say, ring.middleware.resource/wrap-resource) to operate. 118 | 119 | Subsequent checks happen at intervals on a secondary thread. 120 | The thread will shutdown once the Ring request pipeline is GC'ed. 121 | It requires an explicit call to System/gc to ensure that the pipeline is GC'ed 122 | (and the necessary object finalizer called). 123 | 124 | Note that asset exporting is largely intended for development purposes." 125 | [ring-handler 126 | {:keys [assets output-uri] :as configuration} :- ExportsConfiguration] 127 | (if (empty? assets) 128 | ring-handler 129 | (let [token (atom nil) 130 | asset-aliases (build-asset-aliases assets output-uri)] 131 | (fn [request] 132 | (when (nil? @token) 133 | (reset! token (start-exporter-thread (:twixt request) configuration))) 134 | 135 | (ring-handler (assoc-in request [:twixt :asset-aliases] asset-aliases)))))) 136 | -------------------------------------------------------------------------------- /src/io/aviso/twixt/fs_cache.clj: -------------------------------------------------------------------------------- 1 | (ns io.aviso.twixt.fs-cache 2 | "Provide asset pipeline middleware that implements a file-system cache. 3 | The cache will persist between executions of the application; this is used in development to prevent 4 | duplicated, expensive work from being performed after a restart." 5 | (:import [java.util UUID] 6 | [java.io PushbackReader File Writer]) 7 | (:require 8 | [clojure.java.io :as io] 9 | [clojure 10 | [edn :as edn] 11 | [pprint :as pp]] 12 | [medley.core :as medley] 13 | [clojure.tools.logging :as l] 14 | [io.aviso.tracker :as t])) 15 | 16 | 17 | (defn- checksum-matches? 18 | [asset-resolver asset-path cached-checksum] 19 | (-> 20 | asset-path 21 | (asset-resolver nil) 22 | :checksum 23 | (= cached-checksum))) 24 | 25 | (defn- is-valid? 26 | [asset-resolver cached-asset] 27 | (and cached-asset 28 | (every? 29 | (fn [{:keys [asset-path checksum]}] 30 | (checksum-matches? asset-resolver asset-path checksum)) 31 | (-> cached-asset :dependencies vals)))) 32 | 33 | (defn- read-cached-asset-data 34 | [^File file] 35 | (if (.exists file) 36 | (with-open [reader (-> file io/reader PushbackReader.)] 37 | (edn/read reader)))) 38 | 39 | (defn- read-attachments 40 | [asset asset-cache-dir] 41 | (if (-> asset :attachments empty?) 42 | asset 43 | (update-in asset [:attachments] 44 | (fn [attachments] 45 | (medley/map-vals 46 | #(assoc % :content (io/file asset-cache-dir (:content %))) 47 | attachments))))) 48 | 49 | (def ^:private asset-file-name "asset.edn") 50 | (def ^:private content-file-name "content") 51 | 52 | (defn- read-cached-asset 53 | "Attempt to read the cached asset, if it exists. 54 | 55 | `asset-cache-dir` is the directory containing the two files (asset.edn and content)." 56 | [^File asset-cache-dir] 57 | (if (.exists asset-cache-dir) 58 | (t/track 59 | #(format "Reading from asset cache `%s'" asset-cache-dir) 60 | (some-> 61 | (io/file asset-cache-dir asset-file-name) 62 | read-cached-asset-data 63 | (assoc :content (io/file asset-cache-dir content-file-name)) 64 | (read-attachments asset-cache-dir))))) 65 | 66 | 67 | (defn- write-attachment 68 | "Writes the attachment's content to a file, returns the attachment with `:content` modified 69 | to be the simple name of the file written." 70 | [asset-cache-dir name attachment] 71 | (let [file-name (str (UUID/randomUUID) "-" name) 72 | content-file (io/file asset-cache-dir file-name)] 73 | (io/copy (:content attachment) content-file) 74 | (assoc attachment :content file-name))) 75 | 76 | (defn- write-attachments 77 | "Writes attachments as files in the cache dir, updating each's `:content` key into a simple file name." 78 | [asset asset-cache-dir] 79 | (if (-> asset :attachments empty?) 80 | asset 81 | (update-in asset [:attachments] 82 | (fn [attachments] 83 | (into {} 84 | (map (fn [[name attachment]] 85 | [name (write-attachment asset-cache-dir name attachment)]) 86 | attachments)))))) 87 | 88 | (defn- write-cached-asset 89 | [^File asset-cache-dir asset] 90 | (t/track 91 | #(format "Writing to asset cache `%s'" asset-cache-dir) 92 | (.mkdirs asset-cache-dir) 93 | (let [content-file (io/file asset-cache-dir content-file-name) 94 | asset-file (io/file asset-cache-dir asset-file-name)] 95 | ;; Write the content first 96 | (io/copy (:content asset) content-file) 97 | (with-open [^Writer writer (io/writer asset-file)] 98 | (.write writer (-> asset 99 | (write-attachments asset-cache-dir) 100 | ;; The name of the content file is (currently) a fixed value, 101 | ;; so we can just remove the key, rather than replace it with a file name. 102 | (dissoc :content) 103 | ;; Override *print-length* and *print-level* to ensure it is all written out. 104 | ^String (pp/write :length nil :level nil :stream nil))) 105 | (.write writer "\n"))))) 106 | 107 | (defn- delete-dir-and-contents 108 | [^File dir] 109 | (when (.exists dir) 110 | (t/track 111 | #(format "Deleting directory `%s'" dir) 112 | (doseq [file (.listFiles dir)] 113 | (t/track 114 | #(format "Deleting file `%s'" file) 115 | (io/delete-file file))) 116 | (io/delete-file dir)))) 117 | 118 | (defn wrap-with-filesystem-cache 119 | "Used to implement file-system caching of assets (this is typically only used in development, not production). 120 | File system caching improves startup and first-request time by avoiding the cost of recompling assets; instead 121 | the assets are stored on the file system. The asset checksums are used to see if the cache is still valid. 122 | 123 | Only assets that have truthy value for :compiled key will be file-system cached. This is set by the various 124 | compiler and transformers, such as the CoffeeScript to JavaScript transformer. 125 | 126 | Assets that are accessed for aggregation are not cached (the final aggregated asset will be cached). 127 | 128 | asset-handler 129 | : handler to be wrapped 130 | 131 | cache-dir-name 132 | : name of root folder to store cache in (from the Twixt options); the directory will be created as necessary" 133 | [asset-handler cache-dir-name] 134 | (let [cache-dir (io/file cache-dir-name "compiled")] 135 | (l/infof "Caching compiled assets to `%s'." cache-dir) 136 | (.mkdirs cache-dir) 137 | (fn [asset-path {:keys [asset-resolver for-aggregation] :as context}] 138 | (let [asset-cache-dir (io/file cache-dir asset-path) 139 | cached-asset (read-cached-asset asset-cache-dir)] 140 | (if (is-valid? asset-resolver cached-asset) 141 | cached-asset 142 | (do 143 | (delete-dir-and-contents asset-cache-dir) 144 | (let [asset (asset-handler asset-path context)] 145 | (if (and 146 | (:compiled asset) 147 | (not for-aggregation)) 148 | (write-cached-asset asset-cache-dir asset)) 149 | asset))))))) 150 | -------------------------------------------------------------------------------- /src/io/aviso/twixt/jade.clj: -------------------------------------------------------------------------------- 1 | (ns io.aviso.twixt.jade 2 | "Provides asset pipeline middleware for compiling Jade templates to HTML using jade4j." 3 | (:require [clojure.java.io :as io] 4 | [io.aviso.twixt :as twixt] 5 | [io.aviso.tracker :as t] 6 | [io.aviso.twixt.utils :as utils] 7 | [medley.core :as medley]) 8 | (:import [de.neuland.jade4j JadeConfiguration] 9 | [de.neuland.jade4j.exceptions JadeException] 10 | [de.neuland.jade4j.template TemplateLoader])) 11 | 12 | (defn- add-missing-extension 13 | [^String name ext] 14 | (if (.endsWith name ext) 15 | name 16 | (str name ext))) 17 | 18 | (defn- create-template-loader [root-asset asset-resolver dependencies] 19 | (reify TemplateLoader 20 | ;; getLastModified is only needed for caching, and we disable Jade's caching 21 | ;; in favor of Twixt's. 22 | (getLastModified [_ name] -1) 23 | (getReader [_ name] 24 | (if (= name (:asset-path root-asset)) 25 | (-> root-asset :content io/reader) 26 | (let [full-name (add-missing-extension name ".jade")] 27 | (t/track 28 | #(format "Including Jade source from asset `%s'." full-name) 29 | ;; Use the asset-resolver, not the asset pipeline, because we need 30 | ;; the un-compiled version of the asset. 31 | (let [included (asset-resolver full-name nil)] 32 | (utils/nil-check included "Included asset does not exist.") 33 | (swap! dependencies utils/add-asset-as-dependency included) 34 | (-> included 35 | :content 36 | ;; We have to trust that Jade will close the reader. 37 | io/reader)))))))) 38 | 39 | (defn- wrap-asset-pipeline-with-dependency-tracker 40 | [asset-pipeline dependencies] 41 | (fn [asset-path context] 42 | (let [asset (asset-pipeline asset-path context)] 43 | (swap! dependencies utils/add-asset-as-dependency asset) 44 | asset))) 45 | 46 | (defn- create-shared-variables 47 | [asset {{:keys [helpers variables]} :jade :as context} dependencies] 48 | (let [context' (update-in context [:asset-pipeline] 49 | wrap-asset-pipeline-with-dependency-tracker dependencies)] 50 | (-> (medley/map-vals 51 | #(% asset context') 52 | (or helpers {})) 53 | (merge variables)))) 54 | 55 | (defn- ^JadeConfiguration create-configuration 56 | [pretty-print asset {:keys [asset-resolver] :as context} dependencies] 57 | (doto (JadeConfiguration.) 58 | (.setPrettyPrint pretty-print) 59 | (.setSharedVariables (create-shared-variables asset context dependencies)) 60 | (.setCaching false) 61 | (.setTemplateLoader (create-template-loader asset asset-resolver dependencies)))) 62 | 63 | (defn- jade-compiler [asset context] 64 | (let [name (:resource-path asset)] 65 | (t/timer 66 | #(format "Compiled `%s' to HTML in %.2f ms" name %) 67 | (t/track 68 | #(format "Compiling `%s' from Jade to HTML" name) 69 | (try 70 | ;; Seed the dependencies with the Jade source file. Any included 71 | ;; sources will be added to dependencies. 72 | (let [dependencies (atom {name (utils/extract-dependency asset)}) 73 | configuration (create-configuration (-> context :development-mode) asset context dependencies) 74 | template (.getTemplate configuration (:asset-path asset)) 75 | compiled-output (.renderTemplate configuration template {})] 76 | (utils/create-compiled-asset asset "text/html" compiled-output @dependencies)) 77 | (catch JadeException e 78 | (throw (RuntimeException. 79 | (format "Jade Compilation exception on line %d: %s" 80 | (.getLineNumber e) 81 | (or (.getMessage e) (-> e .getClass .getName))) 82 | e)))))))) 83 | 84 | (defn- complete-path 85 | "Computes the complete path for a partial path, relative to an existing asset. 86 | Alternately, if the path starts with a leading slash (an absolute path), the the leading path is stripped. 87 | 88 | The result is a complete asset path that can be passed to [[get-asset-uri]]." 89 | [asset ^String path] 90 | (if (.startsWith path "/") 91 | (.substring path 1) 92 | (utils/compute-relative-path (:asset-path asset) path))) 93 | 94 | (defprotocol TwixtHelper 95 | "A Jade4J helper object that is used to allow a template to resolve asset URIs." 96 | (uri 97 | [this path] 98 | "Used to obtain the URI for a given asset, identified by its asset path. 99 | The path may be relative to the currently compiling asset, or may be absolute (with a leading slash). 100 | 101 | Throws an exception if the asset it not found.") 102 | (uris 103 | [this paths] 104 | "Used to obtain multiple URIs for any number of assets, each identified by a path. 105 | 106 | paths is a seq of asset path strings (eitehr relative to the current asset, or absolute). 107 | 108 | *Added in 0.1.21*")) 109 | 110 | (defn- create-twixt-helper 111 | [asset context] 112 | (reify TwixtHelper 113 | (uri [_ path] 114 | (twixt/get-asset-uri context (complete-path asset path))) 115 | (uris [_ paths] 116 | (->> paths 117 | (map (partial complete-path asset)) 118 | (mapcat (partial twixt/get-asset-uris context)))))) 119 | 120 | (defn register-jade 121 | "Updates the Twixt options with support for compiling Jade into HTML. Pretty printing of the output HTML is enabled 122 | in development mode, but disabled in production (for efficiency)." 123 | [options] 124 | (-> options 125 | (assoc-in [:content-types "jade"] "text/jade") 126 | (assoc-in [:content-transformers "text/jade"] jade-compiler) 127 | (assoc-in [:twixt-template :jade :helpers "twixt"] create-twixt-helper))) -------------------------------------------------------------------------------- /src/io/aviso/twixt/js_minification.clj: -------------------------------------------------------------------------------- 1 | (ns io.aviso.twixt.js-minification 2 | "Provides support for JavaScript minification using the Google Closure compiler." 3 | (:require [clojure.java.io :as io] 4 | [io.aviso.twixt.utils :as utils] 5 | [io.aviso.twixt.schemas :refer [AssetHandler]] 6 | [io.aviso.tracker :as t] 7 | [clojure.string :as str] 8 | [schema.core :as s]) 9 | (:import (com.google.javascript.jscomp CompilerOptions ClosureCodingConvention DiagnosticGroups CheckLevel 10 | CompilationLevel) 11 | [io.aviso.twixt.shims CompilerShim])) 12 | 13 | 14 | 15 | (defn- minimize-javascript-asset 16 | [{file-path :resource-path :as asset} compilation-level] 17 | (t/timer 18 | #(format "Minimized `%s' (%,d bytes) in %.2f ms" 19 | file-path (:size asset) %) 20 | (t/track 21 | #(format "Minimizing `%s' using Google Closure with compilation level %s" file-path compilation-level) 22 | (let [options ^CompilerOptions (doto (CompilerOptions.) 23 | (.setCodingConvention (ClosureCodingConvention.)) 24 | (.setOutputCharset "utf-8") 25 | (.setWarningLevel DiagnosticGroups/CHECK_VARIABLES CheckLevel/WARNING)) 26 | [failures output] (CompilerShim/runCompiler options 27 | compilation-level 28 | file-path 29 | (-> asset :content io/input-stream))] 30 | (if failures 31 | (throw (ex-info (str "JavaScript minimization failed: " 32 | (str/join "; " (seq failures))) 33 | asset)) 34 | (utils/create-compiled-asset asset "text/javascript" output nil)))))) 35 | 36 | (def ^:private optimization-levels {:simple CompilationLevel/SIMPLE_OPTIMIZATIONS 37 | :whitespace CompilationLevel/WHITESPACE_ONLY 38 | :advanced CompilationLevel/ADVANCED_OPTIMIZATIONS}) 39 | 40 | (s/defn wrap-with-javascript-minimizations :- AssetHandler 41 | "Identifies JavaScript assets and, if not aggregating, passes them through the Google Closure compiler." 42 | [asset-handler :- AssetHandler 43 | {:keys [development-mode js-optimizations]}] 44 | (let [js-optimizations' (if (= :default js-optimizations) 45 | (if development-mode :none :simple) 46 | js-optimizations) 47 | level (optimization-levels js-optimizations')] 48 | (if (nil? level) 49 | asset-handler 50 | (fn [asset-path {:keys [for-aggregation] :as context}] 51 | (let [{:keys [content-type] :as asset} (asset-handler asset-path context)] 52 | (if (and (= "text/javascript" content-type) 53 | (not for-aggregation)) 54 | (minimize-javascript-asset asset level) 55 | asset)))))) -------------------------------------------------------------------------------- /src/io/aviso/twixt/less.clj: -------------------------------------------------------------------------------- 1 | (ns io.aviso.twixt.less 2 | "Provides asset pipeline middleware for compiling Less source files to CSS." 3 | (:require [clojure.string :as str] 4 | [io.aviso.tracker :as t] 5 | [io.aviso.twixt.utils :as utils]) 6 | (:import [com.github.sommeri.less4j LessSource LessSource$FileNotFound Less4jException LessCompiler$Problem LessCompiler LessCompiler$CompilationResult LessCompiler$Configuration LessCompiler$SourceMapConfiguration] 7 | [com.github.sommeri.less4j.core DefaultLessCompiler])) 8 | 9 | ;; Putting this logic inside the (proxy) call causes some really awful Clojure compiler problems. 10 | ;; This shim seems to defuse that. 11 | (defn- find-relative 12 | [asset-resolver asset relative-path context] 13 | (-> 14 | asset 15 | :asset-path 16 | (utils/compute-relative-path relative-path) 17 | (asset-resolver context))) 18 | 19 | (defn- ^LessSource create-less-source 20 | [asset-resolver dependencies asset context] 21 | ;; Whenever a LessSource is created, associated the asset as a dependency; this includes the primary source 22 | ;; and all imported sources. 23 | (swap! dependencies utils/add-asset-as-dependency asset) 24 | (proxy [LessSource] [] 25 | (relativeSource [filename] 26 | (if-let [rel (find-relative asset-resolver asset filename context)] 27 | (create-less-source asset-resolver dependencies rel context) 28 | (throw (new LessSource$FileNotFound)))) 29 | 30 | (getName [] (-> asset :asset-path utils/path->name)) 31 | 32 | (toString [] (:resource-path asset)) 33 | 34 | (getContent [] 35 | (-> 36 | asset 37 | :content 38 | utils/as-string 39 | (.replace "\r\n" "\n"))) 40 | 41 | (getBytes [] (:content asset)))) 42 | 43 | (defn- problem-to-string [^LessCompiler$Problem problem] 44 | (let [source (-> problem .getSource .toString) 45 | line (-> problem .getLine) 46 | character (-> problem .getCharacter) 47 | message (-> problem .getMessage)] 48 | (str source 49 | ":" line 50 | ":" character 51 | ": " message))) 52 | 53 | (defn- format-less-exception [^Less4jException e] 54 | (let [problems (->> e .getErrors (map problem-to-string))] 55 | (str 56 | "Less compilation " 57 | (if (= 1 (count problems)) "error" "errors") 58 | ":\n" 59 | (str/join "\n" problems)))) 60 | 61 | 62 | (defn- compile-less 63 | [^LessCompiler less-compiler asset {:keys [asset-resolver] :as context}] 64 | (let [path (:resource-path asset)] 65 | (t/timer 66 | #(format "Compiled `%s' to CSS in %.2f ms" path %) 67 | (t/track 68 | #(format "Compiling `%s' from Less to CSS" path) 69 | (try 70 | (let [dependencies (atom {}) 71 | root-source (create-less-source asset-resolver dependencies asset context) 72 | ^LessCompiler$Configuration options (LessCompiler$Configuration.) 73 | _ (doto ^LessCompiler$SourceMapConfiguration (.getSourceMapConfiguration options) 74 | (.setLinkSourceMap false) 75 | (.setIncludeSourcesContent true)) 76 | ;; A bit of work to trick the Less compiler into writing the right thing into the output CSS. 77 | ^LessCompiler$CompilationResult output (.compile less-compiler root-source options) 78 | complete-css (str 79 | (.getCss output) 80 | "\n/*# sourceMappingURL=" 81 | (utils/path->name path) 82 | "@source.map */\n") 83 | source-map (.getSourceMap output)] 84 | (-> 85 | asset 86 | (utils/create-compiled-asset "text/css" complete-css @dependencies) 87 | (utils/add-attachment "source.map" "application/json" (utils/as-bytes source-map)))) 88 | (catch Less4jException e 89 | (throw (RuntimeException. (format-less-exception e) e)))))))) 90 | 91 | (defn register-less 92 | "Updates the Twixt options with support for compiling Less into CSS." 93 | [options] 94 | (-> options 95 | (assoc-in [:content-types "less"] "text/less") 96 | (assoc-in [:content-transformers "text/less"] (partial compile-less (DefaultLessCompiler.))))) 97 | -------------------------------------------------------------------------------- /src/io/aviso/twixt/memory_cache.clj: -------------------------------------------------------------------------------- 1 | (ns io.aviso.twixt.memory-cache 2 | "Provides asset pipeline middleware implementing an in-memory cache for assets. 3 | This cache is used to bypass the normal loading, compiling, and transforming steps." 4 | (:require [io.aviso.twixt.schemas :refer [Asset AssetHandler]] 5 | [io.aviso.toolchest.macros :refer [cond-let]] 6 | [schema.core :as s]) 7 | (:import [java.util.concurrent.locks ReentrantLock Lock])) 8 | 9 | (s/defn wrap-with-transforming-cache :- AssetHandler 10 | "A cache that operates in terms of a transformation of assets: the canonical example 11 | is GZip compression, where a certain assets can be compressed to form new assets. 12 | 13 | Assets that are obtained for aggregation are _never_ cached; it is assumed that they are 14 | intermediate results and only the final aggregated asset needs to be cached. 15 | 16 | This cache assumes that another cache is present that can quickly obtain the 17 | non-transformed version of the asset. 18 | 19 | asset-handler 20 | : The downstream asset handler, from which the untransformed asset is obtained. 21 | 22 | store-in-cache? 23 | : Function that is passed an Asset and determines if it can be transformed and stored in the cache. 24 | 25 | asset-tranformer 26 | : Function that is passed the untransformed Asset and returns the transformed Asset to be stored 27 | in the cache." 28 | {:added "0.1.20"} 29 | [asset-handler :- AssetHandler 30 | store-in-cache? :- (s/=> s/Bool Asset) 31 | asset-transformer :- (s/=> Asset Asset)] 32 | (let [cache (atom {})] 33 | (fn [asset-path {:keys [for-aggregation] :as context}] 34 | (cond-let 35 | ;; We assume this is cheap because of caching 36 | [prime-asset (asset-handler asset-path context)] 37 | 38 | for-aggregation 39 | prime-asset 40 | 41 | (nil? prime-asset) 42 | nil 43 | 44 | (not (store-in-cache? prime-asset)) 45 | prime-asset 46 | 47 | [{:keys [transformed prime-checksum]} (get cache asset-path)] 48 | 49 | (= prime-checksum (:checksum prime-asset)) 50 | transformed 51 | 52 | :else 53 | (let [transformed (asset-transformer prime-asset)] 54 | (swap! cache assoc asset-path {:transformed transformed 55 | :prime-checksum prime-checksum}) 56 | transformed))))) 57 | 58 | (defmacro ^:private with-lock 59 | [lock & body] 60 | `(let [^Lock lock# ~lock] 61 | (try 62 | (.lock lock#) 63 | ~@body 64 | (finally 65 | (.unlock lock#))))) 66 | 67 | (defn- update-cache 68 | [cache asset-handler asset-path context] 69 | (let [cached-asset (asset-handler asset-path context)] 70 | (swap! cache update-in [asset-path] 71 | assoc :cached-asset cached-asset 72 | :cached-at (System/currentTimeMillis)) 73 | 74 | cached-asset)) 75 | 76 | (s/defn wrap-with-memory-cache :- AssetHandler 77 | "Wraps another asset handler in a simple in-memory cache. 78 | 79 | This is useful for both compiled and raw assets, as it does two things. 80 | 81 | On first access to an asset, a cache entry is created. Subsequent accesses 82 | to the cache entry within the check interval will be served from the cache. 83 | 84 | After the check interval, the delegate asset handler is used to see if the asset has changed, 85 | replacing it in the cache." 86 | {:added "0.1.20"} 87 | [asset-handler :- AssetHandler 88 | check-interval-ms :- s/Num] 89 | (let [cache (atom {}) 90 | cache-lock (ReentrantLock.)] 91 | (fn [asset-path context] 92 | (trampoline 93 | (fn reentry-point [] 94 | (cond-let 95 | [now (System/currentTimeMillis) 96 | {:keys [cached-asset cached-at] :as cache-entry} (get @cache asset-path)] 97 | 98 | (and cached-asset 99 | (< now (+ cached-at check-interval-ms))) 100 | cached-asset 101 | 102 | (nil? cache-entry) 103 | (let [lock (ReentrantLock.)] 104 | (with-lock cache-lock 105 | ;; There can be a race condition where two threads are trying to first 106 | ;; access the same asset. This detects that; the first thread will have 107 | ;; added a cache entry with just the :lock key, then released the cache 108 | ;; lock while obtaining the asset. The second thread will reach this point, 109 | ;; see that there's something in the cache, and loop back 110 | ;; to fall through the normal path, and the per-asset lock. 111 | (if (contains? @cache asset-path) 112 | ;; This releases the cache-lock and falls through the normal 113 | ;; processing. Depending on the timing, either the asset will just be in the cache 114 | ;; and ready to go, or this thread will block while the other thread 115 | ;; is obtaining the asset. 116 | reentry-point 117 | (do 118 | ;; Do this now, with the cache-lock locked 119 | (swap! cache assoc-in [asset-path :lock] lock) 120 | ;; And do this via trampoline, after the cache-lock is unlocked. 121 | #(with-lock lock 122 | (update-cache cache asset-handler asset-path context)))))) 123 | 124 | :else 125 | (with-lock (:lock cache-entry) 126 | (update-cache cache asset-handler asset-path context)))))))) -------------------------------------------------------------------------------- /src/io/aviso/twixt/rhino.clj: -------------------------------------------------------------------------------- 1 | (ns io.aviso.twixt.rhino 2 | "Code to execute Rhino for purposes such as compiling CoffeeScript to JavaScript." 3 | ;; Liberally borrowing from Dieter! 4 | (:import 5 | [org.mozilla.javascript Context Scriptable Function]) 6 | (:require 7 | [clojure.java.io :as io] 8 | [io.aviso.tracker :as t])) 9 | 10 | 11 | (defn- load-script [^Context context scope file] 12 | (with-open [content-reader (-> file io/resource io/reader)] 13 | (t/track 14 | #(str "Loading JavaScript from " file) 15 | (.evaluateReader context scope content-reader file 1 nil)))) 16 | 17 | (defn- invoke-function [^Context context ^Scriptable scope ^String function-name arguments] 18 | (let [^Function js-fn (.get scope function-name scope)] 19 | (.call js-fn context scope nil (into-array arguments)))) 20 | 21 | (defn invoke-javascript 22 | "Invokes a JavaScript function, returning the result. 23 | 24 | script-paths 25 | : JavaScript files to load, as seq of classpath resource paths 26 | 27 | javascript-fn-name 28 | : name of JavaScript function to execute 29 | 30 | arguments 31 | : additional arguments to pass to the function 32 | 33 | Returns the JavaScript result; typically this will be a JavaScript object, and the properties 34 | of it can be accessed via the methods of `java.util.Map`." 35 | [script-paths javascript-fn-name & arguments] 36 | (let [context (Context/enter)] 37 | (try 38 | ;; Apparently, CoffeeScript can blow away a 64K limit pretty easily. 39 | (.setOptimizationLevel context -1) 40 | 41 | (let [scope (.initStandardObjects context)] 42 | 43 | ;; Dieter maintains a pool of configured Context instances so that we don't 44 | ;; have to re-load the scripts on each compilation. That would be nice. 45 | (doseq [file script-paths] 46 | (load-script context scope file)) 47 | 48 | (invoke-function context scope javascript-fn-name arguments)) 49 | 50 | (finally (Context/exit))))) 51 | -------------------------------------------------------------------------------- /src/io/aviso/twixt/ring.clj: -------------------------------------------------------------------------------- 1 | (ns io.aviso.twixt.ring 2 | "Support needed to use Twixt inside a Ring handler pipeline." 3 | (:require [clojure.java.io :as io] 4 | [io.aviso.tracker :as t] 5 | [io.aviso.twixt [asset :as asset]] 6 | [ring.util.response :as r]) 7 | (:import [java.util Calendar TimeZone])) 8 | 9 | (def ^:private far-future 10 | (let [cal (doto (Calendar/getInstance (TimeZone/getTimeZone "UTC")) 11 | (.add Calendar/YEAR 10))] 12 | (.getTime cal))) 13 | 14 | (defn- asset->ring-response 15 | [asset attachment-name] 16 | (let [content-source (if (some? attachment-name) 17 | (get-in asset [:attachments attachment-name]) 18 | asset)] 19 | ;; content-source may be nil if the URL includes an attachment but the attachment doesn't exist. 20 | (if content-source 21 | ;; First the standard stuff ... 22 | (cond-> 23 | (-> 24 | content-source 25 | :content 26 | io/input-stream 27 | r/response 28 | (r/header "Content-Length" (:size content-source)) 29 | ;; Because any change to the content will create a new checksum and a new URL, a change to the content 30 | ;; is really an entirely new resource. The current resource is therefore immutable and can have a far-future expires 31 | ;; header. 32 | (r/header "Expires" far-future) 33 | (r/content-type (:content-type content-source))) 34 | 35 | ;; The optional extras ... 36 | ;; :compressed will never be true for an attachment (that may change in the future). 37 | (:compressed content-source) (r/header "Content-Encoding" "gzip"))))) 38 | 39 | 40 | (defn- match? [^String path-prefix ^String path] 41 | (and 42 | (.startsWith path path-prefix) 43 | (not (or (= path path-prefix) 44 | (.endsWith path "/"))))) 45 | 46 | (defn- split-asset-path-and-attachment-name 47 | [^String path] 48 | (let [atx (.indexOf path "@")] 49 | (if (< atx 0) 50 | [path nil] 51 | [(.substring path 0 atx) 52 | (.substring path (inc atx))]))) 53 | 54 | (defn- parse-path 55 | "Parses the complete request path into a checksum, compressed-flag, asset path and (optional) attachment name." 56 | [^String path-prefix ^String path] 57 | (let [suffix (.substring path (.length path-prefix)) 58 | slashx (.indexOf suffix "/") 59 | full-checksum (.substring suffix 0 slashx) 60 | compressed? (.startsWith full-checksum "z") 61 | checksum (if compressed? (.substring full-checksum 1) full-checksum) 62 | [asset-path attachment-name] (-> suffix (.substring (inc slashx)) split-asset-path-and-attachment-name)] 63 | [checksum 64 | compressed? 65 | asset-path 66 | attachment-name])) 67 | 68 | (defn- asset->redirect-response 69 | [status path-prefix asset] 70 | {:status status 71 | :headers {"Location" (asset/asset->request-path path-prefix asset)} 72 | :body ""}) 73 | 74 | (def ^:private asset->301-response (partial asset->redirect-response 301)) 75 | 76 | (defn- create-asset-response 77 | [path-prefix requested-checksum asset attachment-name] 78 | (cond 79 | (nil? asset) nil 80 | (= requested-checksum (:checksum asset)) (asset->ring-response asset attachment-name) 81 | :else (asset->301-response path-prefix asset))) 82 | 83 | (defn twixt-handler 84 | "A Ring request handler that identifies requests targetted for Twixt assets. Returns a Ring response map 85 | if the request is for an existing asset, otherwise returns nil. 86 | 87 | The path may indicate an attachment to the asset (such as the source.map for a compiled JavaScript file). 88 | The attachment is indicated as a suffix: the `@` symbol and the name of the attachment. 89 | 90 | Asset URLs always include the intended asset's checksum; if the actual asset checksum does not match, then 91 | a 301 (moved permanently) response is sent with the correct asset URL." 92 | [request] 93 | (let [path-prefix (-> request :twixt :path-prefix) 94 | path (:uri request)] 95 | (when (match? path-prefix path) 96 | (t/track 97 | #(format "Handling asset request `%s'" path) 98 | (let [[requested-checksum compressed? asset-path attachment-name] (parse-path path-prefix path) 99 | context (:twixt request) 100 | ;; When actually servicing an asset request, we have to trust the data in the URL 101 | ;; that determines whether to server the normal or gzip'ed resource. 102 | context' (assoc context :gzip-enabled compressed?) 103 | asset-pipeline (:asset-pipeline context)] 104 | (create-asset-response path-prefix 105 | requested-checksum 106 | (asset-pipeline asset-path context') 107 | attachment-name)))))) 108 | 109 | ;;; It's really difficult to start with a path, convert it to a resource, and figure out if it is a folder 110 | ;;; or a file on the classpath. Instead, we just assume that anything that doesn't look like a path that 111 | ;;; ends with a file extension can be ignored. 112 | 113 | (defn- handle-asset-redirect 114 | [^String uri context] 115 | ;; This may be too specific, may need to find a better way to differentiate between a "folder" and a file. 116 | ;; This just checks to see if the path ends with something that looks like an extension. We just have to 117 | ;; assume that none of the folders on classpath look like that! This is also a bit restrictive; it assumes 118 | ;; the extension consists of word characters, or the dash. 119 | (if (re-find #"\.[\w-]+$" uri) 120 | (let [{:keys [asset-pipeline path-prefix]} context 121 | asset-path (.substring uri 1)] 122 | (if-let [asset (asset-pipeline asset-path context)] 123 | 124 | (asset->redirect-response 302 path-prefix asset))))) 125 | 126 | (defn wrap-with-asset-redirector 127 | "In some cases, it is not possible for the client to know what the full asset URI will be, such as when the 128 | URL is composed on the client (in which case, the asset checksum will not be known). 129 | The redirector accepts any request path that maps to an asset and returns a redirect to the asset's true URL. 130 | Non-matching requests are passed through to the provided handler. 131 | 132 | For a file under `META-INF/assets`, such as `META-INF/assets/myapp/icon.png`, the redirector will match 133 | the URI `/myapp/icon.png` and send a redirect to `/assets/123abc/myapp/icon.png`. 134 | 135 | This middleware is __not__ applied by default." 136 | [handler] 137 | (fn [{uri :uri 138 | context :twixt 139 | :as request}] 140 | (or 141 | (handle-asset-redirect uri context) 142 | (handler request)))) 143 | 144 | (defn wrap-with-twixt 145 | "Invokes the twixt-handler and delegates to the provided Ring handler if twixt-handler returns nil. 146 | 147 | This assumes that the resulting handler will then be wrapped with the twixt setup. 148 | 149 | In most cases, you will want to use the `wrap-with-twixt` function in the `io.aviso.twixt.startup` namespace." 150 | [handler] 151 | (fn [request] 152 | (or (twixt-handler request) 153 | (handler request)))) 154 | 155 | (defn wrap-with-twixt-setup 156 | "Wraps a Ring handler with another Ring handler that provides the `:twixt` key in the request object. 157 | 158 | The `:twixt` key is the default asset pipeline context, which is needed by [[get-asset-uri]] in order to resolve asset paths 159 | to an actual asset. 160 | It also contains the keys `:asset-pipeline` (the pipeline used to resolve assets), 161 | `:stack-frame-filter` (which is used by the HTML exception report) and `:development-mode`. 162 | 163 | This provides the information needed by the actual Twixt handler, as well as anything else downstream that needs to 164 | generate Twixt asset URIs." 165 | [handler twixt-options asset-pipeline] 166 | (let [context (-> twixt-options 167 | ;; Pass down only what is needed to generate asset URIs, or to produce the HTML exception report. 168 | (select-keys [:path-prefix :stack-frame-filter :development-mode]) 169 | (assoc :asset-pipeline asset-pipeline) 170 | (merge (:twixt-template twixt-options)))] 171 | (fn [request] 172 | (handler (assoc request :twixt context))))) 173 | -------------------------------------------------------------------------------- /src/io/aviso/twixt/schemas.clj: -------------------------------------------------------------------------------- 1 | (ns io.aviso.twixt.schemas 2 | "Defines schemas for the main types and functions." 3 | {:added "0.1.17"} 4 | (:require [schema.core :as s])) 5 | 6 | (def Content 7 | "Defines the content for an Asset; this is any value compatible with clojure.java.io." 8 | s/Any) 9 | 10 | (def AssetPath 11 | "Path of the asset under the root folder (which is typically `/META-INF/assets/`)." 12 | s/Str) 13 | 14 | (def ResourcePath 15 | "Full path of the underlying resource (on the classpath)." 16 | s/Str) 17 | 18 | (def ContentType 19 | "The MIME type of the content, as determined form the path's extension." 20 | s/Str) 21 | 22 | (def Size 23 | "Size of the content, in bytes." 24 | s/Num) 25 | 26 | (def Checksum 27 | "Computed Adler32 checksum of the content, as a string of hex characters." 28 | s/Str) 29 | 30 | (def ModifiedAt 31 | "Instant at which an asset was last modified; this may not be accurate for assets 32 | stored within a JAR file, but will be adequate for determining if a file's content 33 | has changed." 34 | s/Inst) 35 | 36 | (def AttachmentName 37 | "A unique logical name that represents an attachment to an asset. 38 | Attachments are a limited form of an Asset typically generated as a side-effect 39 | of creating/compiling the asset; the canonical example is a JavaScript source map." 40 | s/Str) 41 | 42 | (s/defschema Dependency 43 | {:asset-path AssetPath 44 | :checksum Checksum 45 | :modified-at ModifiedAt}) 46 | 47 | (s/defschema DependencyMap 48 | {ResourcePath Dependency}) 49 | 50 | (s/defschema Attachment 51 | {:content Content 52 | :size Size 53 | :content-type ContentType}) 54 | 55 | (s/defschema AttachmentMap 56 | {AttachmentName Attachment}) 57 | 58 | (s/defschema Asset 59 | "A server-side resource that may be exposed to a client (such as a web browser). 60 | An Asset may be transformed from raw content, for example to transpile a language 61 | to JavaScript. Other transformations include compression." 62 | {:asset-path AssetPath 63 | :resource-path ResourcePath 64 | :content-type ContentType 65 | :size Size 66 | :checksum Checksum 67 | :modified-at ModifiedAt 68 | :content Content 69 | (s/optional-key :compiled) s/Bool 70 | (s/optional-key :aggregate-asset-paths) [AssetPath] 71 | :dependencies DependencyMap 72 | (s/optional-key :attachments) AttachmentMap 73 | ;; Various plugins and extensions will add their own keys, so we need to not be picky: 74 | s/Any s/Any 75 | }) 76 | 77 | (s/defschema TwixtContext 78 | "Defines the minimal values provided in the Twixt context (the :twixt key of the Ring 79 | request map)." 80 | ;; :asset-pipeline will always be present when requests are processing; there's 81 | ;; just a bit of chicken-or-the-egg at startup time. 82 | {(s/optional-key :asset-pipeline) s/Any ; should be AssetHandler 83 | :path-prefix s/Str 84 | :development-mode s/Bool 85 | (s/optional-key :gzip-enabled) (s/maybe s/Bool) ; often set explicitly to false for aggregation 86 | ;; Other plugins are likely to add thier own data to the context 87 | ;; (via the :twixt-template key of the Twixt options). 88 | s/Any s/Any}) 89 | 90 | (s/defschema AssetHandler 91 | "An asset handler is passed as asset path and the Twixt context and, maybe, returns 92 | an Asset." 93 | (s/=> (s/maybe Asset) AssetPath TwixtContext)) 94 | 95 | (def AssetURI 96 | "A URI that allows an external client to access the client content. AssetURIs typically 97 | include the checksum in the URI path, forming an immutable web resource." 98 | s/Str) 99 | 100 | (s/defschema AssetExport 101 | "Either just an asset path (a String), or two strings: an Asset to export, and an alias 102 | to export it as." 103 | (s/either AssetPath 104 | [(s/one AssetPath 'asset-path) (s/one AssetPath 'output-alias)])) 105 | 106 | (s/defschema ExportsConfiguration 107 | "Defines the details of how assets are exported to the file system. 108 | 109 | :interval-ms 110 | : The time between checks for performing exports. 111 | : Checks only occur during requests, so an external ping process may be needed to keep checks running. 112 | 113 | :output-dir 114 | : The root directory to output files to. The directory will be created as necessary. 115 | 116 | :assets 117 | : The assets to export. 118 | : Keys are the assets to export." 119 | {:interval-ms s/Num 120 | :output-dir s/Str 121 | :output-uri s/Str 122 | :assets [AssetExport]}) -------------------------------------------------------------------------------- /src/io/aviso/twixt/stacks.clj: -------------------------------------------------------------------------------- 1 | (ns io.aviso.twixt.stacks 2 | "Stacks are a way of combining several related files together to form a single aggregate virtual file. 3 | In development, the individual files are used, but in production they are aggregated together so that 4 | the entire stack can be obtained in a single request." 5 | {:since "0.1.13"} 6 | (:require [io.aviso.tracker :as t] 7 | [io.aviso.twixt.utils :as utils] 8 | [clojure.java.io :as io] 9 | [clojure.edn :as edn]) 10 | (:import (java.io PushbackReader ByteArrayOutputStream))) 11 | 12 | (def stack-mime-type 13 | "Defines a MIME type for a Twixt stack, a kind of aggregate resource that combines 14 | multiple other resources." 15 | "application/vnd.twixt-stack+edn") 16 | 17 | (defn- read-stack 18 | [asset] 19 | (-> asset :content io/reader PushbackReader. edn/read)) 20 | 21 | (defn- update-aggregate-asset-paths 22 | [asset-paths component-asset] 23 | (if-let [aggregate-paths (-> component-asset :aggregate-asset-paths seq)] 24 | (concat asset-paths aggregate-paths) 25 | (conj asset-paths (:asset-path component-asset)))) 26 | 27 | (defn- include-component 28 | [{:keys [asset-pipeline] :as context} {:keys [asset-path] :as asset} component-path content-stream] 29 | ;; This may only be correct for relative, not absolute, paths: 30 | (let [complete-path (utils/compute-relative-path asset-path component-path) 31 | component-asset (asset-pipeline complete-path context)] 32 | 33 | (if-not (some? component-asset) 34 | (throw (ex-info (format "Could not locate resource `%s' (a component of `%s')." 35 | complete-path 36 | asset-path) 37 | context))) 38 | 39 | (io/copy (:content component-asset) content-stream) 40 | 41 | ;; Now add the component asset as a dependency by itself, or 42 | ;; add the component asset's dependencies 43 | 44 | (-> asset 45 | (update-in [:dependencies] utils/add-asset-as-dependency component-asset) 46 | (update-in [:aggregate-asset-paths] (fnil update-aggregate-asset-paths []) component-asset)))) 47 | 48 | (defn- aggregate-stack 49 | [{:keys [resource-path] :as asset} context] 50 | (t/track 51 | #(format "Reading stack `%s' and aggregating contents." resource-path) 52 | ;; Make sure that the aggregated asset is not compressed 53 | (let [context' (assoc context :for-aggregation true) 54 | stack (read-stack asset)] 55 | 56 | (assert (-> stack :content-type some?)) 57 | (assert (-> stack :components empty? not)) 58 | 59 | (with-open [content-stream (ByteArrayOutputStream. 5000)] 60 | (-> (reduce (fn [asset component-path] (include-component context' asset component-path content-stream)) asset (:components stack)) 61 | (assoc-in [:dependencies resource-path] (utils/extract-dependency asset)) 62 | (utils/replace-asset-content (:content-type stack) (.toByteArray content-stream)) 63 | (assoc :compiled true)))))) 64 | 65 | (defn register-stacks 66 | "Updates the Twixt options with support for stacks." 67 | [options] 68 | (-> options 69 | (assoc-in [:content-types "stack"] stack-mime-type) 70 | (assoc-in [:content-transformers stack-mime-type] aggregate-stack))) -------------------------------------------------------------------------------- /src/io/aviso/twixt/startup.clj: -------------------------------------------------------------------------------- 1 | (ns io.aviso.twixt.startup 2 | "Breaks out the default logic for initializing Twixt's handlers and middleware." 3 | (:require 4 | [io.aviso.twixt :as t] 5 | [io.aviso.twixt 6 | [coffee-script :as cs] 7 | [compress :as compress] 8 | [exceptions :as te] 9 | [export :as export] 10 | [jade :as jade] 11 | [less :as less] 12 | [ring :as ring] 13 | [stacks :as stacks]])) 14 | 15 | (defn wrap-with-twixt 16 | "The default way to setup Twixt, with exception reporting. 17 | This (currently) enables support for CoffeeScript, Less, and Jade, and Stacks. 18 | 19 | The provided Ring request handler is wrapped in the following stack (outermost to innermost): 20 | 21 | - twixt setup (adds `:twixt` key to the request) 22 | - exception reporting 23 | - compression analyzer (does the client support GZip encoding?) 24 | - asset export logic (exports certain assets to file system when changed 25 | - asset request handling 26 | - the provided handler 27 | 28 | With just a handler, uses the default Twixt options and production mode. 29 | 30 | The two argument version is used to set development-mode, but use default options. 31 | 32 | Otherwise, provide the handler, alternate options and true or false for development mode. 33 | The alternate options are merged with defaults and override them." 34 | ([handler] 35 | (wrap-with-twixt handler false)) 36 | ([handler development-mode] 37 | (wrap-with-twixt handler t/default-options development-mode)) 38 | ([handler opts development-mode] 39 | (let [twixt-options (-> (merge t/default-options opts) 40 | (assoc :development-mode development-mode) 41 | te/register-exception-reporting 42 | cs/register-coffee-script 43 | jade/register-jade 44 | less/register-less 45 | stacks/register-stacks) 46 | asset-pipeline (t/default-asset-pipeline twixt-options)] 47 | (-> 48 | handler 49 | ring/wrap-with-twixt 50 | (export/wrap-with-exporter (:exports twixt-options)) 51 | te/wrap-with-exception-reporting 52 | compress/wrap-with-compression-analyzer 53 | (ring/wrap-with-twixt-setup twixt-options asset-pipeline))))) -------------------------------------------------------------------------------- /src/io/aviso/twixt/utils.clj: -------------------------------------------------------------------------------- 1 | (ns io.aviso.twixt.utils 2 | "Some re-usable utilities. 3 | 4 | Many of these are useful when creating new compilers or translators for Twixt." 5 | (:require [clojure.java.io :as io] 6 | [clojure.string :as str] 7 | [schema.core :as s] 8 | [io.aviso.twixt.schemas :refer :all]) 9 | (:import [java.io CharArrayWriter ByteArrayOutputStream File] 10 | [java.util.zip Adler32] 11 | [java.net URL URI] 12 | [java.util Date])) 13 | 14 | (defn ^String as-string 15 | "Converts a source (compatible with clojure.java.io/IOFactory) into a String using the provided encoding. 16 | 17 | The source is typically a byte array, or a File. 18 | 19 | The default charset is UTF-8." 20 | ([source] 21 | (as-string source "UTF-8")) 22 | ([source charset] 23 | (with-open [reader (io/reader source :encoding charset) 24 | writer (CharArrayWriter. 1000)] 25 | (io/copy reader writer) 26 | (.toString writer)))) 27 | 28 | (defn as-bytes [^String string] 29 | "Converts a string to a byte array. The string should be UTF-8 encoded." 30 | (.getBytes string "UTF-8")) 31 | 32 | (s/defn compute-checksum :- Checksum 33 | "Returns a hex string of the Adler32 checksum of the content." 34 | [^bytes content] 35 | (-> 36 | (doto (Adler32.) 37 | (.update content)) 38 | .getValue 39 | Long/toHexString)) 40 | 41 | (s/defn replace-asset-content :- Asset 42 | "Modifies an Asset new content. 43 | This updates the :size and :checksum properties as well." 44 | [asset :- Asset 45 | content-type :- ContentType 46 | ^bytes content-bytes] 47 | (assoc asset 48 | :content-type content-type 49 | :content content-bytes 50 | :size (alength content-bytes) 51 | :checksum (compute-checksum content-bytes))) 52 | 53 | (s/defn extract-dependency :- Dependency 54 | "Extracts from the asset the keys needed to track dependencies (used by caching logic)." 55 | [asset :- Asset] 56 | (select-keys asset [:checksum :modified-at :asset-path])) 57 | 58 | (s/defn add-asset-as-dependency :- DependencyMap 59 | "Adds the dependencies of the Asset to a dependency map." 60 | [dependencies :- DependencyMap 61 | asset :- Asset] 62 | (merge dependencies (:dependencies asset))) 63 | 64 | (s/defn create-compiled-asset :- Asset 65 | "Used to transform an Asset after it has been compiled from one form to another. 66 | Dependencies is a map of resource path to source asset details, used to check cache validity. 67 | 68 | The source asset's dependencies are merged into any provided dependencies to form the :dependencies entry of the output Asset." 69 | [source-asset :- Asset 70 | content-type :- s/Str 71 | content :- s/Str 72 | dependencies :- (s/maybe DependencyMap)] 73 | (let [merged-dependencies (add-asset-as-dependency (or dependencies {}) source-asset)] 74 | (-> 75 | source-asset 76 | (replace-asset-content content-type (as-bytes content)) 77 | (assoc :compiled true :dependencies merged-dependencies)))) 78 | 79 | (s/defn add-attachment :- Asset 80 | "Adds an attachment to an asset." 81 | {:since "0.1.13"} 82 | [asset :- Asset 83 | name :- AttachmentName 84 | content-type :- ContentType 85 | ^bytes content] 86 | (assoc-in asset [:attachments name] {:content-type content-type 87 | :content content 88 | :size (alength content)})) 89 | 90 | (defn read-content 91 | "Reads the content of a provided source (compatible with `clojure.java.io/input-stream`) as a byte array 92 | 93 | The content is usually a URI or URL." 94 | [source] 95 | (assert source "Unable to read content from nil.") 96 | (with-open [bos (ByteArrayOutputStream.) 97 | in (io/input-stream source)] 98 | (io/copy in bos) 99 | (.toByteArray bos))) 100 | 101 | (defn compute-relative-path 102 | [^String start ^String relative] 103 | ;; Convert the start path into a stack of just the folders 104 | (loop [path-terms (-> (.split start "/") reverse rest) 105 | terms (-> relative (.split "/"))] 106 | (let [[term & remaining] terms] 107 | (cond 108 | (empty? terms) (->> path-terms reverse (str/join "/")) 109 | (or (= term ".") (= term "")) (recur path-terms remaining) 110 | (= term "..") (if (empty? path-terms) 111 | ;; You could rewrite this with reduce, but then generating this exception would be more difficult: 112 | (throw (IllegalArgumentException. (format "Relative path `%s' for `%s' would go above root." relative start))) 113 | (recur (rest path-terms) remaining)) 114 | :else (recur (cons term path-terms) remaining))))) 115 | 116 | (defn- url-to-file 117 | [^URL url] 118 | (-> url .toURI File.)) 119 | 120 | (defn- jar-to-file 121 | "For a URL that points to a file inside a jar, this returns the JAR file itself." 122 | [^URL url] 123 | (-> url 124 | .getPath 125 | (str/split #"!") 126 | first 127 | URI. 128 | File.)) 129 | 130 | ;; Not the same as io/file! 131 | (defn- ^File as-file 132 | "Locates a file from which a last modified date can be extracted." 133 | [^URL url] 134 | (cond 135 | (= "jar" (.getProtocol url)) (jar-to-file url) 136 | :else (url-to-file url))) 137 | 138 | (s/defn modified-at :- Date 139 | "Extracts a last-modified Date" 140 | [url :- URL] 141 | (if-let [^long time-modified (some-> url as-file .lastModified)] 142 | (Date. time-modified))) 143 | 144 | (defn nil-check 145 | {:no-doc true} 146 | [value message] 147 | (or 148 | value 149 | (throw (NullPointerException. message)))) 150 | 151 | (defn path->name 152 | "Converts a path to just the file name, the last term in the path." 153 | [^String path] 154 | (let [slashx (.lastIndexOf path "/")] 155 | (if (< slashx 0) 156 | path 157 | (.substring path (inc slashx))))) 158 | 159 | --------------------------------------------------------------------------------