class GobstonesFeedbackHook < Mumukit::Hook def run!(request, results) content = request.content test_results = results.test_results[0] GobstonesExplainer.new.explain(content, test_results) if test_results.is_a? String end class GobstonesExplainer < Mumukit::Explainer def explain_program_has_a_name(submission, result) if identifier_instead_of_brace? result (submission.match malformed_program_header_with_name).try do |it| { name: it[1] } end end end def explain_program_has_no_opening_curly_brace(submission, result) if identifier_instead_of_brace? result /#{malformed_program_header_with_no_curly_braces}/ =~ submission end end def explain_program_before_closing_structure_when_program(submission, result) if program_instead_of_command?(result) && missing_brace_end?(submission) && function_or_procedure?(submission) (submission.match program).try do |it| { keyword: last_function_or_procedure(submission) } end end end def explain_program_before_closing_structure_when_no_program(submission, result) if program_instead_of_command?(result) && missing_brace_end?(submission) && function_or_procedure?(submission) if /#{program}/ =~ submission nil else { keyword: last_function_or_procedure(submission) } end end end def explain_surplus_closing_brace(_, result) if unbalanced_closing_braces? result (error_line(result)).try do |it| { line: it[1] } end end end def explain_upper_function_typo(submission, result) if upper_identifier_instead_of_definition? result /#{uppercase_function}/ =~ submission end end def explain_upper_procedure_typo(submission, result) if upper_identifier_instead_of_definition? result /#{uppercase_procedure}/ =~ submission end end def explain_upper_program_typo(submission, result) if upper_identifier_instead_of_definition? result /#{uppercase_program}/ =~ submission end end def explain_missing_closing_brace_before_procedure(submission, result) if procedure_instead_of_command?(result) && missing_brace_end?(submission) (error_line(result)).try do |it| { line: it[1] } end end end def explain_lower_builtin_procedure_typo(submission, result) if open_paren_instead_of_assign?(result) (submission.match lower_builtin_procedure).try do |it| { lower: it[1][0...5], upper: it[1][0...5].capitalize } end end end def explain_upper_builtin_function_typo(submission, result) if procedure_invocation_instead_of_expression?(result) (submission.match upper_builtin_function).try do |it| { upper: it[1], lower: it[1].camelize(:lower) } end end end def explain_color_typo(_, result) if roja_not_defined?(result) || negra_not_defined?(result) (result.match color_not_defined).try do |it| { color: it[1], rectified_color: rectified_color(it[1]) } end end end private def malformed_program_header_with_name '.*program +([A-Za-z]\w*)' end def upper_identifier_instead_of_brace?(result) identifier_instead_of_brace?(result, 'may') end def lower_identifier_instead_of_brace?(result) identifier_instead_of_brace?(result, 'min') end def identifier_instead_of_brace?(result, capital='...') result.match? /
\[\d+:\d+\]: Se esperaba una llave izquierda \("{"\).\nSe encontró: un identificador con #{capital}úsculas.<\/pre>/ end def malformed_program_header_with_no_curly_braces '.*program *[\r\n]\s*[^{]\w+' end def program_instead_of_command?(result) result.match? /\[\d+:\d+\]: Se esperaba un comando.\nSe encontró: la palabra clave "program".<\/pre>/ end def missing_brace_end?(submission) submission.count('{') > submission.count('}') end def function_or_procedure?(submission) submission.match? function_or_procedure end def function_or_procedure '(function)\s*\w+\s*\([\w\d\s,]*\)\s*{|(procedure)\s*\w+\s*\([\w\d\s,]*\)\s*{' end def last_function_or_procedure(submission) submission.scan(/#{function_or_procedure}/).last.compact.first end def program 'program\s*{' end def unbalanced_closing_braces?(result) result.match? /\[\d+:\d+\]: Se encontró un "}" pero no había una llave abierta "{".<\/pre>/ end def error_line(result) result.match /\[(\d+):\d+\]:/ end def upper_identifier_instead_of_definition?(result) result.match? /\[\d+:\d+\]: Se esperaba una definición \(de programa, función, procedimiento, o tipo\).\nSe encontró: un identificador con mayúsculas.<\/pre>/ end def uppercase_function 'Function\s' end def uppercase_procedure 'Procedure\s' end def uppercase_program 'Program[\s{]*' end def procedure_instead_of_command?(result) result.match? /\[\d+:\d+\]: Se esperaba un comando.\nSe encontró: la palabra clave "procedure".<\/pre>/ end def open_paren_instead_of_assign?(result) result.match? /\[\d+:\d+\]: Se esperaba un operador de asignación \(":="\).\nSe encontró: un paréntesis izquierdo \("\("\).<\/pre>/ end def lower_builtin_procedure '(mover[\s(]|poner[\s(]|sacar[\s(])' end def procedure_invocation_instead_of_expression?(result) result.match? /\[\d+:\d+\]: Se esperaba una expresión.\nSe encontró: una invocación a un procedimiento.<\/pre>/ end def upper_builtin_function '(PuedeMover|NroBolitas|HayBolitas)' end def roja_not_defined?(result) color_not_defined? result, 'Roja' end def negra_not_defined?(result) color_not_defined? result, 'Negra' end def color_not_defined?(result, color) result.match?(color_not_defined(color)) end def color_not_defined(color='\w+') /\[\d+:\d+\]: El constructor "(#{color})" no está definido.<\/pre>/ end def rectified_color(color) color.chop + "o" end end end