2016-12-29 14:55:35 +02:00
|
|
|
package brew
|
|
|
|
|
|
|
|
import (
|
2017-03-26 20:30:21 +02:00
|
|
|
"bytes"
|
2017-12-25 23:58:07 +02:00
|
|
|
"flag"
|
2018-04-01 19:25:04 +02:00
|
|
|
"fmt"
|
2017-03-26 20:30:21 +02:00
|
|
|
"io/ioutil"
|
|
|
|
"os"
|
|
|
|
"path/filepath"
|
2016-12-29 14:55:35 +02:00
|
|
|
"testing"
|
2016-12-29 14:56:51 +02:00
|
|
|
|
2017-03-23 02:01:29 +02:00
|
|
|
"github.com/goreleaser/goreleaser/config"
|
2017-03-26 20:30:21 +02:00
|
|
|
"github.com/goreleaser/goreleaser/context"
|
2017-12-18 00:14:41 +02:00
|
|
|
"github.com/goreleaser/goreleaser/internal/artifact"
|
2017-08-20 21:50:34 +02:00
|
|
|
"github.com/goreleaser/goreleaser/internal/testlib"
|
2017-01-06 15:13:17 +02:00
|
|
|
"github.com/stretchr/testify/assert"
|
2016-12-29 14:55:35 +02:00
|
|
|
)
|
|
|
|
|
2017-12-25 23:58:07 +02:00
|
|
|
var update = flag.Bool("update", false, "update .golden files")
|
|
|
|
|
2017-03-26 01:43:42 +02:00
|
|
|
func TestDescription(t *testing.T) {
|
2017-12-02 23:53:19 +02:00
|
|
|
assert.NotEmpty(t, Pipe{}.String())
|
2017-03-26 01:43:42 +02:00
|
|
|
}
|
|
|
|
|
2016-12-29 14:55:35 +02:00
|
|
|
func TestNameWithDash(t *testing.T) {
|
|
|
|
assert.Equal(t, formulaNameFor("some-binary"), "SomeBinary")
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestNameWithUnderline(t *testing.T) {
|
|
|
|
assert.Equal(t, formulaNameFor("some_binary"), "SomeBinary")
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestSimpleName(t *testing.T) {
|
|
|
|
assert.Equal(t, formulaNameFor("binary"), "Binary")
|
|
|
|
}
|
2016-12-29 17:14:52 +02:00
|
|
|
|
2016-12-31 21:02:25 +02:00
|
|
|
var defaultTemplateData = templateData{
|
2017-09-27 00:00:24 +02:00
|
|
|
Desc: "Some desc",
|
|
|
|
Homepage: "https://google.com",
|
|
|
|
DownloadURL: "https://github.com",
|
|
|
|
Name: "Test",
|
2017-03-23 02:29:14 +02:00
|
|
|
Repo: config.Repo{
|
|
|
|
Owner: "caarlos0",
|
|
|
|
Name: "test",
|
|
|
|
},
|
|
|
|
Tag: "v0.1.3",
|
|
|
|
Version: "0.1.3",
|
2018-04-05 23:11:31 +02:00
|
|
|
Caveats: []string{},
|
2017-07-02 01:59:16 +02:00
|
|
|
File: "test_Darwin_x86_64.tar.gz",
|
2017-03-23 02:29:14 +02:00
|
|
|
SHA256: "1633f61598ab0791e213135923624eb342196b3494909c91899bcd0560f84c68",
|
2016-12-31 21:02:25 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func assertDefaultTemplateData(t *testing.T, formulae string) {
|
2017-09-27 00:24:49 +02:00
|
|
|
assert.Contains(t, formulae, "class Test < Formula")
|
2017-09-27 13:43:52 +02:00
|
|
|
assert.Contains(t, formulae, `homepage "https://google.com"`)
|
|
|
|
assert.Contains(t, formulae, `url "https://github.com/caarlos0/test/releases/download/v0.1.3/test_Darwin_x86_64.tar.gz"`)
|
|
|
|
assert.Contains(t, formulae, `sha256 "1633f61598ab0791e213135923624eb342196b3494909c91899bcd0560f84c68"`)
|
|
|
|
assert.Contains(t, formulae, `version "0.1.3"`)
|
2016-12-31 21:02:25 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func TestFullFormulae(t *testing.T) {
|
|
|
|
data := defaultTemplateData
|
2018-03-28 16:14:27 +02:00
|
|
|
data.Caveats = []string{"Here are some caveats"}
|
2017-07-16 20:06:32 +02:00
|
|
|
data.Dependencies = []string{"gtk+"}
|
|
|
|
data.Conflicts = []string{"svn"}
|
2017-03-09 19:24:49 +02:00
|
|
|
data.Plist = "it works"
|
|
|
|
data.Install = []string{"custom install script", "another install script"}
|
2017-07-16 20:06:32 +02:00
|
|
|
data.Tests = []string{`system "#{bin}/foo -version"`}
|
2017-01-19 11:04:41 +02:00
|
|
|
out, err := doBuildFormula(data)
|
2017-09-27 00:24:49 +02:00
|
|
|
assert.NoError(t, err)
|
2016-12-31 21:02:25 +02:00
|
|
|
formulae := out.String()
|
2017-07-16 20:06:32 +02:00
|
|
|
|
2017-12-26 00:09:55 +02:00
|
|
|
var golden = "testdata/test.rb.golden"
|
|
|
|
if *update {
|
|
|
|
ioutil.WriteFile(golden, []byte(formulae), 0655)
|
|
|
|
}
|
|
|
|
bts, err := ioutil.ReadFile(golden)
|
2017-09-27 00:24:49 +02:00
|
|
|
assert.NoError(t, err)
|
|
|
|
assert.Equal(t, string(bts), formulae)
|
2016-12-31 21:02:25 +02:00
|
|
|
}
|
|
|
|
|
2017-02-01 19:40:27 +02:00
|
|
|
func TestFormulaeSimple(t *testing.T) {
|
2017-01-19 11:04:41 +02:00
|
|
|
out, err := doBuildFormula(defaultTemplateData)
|
2017-09-27 00:24:49 +02:00
|
|
|
assert.NoError(t, err)
|
2016-12-31 21:02:25 +02:00
|
|
|
formulae := out.String()
|
|
|
|
assertDefaultTemplateData(t, formulae)
|
2017-09-27 00:24:49 +02:00
|
|
|
assert.NotContains(t, formulae, "def caveats")
|
|
|
|
assert.NotContains(t, formulae, "depends_on")
|
|
|
|
assert.NotContains(t, formulae, "def plist;")
|
2016-12-29 17:14:52 +02:00
|
|
|
}
|
2017-03-26 20:30:21 +02:00
|
|
|
|
2017-07-16 21:01:20 +02:00
|
|
|
func TestSplit(t *testing.T) {
|
|
|
|
var parts = split("system \"true\"\nsystem \"#{bin}/foo -h\"")
|
2017-09-27 00:24:49 +02:00
|
|
|
assert.Equal(t, []string{"system \"true\"", "system \"#{bin}/foo -h\""}, parts)
|
2018-04-05 23:11:31 +02:00
|
|
|
parts = split("")
|
|
|
|
assert.Equal(t, []string{}, parts)
|
|
|
|
parts = split("\n ")
|
|
|
|
assert.Equal(t, []string{}, parts)
|
2017-07-16 21:01:20 +02:00
|
|
|
}
|
2017-07-16 21:02:18 +02:00
|
|
|
|
2017-03-26 20:30:21 +02:00
|
|
|
func TestRunPipe(t *testing.T) {
|
2018-04-01 19:25:04 +02:00
|
|
|
for name, fn := range map[string]func(ctx *context.Context){
|
|
|
|
"default": func(ctx *context.Context) {},
|
|
|
|
"github_enterprise_url": func(ctx *context.Context) {
|
|
|
|
ctx.Config.GitHubURLs.Download = "http://github.example.org"
|
2017-07-16 21:01:20 +02:00
|
|
|
},
|
2018-04-01 19:25:04 +02:00
|
|
|
"custom_download_strategy": func(ctx *context.Context) {
|
|
|
|
ctx.Config.Brew.DownloadStrategy = "GitHubPrivateRepositoryReleaseDownloadStrategy"
|
|
|
|
},
|
2018-07-11 09:36:02 +02:00
|
|
|
"binary_overriden": func(ctx *context.Context) {
|
|
|
|
ctx.Config.Archive.Format = "binary"
|
|
|
|
ctx.Config.Archive.FormatOverrides = []config.FormatOverride{
|
|
|
|
{
|
|
|
|
Goos: "darwin",
|
|
|
|
Format: "zip",
|
|
|
|
},
|
|
|
|
}
|
|
|
|
},
|
2018-04-01 19:25:04 +02:00
|
|
|
} {
|
|
|
|
t.Run(name, func(t *testing.T) {
|
|
|
|
folder, err := ioutil.TempDir("", "goreleasertest")
|
|
|
|
assert.NoError(t, err)
|
|
|
|
var ctx = &context.Context{
|
|
|
|
Git: context.GitInfo{
|
|
|
|
CurrentTag: "v1.0.1",
|
2017-09-26 00:01:10 +02:00
|
|
|
},
|
2018-04-01 19:25:04 +02:00
|
|
|
Version: "1.0.1",
|
|
|
|
Artifacts: artifact.New(),
|
|
|
|
Config: config.Project{
|
|
|
|
Dist: folder,
|
|
|
|
ProjectName: name,
|
|
|
|
GitHubURLs: config.GitHubURLs{
|
|
|
|
Download: "https://github.com",
|
|
|
|
},
|
|
|
|
Archive: config.Archive{
|
|
|
|
Format: "tar.gz",
|
|
|
|
},
|
|
|
|
Release: config.Release{
|
|
|
|
GitHub: config.Repo{
|
|
|
|
Owner: "test",
|
|
|
|
Name: "test",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
Brew: config.Homebrew{
|
|
|
|
Name: name,
|
|
|
|
GitHub: config.Repo{
|
|
|
|
Owner: "test",
|
|
|
|
Name: "test",
|
|
|
|
},
|
|
|
|
Description: "A run pipe test formula",
|
|
|
|
Homepage: "https://github.com/goreleaser",
|
|
|
|
Caveats: "don't do this",
|
|
|
|
Test: "system \"true\"\nsystem \"#{bin}/foo -h\"",
|
|
|
|
Plist: `<xml>whatever</xml>`,
|
|
|
|
Dependencies: []string{"zsh", "bash"},
|
|
|
|
Conflicts: []string{"gtk+", "qt"},
|
|
|
|
Install: `bin.install "foo"`,
|
|
|
|
},
|
2017-03-26 20:30:21 +02:00
|
|
|
},
|
2018-04-01 19:25:04 +02:00
|
|
|
}
|
|
|
|
fn(ctx)
|
2018-07-11 09:43:26 +02:00
|
|
|
var format = getFormat(ctx)
|
|
|
|
var path = filepath.Join(folder, "bin."+format)
|
2018-04-01 19:25:04 +02:00
|
|
|
ctx.Artifacts.Add(artifact.Artifact{
|
2018-07-11 09:43:26 +02:00
|
|
|
Name: "bin." + format,
|
2018-04-01 19:25:04 +02:00
|
|
|
Path: path,
|
|
|
|
Goos: "darwin",
|
|
|
|
Goarch: "amd64",
|
|
|
|
Type: artifact.UploadableArchive,
|
|
|
|
})
|
|
|
|
|
|
|
|
_, err = os.Create(path)
|
|
|
|
assert.NoError(t, err)
|
|
|
|
client := &DummyClient{}
|
|
|
|
var distFile = filepath.Join(folder, name+".rb")
|
|
|
|
|
|
|
|
assert.NoError(t, doRun(ctx, client))
|
|
|
|
assert.True(t, client.CreatedFile)
|
|
|
|
var golden = fmt.Sprintf("testdata/%s.rb.golden", name)
|
|
|
|
if *update {
|
2018-07-11 09:43:26 +02:00
|
|
|
assert.NoError(t, ioutil.WriteFile(golden, []byte(client.Content), 0655))
|
2018-04-01 19:25:04 +02:00
|
|
|
}
|
|
|
|
bts, err := ioutil.ReadFile(golden)
|
|
|
|
assert.NoError(t, err)
|
|
|
|
assert.Equal(t, string(bts), client.Content)
|
|
|
|
|
|
|
|
distBts, err := ioutil.ReadFile(distFile)
|
|
|
|
assert.NoError(t, err)
|
|
|
|
assert.Equal(t, string(bts), string(distBts))
|
|
|
|
})
|
2017-03-26 20:30:21 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-04-21 17:02:48 +02:00
|
|
|
func TestRunPipeNoDarwin64Build(t *testing.T) {
|
2017-04-14 20:49:37 +02:00
|
|
|
var ctx = &context.Context{
|
|
|
|
Config: config.Project{
|
|
|
|
Archive: config.Archive{
|
|
|
|
Format: "tar.gz",
|
|
|
|
},
|
|
|
|
Brew: config.Homebrew{
|
|
|
|
GitHub: config.Repo{
|
|
|
|
Owner: "test",
|
|
|
|
Name: "test",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
client := &DummyClient{}
|
2017-09-27 00:24:49 +02:00
|
|
|
assert.Equal(t, ErrNoDarwin64Build, doRun(ctx, client))
|
|
|
|
assert.False(t, client.CreatedFile)
|
2017-04-14 20:49:37 +02:00
|
|
|
}
|
|
|
|
|
2017-12-18 00:14:41 +02:00
|
|
|
func TestRunPipeMultipleDarwin64Build(t *testing.T) {
|
|
|
|
var ctx = context.New(
|
|
|
|
config.Project{
|
|
|
|
Archive: config.Archive{
|
|
|
|
Format: "tar.gz",
|
|
|
|
},
|
|
|
|
Brew: config.Homebrew{
|
|
|
|
GitHub: config.Repo{
|
|
|
|
Owner: "test",
|
|
|
|
Name: "test",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
)
|
|
|
|
ctx.Artifacts.Add(artifact.Artifact{
|
|
|
|
Name: "bin1",
|
|
|
|
Path: "doesnt mather",
|
|
|
|
Goos: "darwin",
|
|
|
|
Goarch: "amd64",
|
|
|
|
Type: artifact.UploadableArchive,
|
|
|
|
})
|
|
|
|
ctx.Artifacts.Add(artifact.Artifact{
|
|
|
|
Name: "bin2",
|
|
|
|
Path: "doesnt mather",
|
|
|
|
Goos: "darwin",
|
|
|
|
Goarch: "amd64",
|
|
|
|
Type: artifact.UploadableArchive,
|
|
|
|
})
|
|
|
|
client := &DummyClient{}
|
|
|
|
assert.Equal(t, ErrTooManyDarwin64Builds, doRun(ctx, client))
|
|
|
|
assert.False(t, client.CreatedFile)
|
|
|
|
}
|
|
|
|
|
2017-04-21 17:02:48 +02:00
|
|
|
func TestRunPipeBrewNotSetup(t *testing.T) {
|
|
|
|
var ctx = &context.Context{
|
2018-02-24 22:59:08 +02:00
|
|
|
Config: config.Project{},
|
2017-04-21 17:02:48 +02:00
|
|
|
}
|
|
|
|
client := &DummyClient{}
|
2017-08-20 21:50:34 +02:00
|
|
|
testlib.AssertSkipped(t, doRun(ctx, client))
|
2017-09-27 00:24:49 +02:00
|
|
|
assert.False(t, client.CreatedFile)
|
2017-04-21 17:02:48 +02:00
|
|
|
}
|
|
|
|
|
2017-07-03 06:24:26 +02:00
|
|
|
func TestRunPipeBinaryRelease(t *testing.T) {
|
2017-12-18 00:14:41 +02:00
|
|
|
var ctx = context.New(
|
|
|
|
config.Project{
|
2017-07-03 06:24:26 +02:00
|
|
|
Archive: config.Archive{
|
|
|
|
Format: "binary",
|
|
|
|
},
|
|
|
|
Brew: config.Homebrew{
|
|
|
|
GitHub: config.Repo{
|
|
|
|
Owner: "test",
|
|
|
|
Name: "test",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
2017-12-18 00:14:41 +02:00
|
|
|
)
|
|
|
|
ctx.Artifacts.Add(artifact.Artifact{
|
|
|
|
Name: "bin",
|
|
|
|
Path: "doesnt mather",
|
|
|
|
Goos: "darwin",
|
|
|
|
Goarch: "amd64",
|
|
|
|
Type: artifact.Binary,
|
|
|
|
})
|
2017-04-14 20:49:37 +02:00
|
|
|
client := &DummyClient{}
|
2017-08-20 21:50:34 +02:00
|
|
|
testlib.AssertSkipped(t, doRun(ctx, client))
|
2017-09-27 00:24:49 +02:00
|
|
|
assert.False(t, client.CreatedFile)
|
2017-04-14 20:49:37 +02:00
|
|
|
}
|
|
|
|
|
2018-01-10 23:22:37 +02:00
|
|
|
func TestRunPipeNoUpload(t *testing.T) {
|
|
|
|
folder, err := ioutil.TempDir("", "goreleasertest")
|
|
|
|
assert.NoError(t, err)
|
|
|
|
var ctx = context.New(config.Project{
|
|
|
|
Dist: folder,
|
|
|
|
ProjectName: "foo",
|
|
|
|
Release: config.Release{},
|
|
|
|
Brew: config.Homebrew{
|
|
|
|
GitHub: config.Repo{
|
|
|
|
Owner: "test",
|
|
|
|
Name: "test",
|
2017-04-22 00:55:25 +02:00
|
|
|
},
|
2017-04-22 00:50:09 +02:00
|
|
|
},
|
2018-01-10 23:22:37 +02:00
|
|
|
})
|
|
|
|
var path = filepath.Join(folder, "whatever.tar.gz")
|
|
|
|
_, err = os.Create(path)
|
|
|
|
assert.NoError(t, err)
|
|
|
|
ctx.Artifacts.Add(artifact.Artifact{
|
|
|
|
Name: "bin",
|
|
|
|
Path: path,
|
|
|
|
Goos: "darwin",
|
|
|
|
Goarch: "amd64",
|
|
|
|
Type: artifact.UploadableArchive,
|
|
|
|
})
|
2017-04-22 00:50:09 +02:00
|
|
|
client := &DummyClient{}
|
2018-01-10 23:22:37 +02:00
|
|
|
|
|
|
|
var assertNoPublish = func(t *testing.T) {
|
|
|
|
testlib.AssertSkipped(t, doRun(ctx, client))
|
|
|
|
assert.False(t, client.CreatedFile)
|
|
|
|
}
|
|
|
|
t.Run("skip upload", func(tt *testing.T) {
|
2018-03-01 06:12:58 +02:00
|
|
|
ctx.Config.Release.Draft = false
|
2018-01-10 23:22:37 +02:00
|
|
|
ctx.Config.Brew.SkipUpload = true
|
2018-03-01 06:12:58 +02:00
|
|
|
ctx.SkipPublish = false
|
2018-01-10 23:22:37 +02:00
|
|
|
assertNoPublish(tt)
|
|
|
|
})
|
2018-03-01 06:12:58 +02:00
|
|
|
t.Run("skip publish", func(tt *testing.T) {
|
|
|
|
ctx.Config.Release.Draft = false
|
|
|
|
ctx.Config.Brew.SkipUpload = false
|
|
|
|
ctx.SkipPublish = true
|
2018-01-10 23:22:37 +02:00
|
|
|
assertNoPublish(tt)
|
|
|
|
})
|
|
|
|
t.Run("draft release", func(tt *testing.T) {
|
|
|
|
ctx.Config.Release.Draft = true
|
2018-03-01 06:12:58 +02:00
|
|
|
ctx.Config.Brew.SkipUpload = false
|
|
|
|
ctx.SkipPublish = false
|
2018-01-10 23:22:37 +02:00
|
|
|
assertNoPublish(tt)
|
|
|
|
})
|
2017-04-22 00:50:09 +02:00
|
|
|
}
|
|
|
|
|
2017-06-08 11:41:09 +02:00
|
|
|
func TestRunPipeFormatBinary(t *testing.T) {
|
2017-06-05 18:28:29 +02:00
|
|
|
var ctx = &context.Context{
|
|
|
|
Config: config.Project{
|
|
|
|
Archive: config.Archive{
|
2017-06-08 11:41:09 +02:00
|
|
|
Format: "binary",
|
2017-06-05 18:28:29 +02:00
|
|
|
},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
client := &DummyClient{}
|
2017-08-20 21:50:34 +02:00
|
|
|
testlib.AssertSkipped(t, doRun(ctx, client))
|
2017-09-27 00:24:49 +02:00
|
|
|
assert.False(t, client.CreatedFile)
|
2017-06-05 18:28:29 +02:00
|
|
|
}
|
|
|
|
|
2017-12-03 16:05:58 +02:00
|
|
|
func TestDefault(t *testing.T) {
|
|
|
|
_, back := testlib.Mktmp(t)
|
|
|
|
defer back()
|
|
|
|
|
|
|
|
var ctx = &context.Context{
|
|
|
|
Config: config.Project{
|
2018-03-08 00:21:01 +02:00
|
|
|
ProjectName: "myproject",
|
2017-12-03 16:05:58 +02:00
|
|
|
Builds: []config.Build{
|
|
|
|
{
|
|
|
|
Binary: "foo",
|
|
|
|
Goos: []string{"linux", "darwin"},
|
|
|
|
Goarch: []string{"386", "amd64"},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
Binary: "bar",
|
|
|
|
Goos: []string{"linux", "darwin"},
|
|
|
|
Goarch: []string{"386", "amd64"},
|
|
|
|
Ignore: []config.IgnoredBuild{
|
|
|
|
{Goos: "darwin", Goarch: "amd64"},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
Binary: "foobar",
|
|
|
|
Goos: []string{"linux"},
|
|
|
|
Goarch: []string{"amd64"},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
assert.NoError(t, Pipe{}.Default(ctx))
|
2018-03-08 00:21:01 +02:00
|
|
|
assert.Equal(t, ctx.Config.ProjectName, ctx.Config.Brew.Name)
|
2017-12-03 16:05:58 +02:00
|
|
|
assert.NotEmpty(t, ctx.Config.Brew.CommitAuthor.Name)
|
|
|
|
assert.NotEmpty(t, ctx.Config.Brew.CommitAuthor.Email)
|
|
|
|
assert.Equal(t, `bin.install "foo"`, ctx.Config.Brew.Install)
|
|
|
|
}
|
|
|
|
|
2017-03-26 20:30:21 +02:00
|
|
|
type DummyClient struct {
|
|
|
|
CreatedFile bool
|
2017-07-02 17:01:59 +02:00
|
|
|
Content string
|
2017-03-26 20:30:21 +02:00
|
|
|
}
|
|
|
|
|
2018-01-26 19:45:38 +02:00
|
|
|
func (client *DummyClient) CreateRelease(ctx *context.Context, body string) (releaseID int64, err error) {
|
2017-03-26 20:30:21 +02:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2018-03-10 19:13:00 +02:00
|
|
|
func (client *DummyClient) CreateFile(ctx *context.Context, commitAuthor config.CommitAuthor, repo config.Repo, content bytes.Buffer, path, msg string) (err error) {
|
2017-03-26 20:30:21 +02:00
|
|
|
client.CreatedFile = true
|
2017-07-02 17:01:59 +02:00
|
|
|
bts, _ := ioutil.ReadAll(&content)
|
|
|
|
client.Content = string(bts)
|
2017-03-26 20:30:21 +02:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2018-01-26 19:45:38 +02:00
|
|
|
func (client *DummyClient) Upload(ctx *context.Context, releaseID int64, name string, file *os.File) (err error) {
|
2017-03-26 20:30:21 +02:00
|
|
|
return
|
|
|
|
}
|