Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Allow optional params #7

Merged
merged 7 commits into from
Jun 24, 2015
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
18 changes: 4 additions & 14 deletions Rakefile
Original file line number Diff line number Diff line change
Expand Up @@ -4,22 +4,12 @@ rescue LoadError
puts 'Cannot load bundler/gem_tasks'
end

task default: :prepare
require 'tasks/libsass'

task prepare: "ext/lib/libsass.so"
task default: :test

file "ext/lib/libsass.so" do
gem_dir = File.expand_path(File.dirname(__FILE__)) + "/"
cd "ext/libsass"
sh 'make lib/libsass.so LDFLAGS="-Wall -O2"'
cd gem_dir
end

task test: :prepare do
desc "Run all tests"
task test: 'libsass:compile' do
$LOAD_PATH.unshift('lib', 'test')
Dir.glob('./test/**/*_test.rb') { |f| require f }
end

task :submodule do
sh "git submodule update --init"
end
3 changes: 3 additions & 0 deletions ext/Rakefile
Original file line number Diff line number Diff line change
@@ -0,0 +1,3 @@
require_relative '../lib/tasks/libsass'

task default: 'libsass:compile'
61 changes: 48 additions & 13 deletions lib/sassc/functions_handler.rb
Original file line number Diff line number Diff line change
Expand Up @@ -10,25 +10,45 @@ def setup(native_options)

list = Native.make_function_list(Script.custom_functions.count)

functs = FunctionWrapper.extend(Script::Functions)
functs.options = @options
functions = FunctionWrapper.extend(Script::Functions)
functions.options = @options

Script.custom_functions.each_with_index do |custom_function, i|
@callbacks[custom_function] = FFI::Function.new(:pointer, [:pointer, :pointer]) do |s_args, cookie|
length = Native.list_get_length(s_args)
@callbacks[custom_function] = FFI::Function.new(:pointer, [:pointer, :pointer]) do |native_argument_list, cookie|
native_argument_list_length = Native.list_get_length(native_argument_list)
custom_function_arguments = []
error_tag = nil

v = Native.list_get_value(s_args, 0)
v = Native.string_get_value(v).dup
(0...native_argument_list_length).each do |i|
native_value = Native.list_get_value(native_argument_list, i)

s = Script::String.new(Script::String.unquote(v), Script::String.type(v))
case value_tag = Native.value_get_tag(native_value)
when :sass_null
# no-op
when :sass_string
native_string = Native.string_get_value(native_value)
argument = Script::String.new(Script::String.unquote(native_string), Script::String.type(native_string))

value = functs.send(custom_function, s)
custom_function_arguments << argument
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

do we want to add an else here that raises something like NotImplemented for now?

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

For values that aren’t a :sass_string? Yeah, that’s a good idea.

I also kinda want to just implement the rest 😄

It looks like the type classes like SassC:Script:String look just like their Sass versions, but with an added #to_native and a couple other tweaks.

How would you feel about temporarily requiring the Sass counterparts like https://github.com/sass/sass/blob/stable/lib/sass/script/value/string.rb and inheriting + extending/overriding those classes? Then we wouldn’t fall behind as Sass grows / our copy-pasted version stays static.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

it's unfortunate to have the sass dependency long term, but right at the moment, sassc-rails requires sass too (because sprockets requires sass). I suppose that would be a good solution for the present.

I can't exactly remember, but i think I may have had to tweak the String value class (i.e. it might not be exactly the same as it is in sass).

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Purely for informational purposes, and not to add an opinion or bias one way or the other, we did a diff. The only things that you seem to have changed are:

  • removed a super call from the constructor
  • added self.type and self.unquote class methods
  • added #to_native
  • removed #plus, #to_sass, and #inspect. Or maybe those just didn't exist yet?

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Aside from that, both keeping sass as a dependency and copy/pasting the Sass::Script counterparts feel equally dirty to me :(

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

do we want to add an else here that raises something like NotImplemented for now?

@bolandrm Since hooking into libsass’ own error handling stuff is a little too tricky for me at the moment, I’ve added some code that upon error, spits out a Sass syntax error and also logs to $stderr.

else
error_tag = error("Sass argument of type #{value_tag} unsupported")
break
end
end

next error_tag if error_tag

if value
value = Script::String.new(Script::String.unquote(value.to_s), value.type)
value.to_native
else
Script::String.new("").to_native
begin
value = functions.send(custom_function, *custom_function_arguments)

if value
value = Script::String.new(Script::String.unquote(value.to_s), value.type)
value.to_native
else
Script::String.new("").to_native
end
rescue StandardError => exception
error(exception.message)
end
end

Expand All @@ -48,6 +68,21 @@ def setup(native_options)

private

def error(message)
value = Native::SassValue.new
value[:unknown] = Native::SassUnknown.new

error = Native::SassError.new
error[:tag] = :sass_error

Native.error_set_message(error, Native.native_string(message))
$stderr.puts "[SassC::FunctionsHandler] #{message}"

value[:unknown][:tag] = :sass_error
value[:error] = error
value.pointer
end

class FunctionWrapper
class << self
attr_accessor :options
Expand Down
3 changes: 2 additions & 1 deletion lib/sassc/native.rb
Original file line number Diff line number Diff line change
Expand Up @@ -48,7 +48,8 @@ def self.return_string_array(ptr)
end

def self.native_string(string)
string += "\0"
string = string.to_s
string << "\0"
data = Native::LibC.malloc(string.bytesize)
data.write_string(string)
data
Expand Down
6 changes: 6 additions & 0 deletions lib/sassc/native/native_functions_api.rb
Original file line number Diff line number Diff line change
Expand Up @@ -19,6 +19,7 @@ module Native

# ADDAPI enum Sass_Tag ADDCALL sass_value_get_tag (const union Sass_Value* v);
attach_function :sass_value_get_tag, [:sass_value_ptr], SassTag
attach_function :sass_value_is_null, [:sass_value_ptr], :bool

# ADDAPI const char* ADDCALL sass_string_get_value (const union Sass_Value* v);
attach_function :sass_string_get_value, [:sass_value_ptr], :string
Expand All @@ -28,6 +29,11 @@ module Native
attach_function :sass_list_get_length, [:sass_value_ptr], :size_t
attach_function :sass_list_get_value, [:sass_value_ptr, :size_t], :sass_value_ptr

# ADDAPI char* ADDCALL sass_error_get_message (const union Sass_Value* v);
# ADDAPI void ADDCALL sass_error_set_message (union Sass_Value* v, char* msg);
attach_function :sass_error_get_message, [:sass_value_ptr], :string
attach_function :sass_error_set_message, [:sass_value_ptr, :pointer], :void
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

🤘


# Getters for custom function descriptors
# ADDAPI const char* ADDCALL sass_function_get_signature (Sass_C_Function_Callback fn);
# ADDAPI Sass_C_Function ADDCALL sass_function_get_function (Sass_C_Function_Callback fn);
Expand Down
4 changes: 2 additions & 2 deletions lib/sassc/native/sass_value.rb
Original file line number Diff line number Diff line change
Expand Up @@ -71,12 +71,12 @@ class SassNull < FFI::Struct

class SassError < FFI::Struct
layout :tag, SassTag,
:messsage, :string
:message, :string
end

class SassWarning < FFI::Struct
layout :tag, SassTag,
:messsage, :string
:message, :string
end

class SassValue # < FFI::Union
Expand Down
5 changes: 2 additions & 3 deletions lib/sassc/script.rb
Original file line number Diff line number Diff line change
Expand Up @@ -8,10 +8,9 @@ def self.custom_functions

def self.formatted_function_name(function_name)
params = Functions.instance_method(function_name).parameters
params = params.select { |param| param[0] == :req }
.map(&:first)
.map { |p| "$#{p}" }
params = params.map { |param_type, name| "$#{name}#{': null' if param_type == :opt}" }
.join(", ")

"#{function_name}(#{params})"
end
end
Expand Down
4 changes: 2 additions & 2 deletions lib/sassc/script/string.rb
Original file line number Diff line number Diff line change
Expand Up @@ -66,12 +66,12 @@ def self.type(contents)
def self.unquote(contents)
s = contents.dup

case contents[0,1]
case contents[0, 1]
when "'", '"', '`'
s[0] = ''
end

case contents[-1,1]
case contents[-1, 1]
when "'", '"', '`'
s[-1] = ''
end
Expand Down
21 changes: 21 additions & 0 deletions lib/tasks/libsass.rb
Original file line number Diff line number Diff line change
@@ -0,0 +1,21 @@
namespace :libsass do
desc "Compile libsass"
task compile: "ext/libsass/lib/libsass.so"

file "ext/libsass/.git" do
sh "git submodule update --init"
end

file "ext/libsass/lib/libsass.so" => "ext/libsass/.git" do
libsass_path = ""
if Dir.pwd.end_with?('/ext')
libsass_path = "libsass"
else
libsass_path = "ext/libsass"
end

cd libsass_path do
sh 'make lib/libsass.so LDFLAGS="-Wall -O2"'
end
end
end
5 changes: 3 additions & 2 deletions sassc.gemspec
Original file line number Diff line number Diff line change
Expand Up @@ -16,9 +16,10 @@ Gem::Specification.new do |spec|
spec.files = `git ls-files -z`.split("\x0")
spec.executables = spec.files.grep(%r{^bin/}) { |f| File.basename(f) }
spec.test_files = spec.files.grep(%r{^(test|spec|features)/})
spec.require_paths = ["lib", "ext"]

spec.extensions = ["Rakefile"]
spec.require_paths = ["lib"]

spec.extensions = ["ext/Rakefile"]

spec.add_development_dependency "rake"
spec.add_development_dependency "minitest", "~> 5.5.1"
Expand Down
112 changes: 89 additions & 23 deletions test/functions_test.rb
Original file line number Diff line number Diff line change
@@ -1,34 +1,16 @@
require_relative "test_helper"
require "stringio"

module SassC
class FunctionsTest < MiniTest::Test
include FixtureHelper

SassString = Struct.new(:value, :type) do
def to_s
value
end
def setup
@real_stderr, $stderr = $stderr, StringIO.new
end

module Script::Functions
def javascript_path(path)
Script::String.new("/js/#{path.value}", :string)
end

def no_return_path(path)
nil
end

def sass_return_path(path)
return SassString.new("'#{path.value}'", :string)
end

module Compass
def stylesheet_path(path)
Script::String.new("/css/#{path.value}", :identifier)
end
end
include Compass
def teardown
$stderr = @real_stderr
end

def test_functions_may_return_sass_string_type
Expand Down Expand Up @@ -70,5 +52,89 @@ def test_function_with_no_return_value
url: url(); }
EOS
end

def test_function_with_optional_arguments
engine = Engine.new("div {url: optional_arguments('first'); url: optional_arguments('second', 'qux')}")
assert_equal <<-EOS, engine.render
div {
url: "first/bar";
url: "second/qux"; }
EOS
end

def test_function_with_unsupported_tag
engine = Engine.new("div {url: function_with_unsupported_tag(red);}")

exception = assert_raises(SassC::SyntaxError) do
engine.render
end

assert_equal "Error: error in C function function_with_unsupported_tag: Sass argument of type sass_color unsupported\n\n Backtrace:\n \tstdin:1, in function `function_with_unsupported_tag`\n \tstdin:1\n on line 1 of stdin\n>> div {url: function_with_unsupported_tag(red);}\n ----------^\n", exception.message

assert_equal "[SassC::FunctionsHandler] Sass argument of type sass_color unsupported", stderr_output
end

def test_function_with_error
engine = Engine.new("div {url: function_that_raises_errors();}")
exception = assert_raises(SassC::SyntaxError) do
engine.render
end

assert_equal "Error: error in C function function_that_raises_errors: Intentional wrong thing happened somewhere inside the custom function

Backtrace:
\tstdin:1, in function `function_that_raises_errors`
\tstdin:1
on line 1 of stdin
>> div {url: function_that_raises_errors();}
----------^
", exception.message

assert_equal "[SassC::FunctionsHandler] Intentional wrong thing happened somewhere inside the custom function", stderr_output
end

private

SassString = Struct.new(:value, :type) do
def to_s
value
end
end

module Script::Functions
def javascript_path(path)
Script::String.new("/js/#{path.value}", :string)
end

def no_return_path(path)
nil
end

def sass_return_path(path)
return SassString.new("'#{path.value}'", :string)
end

def optional_arguments(path, optional = "bar")
return SassString.new("#{path}/#{optional}", :string)
end

def function_that_raises_errors()
raise StandardError, "Intentional wrong thing happened somewhere inside the custom function"
end

def function_with_unsupported_tag(color)
end

module Compass
def stylesheet_path(path)
Script::String.new("/css/#{path.value}", :identifier)
end
end
include Compass
end

def stderr_output
$stderr.string.gsub("\u0000\n", '')
end
end
end