Class: Bddgenx::StepsGenerator
- Inherits:
-
Object
- Object
- Bddgenx::StepsGenerator
- Defined in:
- lib/bddgenx/generators/steps_generator.rb
Constant Summary collapse
- GHERKIN_KEYS_PT =
Palavras-chave Gherkin em Português
%w[Dado Quando Então E Mas].freeze
- GHERKIN_KEYS_EN =
Palavras-chave Gherkin em Inglês
%w[Given When Then And But].freeze
- ALL_KEYS =
Conjunto de todas as palavras-chave reconhecidas
GHERKIN_KEYS_PT + GHERKIN_KEYS_EN
Class Method Summary collapse
-
.camelize(str) ⇒ String
Transforma uma string em camelCase (sem alterar acentuação).
-
.gerar_passos(feature_path) ⇒ Boolean
Gera um arquivo de definição de passos do Cucumber com base em um ‘.feature`.
Class Method Details
.camelize(str) ⇒ String
Transforma uma string em camelCase (sem alterar acentuação).
25 26 27 28 |
# File 'lib/bddgenx/generators/steps_generator.rb', line 25 def self.camelize(str) partes = str.strip.split(/[^a-zA-Z0-9]+/) partes.map.with_index { |palavra, i| i.zero? ? palavra.downcase : palavra.capitalize }.join end |
.gerar_passos(feature_path) ⇒ Boolean
Gera um arquivo de definição de passos do Cucumber com base em um ‘.feature`. Detecta automaticamente o idioma e converte os passos em métodos com placeholders.
37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 |
# File 'lib/bddgenx/generators/steps_generator.rb', line 37 def self.gerar_passos(feature_path) raise ArgumentError, I18n.t('errors.invalid_path', path: feature_path.class) unless feature_path.is_a?(String) linhas = File.readlines(feature_path) # Detecta o idioma a partir da linha `# language:` lang = if (m = linhas.find { |l| l =~ /^#\s*language:\s*(\w+)/i }) m[/^#\s*language:\s*(\w+)/i, 1].downcase else 'pt' end # Define o locale do I18n conforme idioma detectado I18n.locale = lang.to_sym rescue :pt pt_para_en = GHERKIN_KEYS_PT.zip(GHERKIN_KEYS_EN).to_h en_para_pt = GHERKIN_KEYS_EN.zip(GHERKIN_KEYS_PT).to_h # Seleciona apenas as linhas que representam passos linhas_passos = linhas.map(&:strip).select do |linha| ALL_KEYS.any? { |chave| linha.start_with?(chave + ' ') } end return false if linhas_passos.empty? # Cria diretório `steps` no mesmo nível do `.feature` dir_saida = File.join(File.dirname(feature_path), 'steps') FileUtils.mkdir_p(dir_saida) arquivo_saida = File.join(dir_saida, "#{File.basename(feature_path, '.feature')}_steps.rb") # Cabeçalho do arquivo gerado conteudo = +"# encoding: utf-8\n" conteudo << "# #{I18n.t('steps.header', file: File.basename(feature_path))}\n\n" passos_unicos = Set.new linhas_passos.each do |linha| palavra_original, restante = linha.split(' ', 2) # Tradução da palavra-chave inicial chave = case lang when 'en' then pt_para_en[palavra_original] || palavra_original else en_para_pt[palavra_original] || palavra_original end texto_bruto = restante.dup scanner = ::StringScanner.new(restante) padrao = '' tokens = [] # Analisa e parametriza variáveis until scanner.eos? if scanner.check(/"<([^>]+)>"/) scanner.scan(/"<([^>]+)>"/) tokens << scanner[1] padrao << '{string}' elsif scanner.check(/<([^>]+)>/) scanner.scan(/<([^>]+)>/) tokens << scanner[1] padrao << '{int}' elsif scanner.check(/"([^"]+)"/) scanner.scan(/"([^"]+)"/) tokens << scanner[1] padrao << '{string}' elsif scanner.check(/\d+(?:\.\d+)?/) numero = scanner.scan(/\d+(?:\.\d+)?/) tokens << numero padrao << '{int}' else padrao << scanner.getch end end padrao_seguro = padrao.gsub('"', '\\"') # Evita duplicatas de métodos next if passos_unicos.include?(padrao_seguro) passos_unicos << padrao_seguro # Monta assinatura do step assinatura = "#{chave}(\"#{padrao_seguro}\")" if tokens.any? argumentos = tokens.each_index.map { |i| "arg#{i+1}" }.join(', ') assinatura << " do |#{argumentos}|" else assinatura << ' do' end conteudo << "#{assinatura}\n" conteudo << " pending '#{I18n.t('steps.pending', text: texto_bruto)}'\n" conteudo << "end\n\n" end # Escreve o arquivo final File.write(arquivo_saida, conteudo) puts I18n.t('steps.generated', path: arquivo_saida) true end |