Class: Cyclomagic::ParseState

Inherits:
Object
  • Object
show all
Includes:
RubyToken
Defined in:
lib/cyclomagic.rb

Overview

Main class and structure used to compute the cyclomatic complexity of Ruby programs.

Direct Known Subclasses

EndableParseState, ParseComment, ParseSymbol

Constant Summary collapse

@@top_state =
nil
@@token_counter =
TokenCounter.new

Instance Attribute Summary collapse

Class Method Summary collapse

Instance Method Summary collapse

Constructor Details

#initialize(lexer, parent = nil) ⇒ ParseState

Returns a new instance of ParseState.



86
87
88
89
90
91
92
93
94
95
96
# File 'lib/cyclomagic.rb', line 86

def initialize(lexer,parent=nil)
  @name = ""
  @children = Array.new
  @complexity = 0
  @parent = parent
  @lexer = lexer
  @run = true
  # To catch one line def statements, We always have one line.
  @lines = 0
  @last_token_line_and_char = Array.new
end

Instance Attribute Details

#childrenObject

Returns the value of attribute children.



69
70
71
# File 'lib/cyclomagic.rb', line 69

def children
  @children
end

#complexityObject

Returns the value of attribute complexity.



69
70
71
# File 'lib/cyclomagic.rb', line 69

def complexity
  @complexity
end

#linesObject

Returns the value of attribute lines.



69
70
71
# File 'lib/cyclomagic.rb', line 69

def lines
  @lines
end

#nameObject

Returns the value of attribute name.



69
70
71
# File 'lib/cyclomagic.rb', line 69

def name
  @name
end

#parentObject

Returns the value of attribute parent.



69
70
71
# File 'lib/cyclomagic.rb', line 69

def parent
  @parent
end

Class Method Details

.get_token_counterObject



82
83
84
# File 'lib/cyclomagic.rb', line 82

def ParseState.get_token_counter
  @@token_counter
end

.make_top_stateObject



72
73
74
75
76
# File 'lib/cyclomagic.rb', line 72

def ParseState.make_top_state()
  @@top_state = ParseState.new(nil)
  @@top_state.name = "__top__"
  @@top_state
end

.set_token_counter(counter) ⇒ Object



79
80
81
# File 'lib/cyclomagic.rb', line 79

def ParseState.set_token_counter(counter)
  @@token_counter = counter
end

Instance Method Details

#calc_complexityObject



113
114
115
116
117
118
119
# File 'lib/cyclomagic.rb', line 113

def calc_complexity
  complexity = @complexity
  children.each do |child|
    complexity += child.calc_complexity
  end
  complexity
end

#calc_linesObject



121
122
123
124
125
126
127
# File 'lib/cyclomagic.rb', line 121

def calc_lines
  lines = @lines
  children.each do |child|
    lines += child.calc_lines
  end
  lines
end

#compute_state(formater) ⇒ Object



129
130
131
132
133
134
135
136
137
# File 'lib/cyclomagic.rb', line 129

def compute_state(formater)
  if top_state?
    compute_state_for_global(formater)
  end

  @children.each do |s|
    s.compute_state(formater)
  end
end

#compute_state_for_global(formater) ⇒ Object



139
140
141
142
143
144
145
146
147
148
149
150
151
# File 'lib/cyclomagic.rb', line 139

def compute_state_for_global(formater)
  global_def, @children = @children.partition do |s|
    !s.kind_of?(ParseClass)
  end
  return if global_def.empty?
  gx = global_def.inject(0) { |c,s| s.calc_complexity }
  gl = global_def.inject(0) { |c,s| s.calc_lines }
  formater.start_class_compute_state("Global", "", gx, gl)
  global_def.each do |s|
    s.compute_state(formater)
  end
  formater.end_class_compute_state("")
end

#count_tokens?Boolean

Count the tokens parsed if true else ignore them.

Returns:

  • (Boolean)


154
155
156
# File 'lib/cyclomagic.rb', line 154

def count_tokens?
  true
end

#do_begin_token(token) ⇒ Object



199
200
201
# File 'lib/cyclomagic.rb', line 199

def do_begin_token(token)
  make_state(EndableParseState, self)
end

#do_block_token(token) ⇒ Object



236
237
238
# File 'lib/cyclomagic.rb', line 236

def do_block_token(token)
  make_state(ParseBlock,self)
end

#do_case_token(token) ⇒ Object



248
249
250
# File 'lib/cyclomagic.rb', line 248

def do_case_token(token)
  make_state(EndableParseState, self)
end

#do_class_token(token) ⇒ Object



203
204
205
# File 'lib/cyclomagic.rb', line 203

def do_class_token(token)
  make_state(ParseClass,self)
end

#do_comment_token(token) ⇒ Object



268
269
270
# File 'lib/cyclomagic.rb', line 268

def do_comment_token(token)
  make_state(ParseComment, self)
end

#do_conditional_do_control_token(token) ⇒ Object



244
245
246
# File 'lib/cyclomagic.rb', line 244

def do_conditional_do_control_token(token)
  make_state(ParseDoCond,self)
end

#do_conditional_token(token) ⇒ Object



240
241
242
# File 'lib/cyclomagic.rb', line 240

def do_conditional_token(token)
  make_state(ParseCond,self)
end

#do_constant_token(token) ⇒ Object



215
216
217
# File 'lib/cyclomagic.rb', line 215

def do_constant_token(token)
  nil
end

#do_def_token(token) ⇒ Object



211
212
213
# File 'lib/cyclomagic.rb', line 211

def do_def_token(token)
  make_state(ParseDef,self)
end

#do_else_token(token) ⇒ Object



263
264
265
266
# File 'lib/cyclomagic.rb', line 263

def do_else_token(token)
  STDOUT.puts "Ignored/Unknown Token:#{token.class}" if $VERBOSE
  nil
end

#do_end_token(token) ⇒ Object



231
232
233
234
# File 'lib/cyclomagic.rb', line 231

def do_end_token(token)
  end_debug
  nil
end

#do_identifier_token(token) ⇒ Object



219
220
221
222
223
224
225
# File 'lib/cyclomagic.rb', line 219

def do_identifier_token(token)
  if (token.name == "__END__" && token.char_no.to_i == 0)
    # The Ruby code has stopped and the rest is data so cease parsing.
    @run = false
  end
  nil
end

#do_module_token(token) ⇒ Object



207
208
209
# File 'lib/cyclomagic.rb', line 207

def do_module_token(token)
  make_state(ParseModule,self)
end

#do_one_line_conditional_token(token) ⇒ Object



252
253
254
255
256
257
258
259
260
261
# File 'lib/cyclomagic.rb', line 252

def do_one_line_conditional_token(token)
  # This is an if with no end
  @complexity += 1
  #STDOUT.puts "got IF_MOD: #{self.to_yaml}" if $VERBOSE
  #if state.type != "class" && state.type != "def" && state.type != "cond"
  #STDOUT.puts "Changing IF_MOD Parent" if $VERBOSE
  #state = state.parent
  #@run = false
  nil
end

#do_right_brace_token(token) ⇒ Object



227
228
229
# File 'lib/cyclomagic.rb', line 227

def do_right_brace_token(token)
  nil
end

#do_symbol_token(token) ⇒ Object



272
273
274
# File 'lib/cyclomagic.rb', line 272

def do_symbol_token(token)
  make_state(ParseSymbol, self)
end

#end_debugObject



336
337
338
339
340
341
342
343
344
345
346
347
348
# File 'lib/cyclomagic.rb', line 336

def end_debug
  STDOUT.puts "got an end: #{@name} in #{self.class.name}" if $VERBOSE
  if @parent.nil?
    STDOUT.puts "DEBUG: Line #{@lexer.line_no}"
    STDOUT.puts "DEBUG: #{@name}; #{self.class}"
    # to_yaml can cause an infinite loop?
    #STDOUT.puts "TOP: #{@@top_state.to_yaml}"
    #STDOUT.puts "TOP: #{@@top_state.inspect}"

    # This may not be an error?
    #exit 1
  end
end

#lexer=(lexer) ⇒ Object



102
103
104
105
# File 'lib/cyclomagic.rb', line 102

def lexer=(lexer)
  @run = true
  @lexer = lexer
end

#lexer_loop?(token) ⇒ Boolean

Ruby-Lexer can go into a loop if the file does not end with a newline.

Returns:

  • (Boolean)


176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
# File 'lib/cyclomagic.rb', line 176

def lexer_loop?(token)
  return false if @last_token_line_and_char.empty?
  loop_flag = false
  last = @last_token_line_and_char.last
  line = last[0]
  char = last[1]
  ltok = last[2]

  if ( (line == @lexer.line_no.to_i) &&
         (char == @lexer.char_no.to_i) &&
         (ltok.class == token.class) )
    # We are potentially in a loop
    if @last_token_line_and_char.size >= 3
      loop_flag = true
    end
  else
    # Not in a loop so clear stack
    @last_token_line_and_char = Array.new
  end

  loop_flag
end

#make_state(type, parent = nil) ⇒ Object



107
108
109
110
111
# File 'lib/cyclomagic.rb', line 107

def make_state(type,parent = nil)
  cstate = type.new(@lexer,self)
  parent.children<< cstate
  cstate
end

#parseObject



158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
# File 'lib/cyclomagic.rb', line 158

def parse
  while @run do
    tok = @lexer.token
    @run = false if tok.nil?
    if lexer_loop?(tok)
      STDERR.puts "Lexer loop at line : #{@lexer.line_no} char #{@lexer.char_no}."
      @run = false
    end
    @last_token_line_and_char<< [@lexer.line_no.to_i, @lexer.char_no.to_i, tok]
    if $VERBOSE
  puts "DEBUG: #{@lexer.line_no} #{tok.class}:#{tok.name if tok.respond_to?(:name)}"
    end
    @@token_counter.count_token(@lexer.line_no, tok) if count_tokens?
    parse_token(tok)
  end
end

#parse_token(token) ⇒ Object



276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
# File 'lib/cyclomagic.rb', line 276

def parse_token(token)
  state = nil
  case token
  when TkCLASS
    state = do_class_token(token)
  when TkMODULE
    state = do_module_token(token)
  when TkDEF
    state = do_def_token(token)
  when TkCONSTANT
    # Nothing to do with a constant at top level?
    state = do_constant_token(token)
  when TkIDENTIFIER,TkFID
    # Nothing to do at top level?
    state = do_identifier_token(token)
  when TkRBRACE
    # Nothing to do at top level
    state = do_right_brace_token(token)
  when TkEND
    state = do_end_token(token)
    # At top level this might be an error...
  when TkDO,TkfLBRACE
    state = do_block_token(token)
  when TkIF,TkUNLESS
    state = do_conditional_token(token)
  when TkWHILE,TkUNTIL,TkFOR
    state = do_conditional_do_control_token(token)
  when TkELSIF #,TkELSE
    @complexity += 1
  when TkELSE
    # Else does not increase complexity
  when TkCASE
    state = do_case_token(token)
  when TkWHEN
    @complexity += 1
  when TkBEGIN
    state = do_begin_token(token)
  when TkRESCUE
    # Maybe this should add complexity and not begin
    @complexity += 1
  when TkIF_MOD, TkUNLESS_MOD, TkUNTIL_MOD, TkWHILE_MOD, TkQUESTION
    state = do_one_line_conditional_token(token)
  when TkNL
    #
    @lines += 1
  when TkRETURN
    # Early returns do not increase complexity as the condition that
    # calls the return is the one that increases it.
  when TkCOMMENT
    state = do_comment_token(token)
  when TkSYMBEG
    state = do_symbol_token(token)
  when TkError
    STDOUT.puts "Lexer received an error for line #{@lexer.line_no} char #{@lexer.char_no}"
  else
    state = do_else_token(token)
  end
  state.parse if state
end

#top_state?Boolean

Returns:

  • (Boolean)


98
99
100
# File 'lib/cyclomagic.rb', line 98

def top_state?
  self == @@top_state
end