Skip to content

Commit 3050f5c

Browse files
committed
feat(ipns): refactored IPNS package with lean records
1 parent 6f82d77 commit 3050f5c

32 files changed

+1485
-1635
lines changed

CHANGELOG.md

Lines changed: 5 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -18,6 +18,11 @@ The following emojis are used to highlight certain changes:
1818

1919
### Changed
2020

21+
* 🛠 The `ipns` package has been refactored. You should no longer use the direct Protobuf
22+
version of the IPNS Record. Instead, we have a shiny new `ipns.Record` type that wraps
23+
all the required functionality to work the best as possible with IPNS v2 Records. Please
24+
check the [documentation](https://pkg.go.dev/github.com/ipfs/boxo/ipns) for more information.
25+
2126
### Removed
2227

2328
### Fixed

coreiface/options/name.go

Lines changed: 14 additions & 6 deletions
Original file line numberDiff line numberDiff line change
@@ -11,12 +11,11 @@ const (
1111
)
1212

1313
type NamePublishSettings struct {
14-
ValidTime time.Duration
15-
Key string
16-
17-
TTL *time.Duration
18-
19-
AllowOffline bool
14+
ValidTime time.Duration
15+
Key string
16+
TTL *time.Duration
17+
CompatibleWithV1 bool
18+
AllowOffline bool
2019
}
2120

2221
type NameResolveSettings struct {
@@ -104,6 +103,15 @@ func (nameOpts) TTL(ttl time.Duration) NamePublishOption {
104103
}
105104
}
106105

106+
// CompatibleWithV1 is an option for [Name.Publish] which specifies if the
107+
// created record should be backwards compatible with V1 IPNS Records.
108+
func (nameOpts) CompatibleWithV1(compatible bool) NamePublishOption {
109+
return func(settings *NamePublishSettings) error {
110+
settings.CompatibleWithV1 = compatible
111+
return nil
112+
}
113+
}
114+
107115
// Cache is an option for Name.Resolve which specifies if cache should be used.
108116
// Default value is true
109117
func (nameOpts) Cache(cache bool) NameResolveOption {

coreiface/options/namesys/opts.go

Lines changed: 10 additions & 2 deletions
Original file line numberDiff line numberDiff line change
@@ -84,8 +84,9 @@ func ProcessOpts(opts []ResolveOpt) ResolveOpts {
8484

8585
// PublishOptions specifies options for publishing an IPNS record.
8686
type PublishOptions struct {
87-
EOL time.Time
88-
TTL time.Duration
87+
EOL time.Time
88+
TTL time.Duration
89+
CompatibleWithV1 bool
8990
}
9091

9192
// DefaultPublishOptions returns the default options for publishing an IPNS record.
@@ -113,6 +114,13 @@ func PublishWithTTL(ttl time.Duration) PublishOption {
113114
}
114115
}
115116

117+
// PublishCompatibleWithV1 sets compatibility with IPNS Records V1.
118+
func PublishCompatibleWithV1(compatible bool) PublishOption {
119+
return func(o *PublishOptions) {
120+
o.CompatibleWithV1 = compatible
121+
}
122+
}
123+
116124
// ProcessPublishOptions converts an array of PublishOpt into a PublishOpts object.
117125
func ProcessPublishOptions(opts []PublishOption) PublishOptions {
118126
rsopts := DefaultPublishOptions()

coreiface/tests/routing.go

Lines changed: 7 additions & 11 deletions
Original file line numberDiff line numberDiff line change
@@ -5,10 +5,10 @@ import (
55
"testing"
66
"time"
77

8-
"github.com/gogo/protobuf/proto"
98
iface "github.com/ipfs/boxo/coreiface"
109
"github.com/ipfs/boxo/coreiface/options"
11-
ipns_pb "github.com/ipfs/boxo/ipns/pb"
10+
"github.com/ipfs/boxo/ipns"
11+
"github.com/stretchr/testify/require"
1212
)
1313

1414
func (tp *TestSuite) TestRouting(t *testing.T) {
@@ -57,16 +57,12 @@ func (tp *TestSuite) TestRoutingGet(t *testing.T) {
5757
t.Fatal(err)
5858
}
5959

60-
// Checks if values match.
61-
var entry ipns_pb.IpnsEntry
62-
err = proto.Unmarshal(data, &entry)
63-
if err != nil {
64-
t.Fatal(err)
65-
}
60+
rec, err := ipns.UnmarshalRecord(data)
61+
require.NoError(t, err)
6662

67-
if string(entry.GetValue()) != ipnsEntry.Value().String() {
68-
t.Fatalf("routing key has wrong value, expected %s, got %s", ipnsEntry.Value().String(), string(entry.GetValue()))
69-
}
63+
val, err := rec.Value()
64+
require.NoError(t, err)
65+
require.Equal(t, ipnsEntry.Value().String(), val.String())
7066
}
7167

7268
func (tp *TestSuite) TestRoutingPut(t *testing.T) {

examples/gateway/proxy/routing.go

Lines changed: 8 additions & 14 deletions
Original file line numberDiff line numberDiff line change
@@ -7,10 +7,7 @@ import (
77
"net/http"
88
"strings"
99

10-
"github.com/gogo/protobuf/proto"
1110
"github.com/ipfs/boxo/ipns"
12-
ipns_pb "github.com/ipfs/boxo/ipns/pb"
13-
"github.com/libp2p/go-libp2p/core/peer"
1411
"github.com/libp2p/go-libp2p/core/routing"
1512
"go.opentelemetry.io/contrib/instrumentation/net/http/otelhttp"
1613
)
@@ -42,30 +39,28 @@ func (ps *proxyRouting) GetValue(ctx context.Context, k string, opts ...routing.
4239
return nil, routing.ErrNotSupported
4340
}
4441

45-
k = strings.TrimPrefix(k, "/ipns/")
46-
id, err := peer.IDFromBytes([]byte(k))
42+
name, err := ipns.NameFromRoutingKey([]byte(k))
4743
if err != nil {
4844
return nil, err
4945
}
5046

51-
return ps.fetch(ctx, id)
47+
return ps.fetch(ctx, name)
5248
}
5349

5450
func (ps *proxyRouting) SearchValue(ctx context.Context, k string, opts ...routing.Option) (<-chan []byte, error) {
5551
if !strings.HasPrefix(k, "/ipns/") {
5652
return nil, routing.ErrNotSupported
5753
}
5854

59-
k = strings.TrimPrefix(k, "/ipns/")
60-
id, err := peer.IDFromBytes([]byte(k))
55+
name, err := ipns.NameFromRoutingKey([]byte(k))
6156
if err != nil {
6257
return nil, err
6358
}
6459

6560
ch := make(chan []byte)
6661

6762
go func() {
68-
v, err := ps.fetch(ctx, id)
63+
v, err := ps.fetch(ctx, name)
6964
if err != nil {
7065
close(ch)
7166
} else {
@@ -77,8 +72,8 @@ func (ps *proxyRouting) SearchValue(ctx context.Context, k string, opts ...routi
7772
return ch, nil
7873
}
7974

80-
func (ps *proxyRouting) fetch(ctx context.Context, id peer.ID) ([]byte, error) {
81-
urlStr := fmt.Sprintf("%s/ipns/%s", ps.gatewayURL, peer.ToCid(id).String())
75+
func (ps *proxyRouting) fetch(ctx context.Context, name ipns.Name) ([]byte, error) {
76+
urlStr := fmt.Sprintf("%s/%s", ps.gatewayURL, name.String())
8277
req, err := http.NewRequestWithContext(ctx, http.MethodGet, urlStr, nil)
8378
if err != nil {
8479
return nil, err
@@ -104,13 +99,12 @@ func (ps *proxyRouting) fetch(ctx context.Context, id peer.ID) ([]byte, error) {
10499
return nil, err
105100
}
106101

107-
var entry ipns_pb.IpnsEntry
108-
err = proto.Unmarshal(rb, &entry)
102+
rec, err := ipns.UnmarshalRecord(rb)
109103
if err != nil {
110104
return nil, err
111105
}
112106

113-
err = ipns.ValidateWithPeerID(id, &entry)
107+
err = ipns.ValidateWithName(rec, name)
114108
if err != nil {
115109
return nil, err
116110
}

examples/go.mod

Lines changed: 4 additions & 4 deletions
Original file line numberDiff line numberDiff line change
@@ -3,8 +3,7 @@ module github.com/ipfs/boxo/examples
33
go 1.19
44

55
require (
6-
github.com/gogo/protobuf v1.3.2
7-
github.com/ipfs/boxo v0.7.1-0.20230323075409-f4a8dd6614df
6+
github.com/ipfs/boxo v0.8.0
87
github.com/ipfs/go-block-format v0.1.2
98
github.com/ipfs/go-cid v0.4.0
109
github.com/ipfs/go-datastore v0.6.0
@@ -45,6 +44,7 @@ require (
4544
github.com/go-logr/stdr v1.2.2 // indirect
4645
github.com/go-task/slim-sprig v0.0.0-20210107165309-348f09dbbbc0 // indirect
4746
github.com/godbus/dbus/v5 v5.1.0 // indirect
47+
github.com/gogo/protobuf v1.3.2 // indirect
4848
github.com/golang/mock v1.6.0 // indirect
4949
github.com/golang/protobuf v1.5.2 // indirect
5050
github.com/google/gopacket v1.1.19 // indirect
@@ -64,7 +64,6 @@ require (
6464
github.com/ipfs/go-ipld-cbor v0.0.6 // indirect
6565
github.com/ipfs/go-ipld-format v0.5.0 // indirect
6666
github.com/ipfs/go-ipld-legacy v0.2.1 // indirect
67-
github.com/ipfs/go-ipns v0.3.0 // indirect
6867
github.com/ipfs/go-log v1.0.5 // indirect
6968
github.com/ipfs/go-log/v2 v2.5.1 // indirect
7069
github.com/ipfs/go-metrics-interface v0.0.1 // indirect
@@ -82,7 +81,7 @@ require (
8281
github.com/libp2p/go-doh-resolver v0.4.0 // indirect
8382
github.com/libp2p/go-flow-metrics v0.1.0 // indirect
8483
github.com/libp2p/go-libp2p-asn-util v0.2.0 // indirect
85-
github.com/libp2p/go-libp2p-kad-dht v0.21.1 // indirect
84+
github.com/libp2p/go-libp2p-kad-dht v0.23.0 // indirect
8685
github.com/libp2p/go-libp2p-kbucket v0.5.0 // indirect
8786
github.com/libp2p/go-libp2p-record v0.2.0 // indirect
8887
github.com/libp2p/go-msgio v0.3.0 // indirect
@@ -160,6 +159,7 @@ require (
160159
golang.org/x/text v0.7.0 // indirect
161160
golang.org/x/tools v0.3.0 // indirect
162161
golang.org/x/xerrors v0.0.0-20220907171357-04be3eba64a2 // indirect
162+
gonum.org/v1/gonum v0.11.0 // indirect
163163
google.golang.org/genproto v0.0.0-20230110181048-76db0878b65f // indirect
164164
google.golang.org/grpc v1.53.0 // indirect
165165
google.golang.org/protobuf v1.28.1 // indirect

examples/go.sum

Lines changed: 4 additions & 4 deletions
Original file line numberDiff line numberDiff line change
@@ -312,8 +312,6 @@ github.com/ipfs/go-ipld-format v0.5.0 h1:WyEle9K96MSrvr47zZHKKcDxJ/vlpET6PSiQsAF
312312
github.com/ipfs/go-ipld-format v0.5.0/go.mod h1:ImdZqJQaEouMjCvqCe0ORUS+uoBmf7Hf+EO/jh+nk3M=
313313
github.com/ipfs/go-ipld-legacy v0.2.1 h1:mDFtrBpmU7b//LzLSypVrXsD8QxkEWxu5qVxN99/+tk=
314314
github.com/ipfs/go-ipld-legacy v0.2.1/go.mod h1:782MOUghNzMO2DER0FlBR94mllfdCJCkTtDtPM51otM=
315-
github.com/ipfs/go-ipns v0.3.0 h1:ai791nTgVo+zTuq2bLvEGmWP1M0A6kGTXUsgv/Yq67A=
316-
github.com/ipfs/go-ipns v0.3.0/go.mod h1:3cLT2rbvgPZGkHJoPO1YMJeh6LtkxopCkKFcio/wE24=
317315
github.com/ipfs/go-log v1.0.5 h1:2dOuUCB1Z7uoczMWgAyDck5JLb72zHzrMnGnCNNbvY8=
318316
github.com/ipfs/go-log v1.0.5/go.mod h1:j0b8ZoR+7+R99LD9jZ6+AJsrzkPbSXbZfGakb5JPtIo=
319317
github.com/ipfs/go-log/v2 v2.1.3/go.mod h1:/8d0SH3Su5Ooc31QlL1WysJhvyOTDCjcCZ9Axpmri6g=
@@ -393,8 +391,8 @@ github.com/libp2p/go-libp2p v0.26.3 h1:6g/psubqwdaBqNNoidbRKSTBEYgaOuKBhHl8Q5tO+
393391
github.com/libp2p/go-libp2p v0.26.3/go.mod h1:x75BN32YbwuY0Awm2Uix4d4KOz+/4piInkp4Wr3yOo8=
394392
github.com/libp2p/go-libp2p-asn-util v0.2.0 h1:rg3+Os8jbnO5DxkC7K/Utdi+DkY3q/d1/1q+8WeNAsw=
395393
github.com/libp2p/go-libp2p-asn-util v0.2.0/go.mod h1:WoaWxbHKBymSN41hWSq/lGKJEca7TNm58+gGJi2WsLI=
396-
github.com/libp2p/go-libp2p-kad-dht v0.21.1 h1:xpfp8/t9+X2ip1l8Umap1/UGNnJ3RHJgKGAEsnRAlTo=
397-
github.com/libp2p/go-libp2p-kad-dht v0.21.1/go.mod h1:Oy8wvbdjpB70eS5AaFaI68tOtrdo3KylTvXDjikxqFo=
394+
github.com/libp2p/go-libp2p-kad-dht v0.23.0 h1:sxE6LxLopp79eLeV695n7+c77V/Vn4AMF28AdM/XFqM=
395+
github.com/libp2p/go-libp2p-kad-dht v0.23.0/go.mod h1:oO5N308VT2msnQI6qi5M61wzPmJYg7Tr9e16m5n7uDU=
398396
github.com/libp2p/go-libp2p-kbucket v0.5.0 h1:g/7tVm8ACHDxH29BGrpsQlnNeu+6OF1A9bno/4/U1oA=
399397
github.com/libp2p/go-libp2p-kbucket v0.5.0/go.mod h1:zGzGCpQd78b5BNTDGHNDLaTt9aDK/A02xeZp9QeFC4U=
400398
github.com/libp2p/go-libp2p-record v0.2.0 h1:oiNUOCWno2BFuxt3my4i1frNrt7PerzB3queqa1NkQ0=
@@ -963,6 +961,8 @@ golang.org/x/xerrors v0.0.0-20191204190536-9bdfabe68543/go.mod h1:I/5z698sn9Ka8T
963961
golang.org/x/xerrors v0.0.0-20200804184101-5ec99f83aff1/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0=
964962
golang.org/x/xerrors v0.0.0-20220907171357-04be3eba64a2 h1:H2TDz8ibqkAF6YGhCdN3jS9O0/s90v0rJh3X/OLHEUk=
965963
golang.org/x/xerrors v0.0.0-20220907171357-04be3eba64a2/go.mod h1:K8+ghG5WaK9qNqU5K3HdILfMLy1f3aNYFI/wnl100a8=
964+
gonum.org/v1/gonum v0.11.0 h1:f1IJhK4Km5tBJmaiJXtk/PkL4cdVX6J+tGiM187uT5E=
965+
gonum.org/v1/gonum v0.11.0/go.mod h1:fSG4YDCxxUZQJ7rKsQrj0gMOg00Il0Z96/qMA4bVQhA=
966966
google.golang.org/api v0.0.0-20180910000450-7ca32eb868bf/go.mod h1:4mhQ8q/RsB7i+udVvVy5NUi08OU8ZlA0gRVgrF7VFY0=
967967
google.golang.org/api v0.0.0-20181030000543-1d582fd0359e/go.mod h1:4mhQ8q/RsB7i+udVvVy5NUi08OU8ZlA0gRVgrF7VFY0=
968968
google.golang.org/api v0.1.0/go.mod h1:UGEZY7KEX120AnNLIHFMKIo4obdJhkp2tPbaPlQx13Y=

gateway/handler_ipns_record.go

Lines changed: 3 additions & 4 deletions
Original file line numberDiff line numberDiff line change
@@ -49,7 +49,7 @@ func (i *handler) serveIpnsRecord(ctx context.Context, w http.ResponseWriter, r
4949
return false
5050
}
5151

52-
record, err := ipns.UnmarshalIpnsEntry(rawRecord)
52+
record, err := ipns.UnmarshalRecord(rawRecord)
5353
if err != nil {
5454
i.webError(w, r, err, http.StatusInternalServerError)
5555
return false
@@ -69,9 +69,8 @@ func (i *handler) serveIpnsRecord(ctx context.Context, w http.ResponseWriter, r
6969
return false
7070
}
7171

72-
if record.Ttl != nil {
73-
seconds := int(time.Duration(*record.Ttl).Seconds())
74-
w.Header().Set("Cache-Control", fmt.Sprintf("public, max-age=%d", seconds))
72+
if ttl, err := record.TTL(); err == nil {
73+
w.Header().Set("Cache-Control", fmt.Sprintf("public, max-age=%d", int(ttl.Seconds())))
7574
} else {
7675
w.Header().Set("Last-Modified", time.Now().UTC().Format(http.TimeFormat))
7776
}

go.mod

Lines changed: 2 additions & 2 deletions
Original file line numberDiff line numberDiff line change
@@ -38,7 +38,7 @@ require (
3838
github.com/libp2p/go-buffer-pool v0.1.0
3939
github.com/libp2p/go-doh-resolver v0.4.0
4040
github.com/libp2p/go-libp2p v0.26.3
41-
github.com/libp2p/go-libp2p-kad-dht v0.21.1
41+
github.com/libp2p/go-libp2p-kad-dht v0.23.0
4242
github.com/libp2p/go-libp2p-record v0.2.0
4343
github.com/libp2p/go-libp2p-routing-helpers v0.7.0
4444
github.com/libp2p/go-libp2p-testing v0.12.0
@@ -112,7 +112,6 @@ require (
112112
github.com/ipfs/go-ipfs-ds-help v1.1.0 // indirect
113113
github.com/ipfs/go-ipfs-pq v0.0.3 // indirect
114114
github.com/ipfs/go-ipfs-util v0.0.2 // indirect
115-
github.com/ipfs/go-ipns v0.3.0 // indirect
116115
github.com/ipfs/go-log v1.0.5 // indirect
117116
github.com/ipfs/go-unixfs v0.4.5 // indirect
118117
github.com/ipld/go-car/v2 v2.9.1-0.20230325062757-fff0e4397a3d // indirect
@@ -167,6 +166,7 @@ require (
167166
golang.org/x/text v0.7.0 // indirect
168167
golang.org/x/tools v0.3.0 // indirect
169168
golang.org/x/xerrors v0.0.0-20220907171357-04be3eba64a2 // indirect
169+
gonum.org/v1/gonum v0.11.0 // indirect
170170
google.golang.org/appengine v1.6.7 // indirect
171171
google.golang.org/genproto v0.0.0-20230110181048-76db0878b65f // indirect
172172
google.golang.org/grpc v1.53.0 // indirect

go.sum

Lines changed: 4 additions & 4 deletions
Original file line numberDiff line numberDiff line change
@@ -325,8 +325,6 @@ github.com/ipfs/go-ipld-format v0.5.0 h1:WyEle9K96MSrvr47zZHKKcDxJ/vlpET6PSiQsAF
325325
github.com/ipfs/go-ipld-format v0.5.0/go.mod h1:ImdZqJQaEouMjCvqCe0ORUS+uoBmf7Hf+EO/jh+nk3M=
326326
github.com/ipfs/go-ipld-legacy v0.2.1 h1:mDFtrBpmU7b//LzLSypVrXsD8QxkEWxu5qVxN99/+tk=
327327
github.com/ipfs/go-ipld-legacy v0.2.1/go.mod h1:782MOUghNzMO2DER0FlBR94mllfdCJCkTtDtPM51otM=
328-
github.com/ipfs/go-ipns v0.3.0 h1:ai791nTgVo+zTuq2bLvEGmWP1M0A6kGTXUsgv/Yq67A=
329-
github.com/ipfs/go-ipns v0.3.0/go.mod h1:3cLT2rbvgPZGkHJoPO1YMJeh6LtkxopCkKFcio/wE24=
330328
github.com/ipfs/go-log v0.0.1/go.mod h1:kL1d2/hzSpI0thNYjiKfjanbVNU+IIGA/WnNESY9leM=
331329
github.com/ipfs/go-log v1.0.5 h1:2dOuUCB1Z7uoczMWgAyDck5JLb72zHzrMnGnCNNbvY8=
332330
github.com/ipfs/go-log v1.0.5/go.mod h1:j0b8ZoR+7+R99LD9jZ6+AJsrzkPbSXbZfGakb5JPtIo=
@@ -413,8 +411,8 @@ github.com/libp2p/go-libp2p v0.26.3 h1:6g/psubqwdaBqNNoidbRKSTBEYgaOuKBhHl8Q5tO+
413411
github.com/libp2p/go-libp2p v0.26.3/go.mod h1:x75BN32YbwuY0Awm2Uix4d4KOz+/4piInkp4Wr3yOo8=
414412
github.com/libp2p/go-libp2p-asn-util v0.2.0 h1:rg3+Os8jbnO5DxkC7K/Utdi+DkY3q/d1/1q+8WeNAsw=
415413
github.com/libp2p/go-libp2p-asn-util v0.2.0/go.mod h1:WoaWxbHKBymSN41hWSq/lGKJEca7TNm58+gGJi2WsLI=
416-
github.com/libp2p/go-libp2p-kad-dht v0.21.1 h1:xpfp8/t9+X2ip1l8Umap1/UGNnJ3RHJgKGAEsnRAlTo=
417-
github.com/libp2p/go-libp2p-kad-dht v0.21.1/go.mod h1:Oy8wvbdjpB70eS5AaFaI68tOtrdo3KylTvXDjikxqFo=
414+
github.com/libp2p/go-libp2p-kad-dht v0.23.0 h1:sxE6LxLopp79eLeV695n7+c77V/Vn4AMF28AdM/XFqM=
415+
github.com/libp2p/go-libp2p-kad-dht v0.23.0/go.mod h1:oO5N308VT2msnQI6qi5M61wzPmJYg7Tr9e16m5n7uDU=
418416
github.com/libp2p/go-libp2p-kbucket v0.5.0 h1:g/7tVm8ACHDxH29BGrpsQlnNeu+6OF1A9bno/4/U1oA=
419417
github.com/libp2p/go-libp2p-kbucket v0.5.0/go.mod h1:zGzGCpQd78b5BNTDGHNDLaTt9aDK/A02xeZp9QeFC4U=
420418
github.com/libp2p/go-libp2p-record v0.2.0 h1:oiNUOCWno2BFuxt3my4i1frNrt7PerzB3queqa1NkQ0=
@@ -982,6 +980,8 @@ golang.org/x/xerrors v0.0.0-20191204190536-9bdfabe68543/go.mod h1:I/5z698sn9Ka8T
982980
golang.org/x/xerrors v0.0.0-20200804184101-5ec99f83aff1/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0=
983981
golang.org/x/xerrors v0.0.0-20220907171357-04be3eba64a2 h1:H2TDz8ibqkAF6YGhCdN3jS9O0/s90v0rJh3X/OLHEUk=
984982
golang.org/x/xerrors v0.0.0-20220907171357-04be3eba64a2/go.mod h1:K8+ghG5WaK9qNqU5K3HdILfMLy1f3aNYFI/wnl100a8=
983+
gonum.org/v1/gonum v0.11.0 h1:f1IJhK4Km5tBJmaiJXtk/PkL4cdVX6J+tGiM187uT5E=
984+
gonum.org/v1/gonum v0.11.0/go.mod h1:fSG4YDCxxUZQJ7rKsQrj0gMOg00Il0Z96/qMA4bVQhA=
985985
google.golang.org/api v0.0.0-20180910000450-7ca32eb868bf/go.mod h1:4mhQ8q/RsB7i+udVvVy5NUi08OU8ZlA0gRVgrF7VFY0=
986986
google.golang.org/api v0.0.0-20181030000543-1d582fd0359e/go.mod h1:4mhQ8q/RsB7i+udVvVy5NUi08OU8ZlA0gRVgrF7VFY0=
987987
google.golang.org/api v0.1.0/go.mod h1:UGEZY7KEX120AnNLIHFMKIo4obdJhkp2tPbaPlQx13Y=

0 commit comments

Comments
 (0)