lib/faml/indent_tracker.rb in faml-0.2.13 vs lib/faml/indent_tracker.rb in faml-0.2.14
- old
+ new
@@ -1,20 +1,46 @@
require 'faml/error'
module Faml
class IndentTracker
class IndentMismatch < Error
+ attr_reader :current_level, :indent_levels
+
+ def initialize(current_level, indent_levels, lineno)
+ super("Unexpected indent level: #{current_level}: indent_level=#{indent_levels}", lineno)
+ @current_level = current_level
+ @indent_levels = indent_levels
+ end
end
+ class InconsistentIndent < Error
+ attr_reader :previous_size, :current_size
+
+ def initialize(previous_size, current_size, lineno)
+ super("Inconsistent indentation: #{current_size} spaces used for indentation, but the rest of the document was indented using #{previous_size} spaces.", lineno)
+ @previous_size = previous_size
+ @current_size = current_size
+ end
+ end
+
+ class HardTabNotAllowed < Error
+ def initialize(lineno)
+ super('Indentation with hard tabs are not allowed :-p', lineno)
+ end
+ end
+
def initialize(on_enter: nil, on_leave: nil)
@indent_levels = [0]
@on_enter = on_enter || lambda { |level, text| }
@on_leave = on_leave || lambda { |level, text| }
@comment_level = nil
end
def process(line, lineno)
+ if line =~ /\A\t/
+ raise HardTabNotAllowed.new(lineno)
+ end
indent, text = split(line)
indent_level = indent.size
unless text.empty?
track(indent_level, text, lineno)
@@ -37,21 +63,31 @@
def enter_comment!
@comment_level = @indent_levels[-2]
end
+ def check_indent_level!(lineno)
+ if @indent_levels.size >= 3
+ previous_size = @indent_levels[-2] - @indent_levels[-3]
+ current_size = @indent_levels[-1] - @indent_levels[-2]
+ if previous_size != current_size
+ raise InconsistentIndent.new(previous_size, current_size, lineno)
+ end
+ end
+ end
+
private
def track(indent_level, text, lineno)
if indent_level > @indent_levels.last
- indent_enter(indent_level, text)
+ indent_enter(indent_level, text, lineno)
elsif indent_level < @indent_levels.last
indent_leave(indent_level, text, lineno)
end
end
- def indent_enter(indent_level, text)
+ def indent_enter(indent_level, text, lineno)
unless @comment_level
@indent_levels.push(indent_level)
@on_enter.call(indent_level, text)
end
end
@@ -71,10 +107,10 @@
@indent_levels.pop
@on_leave.call(indent_level, text)
end
if indent_level != @indent_levels.last
- raise IndentMismatch.new("Unexpected indent level: #{indent_level}: indent_level=#{@indent_levels}", lineno)
+ raise IndentMismatch.new(indent_level, @indent_levels.dup, lineno)
end
end
end
end