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 +7 -0
- data/.codegen/instructions/ruby.md +145 -0
- data/.env.example +1 -0
- data/.rspec +3 -0
- data/.rubocop.yml +13 -0
- data/.simplecov +6 -0
- data/CHANGELOG.md +6 -0
- data/LICENSE.txt +21 -0
- data/README.md +91 -0
- data/Rakefile +8 -0
- data/config/default/.codegen/instructions/react-vitest.md +86 -0
- data/config/default/.codegen/instructions/rspec-rails.md +31 -0
- data/docs/configuration.md +82 -0
- data/dotcodegen.gemspec +41 -0
- data/lib/dotcodegen/cli.rb +82 -0
- data/lib/dotcodegen/init.rb +25 -0
- data/lib/dotcodegen/test_code_generator.rb +62 -0
- data/lib/dotcodegen/test_file_generator.rb +71 -0
- data/lib/dotcodegen/version.rb +5 -0
- data/lib/dotcodegen.rb +8 -0
- data/sig/dotcodegen.rbs +4 -0
- metadata +141 -0
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
data/.rubocop.yml
ADDED
data/.simplecov
ADDED
data/CHANGELOG.md
ADDED
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,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
|
+
|
data/dotcodegen.gemspec
ADDED
@@ -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
|
data/lib/dotcodegen.rb
ADDED
data/sig/dotcodegen.rbs
ADDED
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: []
|