stylo 0.5
Sign up to get free protection for your applications and to get access to all the features.
- data/.document +5 -0
- data/.gitignore +25 -0
- data/LICENSE +20 -0
- data/README.rdoc +18 -0
- data/Rakefile +46 -0
- data/TODO +4 -0
- data/cucumber.yml +2 -0
- data/features/fixtures/child.css +3 -0
- data/features/fixtures/child.js +3 -0
- data/features/fixtures/grand_parent.css +5 -0
- data/features/fixtures/grand_parent.js +5 -0
- data/features/fixtures/grand_parent_with_parent_with_child.css +11 -0
- data/features/fixtures/grand_parent_with_parent_with_child.js +11 -0
- data/features/fixtures/parent.css +5 -0
- data/features/fixtures/parent.js +5 -0
- data/features/fixtures/parent_with_child.css +7 -0
- data/features/fixtures/parent_with_child.js +7 -0
- data/features/fixtures/processed_sass_child.css +2 -0
- data/features/fixtures/processed_sass_which_uses_mixin.css +3 -0
- data/features/fixtures/sass_child.scss +4 -0
- data/features/fixtures/sass_mixins.scss +3 -0
- data/features/fixtures/sass_which_uses_mixin.scss +6 -0
- data/features/fixtures.rb +5 -0
- data/features/javascripts.feature +19 -0
- data/features/response_headers.feature +13 -0
- data/features/sass_integration.feature +13 -0
- data/features/step_definitions/stylo_steps.rb +24 -0
- data/features/stylesheets.feature +31 -0
- data/features/stylo_cannot_serve_asset.feature +5 -0
- data/features/support/env.rb +24 -0
- data/lib/stylo/asset_loader.rb +9 -0
- data/lib/stylo/combiner.rb +14 -0
- data/lib/stylo/config.rb +14 -0
- data/lib/stylo/pipeline_steps/caching.rb +11 -0
- data/lib/stylo/pipeline_steps/javascript.rb +15 -0
- data/lib/stylo/pipeline_steps/sass.rb +18 -0
- data/lib/stylo/pipeline_steps/stylesheet.rb +15 -0
- data/lib/stylo/processor.rb +47 -0
- data/lib/stylo/rack.rb +20 -0
- data/lib/stylo/railtie.rb +13 -0
- data/lib/stylo/response.rb +41 -0
- data/lib/stylo.rb +5 -0
- data/spec/asset_loader_spec.rb +16 -0
- data/spec/combiner_spec.rb +41 -0
- data/spec/pipeline_steps/caching_spec.rb +26 -0
- data/spec/pipeline_steps/javascript_spec.rb +84 -0
- data/spec/pipeline_steps/sass_spec.rb +94 -0
- data/spec/pipeline_steps/stylesheet_spec.rb +84 -0
- data/spec/rack_spec.rb +75 -0
- data/spec/spec_helper.rb +18 -0
- data/spec/stylo_spec_helpers.rb +30 -0
- data/stylo.gemspec +116 -0
- metadata +198 -0
data/.document
ADDED
data/.gitignore
ADDED
data/LICENSE
ADDED
@@ -0,0 +1,20 @@
|
|
1
|
+
Copyright (c) 2009 mwagg
|
2
|
+
|
3
|
+
Permission is hereby granted, free of charge, to any person obtaining
|
4
|
+
a copy of this software and associated documentation files (the
|
5
|
+
"Software"), to deal in the Software without restriction, including
|
6
|
+
without limitation the rights to use, copy, modify, merge, publish,
|
7
|
+
distribute, sublicense, and/or sell copies of the Software, and to
|
8
|
+
permit persons to whom the Software is furnished to do so, subject to
|
9
|
+
the following conditions:
|
10
|
+
|
11
|
+
The above copyright notice and this permission notice shall be
|
12
|
+
included in all copies or substantial portions of the Software.
|
13
|
+
|
14
|
+
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
15
|
+
EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
16
|
+
MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
17
|
+
NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
|
18
|
+
LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
|
19
|
+
OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
|
20
|
+
WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
data/README.rdoc
ADDED
@@ -0,0 +1,18 @@
|
|
1
|
+
= stylo
|
2
|
+
|
3
|
+
Stylo aims to support server side combining of stylesheets within rails when hosted in an environment
|
4
|
+
such as www.heroku.com where we have the problem of a read only file system.
|
5
|
+
|
6
|
+
== Note on Patches/Pull Requests
|
7
|
+
|
8
|
+
* Fork the project.
|
9
|
+
* Make your feature addition or bug fix.
|
10
|
+
* Add tests for it. This is important so I don't break it in a
|
11
|
+
future version unintentionally.
|
12
|
+
* Commit, do not mess with rakefile, version, or history.
|
13
|
+
(if you want to have your own version, that is fine but bump version in a commit by itself I can ignore when I pull)
|
14
|
+
* Send me a pull request. Bonus points for topic branches.
|
15
|
+
|
16
|
+
== Copyright
|
17
|
+
|
18
|
+
Copyright (c) 2010 mwagg. See LICENSE for details.
|
data/Rakefile
ADDED
@@ -0,0 +1,46 @@
|
|
1
|
+
require 'rubygems'
|
2
|
+
require 'rake'
|
3
|
+
require 'cucumber'
|
4
|
+
require 'cucumber/rake/task'
|
5
|
+
|
6
|
+
begin
|
7
|
+
require 'jeweler'
|
8
|
+
Jeweler::Tasks.new do |gem|
|
9
|
+
gem.name = "stylo"
|
10
|
+
gem.summary = %Q{Server side stylesheet combining for readonly hosting environments}
|
11
|
+
gem.description = %Q{Server side stylesheet combining for readonly hosting environments}
|
12
|
+
gem.email = "michael@guerillatactics.co.uk"
|
13
|
+
gem.homepage = "http://github.com/mwagg/stylo"
|
14
|
+
gem.authors = ["mwagg"]
|
15
|
+
gem.add_development_dependency "rspec", ">= 0"
|
16
|
+
gem.add_development_dependency "cucumber", ">= 0"
|
17
|
+
gem.add_development_dependency "rack-test", ">= 0"
|
18
|
+
gem.add_development_dependency "sinatra", ">= 0"
|
19
|
+
gem.add_runtime_dependency "haml", ">= 3.0.21"
|
20
|
+
gem.version = "0.5"
|
21
|
+
# gem is a Gem::Specification... see http://www.rubygems.org/read/chapter/20 for additional settings
|
22
|
+
end
|
23
|
+
Jeweler::GemcutterTasks.new
|
24
|
+
rescue LoadError
|
25
|
+
puts "Jeweler (or a dependency) not available. Install it with: gem install jeweler"
|
26
|
+
end
|
27
|
+
|
28
|
+
require 'rspec/core/rake_task'
|
29
|
+
desc "Run all examples"
|
30
|
+
RSpec::Core::RakeTask.new()
|
31
|
+
|
32
|
+
task :default => [:spec, :features]
|
33
|
+
|
34
|
+
require 'rake/rdoctask'
|
35
|
+
Rake::RDocTask.new do |rdoc|
|
36
|
+
version = File.exist?('VERSION') ? File.read('VERSION') : ""
|
37
|
+
|
38
|
+
rdoc.rdoc_dir = 'rdoc'
|
39
|
+
rdoc.title = "stylo #{version}"
|
40
|
+
rdoc.rdoc_files.include('README*')
|
41
|
+
rdoc.rdoc_files.include('lib/**/*.rb')
|
42
|
+
end
|
43
|
+
|
44
|
+
Cucumber::Rake::Task.new(:features) do |t|
|
45
|
+
t.profile = :default
|
46
|
+
end
|
data/TODO
ADDED
data/cucumber.yml
ADDED
@@ -0,0 +1,19 @@
|
|
1
|
+
Feature: Javascript serving
|
2
|
+
|
3
|
+
Scenario: Simple javascript serving
|
4
|
+
Given "child.js" is located at "javascripts" in the asset location
|
5
|
+
When a request is made for "javascripts/child.js"
|
6
|
+
Then the response body should look like "child.js"
|
7
|
+
|
8
|
+
Scenario: Simple javascript combining
|
9
|
+
Given "parent.js" is located at "javascripts" in the asset location
|
10
|
+
And "child.js" is located at "javascripts" in the asset location
|
11
|
+
When a request is made for "javascripts/parent.js"
|
12
|
+
Then the response body should look like "parent_with_child.js"
|
13
|
+
|
14
|
+
Scenario: Nested stylesheet combining
|
15
|
+
Given "grand_parent.js" is located at "javascripts" in the asset location
|
16
|
+
And "parent.js" is located at "javascripts" in the asset location
|
17
|
+
And "child.js" is located at "javascripts" in the asset location
|
18
|
+
When a request is made for "javascripts/grand_parent.js"
|
19
|
+
Then the response body should look like "grand_parent_with_parent_with_child.js"
|
@@ -0,0 +1,13 @@
|
|
1
|
+
Feature: Response headers
|
2
|
+
|
3
|
+
Scenario: Responses should have the correct headers
|
4
|
+
Given "child.js" is located at "javascripts" in the asset location
|
5
|
+
When a request is made for "javascripts/child.js"
|
6
|
+
Then the response code should be "200"
|
7
|
+
And the "Content-Length" header should be "40"
|
8
|
+
And the "Content-Type" header should be "text/javascript"
|
9
|
+
|
10
|
+
Scenario: Content should be cached
|
11
|
+
Given "child.js" is located at "javascripts" in the asset location
|
12
|
+
When a request is made for "javascripts/child.js"
|
13
|
+
Then the "Cache-Control" header should be "public, max-age=86400"
|
@@ -0,0 +1,13 @@
|
|
1
|
+
Feature: Sass integration
|
2
|
+
|
3
|
+
Scenario: Simple sass stylesheet serving
|
4
|
+
Given "sass_child.scss" is located at "stylesheets" in the asset location
|
5
|
+
When a request is made for "stylesheets/sass_child.css"
|
6
|
+
Then the response body should look like "processed_sass_child.css"
|
7
|
+
|
8
|
+
Scenario: Combined sass stylesheet serving
|
9
|
+
Given "sass_mixins.scss" is located at "stylesheets" in the asset location
|
10
|
+
And "sass_which_uses_mixin.scss" is located at "stylesheets" in the asset location
|
11
|
+
When a request is made for "stylesheets/sass_which_uses_mixin.css"
|
12
|
+
Then the response body should look like "processed_sass_which_uses_mixin.css"
|
13
|
+
|
@@ -0,0 +1,24 @@
|
|
1
|
+
Given /^"([^"]*)" is located at "([^"]*)" in the asset location$/ do |filename, folder|
|
2
|
+
cp fixture_path(filename), temp_path(folder)
|
3
|
+
end
|
4
|
+
|
5
|
+
When /^a request is made for "([^"]*)"$/ do |path|
|
6
|
+
get path
|
7
|
+
end
|
8
|
+
|
9
|
+
Then /^the response body should look like "([^"]*)"$/ do |filename|
|
10
|
+
last_response.body.should == load_fixture(filename)
|
11
|
+
end
|
12
|
+
|
13
|
+
Then /^the response code should be "([^\"]*)"$/ do |response_code|
|
14
|
+
last_response.status.should == response_code.to_i
|
15
|
+
end
|
16
|
+
|
17
|
+
When /^the "([^\"]*)" header should be "([^\"]*)"$/ do |header_key, expected_value|
|
18
|
+
last_response.headers[header_key].should == expected_value
|
19
|
+
end
|
20
|
+
|
21
|
+
Then /^the response should be from a call back into rack$/ do
|
22
|
+
last_response.status.should == 404
|
23
|
+
last_response.body.should include("Sinatra doesn't know this ditty.")
|
24
|
+
end
|
@@ -0,0 +1,31 @@
|
|
1
|
+
Feature: Stylesheet serving
|
2
|
+
|
3
|
+
Scenario: Simple stylesheet serving
|
4
|
+
Given "child.css" is located at "stylesheets" in the asset location
|
5
|
+
When a request is made for "stylesheets/child.css"
|
6
|
+
Then the response body should look like "child.css"
|
7
|
+
|
8
|
+
Scenario: Simple stylesheet combining
|
9
|
+
Given "parent.css" is located at "stylesheets" in the asset location
|
10
|
+
And "child.css" is located at "stylesheets" in the asset location
|
11
|
+
When a request is made for "stylesheets/parent.css"
|
12
|
+
Then the response body should look like "parent_with_child.css"
|
13
|
+
|
14
|
+
Scenario: Nested stylesheet combining
|
15
|
+
Given "grand_parent.css" is located at "stylesheets" in the asset location
|
16
|
+
And "parent.css" is located at "stylesheets" in the asset location
|
17
|
+
And "child.css" is located at "stylesheets" in the asset location
|
18
|
+
When a request is made for "stylesheets/grand_parent.css"
|
19
|
+
Then the response body should look like "grand_parent_with_parent_with_child.css"
|
20
|
+
|
21
|
+
Scenario: Stylesheet responses should have the correct headers
|
22
|
+
Given "child.css" is located at "stylesheets" in the asset location
|
23
|
+
When a request is made for "stylesheets/child.css"
|
24
|
+
Then the response code should be "200"
|
25
|
+
And the "Content-Length" header should be "35"
|
26
|
+
And the "Content-Type" header should be "text/css"
|
27
|
+
|
28
|
+
Scenario: Stylesheets should be cached
|
29
|
+
Given "child.css" is located at "stylesheets" in the asset location
|
30
|
+
When a request is made for "stylesheets/child.css"
|
31
|
+
Then the "Cache-Control" header should be "public, max-age=86400"
|
@@ -0,0 +1,24 @@
|
|
1
|
+
require 'rack/test'
|
2
|
+
require 'sinatra'
|
3
|
+
|
4
|
+
require File.join(File.dirname(__FILE__), '../../spec/spec_helper')
|
5
|
+
|
6
|
+
class MyApp < Sinatra::Application
|
7
|
+
|
8
|
+
end
|
9
|
+
|
10
|
+
World do
|
11
|
+
include Rack::Test::Methods
|
12
|
+
include StyloSpecHelpers
|
13
|
+
include FileUtils
|
14
|
+
|
15
|
+
def app
|
16
|
+
@app ||= Rack::Builder.new do
|
17
|
+
use Stylo::Rack
|
18
|
+
run MyApp
|
19
|
+
end
|
20
|
+
end
|
21
|
+
|
22
|
+
reset_paths
|
23
|
+
end
|
24
|
+
|
@@ -0,0 +1,14 @@
|
|
1
|
+
module Stylo
|
2
|
+
class Combiner
|
3
|
+
def initialize(asset_directory, require_pattern)
|
4
|
+
@require_pattern = require_pattern
|
5
|
+
@asset_directory = asset_directory
|
6
|
+
end
|
7
|
+
|
8
|
+
def process(content)
|
9
|
+
content.gsub @require_pattern do |match|
|
10
|
+
process(AssetLoader.load_content(File.join(@asset_directory, $1)))
|
11
|
+
end
|
12
|
+
end
|
13
|
+
end
|
14
|
+
end
|
data/lib/stylo/config.rb
ADDED
@@ -0,0 +1,14 @@
|
|
1
|
+
module Stylo
|
2
|
+
class Config
|
3
|
+
class << self
|
4
|
+
attr_accessor :asset_location
|
5
|
+
|
6
|
+
def pipeline
|
7
|
+
[Stylo::PipelineSteps::Stylesheet.new,
|
8
|
+
Stylo::PipelineSteps::Javascript.new,
|
9
|
+
Stylo::PipelineSteps::Sass.new,
|
10
|
+
Stylo::PipelineSteps::Caching.new]
|
11
|
+
end
|
12
|
+
end
|
13
|
+
end
|
14
|
+
end
|
@@ -0,0 +1,15 @@
|
|
1
|
+
module Stylo
|
2
|
+
module PipelineSteps
|
3
|
+
class Javascript
|
4
|
+
REQUIRE_PATTERN = /\/\/\/require "(.*)";/
|
5
|
+
|
6
|
+
def call(response)
|
7
|
+
return if response.has_content? || response.extension != '.js'
|
8
|
+
return unless content = AssetLoader.load_content(response.path)
|
9
|
+
combined_content = Combiner.new(File.dirname(response.path), REQUIRE_PATTERN).process(content)
|
10
|
+
|
11
|
+
response.set_body combined_content, :javascript
|
12
|
+
end
|
13
|
+
end
|
14
|
+
end
|
15
|
+
end
|
@@ -0,0 +1,18 @@
|
|
1
|
+
require 'sass'
|
2
|
+
|
3
|
+
module Stylo
|
4
|
+
module PipelineSteps
|
5
|
+
class Sass
|
6
|
+
def call(response)
|
7
|
+
return if response.has_content? || response.extension != '.css'
|
8
|
+
content = AssetLoader.load_content(response.path.gsub('.css', '.scss'))
|
9
|
+
return if content.nil?
|
10
|
+
|
11
|
+
combined_content = Combiner.new(File.dirname(response.path), REQUIRE_PATTERN).process(content)
|
12
|
+
processed_content = ::Sass::Engine.new(combined_content, {:syntax => :scss}).render
|
13
|
+
|
14
|
+
response.set_body(processed_content, :css)
|
15
|
+
end
|
16
|
+
end
|
17
|
+
end
|
18
|
+
end
|
@@ -0,0 +1,15 @@
|
|
1
|
+
module Stylo
|
2
|
+
module PipelineSteps
|
3
|
+
REQUIRE_PATTERN = /@import "(.*)";/
|
4
|
+
|
5
|
+
class Stylesheet
|
6
|
+
def call(response)
|
7
|
+
return if response.has_content? || response.extension != '.css'
|
8
|
+
return unless content = AssetLoader.load_content(response.path)
|
9
|
+
combined_content = Combiner.new(File.dirname(response.path), REQUIRE_PATTERN).process(content)
|
10
|
+
|
11
|
+
response.set_body combined_content, :css
|
12
|
+
end
|
13
|
+
end
|
14
|
+
end
|
15
|
+
end
|
@@ -0,0 +1,47 @@
|
|
1
|
+
module Stylo
|
2
|
+
class Processor
|
3
|
+
def initialize
|
4
|
+
end
|
5
|
+
|
6
|
+
def get_stylesheet_path(asset)
|
7
|
+
File.join(Stylo::Config.asset_location, asset)
|
8
|
+
end
|
9
|
+
|
10
|
+
def process_asset(asset)
|
11
|
+
asset_path = get_stylesheet_path(asset)
|
12
|
+
asset_extension = File.extname(asset_path)
|
13
|
+
|
14
|
+
file_content = get_file_content(asset_path)
|
15
|
+
if !file_content.nil?
|
16
|
+
asset_dir = File.dirname(asset_path)
|
17
|
+
|
18
|
+
import_regex = asset_extension == '.css' ? /@import "(.*)";/ : /\/\/\/require "(.*)";/
|
19
|
+
|
20
|
+
process_requires file_content, asset_dir, import_regex
|
21
|
+
else
|
22
|
+
nil
|
23
|
+
end
|
24
|
+
end
|
25
|
+
|
26
|
+
def get_file_content(asset_path)
|
27
|
+
return nil if !File.exist?(asset_path)
|
28
|
+
|
29
|
+
File.read(asset_path)
|
30
|
+
end
|
31
|
+
|
32
|
+
private
|
33
|
+
|
34
|
+
def process_requires(contents, base_path, import_regex)
|
35
|
+
contents.gsub import_regex do |match|
|
36
|
+
import_path = File.join(base_path, $1)
|
37
|
+
import_content = get_file_content(import_path)
|
38
|
+
|
39
|
+
if !import_content.nil?
|
40
|
+
process_requires(import_content, base_path, import_regex)
|
41
|
+
else
|
42
|
+
raise "Cannot find file to import '#{$1}'."
|
43
|
+
end
|
44
|
+
end
|
45
|
+
end
|
46
|
+
end
|
47
|
+
end
|
data/lib/stylo/rack.rb
ADDED
@@ -0,0 +1,20 @@
|
|
1
|
+
module Stylo
|
2
|
+
class Rack
|
3
|
+
def initialize(app)
|
4
|
+
@app = app
|
5
|
+
end
|
6
|
+
|
7
|
+
def call(env)
|
8
|
+
path = env["PATH_INFO"]
|
9
|
+
|
10
|
+
response = Response.new(path)
|
11
|
+
Stylo::Config.pipeline.each { |step| step.call(response) }
|
12
|
+
|
13
|
+
if response.has_content?
|
14
|
+
response.build
|
15
|
+
else
|
16
|
+
@app.call(env)
|
17
|
+
end
|
18
|
+
end
|
19
|
+
end
|
20
|
+
end
|
@@ -0,0 +1,13 @@
|
|
1
|
+
begin
|
2
|
+
require 'rails'
|
3
|
+
|
4
|
+
module Stylo
|
5
|
+
class Railtie < Rails::Railtie
|
6
|
+
initializer "stylo.initialize" do |app|
|
7
|
+
Stylo::Config.asset_location ||= File.join(app.root, 'public')
|
8
|
+
app.middleware.insert_before ActionDispatch::Static, Stylo::Rack
|
9
|
+
end
|
10
|
+
end
|
11
|
+
end
|
12
|
+
rescue LoadError
|
13
|
+
end
|
@@ -0,0 +1,41 @@
|
|
1
|
+
module Stylo
|
2
|
+
class Response
|
3
|
+
def initialize(path)
|
4
|
+
@headers = {}
|
5
|
+
@path = path
|
6
|
+
end
|
7
|
+
|
8
|
+
attr_reader :body, :headers, :path
|
9
|
+
|
10
|
+
def has_content?
|
11
|
+
!body.nil?
|
12
|
+
end
|
13
|
+
|
14
|
+
def extension
|
15
|
+
File.extname @path
|
16
|
+
end
|
17
|
+
|
18
|
+
def set_body(content, content_type)
|
19
|
+
@body = content
|
20
|
+
|
21
|
+
set_header 'Content-Length', content.length.to_s
|
22
|
+
set_header "Content-Type", case content_type
|
23
|
+
when :css
|
24
|
+
'text/css'
|
25
|
+
when :javascript
|
26
|
+
'text/javascript'
|
27
|
+
else
|
28
|
+
raise "Unknown content type #{content_type}"
|
29
|
+
end
|
30
|
+
end
|
31
|
+
|
32
|
+
def set_header(name, value)
|
33
|
+
@headers[name] = value
|
34
|
+
end
|
35
|
+
|
36
|
+
def build
|
37
|
+
[200, headers, body]
|
38
|
+
end
|
39
|
+
end
|
40
|
+
|
41
|
+
end
|
data/lib/stylo.rb
ADDED
@@ -0,0 +1,16 @@
|
|
1
|
+
require 'spec_helper'
|
2
|
+
|
3
|
+
describe Stylo::AssetLoader do
|
4
|
+
describe "load_content" do
|
5
|
+
it "should return nil if the content does not exist" do
|
6
|
+
Stylo::AssetLoader.load_content('javascripts/i_dont_exist.js').should be_nil
|
7
|
+
end
|
8
|
+
|
9
|
+
it "should return the file content if the content exists" do
|
10
|
+
content = "alert('hello');"
|
11
|
+
write_content(temp_path('javascripts/i_exist.js'), content)
|
12
|
+
|
13
|
+
Stylo::AssetLoader.load_content('javascripts/i_exist.js').should == content
|
14
|
+
end
|
15
|
+
end
|
16
|
+
end
|