Skip to content

Commit c1264c9

Browse files
committed
Initial commit
0 parents  commit c1264c9

31 files changed

+1411
-0
lines changed

.github/dependabot.yml

+6
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,6 @@
1+
version: 2
2+
updates:
3+
- package-ecosystem: "bundler"
4+
directory: "/"
5+
schedule:
6+
interval: "daily"

.github/workflows/main.yml

+32
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,32 @@
1+
name: Main
2+
on:
3+
- push
4+
- pull_request_target
5+
jobs:
6+
ci:
7+
name: CI
8+
runs-on: ubuntu-latest
9+
env:
10+
CI: true
11+
steps:
12+
- uses: actions/checkout@master
13+
- uses: ruby/setup-ruby@v1
14+
with:
15+
bundler-cache: true
16+
ruby-version: '3.1'
17+
- name: Test
18+
run: bundle exec rake test
19+
automerge:
20+
name: AutoMerge
21+
needs: ci
22+
runs-on: ubuntu-latest
23+
if: github.event_name == 'pull_request_target' && github.actor == 'dependabot[bot]'
24+
steps:
25+
- uses: actions/github-script@v3
26+
with:
27+
script: |
28+
github.pulls.merge({
29+
owner: context.payload.repository.owner.login,
30+
repo: context.payload.repository.name,
31+
pull_number: context.payload.pull_request.number
32+
})

.gitignore

+2
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,2 @@
1+
coverage
2+
test.haml

CODE_OF_CONDUCT.md

+74
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,74 @@
1+
# Contributor Covenant Code of Conduct
2+
3+
## Our Pledge
4+
5+
In the interest of fostering an open and welcoming environment, we as
6+
contributors and maintainers pledge to making participation in our project and
7+
our community a harassment-free experience for everyone, regardless of age, body
8+
size, disability, ethnicity, gender identity and expression, level of experience,
9+
nationality, personal appearance, race, religion, or sexual identity and
10+
orientation.
11+
12+
## Our Standards
13+
14+
Examples of behavior that contributes to creating a positive environment
15+
include:
16+
17+
* Using welcoming and inclusive language
18+
* Being respectful of differing viewpoints and experiences
19+
* Gracefully accepting constructive criticism
20+
* Focusing on what is best for the community
21+
* Showing empathy towards other community members
22+
23+
Examples of unacceptable behavior by participants include:
24+
25+
* The use of sexualized language or imagery and unwelcome sexual attention or
26+
advances
27+
* Trolling, insulting/derogatory comments, and personal or political attacks
28+
* Public or private harassment
29+
* Publishing others' private information, such as a physical or electronic
30+
address, without explicit permission
31+
* Other conduct which could reasonably be considered inappropriate in a
32+
professional setting
33+
34+
## Our Responsibilities
35+
36+
Project maintainers are responsible for clarifying the standards of acceptable
37+
behavior and are expected to take appropriate and fair corrective action in
38+
response to any instances of unacceptable behavior.
39+
40+
Project maintainers have the right and responsibility to remove, edit, or
41+
reject comments, commits, code, wiki edits, issues, and other contributions
42+
that are not aligned to this Code of Conduct, or to ban temporarily or
43+
permanently any contributor for other behaviors that they deem inappropriate,
44+
threatening, offensive, or harmful.
45+
46+
## Scope
47+
48+
This Code of Conduct applies both within project spaces and in public spaces
49+
when an individual is representing the project or its community. Examples of
50+
representing a project or community include using an official project e-mail
51+
address, posting via an official social media account, or acting as an appointed
52+
representative at an online or offline event. Representation of a project may be
53+
further defined and clarified by project maintainers.
54+
55+
## Enforcement
56+
57+
Instances of abusive, harassing, or otherwise unacceptable behavior may be
58+
reported by contacting the project team at [email protected]. All
59+
complaints will be reviewed and investigated and will result in a response that
60+
is deemed necessary and appropriate to the circumstances. The project team is
61+
obligated to maintain confidentiality with regard to the reporter of an incident.
62+
Further details of specific enforcement policies may be posted separately.
63+
64+
Project maintainers who do not follow or enforce the Code of Conduct in good
65+
faith may face temporary or permanent repercussions as determined by other
66+
members of the project's leadership.
67+
68+
## Attribution
69+
70+
This Code of Conduct is adapted from the [Contributor Covenant][homepage], version 1.4,
71+
available at [https://contributor-covenant.org/version/1/4][version]
72+
73+
[homepage]: https://contributor-covenant.org
74+
[version]: https://contributor-covenant.org/version/1/4/

Gemfile

+5
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,5 @@
1+
# frozen_string_literal: true
2+
3+
source "https://rubygems.org"
4+
5+
gemspec

Gemfile.lock

+38
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,38 @@
1+
PATH
2+
remote: .
3+
specs:
4+
syntax_tree-haml (0.1.0)
5+
haml (>= 5.2)
6+
syntax_tree
7+
8+
GEM
9+
remote: https://rubygems.org/
10+
specs:
11+
docile (1.4.0)
12+
haml (5.2.2)
13+
temple (>= 0.8.0)
14+
tilt
15+
minitest (5.15.0)
16+
rake (13.0.6)
17+
simplecov (0.21.2)
18+
docile (~> 1.1)
19+
simplecov-html (~> 0.11)
20+
simplecov_json_formatter (~> 0.1)
21+
simplecov-html (0.12.3)
22+
simplecov_json_formatter (0.1.4)
23+
syntax_tree (1.2.0)
24+
temple (0.8.2)
25+
tilt (2.0.10)
26+
27+
PLATFORMS
28+
x86_64-darwin-21
29+
30+
DEPENDENCIES
31+
bundler
32+
minitest
33+
rake
34+
simplecov
35+
syntax_tree-haml!
36+
37+
BUNDLED WITH
38+
2.4.0.dev

README.md

+68
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,68 @@
1+
# SyntaxTree::Haml
2+
3+
[![Build Status](https://github.com/ruby-syntax-tree/syntax_tree-haml/workflows/Main/badge.svg)](https://github.com/ruby-syntax-tree/syntax_tree-haml/actions)
4+
[![Gem Version](https://img.shields.io/gem/v/syntax_tree-haml.svg)](https://rubygems.org/gems/syntax_tree-haml)
5+
6+
[Syntax Tree](https://github.com/ruby-syntax-tree/syntax_tree) support for the [Haml template language](https://haml.info/).
7+
8+
## Installation
9+
10+
Add this line to your application's Gemfile:
11+
12+
```ruby
13+
gem "syntax_tree-haml"
14+
```
15+
16+
And then execute:
17+
18+
$ bundle install
19+
20+
Or install it yourself as:
21+
22+
$ gem install syntax_tree-haml
23+
24+
## Usage
25+
26+
From code:
27+
28+
```ruby
29+
require "syntax_tree/haml"
30+
31+
pp SyntaxTree::Haml.parse(source) # print out the AST
32+
puts SyntaxTree::Haml.format(source) # format the AST
33+
```
34+
35+
From the CLI:
36+
37+
```sh
38+
$ stree ast template.haml
39+
(root children=[(tag attributes={} value="Hello, world!")])
40+
```
41+
42+
or
43+
44+
```sh
45+
$ stree format template.haml
46+
%span Hello, world!
47+
```
48+
49+
or
50+
51+
```sh
52+
$ stree write template.haml
53+
template.haml 1ms
54+
```
55+
56+
## Development
57+
58+
After checking out the repo, run `bin/setup` to install dependencies. Then, run `rake test` to run the tests. You can also run `bin/console` for an interactive prompt that will allow you to experiment.
59+
60+
To install this gem onto your local machine, run `bundle exec rake install`. To release a new version, update the version number in `version.rb`, and then run `bundle exec rake release`, which will create a git tag for the version, push git commits and tags, and push the `.gem` file to [rubygems.org](https://rubygems.org).
61+
62+
## Contributing
63+
64+
Bug reports and pull requests are welcome on GitHub at https://github.com/ruby-syntax-tree/syntax_tree-haml.
65+
66+
## License
67+
68+
The gem is available as open source under the terms of the [MIT License](https://opensource.org/licenses/MIT).

Rakefile

+12
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,12 @@
1+
# frozen_string_literal: true
2+
3+
require "bundler/gem_tasks"
4+
require "rake/testtask"
5+
6+
Rake::TestTask.new(:test) do |t|
7+
t.libs << "test"
8+
t.libs << "lib"
9+
t.test_files = FileList["test/**/*_test.rb"]
10+
end
11+
12+
task default: :test

bin/format

+9
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,9 @@
1+
#!/usr/bin/env ruby
2+
# frozen_string_literal: true
3+
4+
require "bundler/setup"
5+
6+
$:.unshift(File.expand_path("../lib", __dir__))
7+
require "syntax_tree/haml"
8+
9+
puts SyntaxTree::Haml.format(ARGF.read)

bin/parse

+9
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,9 @@
1+
#!/usr/bin/env ruby
2+
# frozen_string_literal: true
3+
4+
require "bundler/setup"
5+
6+
$:.unshift(File.expand_path("../lib", __dir__))
7+
require "syntax_tree/haml"
8+
9+
pp SyntaxTree::Haml.parse(ARGF.read)

lib/syntax_tree/haml.rb

+74
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,74 @@
1+
# frozen_string_literal: true
2+
3+
require "haml"
4+
require "syntax_tree"
5+
6+
require "syntax_tree/haml/comment"
7+
require "syntax_tree/haml/doctype"
8+
require "syntax_tree/haml/filter"
9+
require "syntax_tree/haml/haml_comment"
10+
require "syntax_tree/haml/plain"
11+
require "syntax_tree/haml/root"
12+
require "syntax_tree/haml/script"
13+
require "syntax_tree/haml/silent_script"
14+
require "syntax_tree/haml/tag"
15+
16+
class Haml::Parser::ParseNode
17+
def format(q)
18+
syntax_tree.format(q)
19+
end
20+
21+
def pretty_print(q)
22+
syntax_tree.pretty_print(q)
23+
end
24+
25+
private
26+
27+
def syntax_tree
28+
case type
29+
when :comment then SyntaxTree::Haml::Comment.new(self)
30+
when :doctype then SyntaxTree::Haml::Doctype.new(self)
31+
when :filter then SyntaxTree::Haml::Filter.new(self)
32+
when :haml_comment then SyntaxTree::Haml::HamlComment.new(self)
33+
when :plain then SyntaxTree::Haml::Plain.new(self)
34+
when :root then SyntaxTree::Haml::Root.new(self)
35+
when :script then SyntaxTree::Haml::Script.new(self)
36+
when :silent_script then SyntaxTree::Haml::SilentScript.new(self)
37+
when :tag then SyntaxTree::Haml::Tag.new(self)
38+
else
39+
raise ArgumentError, "Unsupported type: #{type}"
40+
end
41+
end
42+
end
43+
44+
class SyntaxTree < Ripper
45+
module Haml
46+
def self.parse(source)
47+
::Haml::Parser.new({}).call(source)
48+
end
49+
50+
def self.format(source)
51+
formatter = PP.new([])
52+
parse(source).format(formatter)
53+
54+
formatter.flush
55+
formatter.output.join
56+
end
57+
58+
def self.with_children(node, q)
59+
if node.children.empty?
60+
q.group { yield }
61+
else
62+
q.group do
63+
q.group { yield }
64+
q.indent do
65+
node.children.each do |child|
66+
q.breakable(force: true)
67+
child.format(q)
68+
end
69+
end
70+
end
71+
end
72+
end
73+
end
74+
end

lib/syntax_tree/haml/comment.rb

+52
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,52 @@
1+
# frozen_string_literal: true
2+
3+
class SyntaxTree < Ripper
4+
module Haml
5+
# https://haml.info/docs/yardoc/file.REFERENCE.html#html-comments-
6+
class Comment
7+
attr_reader :node
8+
9+
def initialize(node)
10+
@node = node
11+
end
12+
13+
def format(q)
14+
Haml.with_children(node, q) do
15+
q.text("/")
16+
q.text("!") if node.value[:revealed]
17+
18+
if node.value[:conditional]
19+
q.text(node.value[:conditional])
20+
elsif node.value[:text]
21+
q.text(" #{node.value[:text]}")
22+
end
23+
end
24+
end
25+
26+
def pretty_print(q)
27+
q.group(2, "(comment", ")") do
28+
q.breakable
29+
30+
if node.value[:conditional]
31+
q.text("conditional=")
32+
q.pp(node.value[:conditional])
33+
elsif node.value[:text]
34+
q.text("text=")
35+
q.pp(node.value[:text])
36+
end
37+
38+
if node.value[:revealed]
39+
q.breakable
40+
q.text("revealed")
41+
end
42+
43+
if node.children.any?
44+
q.breakable
45+
q.text("children=")
46+
q.pp(node.children)
47+
end
48+
end
49+
end
50+
end
51+
end
52+
end

0 commit comments

Comments
 (0)