sweet-js/sweet-core

Optional tokens

Closed this issue · 7 comments

Hi, Ive been trying to implement a "def" macro that supports default parameters. What I could come up with is this:

macro def {
    case ($($p:ident = $v:expr) (,) ...) { $body:SourceElements } => {
        def($p (,) ...) {
            $($p = typeof $p === 'undefined' ? $v : $p) (;) ...;
            $body
        };
    }

    case ($params:ident (,) ...) { $body:SourceElements } => {
        $(function($params (,) ...) {
            $body
        });
    }
}

So, it works for def(a, b) { } and def(a = 4, b = a) { } but not def(a, b = 4) { }. What would be a good way to do this? I thought if I could denote the = $v:expr part as optional, it would be ok but have no idea about the syntax.

Is there a better way to do this? Maybe a recursive solution?

I think what you might want is some helper macros:

macro helper_params_names {
  // extract just the param names
}

macro helper_assign_default {
  // do default assignment
}

macro def {
    case ($params ...) { $body } => {
      (function(helper_param_names($params ...)) {
        helper_assign_default($params ...);
        $body
      });
    }
}

I receive this error when I try to place a macro in function parameter definitions:

15    case ($params:ident (,) ...) { $body:SourceElements } => {
16        (function(helper_expand_params($params (,) ...)) {
17            $body
18        });
19    }
Error: Line 16: Unexpected token (
    at throwError (/usr/local/lib/node_modules/sweet.js/lib/sweet.js:1172:21)
    at throwUnexpected (/usr/local/lib/node_modules/sweet.js/lib/sweet.js:1233:9)
    at parseVariableIdentifier (/usr/local/lib/node_modules/sweet.js/lib/sweet.js:2172:13)
    at parseFunctionExpression (/usr/local/lib/node_modules/sweet.js/lib/sweet.js:3062:25)
    at parsePrimaryExpression (/usr/local/lib/node_modules/sweet.js/lib/sweet.js:1540:24)
    at parseLeftHandSideExpressionAllowCall (/usr/local/lib/node_modules/sweet.js/lib/sweet.js:1734:48)
    at parsePostfixExpression (/usr/local/lib/node_modules/sweet.js/lib/sweet.js:1781:20)
    at parseUnaryExpression (/usr/local/lib/node_modules/sweet.js/lib/sweet.js:1851:16)
    at parseMultiplicativeExpression (/usr/local/lib/node_modules/sweet.js/lib/sweet.js:1857:20)
    at parseAdditiveExpression (/usr/local/lib/node_modules/sweet.js/lib/sweet.js:1874:20)

Can you post your entire code?

I did not went on to recursive param helpers but this much should work as far as I can tell;

macro expand_params {
    case ($param ...) => {
        $param ...
    }
}

macro def {
    case ($params ...) { $body:SourceElements } => {
        (function(expand_params($params ...)) {
            $body
        });
    }
}

def(a, b) {
    return a + b;
}
/usr/local/lib/node_modules/sweet.js/lib/sweet.js:5038
            throw e;
                  ^
Error: Line 9: Unexpected token (
    at throwError (/usr/local/lib/node_modules/sweet.js/lib/sweet.js:1172:21)
    at throwUnexpected (/usr/local/lib/node_modules/sweet.js/lib/sweet.js:1233:9)
    at parseVariableIdentifier (/usr/local/lib/node_modules/sweet.js/lib/sweet.js:2172:13)
    at parseFunctionExpression (/usr/local/lib/node_modules/sweet.js/lib/sweet.js:3062:25)
    at parsePrimaryExpression (/usr/local/lib/node_modules/sweet.js/lib/sweet.js:1540:24)
    at parseLeftHandSideExpressionAllowCall (/usr/local/lib/node_modules/sweet.js/lib/sweet.js:1734:48)
    at parsePostfixExpression (/usr/local/lib/node_modules/sweet.js/lib/sweet.js:1781:20)
    at parseUnaryExpression (/usr/local/lib/node_modules/sweet.js/lib/sweet.js:1851:16)
    at parseMultiplicativeExpression (/usr/local/lib/node_modules/sweet.js/lib/sweet.js:1857:20)
    at parseAdditiveExpression (/usr/local/lib/node_modules/sweet.js/lib/sweet.js:1874:20)

Ah! Looks like I led you down the wrong path. We don't currently expand macros in the parameter position of a function, and I don't think we ever will but need to think about that more. Clearly we need a better error message though.

You might think we could overcome this by using another helper macro, something like:

macro expand_params {
    case ($param ...) => {
        $param ...
    }
}

macro helper_def {
  case ($params ...) { $body ... } => {
    (function ($params ...) { $body ... })
  }
}

macro def {
    case ($params ...) { $body ... } => {
      helper_def (expand_params($params ...)) { $body ... }
    }
}

def(a, b) {
    return a + b;
}

And this should work but a bug is preventing nested macros from being expanded (I'm currently working on fixing this).

Hey, thanks for the info. I actually tried this one too, facing some error, backed up to former one.

BTW, this is the sole "compile to javascript" project that makes a lot of sense to me. Great work, hope it matures as fast as possible, waiting to use it in production :)

Great the hear! Thanks for being willing to bang your head against it for a little bit :)