dotcodegen 0.1.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
checksums.yaml ADDED
@@ -0,0 +1,7 @@
1
+ ---
2
+ SHA256:
3
+ metadata.gz: 3df431650d3b7c3736d9f9c8c47b7eb70fae00c78368d21954d5ef139170f4bc
4
+ data.tar.gz: 2ba426662ea615dfdce8f7eebee074deb5165072808d08f0c0cc13c54bcdba82
5
+ SHA512:
6
+ metadata.gz: 37ad7696e16dbfbdda40ce3b75796bbf1d7b0caa02cc7419283c66d89bf61ba05b0025fc622e6921e1a141dedb2602246602e56c49a6f8479b7d0371d1b883d8
7
+ data.tar.gz: 1cdbc250648762a3f688749f944622e83263686eb95b755ba334aacfabed1b3ebb172a74976db87461566577e97482e55a11309b80ec3158a6a9c2dfc179c3ac
@@ -0,0 +1,145 @@
1
+ ---
2
+ regex: 'lib/.*\.rb'
3
+ root_path: 'lib'
4
+ test_root_path: 'spec'
5
+ test_file_suffix: '_spec.rb'
6
+ ---
7
+
8
+ When writing a test, you should follow these steps:
9
+
10
+ 1. Avoid typos.
11
+ 2. Avoid things that could be infinite loops.
12
+ 3. This codebase is a Ruby gem, try to follow the conventions of the Ruby community.
13
+ 4. Avoid things that could be security vulnerabilities.
14
+ 5. Keep the codebase clean and easy to understand.
15
+ 6. Use Rspec for tests, don't use any other testing framework.
16
+ 7. Don't include ANY dependencies that are not already in the files you are provided.
17
+ 8. Don't start your tests with ``` or other strings, as your reply will be run as a Ruby file.
18
+
19
+ Here's an example of a good test you should reply with:
20
+
21
+ ```ruby
22
+ # frozen_string_literal: true
23
+
24
+ require 'dotcodegen/test_file_generator'
25
+
26
+ RSpec.describe Dotcodegen::TestFileGenerator do
27
+ let(:file_path) { 'client/app/components/feature.tsx' }
28
+ let(:api_matcher) do
29
+ {
30
+ 'regex' => 'api/.*\.rb',
31
+ 'root_path' => 'api/app/',
32
+ 'test_root_path' => 'api/spec/',
33
+ 'test_file_suffix' => '_spec.rb'
34
+ }
35
+ end
36
+ let(:client_matcher) do
37
+ {
38
+ 'regex' => 'client/app/.*\.tsx',
39
+ 'test_file_suffix' => '.test.tsx'
40
+ }
41
+ end
42
+ let(:matchers) { [api_matcher, client_matcher] }
43
+ let(:openai_key) { 'test_openai_key' }
44
+ let(:codegen_instance) { instance_double(Dotcodegen::TestFileGenerator) }
45
+
46
+ subject { described_class.new(file_path:, matchers:, openai_key:) }
47
+
48
+ describe '#run' do
49
+ after(:each) { FileUtils.remove_dir('client/', force: true) }
50
+ let(:file_path) { 'spec/fixtures/feature.tsx' }
51
+ let(:client_matcher) do
52
+ {
53
+ 'regex' => 'spec/fixtures/.*\.tsx',
54
+ 'test_file_suffix' => '.test.tsx',
55
+ 'root_path' => 'spec/fixtures/',
56
+ 'test_root_path' => 'tmp/codegen_spec/',
57
+ 'instructions' => 'instructions/react.md'
58
+ }
59
+ end
60
+
61
+ context 'when test file does not exist' do
62
+ it 'creates a test file and writes generated code once' do
63
+ allow(File).to receive(:exist?).with('tmp/codegen_spec/feature.test.tsx').and_return(false)
64
+ expect(FileUtils).to receive(:mkdir_p).with('tmp/codegen_spec')
65
+ allow(Dotcodegen::TestCodeGenerator).to receive_message_chain(:new, :generate_test_code).and_return('Mocked generated code')
66
+ expect(File).to receive(:write).with('tmp/codegen_spec/feature.test.tsx', '').once
67
+ expect(File).to receive(:write).with('tmp/codegen_spec/feature.test.tsx', 'Mocked generated code').once
68
+ subject.run
69
+ end
70
+ end
71
+
72
+ context 'when test file already exists' do
73
+ it 'does not create a test file but writes generated code' do
74
+ allow(File).to receive(:exist?).with('tmp/codegen_spec/feature.test.tsx').and_return(true)
75
+ expect(FileUtils).not_to receive(:mkdir_p)
76
+ allow(Dotcodegen::TestCodeGenerator).to receive_message_chain(:new, :generate_test_code).and_return('Mocked generated code')
77
+ expect(File).to receive(:write).with('tmp/codegen_spec/feature.test.tsx', 'Mocked generated code').once
78
+ subject.run
79
+ end
80
+ end
81
+ end
82
+
83
+ describe '#matcher' do
84
+ it 'returns the matching regex for the frontend' do
85
+ expect(subject.matcher).to eq(client_matcher)
86
+ end
87
+
88
+ context 'when file path is a ruby file' do
89
+ let(:file_path) { 'api/app/models/app.rb' }
90
+ it 'returns the matching regex for the backend' do
91
+ expect(subject.matcher).to eq(api_matcher)
92
+ end
93
+ end
94
+
95
+ context 'when there are no matches' do
96
+ let(:file_path) { 'terraform/models/app.rb' }
97
+ it 'returns nil' do
98
+ expect(subject.matcher).to be_nil
99
+ end
100
+ end
101
+
102
+ context 'when file path does not match any regex' do
103
+ let(:file_path) { 'api/models/app.go' }
104
+ it 'returns nil' do
105
+ expect(subject.matcher).to be_nil
106
+ end
107
+ end
108
+ end
109
+
110
+ describe '#test_file_path' do
111
+ it 'returns the test file path for the frontend' do
112
+ expect(subject.test_file_path).to eq('client/app/components/feature.test.tsx')
113
+ end
114
+
115
+ context 'when file path is a ruby file' do
116
+ let(:file_path) { 'api/app/models/app.rb' }
117
+ it 'returns the test file path for the backend' do
118
+ expect(subject.test_file_path).to eq('api/spec/models/app_spec.rb')
119
+ end
120
+ end
121
+ end
122
+ end
123
+ ```
124
+
125
+ Here's the skeleton of a test you can start from:
126
+
127
+ ```ruby
128
+ # frozen_string_literal: true
129
+
130
+ require 'dotcodegen/__file_path__'
131
+
132
+ RSpec.describe Dotcodegen::__CLASS_NAME__ do
133
+ let(:params) do
134
+ {
135
+ # Add params here
136
+ }
137
+ end
138
+ subject { described_class.new(params) }
139
+
140
+
141
+ it 'runs' do
142
+ # Add assertions here
143
+ end
144
+ end
145
+ ```
data/.env.example ADDED
@@ -0,0 +1 @@
1
+ OPENAI_KEY=your-api-key
data/.rspec ADDED
@@ -0,0 +1,3 @@
1
+ --format documentation
2
+ --color
3
+ --require spec_helper
data/.rubocop.yml ADDED
@@ -0,0 +1,13 @@
1
+ AllCops:
2
+ TargetRubyVersion: 3.3.0
3
+ Exclude:
4
+ - "spec/**/*"
5
+
6
+ Documentation:
7
+ Enabled: false
8
+
9
+ Style/GlobalVars:
10
+ Enabled: false
11
+
12
+ Layout/LineLength:
13
+ Enabled: false
data/.simplecov ADDED
@@ -0,0 +1,6 @@
1
+ # frozen_string_literal: true
2
+
3
+ SimpleCov.start do
4
+ add_filter '/spec/'
5
+ add_filter '/vendor/bundle/'
6
+ end
data/CHANGELOG.md ADDED
@@ -0,0 +1,6 @@
1
+ ## [Unreleased]
2
+
3
+ ## [0.1.0] - 2024-03-10
4
+
5
+ - Initial release
6
+ - Support for Ruby 3.3.0
data/LICENSE.txt ADDED
@@ -0,0 +1,21 @@
1
+ The MIT License (MIT)
2
+
3
+ Copyright (c) 2024 Ferruccio Balestreri
4
+
5
+ Permission is hereby granted, free of charge, to any person obtaining a copy
6
+ of this software and associated documentation files (the "Software"), to deal
7
+ in the Software without restriction, including without limitation the rights
8
+ to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
9
+ copies of the Software, and to permit persons to whom the Software is
10
+ furnished to do so, subject to the following conditions:
11
+
12
+ The above copyright notice and this permission notice shall be included in
13
+ all copies or substantial portions of the Software.
14
+
15
+ THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16
+ IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17
+ FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
18
+ AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19
+ LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
20
+ OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
21
+ THE SOFTWARE.
data/README.md ADDED
@@ -0,0 +1,91 @@
1
+ # Automatic test generation for monoliths
2
+
3
+ Never write a test from scratch again. Automatically generate tests for any file you open in your codebase.
4
+
5
+ Keep your team up to date with the latest best practices and conventions you adopt. Customize the templates to fit your team's needs.
6
+
7
+ We're using this tool to speed up writing tests for our monolith at [June](https://june.so). We open sourced it so you can use it too.
8
+
9
+ ## Get started
10
+
11
+
12
+ 1. Install our CLI by running:
13
+
14
+ ```bash
15
+ brew tap ferrucc-io/dotcodegen
16
+ brew install dotcodegen
17
+ ```
18
+
19
+ 2. Initialise the `.codegen` directory in your codebase:
20
+
21
+ ```bash
22
+ codegen init
23
+ ```
24
+
25
+ 3. Configure the templates to fit your team's needs. See the [configuration](./docs/configuration.md) section for more details.
26
+
27
+ 4. Run the codegen command in your terminal:
28
+
29
+ ```bash
30
+ codegen path/to/the/file/you/want/to/test --openai_key <your_openai_key>
31
+ ```
32
+
33
+
34
+ 5. That's it! You're ready to start generating tests for your codebase.
35
+
36
+
37
+ **Extra**:
38
+
39
+ This code becomes very powerful when you integrate it with your editor, so you can open a file and generate tests with a single command.
40
+
41
+ In order to do that you can add a task to your `tasks.json` file in the `.vscode` directory of your project:
42
+
43
+ ```json
44
+ {
45
+ "version": "2.0.0",
46
+ "tasks": [
47
+ {
48
+ "label": "Generate test for the current file",
49
+ "type": "shell",
50
+ // Alternatively you can specify the OPENAI_KEY environment variable in your .env file
51
+ "command": "codegen ${relativeFile} --openai_key your_openai_key",
52
+ "group": "test",
53
+ "problemMatcher": [],
54
+ "options": {
55
+ "cwd": "${workspaceFolder}"
56
+ },
57
+ }
58
+ ]
59
+ }
60
+ ```
61
+
62
+ We're currently building a VSCode extension to be able to generate tests directly from your editor. Stay tuned!
63
+
64
+ ## How it works
65
+
66
+ The extension uses AI to generate tests for any file you open in your codebase. It uses a set of customizable templates to generate the tests. You can customize the templates to fit your team's needs.
67
+
68
+ ## Features
69
+
70
+ - **Easy to learn**: Get started in minutes.
71
+ - **AI powered scaffolding**: Generate smart tests for any file you open in your codebase.
72
+ - **Fully customisable**: Customize the templates to fit your team's needs.
73
+ - **Bring your own API key**: Use your own OpenAI API key to generate the tests.
74
+ - **🚧 Integrates with VSCode**: Use the extension to generate tests directly from your editor.
75
+
76
+ ## Contributing
77
+
78
+ If you want to add some default templates for your language, feel free to open a PR. See the [config/default](./config/default) directory for the ones we have already.
79
+
80
+ Bug reports and pull requests are welcome on GitHub at https://github.com/ferrucc-io/dotcodegen.
81
+
82
+ ## Development
83
+
84
+ After checking out the repo, run `bin/setup` to install dependencies. Then, run `rake spec` to run the tests. You can also run `bin/console` for an interactive prompt that will allow you to experiment.
85
+
86
+ To install this gem onto your local machine, run `bundle exec rake install`. To release a new version, update the version number in `version.rb`, and then run `bundle exec rake release`, which will create a git tag for the version, push git commits and the created tag, and push the `.gem` file to [rubygems.org](https://rubygems.org).
87
+
88
+
89
+ ## License
90
+
91
+ The gem is available as open source under the terms of the [MIT License](https://opensource.org/licenses/MIT).
data/Rakefile ADDED
@@ -0,0 +1,8 @@
1
+ # frozen_string_literal: true
2
+
3
+ require 'bundler/gem_tasks'
4
+ require 'rspec/core/rake_task'
5
+
6
+ RSpec::Core::RakeTask.new(:spec)
7
+
8
+ task default: :spec
@@ -0,0 +1,86 @@
1
+ ---
2
+ regex: 'src/.*\.tsx'
3
+ test_file_suffix: '.test.tsx'
4
+ ---
5
+
6
+
7
+ When writing a test, you should follow these steps:
8
+
9
+ 1. Avoid typos.
10
+ 2. Avoid things that could be infinite loops.
11
+ 3. This codebase is React and Typescript, try to follow the conventions and patterns of React and Typescript.
12
+ 4. Avoid dangerous stuff, like things that would show up as a CVE somewhere.
13
+ 5. Use vitest for tests. It's a testing library that is used in this codebase.
14
+ 6. Always import tests from test-utils/
15
+
16
+ Here's the skeleton of a test:
17
+
18
+ ```Typescript
19
+
20
+ import "@testing-library/jest-dom";
21
+ import userEvent from "@testing-library/user-event";
22
+ import { Pagination } from "./Pagination";
23
+ import { render, screen } from "../../../../tests/testUtils";
24
+
25
+ describe("core/components/List/Pagination", () => {
26
+ const mockNextPage = vi.fn();
27
+ const mockPreviousPage = vi.fn();
28
+ const setup = (currentPage = 1, lastPage = 5) => {
29
+ const pagy = {
30
+ page: currentPage,
31
+ items: 10,
32
+ count: 50,
33
+ last: lastPage,
34
+ };
35
+ const pagination = {
36
+ currentPage: currentPage,
37
+ nextPage: mockNextPage,
38
+ previousPage: mockPreviousPage,
39
+ };
40
+ render(
41
+ <Pagination pagination={pagination} pagy={pagy} resourceName="items" />,
42
+ );
43
+ };
44
+
45
+ test("renders pagination component correctly", () => {
46
+ setup();
47
+ expect(screen.getByText(/previous page/i)).toBeInTheDocument();
48
+ expect(screen.getByText(/next page/i)).toBeInTheDocument();
49
+ expect(screen.getByText(/1-10/i)).toBeInTheDocument();
50
+ expect(screen.getByText(/out of 50/i)).toBeInTheDocument();
51
+ });
52
+
53
+ test("disables 'Previous page' button on first page", () => {
54
+ setup(1);
55
+ expect(screen.getByText(/previous page/i)).toBeDisabled();
56
+ });
57
+
58
+ test("enables 'Previous page' button on page greater than 1", () => {
59
+ setup(2);
60
+ expect(screen.getByText(/previous page/i)).toBeEnabled();
61
+ });
62
+
63
+ test("disables 'Next page' button on last page", () => {
64
+ setup(5, 5);
65
+ expect(screen.getByText(/next page/i)).toBeDisabled();
66
+ });
67
+
68
+ test("enables 'Next page' button before last page", () => {
69
+ setup(4, 5);
70
+ expect(screen.getByText(/next page/i)).toBeEnabled();
71
+ });
72
+
73
+ test("calls nextPage function when 'Next page' button is clicked", async () => {
74
+ setup(1, 5);
75
+ await userEvent.click(screen.getByText(/next page/i));
76
+ expect(mockNextPage).toHaveBeenCalled();
77
+ });
78
+
79
+ test("calls previousPage function when 'Previous page' button is clicked", async () => {
80
+ setup(2);
81
+ await userEvent.click(screen.getByText(/previous page/i));
82
+ expect(mockPreviousPage).toHaveBeenCalled();
83
+ });
84
+ });
85
+
86
+ ```
@@ -0,0 +1,31 @@
1
+ ---
2
+ regex: 'app/.*\.rb'
3
+ root_path: 'app'
4
+ test_root_path: 'spec'
5
+ test_file_suffix: '_spec.rb'
6
+ ---
7
+
8
+ When writing a test, you should follow these steps:
9
+
10
+ 1. Avoid typos.
11
+ 2. Avoid things that could be infinite loops.
12
+ 3. This codebase is Rails, try to follow the conventions of Rails.
13
+ 4. Write tests using RSpec like in the example I included
14
+ 5. If you're in doubt, just write the parts you're sure of
15
+ 6. No comments in the test file, just the test code
16
+
17
+ Use FactoryBot factories for tests, so you should always create a factory for the model you are testing. This will help you create test data quickly and easily.
18
+
19
+ Here's the skeleton of a test:
20
+
21
+ ```ruby
22
+ # frozen_string_literal: true
23
+
24
+ require 'rails_helper'
25
+
26
+ RSpec.describe __FULL_TEST_NAME__ do
27
+ let(:app) { create(:app) }
28
+
29
+ # Tests go here
30
+ end
31
+ ```
@@ -0,0 +1,82 @@
1
+ # Configuration
2
+
3
+ ## Introduction
4
+
5
+ To make sure the tests generated by `dotcodegen` fit your team's needs, you can customize the instructions used to generate the tests. This section will guide you through the process of customizing the instructions.
6
+
7
+ The instructions are markdown files in plain English. They are used to generate the tests for your codebase. You can customize the instructions to fit your team's needs.
8
+
9
+ ## Writing your first instruction
10
+
11
+ To write your first instruction make sure you have `codegen` installed. If you don't have it installed, follow the [installation](../README.md#installation) instructions.
12
+
13
+ Create a new file in the `.codegen/instructions` directory called `react.md`. This file will contain the instructions used to generate the tests.
14
+
15
+ ```bash
16
+ mkdir -p .codegen/instructions
17
+ touch .codegen/instructions/react.md
18
+ ```
19
+
20
+ Then for the content of the file, you'll want to specify:
21
+
22
+ - The regex to match the files you want to generate the test for. (e.g. `.*\.tsx` or `api/.*\.rb`)
23
+ - The suffix of the test file. (e.g. `.test.tsx` or `_spec.rb`)
24
+ - The instructions to generate the tests.
25
+
26
+ Here's an example of a `react.md` file:
27
+
28
+ ```markdown
29
+ ---
30
+ regex: '.*\.tsx'
31
+ test_file_suffix: '.test.tsx'
32
+ ---
33
+
34
+ When writing a test, you should follow these steps:
35
+
36
+ 1. Avoid typos.
37
+ 2. Avoid things that could be infinite loops.
38
+ 3. This codebase is a React codebase, try to follow the conventions of the React community.
39
+
40
+ Here's an example of a good test you should reply with:
41
+
42
+ <!-- Copy paste one of your existing tests here -->
43
+
44
+ ```
45
+
46
+ ## Customizing the file paths
47
+
48
+ In some programming languages, tests live in specific directories. For example, in Ruby, tests live in a `spec` directory. You can specify the directory where the tests should be generated by adding a `test_file_directory` key to the instruction file.
49
+
50
+ Here's an example of a `rails-controller.md` file, that specifies `spec/controllers` the directory where the tests should be generated and removes the `app/controllers` prefix from the starting file path:
51
+
52
+ ```markdown
53
+ ---
54
+ regex: 'app/controllers/.*\.rb'
55
+ root_path: 'app/controllers'
56
+ test_root_path: 'spec/controllers'
57
+ test_file_suffix: '_spec.rb'
58
+ ---
59
+
60
+ Follow these steps when writing a test for a Rails controller:
61
+
62
+ 1. Avoid typos.
63
+ 2. Avoid things that could be infinite loops.
64
+ 3. This codebase is a Rails codebase, try to follow the conventions of the Rails community.
65
+ ```
66
+
67
+
68
+ ## Instructions reference
69
+
70
+ The instructions need to be .md files in the `.codegen/instructions` directory. For each instruction file, you can specify the following keys:
71
+
72
+ - `regex`: The regex to match the files you want to generate the test for. (e.g. `.*\.tsx` or `api/.*\.rb`)
73
+ - `test_root_path`: The root path that codegen will pre-pend to all test files. (e.g. `spec/controllers` or `api`)
74
+ - `root_path`: The root path of your which you want to remove when adding the test files `test_root_path`. (e.g. `app/controllers` or `api`)
75
+ - `test_file_suffix`: The suffix of the test file. (e.g. `.test.tsx` or `_spec.rb`) We will automatically add this suffix to the file name and remove the .extension from your original file name.
76
+
77
+ ## The .codegen directory
78
+
79
+ The `.codegen` directory is where you can customize the instructions used to generate the tests. The directory should be placed at the root of your project.
80
+
81
+
82
+
@@ -0,0 +1,41 @@
1
+ # frozen_string_literal: true
2
+
3
+ require_relative 'lib/dotcodegen/version'
4
+
5
+ Gem::Specification.new do |spec|
6
+ spec.name = 'dotcodegen'
7
+ spec.version = Dotcodegen::VERSION
8
+ spec.authors = ['Ferruccio Balestreri']
9
+ spec.email = ['ferruccio.balestreri@gmail.com']
10
+
11
+ spec.summary = 'Generate tests for your code using LLMs.'
12
+ spec.description = 'Generate tests for your code using LLMs. This gem is a CLI tool that uses OpenAI to generate test code for your code. It uses a configuration file to match files with the right test code generation instructions. It is designed to be used with Ruby on Rails, but it can be used with any codebase. It is a work in progress.'
13
+ spec.homepage = 'https://github.com/ferrucc-io/dotcodegen'
14
+ spec.license = 'MIT'
15
+ spec.required_ruby_version = '= 3.3.0'
16
+
17
+ spec.metadata['allowed_push_host'] = 'https://rubygems.org'
18
+
19
+ spec.metadata['homepage_uri'] = spec.homepage
20
+ spec.metadata['source_code_uri'] = 'https://github.com/ferrucc-io/dotcodegen'
21
+ spec.metadata['changelog_uri'] = 'https://github.com/ferrucc-io/dotcodegen/blob/main/CHANGELOG.md'
22
+
23
+ spec.files = Dir.chdir(__dir__) do
24
+ `git ls-files -z`.split("\x0").reject do |f|
25
+ (File.expand_path(f) == __FILE__) ||
26
+ f.start_with?(*%w[bin/ /lib test/ spec/ features/ .git .github appveyor Gemfile])
27
+ end
28
+ end
29
+ spec.bindir = 'exe'
30
+ spec.executables = spec.files.grep(%r{\Aexe/}) { |f| File.basename(f) }
31
+ spec.require_paths = ['lib']
32
+
33
+ spec.add_dependency 'dotenv'
34
+ spec.add_dependency 'front_matter_parser'
35
+ spec.add_dependency 'optparse'
36
+ spec.add_dependency 'ostruct'
37
+ spec.add_dependency 'ruby-openai'
38
+
39
+ # For more information and examples about making a new gem, check out our
40
+ # guide at: https://bundler.io/guides/creating_gem.html
41
+ end
@@ -0,0 +1,82 @@
1
+ # frozen_string_literal: true
2
+
3
+ require 'optparse'
4
+ require 'ostruct'
5
+ require 'front_matter_parser'
6
+ require 'fileutils'
7
+ require_relative 'test_file_generator'
8
+ require_relative 'version'
9
+ require_relative 'init'
10
+ require 'dotenv' unless defined?($running_tests) && $running_tests
11
+
12
+ module Dotcodegen
13
+ class CLI
14
+ def self.parse(args)
15
+ options = OpenStruct.new
16
+ opt_parser = build_option_parser(options)
17
+ handle_arguments(args, opt_parser, options)
18
+ options
19
+ end
20
+
21
+ def self.run(args)
22
+ Dotenv.load unless defined?($running_tests) && $running_tests
23
+ options = parse(args)
24
+ return version if options.version
25
+
26
+ return Init.run if options.init
27
+
28
+ matchers = load_matchers('.codegen/instructions')
29
+ TestFileGenerator.new(file_path: options.file_path, matchers:, openai_key: options.openai_key).run
30
+ end
31
+
32
+ def self.version
33
+ puts "Dotcodegen version #{Dotcodegen::VERSION}"
34
+ exit
35
+ end
36
+
37
+ def self.load_matchers(instructions_path)
38
+ Dir.glob("#{instructions_path}/*.md").map do |file|
39
+ parsed = FrontMatterParser::Parser.parse_file(file)
40
+ parsed.front_matter.merge({ content: parsed.content })
41
+ end
42
+ end
43
+
44
+ def self.build_option_parser(options)
45
+ OptionParser.new do |opts|
46
+ opts.banner = 'Usage: dotcodegen [options] file_path'
47
+
48
+ opts.on('--openai_key KEY', 'OpenAI API Key') { |key| options.openai_key = key }
49
+ opts.on('--init', 'Initialize a .codegen configuration in the current directory') { options.init = true }
50
+ opts.on('--version', 'Show version') { options.version = true }
51
+ opts.on_tail('-h', '--help', 'Show this message') do
52
+ puts opts
53
+ exit
54
+ end
55
+ end
56
+ end
57
+
58
+ def self.handle_arguments(args, opt_parser, options)
59
+ opt_parser.parse!(args)
60
+ return if options.version || options.init
61
+
62
+ validate_file_path(args, opt_parser)
63
+ options.file_path = args.shift
64
+
65
+ options.openai_key ||= ENV['OPENAI_KEY']
66
+
67
+ return unless options.openai_key.to_s.strip.empty?
68
+
69
+ puts 'Error: Missing --openai_key flag or OPENAI_KEY environment variable.'
70
+ puts opt_parser
71
+ exit 1
72
+ end
73
+
74
+ def self.validate_file_path(args, opt_parser)
75
+ return unless args.empty?
76
+
77
+ puts 'Error: Missing file path.'
78
+ puts opt_parser
79
+ exit 1
80
+ end
81
+ end
82
+ end
@@ -0,0 +1,25 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Dotcodegen
4
+ class Init
5
+ # rubocop:disable Metrics/MethodLength
6
+ def self.run
7
+ source_dir = File.expand_path('../../config/default/.codegen', __dir__)
8
+ destination_dir = File.expand_path('.codegen', Dir.pwd)
9
+
10
+ FileUtils.mkdir_p(destination_dir) unless Dir.exist?(destination_dir)
11
+ FileUtils.cp_r("#{source_dir}/.", destination_dir)
12
+
13
+ instructions_dir = File.expand_path('instructions', destination_dir)
14
+ FileUtils.mkdir_p(instructions_dir) unless Dir.exist?(instructions_dir)
15
+
16
+ Dir.glob("#{source_dir}/instructions/*.md").each do |md_file|
17
+ FileUtils.cp(md_file, instructions_dir)
18
+ end
19
+
20
+ puts 'Codegen initialized.'
21
+ exit
22
+ end
23
+ # rubocop:enable Metrics/MethodLength
24
+ end
25
+ end
@@ -0,0 +1,62 @@
1
+ # frozen_string_literal: true
2
+
3
+ require 'openai'
4
+ module Dotcodegen
5
+ class TestCodeGenerator
6
+ attr_reader :config, :file_to_test_path, :openai_key
7
+
8
+ def initialize(config:, file_to_test_path:, openai_key:)
9
+ @config = config
10
+ @file_to_test_path = file_to_test_path
11
+ @openai_key = openai_key
12
+ end
13
+
14
+ def generate_test_code
15
+ response = openai_client.chat(
16
+ parameters: {
17
+ model: 'gpt-4-turbo-preview',
18
+ messages: [{ role: 'user', content: test_prompt_text }], # Required.
19
+ temperature: 0.7
20
+ }
21
+ )
22
+ response.dig('choices', 0, 'message', 'content')
23
+ end
24
+
25
+ def test_prompt_text
26
+ [{ "type": 'text', "text": test_prompt }]
27
+ end
28
+
29
+ # rubocop:disable Metrics/MethodLength
30
+ def test_prompt
31
+ [
32
+ 'You are an expert programmer. You have been given a task to write a test file for a given file following some instructions.',
33
+ 'This is the file you want to test:',
34
+ '--start--',
35
+ test_file_content,
36
+ '--end--',
37
+ 'Here are the instructions on how to write the test file:',
38
+ '--start--',
39
+ test_instructions,
40
+ '--end--',
41
+ "Your answer will be directly written in the file you want to test. Don't include any explanation or comments in your answer that isn't code.",
42
+ 'You can use the comment syntax to write comments in your answer.'
43
+ ].join("\n")
44
+ end
45
+ # rubocop:enable Metrics/MethodLength
46
+
47
+ def test_file_content
48
+ File.open(file_to_test_path).read
49
+ end
50
+
51
+ def test_instructions
52
+ config['content']
53
+ end
54
+
55
+ def openai_client
56
+ @openai_client ||= OpenAI::Client.new(
57
+ access_token: openai_key,
58
+ organization_id: 'org-4nA9FJ8NajsLJ2fbHRAw7MLI'
59
+ )
60
+ end
61
+ end
62
+ end
@@ -0,0 +1,71 @@
1
+ # frozen_string_literal: true
2
+
3
+ require 'fileutils'
4
+ require_relative 'test_code_generator'
5
+
6
+ module Dotcodegen
7
+ class TestFileGenerator
8
+ attr_reader :file_path, :matchers, :openai_key
9
+
10
+ def initialize(file_path:, matchers:, openai_key:)
11
+ @file_path = file_path
12
+ @matchers = matchers
13
+ @openai_key = openai_key
14
+ end
15
+
16
+ def run
17
+ puts "Finding matcher for #{file_path}..."
18
+ return puts "No matcher found for #{file_path}" unless matcher
19
+
20
+ puts "Test file path: #{test_file_path}"
21
+ ensure_test_file_presence
22
+
23
+ write_generated_code_to_test_file
24
+ open_test_file_in_editor unless $running_tests
25
+
26
+ puts 'Running codegen...'
27
+ end
28
+
29
+ def ensure_test_file_presence
30
+ puts "Creating test file if it doesn't exist..."
31
+ return if File.exist?(test_file_path)
32
+
33
+ FileUtils.mkdir_p(File.dirname(test_file_path))
34
+ File.write(test_file_path, '')
35
+ end
36
+
37
+ def write_generated_code_to_test_file
38
+ generated_code = Dotcodegen::TestCodeGenerator.new(config: matcher, file_to_test_path: file_path,
39
+ openai_key:).generate_test_code
40
+ File.write(test_file_path, generated_code)
41
+ end
42
+
43
+ def open_test_file_in_editor
44
+ system("code #{test_file_path}")
45
+ end
46
+
47
+ def matcher
48
+ @matcher ||= matchers.find { |m| file_path.match?(m['regex']) }
49
+ end
50
+
51
+ def test_file_path
52
+ @test_file_path ||= "#{test_root_path}#{relative_file_name}#{test_file_suffix}"
53
+ end
54
+
55
+ def relative_file_name
56
+ file_path.sub(root_path, '').sub(/\.\w+$/, '')
57
+ end
58
+
59
+ def test_root_path
60
+ matcher['test_root_path'] || ''
61
+ end
62
+
63
+ def root_path
64
+ matcher['root_path'] || ''
65
+ end
66
+
67
+ def test_file_suffix
68
+ matcher['test_file_suffix'] || ''
69
+ end
70
+ end
71
+ end
@@ -0,0 +1,5 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Dotcodegen
4
+ VERSION = '0.1.0'
5
+ end
data/lib/dotcodegen.rb ADDED
@@ -0,0 +1,8 @@
1
+ # frozen_string_literal: true
2
+
3
+ require_relative 'dotcodegen/version'
4
+ require_relative 'dotcodegen/test_file_generator'
5
+
6
+ module Dotcodegen
7
+ class Error < StandardError; end
8
+ end
@@ -0,0 +1,4 @@
1
+ module Dotcodegen
2
+ VERSION: String
3
+ # See the writing guide of rbs: https://github.com/ruby/rbs#guides
4
+ end
metadata ADDED
@@ -0,0 +1,141 @@
1
+ --- !ruby/object:Gem::Specification
2
+ name: dotcodegen
3
+ version: !ruby/object:Gem::Version
4
+ version: 0.1.0
5
+ platform: ruby
6
+ authors:
7
+ - Ferruccio Balestreri
8
+ autorequire:
9
+ bindir: exe
10
+ cert_chain: []
11
+ date: 2024-03-10 00:00:00.000000000 Z
12
+ dependencies:
13
+ - !ruby/object:Gem::Dependency
14
+ name: dotenv
15
+ requirement: !ruby/object:Gem::Requirement
16
+ requirements:
17
+ - - ">="
18
+ - !ruby/object:Gem::Version
19
+ version: '0'
20
+ type: :runtime
21
+ prerelease: false
22
+ version_requirements: !ruby/object:Gem::Requirement
23
+ requirements:
24
+ - - ">="
25
+ - !ruby/object:Gem::Version
26
+ version: '0'
27
+ - !ruby/object:Gem::Dependency
28
+ name: front_matter_parser
29
+ requirement: !ruby/object:Gem::Requirement
30
+ requirements:
31
+ - - ">="
32
+ - !ruby/object:Gem::Version
33
+ version: '0'
34
+ type: :runtime
35
+ prerelease: false
36
+ version_requirements: !ruby/object:Gem::Requirement
37
+ requirements:
38
+ - - ">="
39
+ - !ruby/object:Gem::Version
40
+ version: '0'
41
+ - !ruby/object:Gem::Dependency
42
+ name: optparse
43
+ requirement: !ruby/object:Gem::Requirement
44
+ requirements:
45
+ - - ">="
46
+ - !ruby/object:Gem::Version
47
+ version: '0'
48
+ type: :runtime
49
+ prerelease: false
50
+ version_requirements: !ruby/object:Gem::Requirement
51
+ requirements:
52
+ - - ">="
53
+ - !ruby/object:Gem::Version
54
+ version: '0'
55
+ - !ruby/object:Gem::Dependency
56
+ name: ostruct
57
+ requirement: !ruby/object:Gem::Requirement
58
+ requirements:
59
+ - - ">="
60
+ - !ruby/object:Gem::Version
61
+ version: '0'
62
+ type: :runtime
63
+ prerelease: false
64
+ version_requirements: !ruby/object:Gem::Requirement
65
+ requirements:
66
+ - - ">="
67
+ - !ruby/object:Gem::Version
68
+ version: '0'
69
+ - !ruby/object:Gem::Dependency
70
+ name: ruby-openai
71
+ requirement: !ruby/object:Gem::Requirement
72
+ requirements:
73
+ - - ">="
74
+ - !ruby/object:Gem::Version
75
+ version: '0'
76
+ type: :runtime
77
+ prerelease: false
78
+ version_requirements: !ruby/object:Gem::Requirement
79
+ requirements:
80
+ - - ">="
81
+ - !ruby/object:Gem::Version
82
+ version: '0'
83
+ description: Generate tests for your code using LLMs. This gem is a CLI tool that
84
+ uses OpenAI to generate test code for your code. It uses a configuration file to
85
+ match files with the right test code generation instructions. It is designed to
86
+ be used with Ruby on Rails, but it can be used with any codebase. It is a work in
87
+ progress.
88
+ email:
89
+ - ferruccio.balestreri@gmail.com
90
+ executables: []
91
+ extensions: []
92
+ extra_rdoc_files: []
93
+ files:
94
+ - ".codegen/instructions/ruby.md"
95
+ - ".env.example"
96
+ - ".rspec"
97
+ - ".rubocop.yml"
98
+ - ".simplecov"
99
+ - CHANGELOG.md
100
+ - LICENSE.txt
101
+ - README.md
102
+ - Rakefile
103
+ - config/default/.codegen/instructions/react-vitest.md
104
+ - config/default/.codegen/instructions/rspec-rails.md
105
+ - docs/configuration.md
106
+ - dotcodegen.gemspec
107
+ - lib/dotcodegen.rb
108
+ - lib/dotcodegen/cli.rb
109
+ - lib/dotcodegen/init.rb
110
+ - lib/dotcodegen/test_code_generator.rb
111
+ - lib/dotcodegen/test_file_generator.rb
112
+ - lib/dotcodegen/version.rb
113
+ - sig/dotcodegen.rbs
114
+ homepage: https://github.com/ferrucc-io/dotcodegen
115
+ licenses:
116
+ - MIT
117
+ metadata:
118
+ allowed_push_host: https://rubygems.org
119
+ homepage_uri: https://github.com/ferrucc-io/dotcodegen
120
+ source_code_uri: https://github.com/ferrucc-io/dotcodegen
121
+ changelog_uri: https://github.com/ferrucc-io/dotcodegen/blob/main/CHANGELOG.md
122
+ post_install_message:
123
+ rdoc_options: []
124
+ require_paths:
125
+ - lib
126
+ required_ruby_version: !ruby/object:Gem::Requirement
127
+ requirements:
128
+ - - '='
129
+ - !ruby/object:Gem::Version
130
+ version: 3.3.0
131
+ required_rubygems_version: !ruby/object:Gem::Requirement
132
+ requirements:
133
+ - - ">="
134
+ - !ruby/object:Gem::Version
135
+ version: '0'
136
+ requirements: []
137
+ rubygems_version: 3.5.6
138
+ signing_key:
139
+ specification_version: 4
140
+ summary: Generate tests for your code using LLMs.
141
+ test_files: []