Huge News!Announcing our $40M Series B led by Abstract Ventures.Learn More
Socket
Sign inDemoInstall
Socket

sirop

Package Overview
Dependencies
Maintainers
1
Alerts
File Explorer

Advanced tools

Socket logo

Install Socket

Detect and block malicious and high-risk dependencies

Install

sirop

  • 0.4
  • Rubygems
  • Socket score

Version published
Maintainers
1
Created
Source

Sirop

Sirop is a Ruby gem for manipulating Ruby source code. Sirop is very young, so the following information might be incomplete, out of date, or simply wrong!

Use Cases

Some of the use cases addressed by Sirop are:

  • Compile DSLs into optimized Ruby code. This is especially interesting for HTML templating DSLs in libraries like Phlex, Papercraft etc. Example
  • Get the source of a given block or method.
  • Rewrite parts of Ruby code, for implementing Ruby macros (and why not?).

Limitations

  • Sirop supports Ruby 3.2 or newer.
  • Sirop can be used only on blocks and methods defined in a file, so cannot really be used on dynamically eval'd Ruby code, or in an IRB/Pry session.

Getting the AST/source of a Ruby proc or method

To get the AST of a proc or a method, use Sirop.to_ast:

# for a proc
mul = ->(x, y) { x * y }
Sirop.to_ast(mul) #=> ...

# for a method
def foo; :bar; end
Sirop.to_ast(method(:foo)) #=> ...

To get the source of a proc or a method, use Sirop.to_source:

mul = ->(x, y) { x * y }
Sirop.to_source(mul) #=> "->(x, y) { x * y }"

def foo; :bar; end
Sirop.to_source(method(:foo)) #=> "def foo; :bar; end"

Rewriting Ruby code

You can consult the DSL compiler example. This example intercepts method calls by defining a visit_call_node method:

# Annotated with some explanations
def visit_call_node(node)
  # don't rewrite if the call has a receiver
  return super if node.receiver

  # set HTML location start
  @html_location_start ||= node.location
  # get method arguments...
  inner_text, attrs = tag_args(node)
  # and block
  block = node.block

  # emit HTML tag according to given arguments
  if inner_text
    emit_tag_open(node, attrs)
    emit_tag_inner_text(inner_text)
    emit_tag_close(node)
  elsif block
    emit_tag_open(node, attrs)
    visit(block.body)
    emit_tag_close(node)
  else
    emit_tag_open_close(node, attrs)
  end
  # set HTML location end
  @html_location_end = node.location
end

Future directions

  • Implement a macro expander with support for quote/unquote:

    trace_macro = Sirop.macro do |ast|
      source = Sirop.to_source(ast)
      quote do
        result = unquote(ast)
        puts "The result of #{source} is: #{result}"
        result
      end
    end
    
    def add(x, y)
      trace(x + y)
    end
    
    Sirop.expand_macros(method(:add), trace: trace_macro)
    
  • Implement a DSL compiler with hooks for easier usage in DSL libraries.

Contributing

We gladly welcome contributions from anyone! Some areas that need work currently are:

Please feel free to contribute PR's and issues

FAQs

Package last updated on 29 Apr 2024

Did you know?

Socket

Socket for GitHub automatically highlights issues in each pull request and monitors the health of all your open source dependencies. Discover the contents of your packages and block harmful activity before you install or update your dependencies.

Install

Related posts

SocketSocket SOC 2 Logo

Product

  • Package Alerts
  • Integrations
  • Docs
  • Pricing
  • FAQ
  • Roadmap
  • Changelog

Packages

npm

Stay in touch

Get open source security insights delivered straight into your inbox.


  • Terms
  • Privacy
  • Security

Made with ⚡️ by Socket Inc