-
Notifications
You must be signed in to change notification settings - Fork 743
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
Add support for crystal language #441
Conversation
@jneen Do you need more info/support from me to get this PR merged? :-) |
Not the author, but this seems nearly identical to the Ruby lexer so you might be better off subclassing it to prevent unnecessary duplication. The C/C++/ObjC lexers are good examples of how to do this. |
@ryphill You're right. Both lexers are almost identical because Crystal syntax is very similar to Ruby's. I've copied the lexer on purpose because Crystal is still under heavy development so syntax changes are likely. Once Crystal's syntax has settled one can think of re-using the Ruby lexer :-) |
Hi @splattael, thanks for the pull request. In this case, given the sheer complexity of the ruby lexer, I'd prefer re-using the ruby lexer and breaking off a new lexer for crystal if enough breaking syntax changes are made. |
Hey @jneen! Yes, re-using the Ruby lexer is definitely an option! In order to make the Ruby lexer re-usable, we have to tweak it, I guess. What I am confident about is to clean up the current Crystal lexer a bit (removing some items from What do you think? Making the Ruby lexer re-usable would definitely take longer and need some assistance from you (and/or from other contributors), I guess. Thanks 💚, |
Hm. I wonder, how much of ruby syntax does Crystal support? e: also, could you give me a summary of the differences between this lexer and the ruby lexer? |
AFAIK, Crystal started as a subset of Ruby's syntax. Since then, the syntax has diverged (map-syntax, annotations, variable types) so now it's NOT a subset any more The differences (for now) in the Crystal lexer are mostly in Here's the (shortened) diff: keywords = %w(
BEGIN END alias begin break case defined\? do else elsif end
- ensure for if in next redo rescue raise retry return super then
- undef unless until when while yield
+ ensure for if ifdef in next redo rescue raise retry return super then
+ undef unless until when while yield lib fun type of as
)
keywords_pseudo = %w(
initialize new loop include extend raise attr_reader attr_writer
attr_accessor alias_method attr catch throw private module_function
public protected true false nil __FILE__ __LINE__
+ getter getter? getter! property property? property! struct record
)
builtins_g = %w(
@@ -166,6 +165,8 @@
rule /0b[01]+(?:_[01]+)*/, Num::Bin
rule /[\d]+(?:_\d+)*/, Num::Integer
+ rule /@\[([^\]]+)\]/, Name::Decorator
+ Again, the Crystal lexer still needs some ❤️ (mostly tweaking |
Okay, so the process to use subclassing instead of copy-pasta would be this:
This should be doable without too much effort, yes? |
@jneen If you wish to subclass instead of copy-paste I'll try it. Stay tuned :-) |
Hey @jneen! Sorry for the delay. I'm trying to implement the Crystal lexer as a subclass of the Ruby lexer. As suggested. I am having problems defining e.g. I'm talking about this (https://github.com/jneen/rouge/blob/c8a9194b4637eec6595182e6ded9766c7e9b0e0e/lib/rouge/lexers/ruby.rb#L179-L180): rule /(?:#{keywords.join('|')})\b/, Keyword, :expr_start
rule /(?:#{keywords_pseudo.join('|')})\b/, Keyword::Pseudo, :expr_start Defining rule /(?:#{self.class.keywords.join('|')})\b/, Keyword, :expr_start
rule /(?:#{self.class.keywords_pseudo.join('|')})\b/, Keyword::Pseudo, :expr_start (Note the Defining a generic rule like the C lexer does (https://github.com/jneen/rouge/blob/c8a9194b4637eec6595182e6ded9766c7e9b0e0e/lib/rouge/lexers/c.rb#L116-L130) won't work either because AFAIK a I fear I need help :-( I can push my not-working code if you need more insights. Kind regards, |
Hi @splattael! Thanks for getting this far :]. Of those options, I'd prefer the generic rule like the C lexer (it's what ruby should have used anyways...) A rule block can call the rule /someregex/ do
# ...
push :state_name
end It can also call |
Thanks! I will try it. Expect broken code soonish %-) |
Mention here: https://gitlab.com/gitlab-org/gitlab-ce/issues/35409 |
300986f
to
c68a152
Compare
@faustinoaq Thanks for waking me up ;) So, yes, I did not have time to implement @jneen suggestions from #441 (comment) 😞 I'd also love ❤️ to see Crystal being highlighted beautifully on GitLab 😄 as I also use Crystal on GitLab a lot 😊 @jneen Is there a way to merge 🔀 this PR without subclassing the Ruby lexer FOR NOW? 😃 🙏 The changes between Ruby lexer and the Crystal lexer are mostly keywords and highlighting annotations like --- lib/rouge/lexers/ruby.rb 2017-07-25 08:54:39.629975025 +0200
+++ lib/rouge/lexers/crystal.rb 2017-07-25 09:06:11.014454805 +0200
@@ -2,19 +2,17 @@
module Rouge
module Lexers
- class Ruby < RegexLexer
- title "Ruby"
- desc "The Ruby programming language (ruby-lang.org)"
- tag 'ruby'
- aliases 'rb'
- filenames '*.rb', '*.ruby', '*.rbw', '*.rake', '*.gemspec', '*.podspec',
- 'Rakefile', 'Guardfile', 'Gemfile', 'Capfile', 'Podfile',
- 'Vagrantfile', '*.ru', '*.prawn', 'Berksfile'
+ class Crystal < RegexLexer
+ title "Crystal"
+ desc "Crystal The Programming Language (crystal-lang.org)"
+ tag 'crystal'
+ aliases 'cr'
+ filenames '*.cr'
- mimetypes 'text/x-ruby', 'application/x-ruby'
+ mimetypes 'text/x-crystal', 'application/x-crystal'
def self.analyze_text(text)
- return 1 if text.shebang? 'ruby'
+ return 1 if text.shebang? 'crystal'
end
state :symbols do
@@ -103,41 +101,33 @@
keywords = %w(
BEGIN END alias begin break case defined\? do else elsif end
- ensure for if in next redo rescue raise retry return super then
- undef unless until when while yield
+ ensure for if ifdef in next redo rescue raise retry return super then
+ undef unless until when while yield lib fun type of as
)
keywords_pseudo = %w(
- loop include extend raise
- alias_method attr catch throw private module_function
+ initialize new loop include extend raise attr_reader attr_writer
+ attr_accessor alias_method attr catch throw private module_function
public protected true false nil __FILE__ __LINE__
+ getter getter? getter! property property? property! struct record
)
builtins_g = %w(
- attr_reader attr_writer attr_accessor
-
- __id__ __send__ abort ancestors at_exit autoload binding callcc
- caller catch chomp chop class_eval class_variables clone
- const_defined\? const_get const_missing const_set constants
+ abort ancestors at_exit
+ caller catch chomp chop clone constants
display dup eval exec exit extend fail fork format freeze
getc gets global_variables gsub hash id included_modules
inspect instance_eval instance_method instance_methods
- instance_variable_get instance_variable_set instance_variables
- lambda load local_variables loop method method_missing
+ lambda loop method method_missing
methods module_eval name object_id open p print printf
- private_class_method private_instance_methods private_methods proc
- protected_instance_methods protected_methods public_class_method
- public_instance_methods public_methods putc puts raise rand
- readline readlines require require_relative scan select self send set_trace_func
- singleton_methods sleep split sprintf srand sub syscall system
- taint test throw to_a to_s trace_var trap untaint untrace_var warn
+ proc putc puts raise rand
+ readline readlines require scan select self send
+ sleep split sprintf srand sub syscall system
+ test throw to_a to_s trap warn
)
builtins_q = %w(
- autoload block_given const_defined eql equal frozen
- include instance_of is_a iterator kind_of method_defined
- nil private_method_defined protected_method_defined
- public_method_defined respond_to tainted
+ eql equal include is_a iterator kind_of nil
)
builtins_b = %w(chomp chop exit gsub sub)
@@ -169,6 +159,8 @@
rule /\d+\.\d+(e[\+\-]?\d+)?/, Num::Float
rule /[\d]+(?:_\d+)*/, Num::Integer
+ rule /@\[([^\]]+)\]/, Name::Decorator
+
# names
rule /@@[a-z_]\w*/i, Name::Variable::Class
rule /@[a-z_]\w*/i, Name::Variable::Instance @jneen WDYT? :) |
Hi @jneen do you think we can merge this PR? 😅 |
Any news here? |
👋 Hi there! Just jumping in to offer some help to get this merged. Crystal's own website uses this gem - so Crystal's site can't highlight Crystal yet 😨 |
@jneen Thanks for this wonderful gem. I personally think that subclassing the ruby lexer is the wrong direction since unlike C++ being a superset of C, crystal isn't a superset of ruby (anymore). This sort of DRY code should be avoided since it introduces coupling and direct dependency for technical reasons only. But subclassing the Lexer feels kinda wrong for me unless the Language is an actual superset since future changes in the ruby syntax will most likely not be present in crystal in the same form. But those are just my 2 cents. |
Crystal isn't ruby and it doesn't aim for compatability. Asking for the crystal lexer to subclass the ruby lexer is like asking for the java lexer to subclass the C lexer because they share syntax heritage. In fact, there's little point since crystal will not be adopting any future ruby syntax features. Please merge this PR (or #863), it's far better than the nothing we have now on Gitlab, the crystal blog, etc. |
Heavily based on the Ruby lexer.
In the meantime... As @ysbaddaden suggested I've (finally) extracted the Crystal lexer into an own gem: Put in your Gemfile: gem "rouge"
gem "rouge-lexers-crystal" The lexer for Crystal should be available then. Additionally, I've created a merge request on GitLab to use this plugin in order to make Crystal code look great 🤞 Wish us luck 😁 |
I will review this week. Given the discussion I think not subclassing Ruby is the right solution. I also would rather merge this than see GitLab add a dependency on another gem. |
@dblessing Awesome, thank you 💜 |
Hi @splattael I was checking to use this in crystal's website, I've found a couple of keywords missing like |
@bcardiff Crystal's lexer is heavily based on Ruby's lexer. I suspect that I'd add I hope this helps! |
Heavily based on the Ruby lexer.