Skip to content

Commit ca81c3b

Browse files
Add support for flattened value groups (#700)
Fixes #693. Co-authored-by: Abhinav Gupta <[email protected]>
1 parent 3f21605 commit ca81c3b

7 files changed

+45
-7
lines changed

CHANGELOG.md

+4-1
Original file line numberDiff line numberDiff line change
@@ -5,7 +5,10 @@ The format is based on [Keep a Changelog](http://keepachangelog.com/en/1.0.0/)
55
and this project adheres to [Semantic Versioning](http://semver.org/spec/v2.0.0.html).
66

77
## [Unreleased]
8-
- Back to development.
8+
### Added
9+
- Value groups can use the `flatten` option to indicate values in a slice should
10+
be provided individually rather than providing the slice itself. See package
11+
documentation for details.
912

1013
## [1.10.0] - 2019-11-20
1114
### Added

annotated.go

+4
Original file line numberDiff line numberDiff line change
@@ -64,6 +64,10 @@ type Annotated struct {
6464
// by the constructor. For more information on value groups, see the package documentation.
6565
//
6666
// A group option may not be provided if a name option is provided.
67+
//
68+
// Similar to group tags, the group name may be followed by a `,flatten`
69+
// option to indicate that each element in the slice returned by the
70+
// constructor should be injected into the value group individually.
6771
Group string
6872

6973
// Target is the constructor being annotated with fx.Annotated.

app_test.go

+21-1
Original file line numberDiff line numberDiff line change
@@ -141,6 +141,26 @@ func TestNewApp(t *testing.T) {
141141
require.NoError(t, app.Err())
142142
})
143143

144+
t.Run("ProvidesWithAnnotateFlattened", func(t *testing.T) {
145+
app := fxtest.New(t,
146+
Provide(Annotated{
147+
Target: func() []int { return []int{1} },
148+
Group: "foo,flatten",
149+
}),
150+
Invoke(
151+
func(b struct {
152+
In
153+
Foos []int `group:"foo"`
154+
}) {
155+
assert.Len(t, b.Foos, 1)
156+
},
157+
),
158+
)
159+
160+
defer app.RequireStart().RequireStop()
161+
require.NoError(t, app.Err())
162+
})
163+
144164
t.Run("ProvidesWithEmptyAnnotate", func(t *testing.T) {
145165
type A struct{}
146166

@@ -250,7 +270,7 @@ func TestInvokes(t *testing.T) {
250270
)
251271
err := app.Err()
252272
require.Error(t, err)
253-
assert.Contains(t, err.Error(), "fx_test.A is not in the container")
273+
assert.Contains(t, err.Error(), "missing type: fx_test.A")
254274
})
255275

256276
t.Run("ErrorHooksAreCalled", func(t *testing.T) {

go.mod

+1-1
Original file line numberDiff line numberDiff line change
@@ -4,7 +4,7 @@ go 1.13
44

55
require (
66
github.com/stretchr/testify v1.4.0
7-
go.uber.org/dig v1.8.0
7+
go.uber.org/dig v1.9.0
88
go.uber.org/goleak v0.10.0
99
go.uber.org/multierr v1.4.0
1010
golang.org/x/lint v0.0.0-20190930215403-16217165b5de

go.sum

+2-2
Original file line numberDiff line numberDiff line change
@@ -20,8 +20,8 @@ github.com/stretchr/testify v1.4.0 h1:2E4SXV/wtOkTonXsotYi4li6zVWxYlZuYNCXe9XRJy
2020
github.com/stretchr/testify v1.4.0/go.mod h1:j7eGeouHqKxXV5pUuKE4zz7dFj8WfuZ+81PSLYec5m4=
2121
go.uber.org/atomic v1.5.0 h1:OI5t8sDa1Or+q8AeE+yKeB/SDYioSHAgcVljj9JIETY=
2222
go.uber.org/atomic v1.5.0/go.mod h1:sABNBOSYdrvTF6hTgEIbc7YasKWGhgEQZyfxyTvoXHQ=
23-
go.uber.org/dig v1.8.0 h1:1rR6hnL/bu1EVcjnRDN5kx1vbIjEJDTGhSQ2B3ddpcI=
24-
go.uber.org/dig v1.8.0/go.mod h1:X34SnWGr8Fyla9zQNO2GSO2D+TIuqB14OS8JhYocIyw=
23+
go.uber.org/dig v1.9.0 h1:pJTDXKEhRqBI8W7rU7kwT5EgyRZuSMVSFcZolOvKK9U=
24+
go.uber.org/dig v1.9.0/go.mod h1:X34SnWGr8Fyla9zQNO2GSO2D+TIuqB14OS8JhYocIyw=
2525
go.uber.org/goleak v0.10.0 h1:G3eWbSNIskeRqtsN/1uI5B+eP73y3JUuBsv9AZjehb4=
2626
go.uber.org/goleak v0.10.0/go.mod h1:VCZuO8V8mFPlL0F5J5GK1rtHV3DrFcQ1R8ryq7FK0aI=
2727
go.uber.org/multierr v1.4.0 h1:f3WCSC2KzAcBXGATIxAB1E2XuCpNU255wNKZ505qi3E=

inout.go

+11
Original file line numberDiff line numberDiff line change
@@ -251,4 +251,15 @@ type In struct{ dig.In }
251251
// value groups require parameter and result structs to use fields with
252252
// different types: if a group of constructors each returns type T, parameter
253253
// structs consuming the group must use a field of type []T.
254+
//
255+
// To provide multiple values for a group from a result struct, produce a
256+
// slice and use the `,flatten` option on the group tag. This indicates that
257+
// each element in the slice should be injected into the group individually.
258+
//
259+
// type IntResult struct {
260+
// fx.Out
261+
//
262+
// Handler []int `group:"server"` // Consume as [][]int
263+
// Handler []int `group:"server,flatten"` // Consume as []int
264+
// }
254265
type Out struct{ dig.Out }

populate_test.go

+2-2
Original file line numberDiff line numberDiff line change
@@ -208,7 +208,7 @@ func TestPopulateErrors(t *testing.T) {
208208
{
209209
msg: "container pointer without fx.In",
210210
opt: Populate(&containerNoIn{}),
211-
wantErr: "is not in the container",
211+
wantErr: "missing type: fx_test.containerNoIn",
212212
},
213213
{
214214
msg: "function",
@@ -218,7 +218,7 @@ func TestPopulateErrors(t *testing.T) {
218218
{
219219
msg: "function pointer",
220220
opt: Populate(&fn),
221-
wantErr: "is not in the container",
221+
wantErr: "missing type: func()",
222222
},
223223
{
224224
msg: "invalid last argument",

0 commit comments

Comments
 (0)