Facto is a fixtures library with a definition syntax. It aims to allow Go developers to define fixtures in a declarative way and using the Go language as the means to describe the fixtures instead of TOML/JSON/YAML. It is inspired by factory_bot.
Facto exposes a small API to register and use factories. The following example shows how to create a factory named User
:
// in factories/user.go
package factories
import (
"github.com/wawandco/facto"
)
func UserFactory(h facto.Helper) facto.Product {
user := User{
Name: h.Faker.Name(),
}
return facto.Product(user)
}
One these are added we can use our factories in our tests, e.g:
// in users/user_test.go
package user_test
import (
"your/package/factories"
)
func TestUser(t *testing.T) {
user := facto.Build(factories.UserFactory).(models.User)
err := db.Create(&user)
// use user for test purposes ...
// Another alternative is to use create, which will attempt at creating the object directly in your database.
// In which case the user variable has already been stored in the database.
p, err := facto.Create("User")
user := p.(models.User)
// use user for test purposes ...
}
Sometimes you need to build more than one instance of an object, in that case. Facto provides 2 functions to do that: BuildN
and CreateN
.
// in test/user_test.go
package user_test
func TestUser(t *testing.T) {
users := facto.BuildN(factories.User, 10).([]models.User)
// use users for test purposes ...
// you can also create N Users
p, err := facto.CreateN(factories.User, 10)
// make sure you check the error
users := p.([]models.User)
}
When running BuildN
you can use the Index variable to get the index of the object being built. This is useful when you need to build a list of objects and you want to know which one is being built.
// in factories/user.go
package factories
import (
"fmt"
"github.com/wawandco/facto"
)
func UserFactory(h facto.Helper) facto.Product {
user := User{
Name: fmt.Sprintf("User %d", f.Index),
}
return facto.Product(user)
}
When the factory gets called individually Index will be 0.
Another case is when you need to build an object that depends on another object. You can use the factory.Helper
parameter on your factory to get the object you need:
// in factories/event.go
package factories
import (
"github.com/wawandco/facto"
)
func UserFactory(h facto.Helper) facto.Product {
user := User{
Name: h.Faker.FirstName(),
}
return facto.Product(user)
}
func EventFactory(h facto.Helper) facto.Product {
event := Event{
// Here we pull the User from the helper given we know there is a
// factory for it. can use it like this:
User: facto.Build(UserFactory).(User),
Type: "Something",
}
return facto.Product(event)
})
Another case here is when you need to reference the ID of the previous object. You can use the factory.Helper
parameter on your factory to build a Named UUID:
// in factories/user.go
package factories
import (
"github.com/wawandco/facto"
)
func UserFactory(h facto.Helper) facto.Product {
user := User{
// owner_id will be assigned the generated UUID
// and any
ID: h.NamedUUID("owner_id"),
Name: h.Faker.FirstName(),
}
return facto.Product(user)
}
// in factories/event.go
package factories
import (
"github.com/wawandco/facto"
)
func EventFactory(h facto.Helper) facto.Product {
event := Event{
// Here we pull the User from the helper given we know there is a
// factory that adds it. If there was not one it would be generated new.
UserID: h.NamedUUID("owner_id"),
Type: "Something",
}
return facto.Product(event)
})
Sometimes you need to generate data that is not real but at least looks similar to what the real data will be. To solve that need Facto provides the Fake
method within the facto.Helper, it can be used within the factory it to generate fake data. for example:
// in factories/event.go
package factories
import (
"github.com/wawandco/facto"
)
func EventFactory(h facto.Helper) facto.Product {
event := Event{
Name: h.Faker.FirstName(),
Type: "Sports",
ContactEmail: h.Email(),
Company: h.Faker.Company(),
Address: h.Faker.Address(),
}
return facto.Product(event)
})
The fake data generation comes by courtesy of Gofakeit
The full list of available fake data generators can be found in here.
Another thing you could do with Facto is randomize the selection from a list of passed elements. For example:
// in factories/event.go
package factories
import (
"github.com/wawandco/facto"
)
func EventFactory(h facto.Helper) facto.Product {
event := Event{
Name: h.Faker.FirstName(),
// You can pass here a list of elements to randomly select
// from and the facto helper will pick one of these.
Type: f.OneOf(TypeSports, TypeMusic, TypeConcert).(EventType),
...
}
return facto.Product(event)
})
The Facto CLI is a simple command line tool that allows you to generate fixtures files. To install it you can use the following command:
go get github.com/wawandco/facto/cmd/facto@latest
The CLI allows you to generate factory files based on a given name. For example:
facto generate user
# Generates factories/user.go
Facto provides a plugin for the Ox CLI plugin system. To use it add the following to your cmd/ox/main.go
file:
import (
...
fox "github.com/wawandco/facto/ox" // Add the facto ox package, we called it fox here 😉
)
func main() {
// You may have other plugins here.
//...
cli.Use(fox.Plugin{}) // Add the facto Ox plugin.
err := cli.Run(context.Background(), os.Args)
if err != nil {
log.Fatal(err)
}
}
Once added you can use facto with Ox within the generate
command, you can see that invoking the generate
command will list factory
as a generator.
$ ox generate
[info] Using cmd/ox/main.go
Available Generators:
Name Plugin
---- ------
ox ox/generate-cli-main
template buffalo/generate-template
model buffalo/generate-model
action buffalo/generate-action
resource buffalo/generate-resource
task grift/generate-task
migration liquo/generate-migration
factory facto ## (this is the one you are looking for)
Which you can then use to generate your factories with:
$ ox generate factory [name]
- Create API: Create a new object in the database.
- Review terminology from Factory bot.
- Explain the "Magic" constraints and some principles.