Skip to content

Commit

Permalink
Add possibility to define a custom folder for temporary files.
Browse files Browse the repository at this point in the history
  • Loading branch information
gsanchezgavier authored and alvarocabanas committed Feb 9, 2024
1 parent bb90381 commit a681c65
Show file tree
Hide file tree
Showing 13 changed files with 205 additions and 72 deletions.
29 changes: 29 additions & 0 deletions .github/workflows/test.yml
Original file line number Diff line number Diff line change
@@ -0,0 +1,29 @@
name: UnitTesting

on: [push, pull_request]

env:
GO111MODULE: off

jobs:

build:
name: Build
runs-on: ubuntu-latest
defaults:
run:
working-directory: src/github.com/newrelic/infra-integrations-sdk
steps:

- name: Set up Go
uses: actions/setup-go@v2

- name: Check out code into the Go module directory
uses: actions/checkout@v2
with:
path: src/github.com/newrelic/infra-integrations-sdk

- name: Test
env:
GOPATH: "${{ github.workspace }}"
run: env PATH="$PATH:$GOPATH/bin" make test
13 changes: 7 additions & 6 deletions args/args.go
Original file line number Diff line number Diff line change
Expand Up @@ -23,6 +23,7 @@ type DefaultArgumentList struct {
NriAddHostname bool `default:"false" help:"Add hostname attribute to the samples."`
NriCluster string `default:"" help:"Optional. Cluster name"`
NriService string `default:"" help:"Optional. Service name"`
TempDir string `default:"" help:"Optional. Integrations path to store temporal data (defaults to os.tempDir if left empty)."`
}

// All returns if all data should be published
Expand Down Expand Up @@ -81,12 +82,12 @@ func underscore(s string) string {
// fields it defines. Each of the fields in the struct can define their defaults
// and help string by using tags:
//
// type Arguments struct {
// DefaultArgumentList
// Argument1 bool `default:"false" help:"This is the help we will print"`
// Argument2 int `default:"1" help:"This is the help we will print"`
// Argument3 string `default:"value" help:"This is the help we will print"`
// }
// type Arguments struct {
// DefaultArgumentList
// Argument1 bool `default:"false" help:"This is the help we will print"`
// Argument2 int `default:"1" help:"This is the help we will print"`
// Argument3 string `default:"value" help:"This is the help we will print"`
// }
//
// The fields in the struct will be populated with the values set either from
// the command line or from environment variables.
Expand Down
4 changes: 2 additions & 2 deletions data/metric/metrics_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -130,7 +130,7 @@ func TestSet_SetMetricsRatesAndDeltas(t *testing.T) {
}

for _, tc := range testCases {
t.Run(string(tc.sourceType), func(t *testing.T) {
t.Run(tc.sourceType.String(), func(t *testing.T) {

persist.SetNow(growingTime)

Expand All @@ -145,7 +145,7 @@ func TestSet_SetMetricsRatesAndDeltas(t *testing.T) {

func TestSet_SetMetricPositivesThrowsOnNegativeValues(t *testing.T) {
for _, sourceType := range []SourceType{PDELTA, PRATE} {
t.Run(string(sourceType), func(t *testing.T) {
t.Run(sourceType.String(), func(t *testing.T) {
persist.SetNow(growingTime)
ms := NewSet(
"some-event-type",
Expand Down
2 changes: 1 addition & 1 deletion data/metric/source_type_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -25,7 +25,7 @@ func TestSourceType_Positive(t *testing.T) {
}

for _, tc := range testCases {
t.Run(string(tc.sourceType), func(t *testing.T) {
t.Run(tc.sourceType.String(), func(t *testing.T) {
assert.Equal(t, tc.isPositive, tc.sourceType.IsPositive())
})
}
Expand Down
1 change: 1 addition & 0 deletions docs/toolset/args.md
Original file line number Diff line number Diff line change
Expand Up @@ -34,6 +34,7 @@ consider. All of them are `bool` and default to `false`. They are described foll
* `NriAddHostname`: if true, agent will decorate all the metrics with the `hostname`.
* `NriCluster`: if any value is provided, all the metrics will be decorated with `clusterName: value`.
* `NriService`: if any value is provided, all the metrics will be decorated with `serviceName: value`.
* `TempDir`: Integrations path to store temporal data (it defaults to `os.TempDir()` if it's left empty).

An example of

Expand Down
2 changes: 1 addition & 1 deletion integration/integration.go
Original file line number Diff line number Diff line change
Expand Up @@ -100,7 +100,7 @@ func New(name, version string, opts ...Option) (i *Integration, err error) {
}

if i.storer == nil {
storePath, err := persist.NewStorePath(i.Name, i.CreateUniqueID(), i.logger, persist.DefaultTTL)
storePath, err := persist.NewStorePath(i.Name, i.CreateUniqueID(), defaultArgs.TempDir, i.logger, persist.DefaultTTL)
if err != nil {
return nil, fmt.Errorf("can't create temporary directory for store: %s", err)
}
Expand Down
4 changes: 2 additions & 2 deletions integration/integration_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -464,7 +464,7 @@ func TestIntegration_CreateUniqueID_Default(t *testing.T) {
i, err := New("testIntegration", "0.0.0", Args(&al))
assert.NoError(t, err)

assert.Equal(t, i.CreateUniqueID(), "3071eb6863e28435e6c7e0c2bbe55ecd")
assert.Equal(t, i.CreateUniqueID(), "f2fbf79d935a14908cb886e98cff0879")
}

func TestIntegration_CreateUniqueID_EnvironmentVar(t *testing.T) {
Expand All @@ -481,7 +481,7 @@ func TestIntegration_CreateUniqueID_EnvironmentVar(t *testing.T) {
i, err := New("testIntegration", "0.0.0", Args(&al))
assert.NoError(t, err)

assert.Equal(t, i.CreateUniqueID(), "2d998100982b7de9b4e446c85c3bed78")
assert.Equal(t, i.CreateUniqueID(), "3d3c4d31fdec9ccd1250c9d080ca514f")
}

type testWriter struct {
Expand Down
2 changes: 1 addition & 1 deletion integration/options_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -72,7 +72,7 @@ func TestItStoresOnDiskByDefault(t *testing.T) {
assert.NoError(t, i.Publish())

// assert data has been flushed to disk
storePath, err := persist.NewStorePath(i.Name, i.CreateUniqueID(), i.logger, persist.DefaultTTL)
storePath, err := persist.NewStorePath(i.Name, i.CreateUniqueID(), "", i.logger, persist.DefaultTTL)
assert.NoError(t, err)

c, err := persist.NewFileStore(storePath.GetFilePath(), log.NewStdErr(true), persist.DefaultTTL)
Expand Down
6 changes: 2 additions & 4 deletions jmx/jmx_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -123,8 +123,6 @@ func TestQuery_WithSSL(t *testing.T) {
}

func TestOpen_WithNrjmx(t *testing.T) {
defer Close()

aux := os.Getenv("NR_JMX_TOOL")
require.NoError(t, os.Unsetenv("NR_JMX_TOOL"))

Expand Down Expand Up @@ -242,14 +240,14 @@ func Test_DefaultPath_IsCorrectForOs(t *testing.T) {
}

func TestHostName(t *testing.T) {
defer Close()
cmd = nil
host := "a-host"
assert.NoError(t, OpenNoAuth(host, ""))
assert.Equal(t, host, HostName())
}

func TestPort(t *testing.T) {
defer Close()
cmd = nil
port := "6666"
assert.NoError(t, OpenNoAuth("", port))
assert.Equal(t, port, Port())
Expand Down
4 changes: 2 additions & 2 deletions persist/store_path.go
Original file line number Diff line number Diff line change
Expand Up @@ -28,7 +28,7 @@ type storePath struct {
}

// NewStorePath create a new instance of StorePath
func NewStorePath(integrationName, integrationID string, ilog log.Logger, ttl time.Duration) (StorePath, error) {
func NewStorePath(integrationName, integrationID, customTempDir string, ilog log.Logger, ttl time.Duration) (StorePath, error) {
if integrationName == "" {
return nil, fmt.Errorf("integration name not specified")
}
Expand All @@ -42,7 +42,7 @@ func NewStorePath(integrationName, integrationID string, ilog log.Logger, ttl ti
}

return &storePath{
dir: tmpIntegrationDir(),
dir: tmpIntegrationDir(customTempDir),
integrationName: integrationName,
integrationID: integrationID,
ilog: ilog,
Expand Down
35 changes: 27 additions & 8 deletions persist/store_path_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -15,7 +15,7 @@ func setupTestCase(t *testing.T) func(t *testing.T) {
t.Log("setup test case")

assert.NoError(t, os.RemoveAll(filepath.Join(os.TempDir(), integrationsDir)))
tmpDir = tmpIntegrationDir()
tmpDir = tmpIntegrationDir("")

files := []struct {
name string
Expand Down Expand Up @@ -61,7 +61,7 @@ func TestStorePath_CleanOldFiles(t *testing.T) {
defer tearDownFn(t)

// WHEN new store file is generated
newPath, err := NewStorePath("com.newrelic.fake", "c", log.Discard, 1*time.Minute)
newPath, err := NewStorePath("com.newrelic.fake", "c", "", log.Discard, 1*time.Minute)
assert.NoError(t, err)

// THEN only old files with different integration ID are removed
Expand All @@ -79,20 +79,39 @@ func TestStorePath_CleanOldFiles(t *testing.T) {
}

func TestStorePath_GetFilePath(t *testing.T) {
storeFile, err := NewStorePath("com.newrelic.fake", "c", log.Discard, 1*time.Minute)
assert.NoError(t, err)
cases := []struct {
tempDir string
expected string
}{
{
tempDir: "",
expected: filepath.Join(tmpIntegrationDir(""), "com.newrelic.fake-c.json"),
},
{
tempDir: "",
expected: filepath.Join(tmpIntegrationDir(os.TempDir()), "com.newrelic.fake-c.json"),
},
{
tempDir: "custom-tmp",
expected: filepath.Join(tmpIntegrationDir("custom-tmp"), "com.newrelic.fake-c.json"),
},
}

expected := filepath.Join(tmpIntegrationDir(), "com.newrelic.fake-c.json")
assert.Equal(t, expected, storeFile.GetFilePath())
for _, tt := range cases {
storeFile, err := NewStorePath("com.newrelic.fake", "c", tt.tempDir, log.Discard, 1*time.Minute)
assert.NoError(t, err)

assert.Equal(t, tt.expected, storeFile.GetFilePath())
}
}

func TestStorePath_glob(t *testing.T) {
storeFile, err := NewStorePath("com.newrelic.fake", "c", log.Discard, 1*time.Minute)
storeFile, err := NewStorePath("com.newrelic.fake", "c", "", log.Discard, 1*time.Minute)
assert.NoError(t, err)

tmp, ok := storeFile.(*storePath)
assert.True(t, ok)

expected := filepath.Join(tmpIntegrationDir(), "com.newrelic.fake-*.json")
expected := filepath.Join(tmpIntegrationDir(""), "com.newrelic.fake-*.json")
assert.Equal(t, expected, tmp.glob())
}
56 changes: 48 additions & 8 deletions persist/storer.go
Original file line number Diff line number Diff line change
Expand Up @@ -23,10 +23,8 @@ const (
integrationsDir = "nr-integrations"
)

var (
// ErrNotFound defines an error that will be returned when trying to access a storage entry that can't be found
ErrNotFound = errors.New("key not found")
)
// ErrNotFound defines an error that will be returned when trying to access a storage entry that can't be found
var ErrNotFound = errors.New("key not found")

var now = time.Now

Expand Down Expand Up @@ -67,6 +65,7 @@ type fileStore struct {
inMemoryStore
path string
ilog log.Logger
ttl time.Duration
}

// SetNow forces a different "current time" for the Storer.
Expand All @@ -78,17 +77,21 @@ func SetNow(newNow func() time.Time) {
// DefaultPath returns a default folder/filename dir to a Storer for an integration from the given name. The name of
// the file will be the name of the integration with the .json extension.
func DefaultPath(integrationName string) string {
dir := tmpIntegrationDir()
dir := tmpIntegrationDir("")
file := filepath.Join(dir, integrationName+".json")

return file
}

func tmpIntegrationDir() string {
dir := filepath.Join(os.TempDir(), integrationsDir)
func tmpIntegrationDir(tempDir string) string {
if tempDir == "" {
tempDir = os.TempDir()
}

dir := filepath.Join(tempDir, integrationsDir)
// Create integrations Storer directory
if os.MkdirAll(dir, dirFilePerm) != nil {
dir = os.TempDir()
dir = tempDir
}
return dir
}
Expand All @@ -110,6 +113,7 @@ func NewFileStore(storagePath string, ilog log.Logger, ttl time.Duration) (Store
path: storagePath,
ilog: ilog,
inMemoryStore: *ms,
ttl: ttl,
}

if stat, err := os.Stat(storagePath); err == nil {
Expand All @@ -122,6 +126,7 @@ func NewFileStore(storagePath string, ilog log.Logger, ttl time.Duration) (Store
if err != nil {
ilog.Debugf(err.Error())
}

} else if os.IsNotExist(err) {
folder := path.Dir(storagePath)
err := os.MkdirAll(folder, dirFilePerm)
Expand All @@ -142,13 +147,20 @@ func (j *fileStore) Save() error {
return err
}

// Removes any entry that has not been updated duiring the integration cycle and with
// an expired timestamp.
if err := j.deleteOldEntries(); err != nil {
j.ilog.Debugf("Error deleting old entries in store file %q: %w. ", j.path, err)
}

j.locker.Lock()
defer j.locker.Unlock()

bytes, err := json.Marshal(j)
if err != nil {
return err
}

return ioutil.WriteFile(j.path, bytes, filePerm)
}

Expand Down Expand Up @@ -238,6 +250,34 @@ func (j *fileStore) loadFromDisk() error {
return nil
}

// deleteOldEntries traverse the stored data removing entires with timestamp greater than ttl.
// There is an implicit filter by integration for the removed entires since each storer file
// contains only the metrics for a specifc integration instance.
func (j *fileStore) deleteOldEntries() error {
j.locker.Lock()
defer j.locker.Unlock()

// Just decode Timestamp for performance reasons.
var entry struct {
Timestamp int64
}

nowTime := now()

for key, val := range j.Data {
if err := json.Unmarshal(val, &entry); err != nil {
return fmt.Errorf("decoding storer entry: %w", err)
}

if nowTime.Sub(time.Unix(entry.Timestamp, 0)) > j.ttl {
delete(j.Data, key)
}

}

return nil
}

// Delete removes the cached data for the given key. If the data does not exist, the system does not return
// any error.
func (j inMemoryStore) Delete(key string) error {
Expand Down
Loading

0 comments on commit a681c65

Please sign in to comment.