123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507 |
- #ifdef _WIN32
- #define PATH_SEP ';'
- #else
- #define PATH_SEP ':'
- #endif
- #ifndef SASS_AST
- #include "ast.hpp"
- #endif
- #include "context.hpp"
- #include "constants.hpp"
- #include "parser.hpp"
- #include "file.hpp"
- #include "inspect.hpp"
- #include "output_nested.hpp"
- #include "output_compressed.hpp"
- #include "expand.hpp"
- #include "eval.hpp"
- #include "contextualize.hpp"
- #include "extend.hpp"
- #include "remove_placeholders.hpp"
- #include "copy_c_str.hpp"
- #include "color_names.hpp"
- #include "functions.hpp"
- #include "backtrace.hpp"
- #include "sass2scss.h"
- #ifndef SASS_PRELEXER
- #include "prelexer.hpp"
- #endif
- #include <iomanip>
- #include <iostream>
- #include <cstring>
- #include <sstream>
- namespace Sass {
- using namespace Constants;
- using namespace File;
- using std::cerr;
- using std::endl;
- Sass_Queued::Sass_Queued(const string& load_path, const string& abs_path, const char* source)
- {
- this->load_path = load_path;
- this->abs_path = abs_path;
- this->source = source;
- }
- Context::Context(Context::Data initializers)
- : mem(Memory_Manager<AST_Node>()),
- source_c_str (initializers.source_c_str()),
- sources (vector<const char*>()),
- include_paths (initializers.include_paths()),
- queue (vector<Sass_Queued>()),
- style_sheets (map<string, Block*>()),
- source_map (resolve_relative_path(initializers.output_path(), initializers.source_map_file(), get_cwd())),
- c_functions (vector<Sass_C_Function_Callback>()),
- image_path (initializers.image_path()),
- input_path (make_canonical_path(initializers.input_path())),
- output_path (make_canonical_path(initializers.output_path())),
- source_comments (initializers.source_comments()),
- output_style (initializers.output_style()),
- source_map_file (make_canonical_path(initializers.source_map_file())),
- source_map_embed (initializers.source_map_embed()),
- source_map_contents (initializers.source_map_contents()),
- omit_source_map_url (initializers.omit_source_map_url()),
- is_indented_syntax_src (initializers.is_indented_syntax_src()),
- importer (initializers.importer()),
- names_to_colors (map<string, Color*>()),
- colors_to_names (map<int, string>()),
- precision (initializers.precision()),
- _skip_source_map_update (initializers._skip_source_map_update()),
- subset_map (Subset_Map<string, pair<Complex_Selector*, Compound_Selector*> >())
- {
- cwd = get_cwd();
- // enforce some safe defaults
- // used to create relative file links
- if (input_path == "") input_path = "stdin";
- if (output_path == "") output_path = "stdout";
- include_paths.push_back(cwd);
- collect_include_paths(initializers.include_paths_c_str());
- collect_include_paths(initializers.include_paths_array());
- setup_color_map();
- string entry_point = initializers.entry_point();
- if (!entry_point.empty()) {
- string result(add_file(entry_point));
- if (result.empty()) {
- throw "File to read not found or unreadable: " + entry_point;
- }
- }
- }
- Context::~Context()
- {
- // everything that gets put into sources will be freed by us
- for (size_t i = 0; i < sources.size(); ++i) delete[] sources[i];
- for (size_t n = 0; n < import_stack.size(); ++n) sass_delete_import(import_stack[n]);
- sources.clear(); import_stack.clear();
- }
- void Context::setup_color_map()
- {
- size_t i = 0;
- while (color_names[i]) {
- string name(color_names[i]);
- Color* value = new (mem) Color("[COLOR TABLE]", Position(),
- color_values[i*4],
- color_values[i*4+1],
- color_values[i*4+2],
- color_values[i*4+3]);
- names_to_colors[name] = value;
- // only map fully opaque colors
- if (color_values[i*4+3] >= 1) {
- int numval = color_values[i*4]*0x10000;
- numval += color_values[i*4+1]*0x100;
- numval += color_values[i*4+2];
- colors_to_names[numval] = name;
- }
- ++i;
- }
- }
- void Context::collect_include_paths(const char* paths_str)
- {
- if (paths_str) {
- const char* beg = paths_str;
- const char* end = Prelexer::find_first<PATH_SEP>(beg);
- while (end) {
- string path(beg, end - beg);
- if (!path.empty()) {
- if (*path.rbegin() != '/') path += '/';
- include_paths.push_back(path);
- }
- beg = end + 1;
- end = Prelexer::find_first<PATH_SEP>(beg);
- }
- string path(beg);
- if (!path.empty()) {
- if (*path.rbegin() != '/') path += '/';
- include_paths.push_back(path);
- }
- }
- }
- void Context::collect_include_paths(const char** paths_array)
- {
- if (*include_paths.back().rbegin() != '/') include_paths.back() += '/';
- if (paths_array) {
- for (size_t i = 0; paths_array[i]; i++) {
- collect_include_paths(paths_array[i]);
- }
- }
- }
- void Context::add_source(string load_path, string abs_path, const char* contents)
- {
- sources.push_back(contents);
- included_files.push_back(abs_path);
- queue.push_back(Sass_Queued(load_path, abs_path, contents));
- source_map.source_index.push_back(sources.size() - 1);
- include_links.push_back(resolve_relative_path(abs_path, source_map_file, cwd));
- }
- string Context::add_file(string path)
- {
- using namespace File;
- char* contents = 0;
- string real_path;
- path = make_canonical_path(path);
- for (size_t i = 0, S = include_paths.size(); i < S; ++i) {
- string full_path(join_paths(include_paths[i], path));
- if (style_sheets.count(full_path)) return full_path;
- contents = resolve_and_load(full_path, real_path);
- if (contents) {
- add_source(full_path, real_path, contents);
- style_sheets[full_path] = 0;
- return full_path;
- }
- }
- return string();
- }
- string Context::add_file(string dir, string rel_filepath)
- {
- using namespace File;
- char* contents = 0;
- string real_path;
- rel_filepath = make_canonical_path(rel_filepath);
- string full_path(join_paths(dir, rel_filepath));
- if (style_sheets.count(full_path)) return full_path;
- contents = resolve_and_load(full_path, real_path);
- if (contents) {
- add_source(full_path, real_path, contents);
- style_sheets[full_path] = 0;
- return full_path;
- }
- for (size_t i = 0, S = include_paths.size(); i < S; ++i) {
- string full_path(join_paths(include_paths[i], rel_filepath));
- if (style_sheets.count(full_path)) return full_path;
- contents = resolve_and_load(full_path, real_path);
- if (contents) {
- add_source(full_path, real_path, contents);
- style_sheets[full_path] = 0;
- return full_path;
- }
- }
- return string();
- }
- void register_function(Context&, Signature sig, Native_Function f, Env* env);
- void register_function(Context&, Signature sig, Native_Function f, size_t arity, Env* env);
- void register_overload_stub(Context&, string name, Env* env);
- void register_built_in_functions(Context&, Env* env);
- void register_c_functions(Context&, Env* env, Sass_C_Function_List);
- void register_c_function(Context&, Env* env, Sass_C_Function_Callback);
- char* Context::compile_block(Block* root)
- {
- char* result = 0;
- if (!root) return 0;
- switch (output_style) {
- case COMPRESSED: {
- Output_Compressed output_compressed(this);
- root->perform(&output_compressed);
- string output = output_compressed.get_buffer();
- if (source_map_file != "" && !omit_source_map_url) {
- output += format_source_mapping_url(source_map_file);
- }
- result = copy_c_str(output.c_str());
- } break;
- default: {
- Output_Nested output_nested(source_comments, this);
- root->perform(&output_nested);
- string output = output_nested.get_buffer();
- if (source_map_file != "" && !omit_source_map_url) {
- output += "\n" + format_source_mapping_url(source_map_file);
- }
- result = copy_c_str(output.c_str());
- } break;
- }
- return result;
- }
- Block* Context::parse_file()
- {
- Block* root = 0;
- for (size_t i = 0; i < queue.size(); ++i) {
- struct Sass_Import* import = sass_make_import(
- queue[i].load_path.c_str(),
- queue[i].abs_path.c_str(),
- 0, 0
- );
- import_stack.push_back(import);
- Parser p(Parser::from_c_str(queue[i].source, *this, queue[i].abs_path, Position(1 + i, 1, 1)));
- Block* ast = p.parse();
- sass_delete_import(import_stack.back());
- import_stack.pop_back();
- if (i == 0) root = ast;
- style_sheets[queue[i].load_path] = ast;
- }
- if (root == 0) return 0;
- Env tge;
- Backtrace backtrace(0, "", Position(), "");
- register_built_in_functions(*this, &tge);
- for (size_t i = 0, S = c_functions.size(); i < S; ++i) {
- register_c_function(*this, &tge, c_functions[i]);
- }
- Eval eval(*this, &tge, &backtrace);
- Contextualize contextualize(*this, &eval, &tge, &backtrace);
- Expand expand(*this, &eval, &contextualize, &tge, &backtrace);
- // Inspect inspect(this);
- // Output_Nested output_nested(*this);
- root = root->perform(&expand)->block();
- if (!subset_map.empty()) {
- Extend extend(*this, subset_map);
- root->perform(&extend);
- }
- Remove_Placeholders remove_placeholders(*this);
- root->perform(&remove_placeholders);
- return root;
- }
- Block* Context::parse_string()
- {
- if (!source_c_str) return 0;
- queue.clear();
- if(is_indented_syntax_src) {
- char * contents = sass2scss(source_c_str, SASS2SCSS_PRETTIFY_1);
- add_source(input_path, input_path, contents);
- return parse_file();
- }
- add_source(input_path, input_path, strdup(source_c_str));
- return parse_file();
- }
- char* Context::compile_file()
- {
- // returns NULL if something fails
- return compile_block(parse_file());
- }
- char* Context::compile_string()
- {
- // returns NULL if something fails
- return compile_block(parse_string());
- }
- string Context::format_source_mapping_url(const string& file)
- {
- string url = resolve_relative_path(file, output_path, cwd);
- if (source_map_embed) {
- string map = source_map.generate_source_map(*this);
- istringstream is( map );
- ostringstream buffer;
- base64::encoder E;
- E.encode(is, buffer);
- url = "data:application/json;base64," + buffer.str();
- url.erase(url.size() - 1);
- }
- return "/*# sourceMappingURL=" + url + " */";
- }
- char* Context::generate_source_map()
- {
- if (source_map_file == "") return 0;
- char* result = 0;
- string map = source_map.generate_source_map(*this);
- result = copy_c_str(map.c_str());
- return result;
- }
- std::vector<std::string> Context::get_included_files(size_t skip)
- {
- vector<string> includes = included_files;
- std::sort( includes.begin() + skip, includes.end() );
- includes.erase( std::unique( includes.begin(), includes.end() ), includes.end() );
- // the skip solution seems more robust, as we may have real files named stdin
- // includes.erase( std::remove( includes.begin(), includes.end(), "stdin" ), includes.end() );
- return includes;
- }
- string Context::get_cwd()
- {
- return Sass::File::get_cwd();
- }
- void register_function(Context& ctx, Signature sig, Native_Function f, Env* env)
- {
- Definition* def = make_native_function(sig, f, ctx);
- def->environment(env);
- (*env)[def->name() + "[f]"] = def;
- }
- void register_function(Context& ctx, Signature sig, Native_Function f, size_t arity, Env* env)
- {
- Definition* def = make_native_function(sig, f, ctx);
- stringstream ss;
- ss << def->name() << "[f]" << arity;
- def->environment(env);
- (*env)[ss.str()] = def;
- }
- void register_overload_stub(Context& ctx, string name, Env* env)
- {
- Definition* stub = new (ctx.mem) Definition("[built-in function]",
- Position(),
- 0,
- name,
- 0,
- 0,
- true);
- (*env)[name + "[f]"] = stub;
- }
- void register_built_in_functions(Context& ctx, Env* env)
- {
- using namespace Functions;
- // RGB Functions
- register_function(ctx, rgb_sig, rgb, env);
- register_overload_stub(ctx, "rgba", env);
- register_function(ctx, rgba_4_sig, rgba_4, 4, env);
- register_function(ctx, rgba_2_sig, rgba_2, 2, env);
- register_function(ctx, red_sig, red, env);
- register_function(ctx, green_sig, green, env);
- register_function(ctx, blue_sig, blue, env);
- register_function(ctx, mix_sig, mix, env);
- // HSL Functions
- register_function(ctx, hsl_sig, hsl, env);
- register_function(ctx, hsla_sig, hsla, env);
- register_function(ctx, hue_sig, hue, env);
- register_function(ctx, saturation_sig, saturation, env);
- register_function(ctx, lightness_sig, lightness, env);
- register_function(ctx, adjust_hue_sig, adjust_hue, env);
- register_function(ctx, lighten_sig, lighten, env);
- register_function(ctx, darken_sig, darken, env);
- register_function(ctx, saturate_sig, saturate, env);
- register_function(ctx, desaturate_sig, desaturate, env);
- register_function(ctx, grayscale_sig, grayscale, env);
- register_function(ctx, complement_sig, complement, env);
- register_function(ctx, invert_sig, invert, env);
- // Opacity Functions
- register_function(ctx, alpha_sig, alpha, env);
- register_function(ctx, opacity_sig, alpha, env);
- register_function(ctx, opacify_sig, opacify, env);
- register_function(ctx, fade_in_sig, opacify, env);
- register_function(ctx, transparentize_sig, transparentize, env);
- register_function(ctx, fade_out_sig, transparentize, env);
- // Other Color Functions
- register_function(ctx, adjust_color_sig, adjust_color, env);
- register_function(ctx, scale_color_sig, scale_color, env);
- register_function(ctx, change_color_sig, change_color, env);
- register_function(ctx, ie_hex_str_sig, ie_hex_str, env);
- // String Functions
- register_function(ctx, unquote_sig, sass_unquote, env);
- register_function(ctx, quote_sig, sass_quote, env);
- register_function(ctx, str_length_sig, str_length, env);
- register_function(ctx, str_insert_sig, str_insert, env);
- register_function(ctx, str_index_sig, str_index, env);
- register_function(ctx, str_slice_sig, str_slice, env);
- register_function(ctx, to_upper_case_sig, to_upper_case, env);
- register_function(ctx, to_lower_case_sig, to_lower_case, env);
- // Number Functions
- register_function(ctx, percentage_sig, percentage, env);
- register_function(ctx, round_sig, round, env);
- register_function(ctx, ceil_sig, ceil, env);
- register_function(ctx, floor_sig, floor, env);
- register_function(ctx, abs_sig, abs, env);
- register_function(ctx, min_sig, min, env);
- register_function(ctx, max_sig, max, env);
- register_function(ctx, random_sig, random, env);
- // List Functions
- register_function(ctx, length_sig, length, env);
- register_function(ctx, nth_sig, nth, env);
- register_function(ctx, set_nth_sig, set_nth, env);
- register_function(ctx, index_sig, index, env);
- register_function(ctx, join_sig, join, env);
- register_function(ctx, append_sig, append, env);
- register_function(ctx, compact_sig, compact, env);
- register_function(ctx, zip_sig, zip, env);
- register_function(ctx, list_separator_sig, list_separator, env);
- // Map Functions
- register_function(ctx, map_get_sig, map_get, env);
- register_function(ctx, map_merge_sig, map_merge, env);
- register_function(ctx, map_remove_sig, map_remove, env);
- register_function(ctx, map_keys_sig, map_keys, env);
- register_function(ctx, map_values_sig, map_values, env);
- register_function(ctx, map_has_key_sig, map_has_key, env);
- register_function(ctx, keywords_sig, keywords, env);
- // Introspection Functions
- register_function(ctx, type_of_sig, type_of, env);
- register_function(ctx, unit_sig, unit, env);
- register_function(ctx, unitless_sig, unitless, env);
- register_function(ctx, comparable_sig, comparable, env);
- register_function(ctx, variable_exists_sig, variable_exists, env);
- register_function(ctx, global_variable_exists_sig, global_variable_exists, env);
- register_function(ctx, function_exists_sig, function_exists, env);
- register_function(ctx, mixin_exists_sig, mixin_exists, env);
- register_function(ctx, feature_exists_sig, feature_exists, env);
- register_function(ctx, call_sig, call, env);
- // Boolean Functions
- register_function(ctx, not_sig, sass_not, env);
- register_function(ctx, if_sig, sass_if, env);
- // Path Functions
- register_function(ctx, image_url_sig, image_url, env);
- // Misc Functions
- register_function(ctx, inspect_sig, inspect, env);
- register_function(ctx, unique_id_sig, unique_id, env);
- }
- void register_c_functions(Context& ctx, Env* env, Sass_C_Function_List descrs)
- {
- while (descrs && *descrs) {
- register_c_function(ctx, env, *descrs);
- ++descrs;
- }
- }
- void register_c_function(Context& ctx, Env* env, Sass_C_Function_Callback descr)
- {
- Definition* def = make_c_function(
- sass_function_get_signature(descr),
- sass_function_get_function(descr),
- sass_function_get_cookie(descr),
- ctx
- );
- def->environment(env);
- (*env)[def->name() + "[f]"] = def;
- }
- }
|