Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

[RSDK-9498] Search for meta.json File in Top Level of Online Module #4689

Merged
merged 50 commits into from
Jan 16, 2025

Conversation

bashar-515
Copy link
Member

Search for a registry module's meta.json file in the top level of the module and in the parent directory of its executable entry point.

Testing

Running the robot linked in the Jira ticket no longer outputs the WARN log listed. first_run.sh is successfully sourced and executed, and a dist.first_run_succeeded file is created.
Note: This has only been tested with a registry module, not a local module.

@viambot viambot added the safe to test This pull request is marked safe to test from a trusted zone label Jan 8, 2025
@viambot viambot added safe to test This pull request is marked safe to test from a trusted zone and removed safe to test This pull request is marked safe to test from a trusted zone labels Jan 8, 2025
config/module.go Outdated
@@ -282,7 +282,13 @@ func (m *Module) FirstRun(
logger.Debug("no first run script specified, skipping first run")
return nil
}
relFirstRunPath, err := utils.SafeJoinDir(unpackedModDir, meta.FirstRun)
var firstRunTopLevelDir string
Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Before, meta.FirstRun was used relative to the directory specified by unpackedModDir. In the case that the module has a type of registry, this would just be the parent directory of the module's entry point executable - not necessarily the top level of its directory. Here, I've changed it so that if moduleWorkingDirectory is not the empty string (which is equivalent to saying that the module's type is registry and the VIAM_ROOT_MODULE environment variable has been set), then the module's top level directory is used to find and execute the specified "first run" script.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

why not just have moduleWorkingDirectory get returned all the time?

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Good point, this will clean up the code a bit.

config/module.go Outdated
return meta, "", err
}

if m.Type == ModuleTypeRegistry {
Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Before, this function would automatically return an error and no meta information when m.Type == ModuleTypeRegistry

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I feel like this case should be at the top then, and also only check in the moduleWorkingDirectory.
Since if m.Type == ModuleTypeRegistry then this will go through the !localNonTarball branch automatically

@bashar-515 bashar-515 requested a review from cheukt January 8, 2025 01:08
@bashar-515 bashar-515 marked this pull request as ready for review January 8, 2025 01:08
Copy link
Member

@cheukt cheukt left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

have some comments, and we should have at least 1 test for this

config/module.go Outdated
return nil, err
}
_, err = os.Stat(metaPath)
if err == nil {
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

have the conditional be err != nil so we can early return

config/module.go Outdated
@@ -381,38 +387,75 @@ func (m *Module) FirstRun(
// 1. if there is a meta.json in the exe dir, use that, except in local non-tarball case.
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

the comment should be rewritten to reflect the current state of things.

config/module.go Outdated Show resolved Hide resolved
@@ -381,38 +387,75 @@ func (m *Module) FirstRun(
// 1. if there is a meta.json in the exe dir, use that, except in local non-tarball case.
// 2. if this is a local tarball and there's a meta.json next to the tarball, use that.
// Note: the working directory must be the unpacked tarball directory or local exec directory.
func (m Module) getJSONManifest(unpackedModDir string) (*JSONManifest, error) {
func (m Module) getJSONManifest(unpackedModDir string, env map[string]string) (*JSONManifest, string, error) {
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

would like to see a test for this

@@ -267,7 +267,7 @@ func (m *Module) FirstRun(

// Load the module's meta.json. If it doesn't exist DEBUG log and exit quietly.
// For all other errors WARN log and exit.
meta, err := m.getJSONManifest(unpackedModDir)
meta, moduleWorkingDirectory, err := m.getJSONManifest(unpackedModDir, env)
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

want to see an integration test around this whole function

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Do you mean around getJSONManifest()? or around the FirstRun() which calls getJSONManifest()?

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

around FirstRun() at the very least, would recommend trying to unit test getJSONManifest() as well

config/module.go Outdated
return meta, "", err
}

if m.Type == ModuleTypeRegistry {
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I feel like this case should be at the top then, and also only check in the moduleWorkingDirectory.
Since if m.Type == ModuleTypeRegistry then this will go through the !localNonTarball branch automatically

config/module.go Outdated
@@ -282,7 +282,13 @@ func (m *Module) FirstRun(
logger.Debug("no first run script specified, skipping first run")
return nil
}
relFirstRunPath, err := utils.SafeJoinDir(unpackedModDir, meta.FirstRun)
var firstRunTopLevelDir string
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

why not just have moduleWorkingDirectory get returned all the time?

config/module.go Outdated
if !ok {
return nil, "", errors.Errorf("VIAM_MODULE_ROOT not set. Failed to find meta.json in executable directory %s", unpackedModDir)
}
return nil, "", errors.Errorf("failed to find meta.json. Searched in executable directory %s and path set by VIAM_MODULE_ROOT %s",
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

can we ever reach this line?

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think yes. As this function is currently written, we only search for the meta.json file in two locations 1) the top level directory of the online module and 2) in the same sub-directory of the module's executable. If the meta.json lives anywhere else, it won't be found - even if the VIAM_MODULE_ROOT env. variable is set. I could be wrong re: how modules are unpacked/sourced/retrieved.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I ask because I'm pretty sure I saw both terminal ok and !ok cases at the time

@viambot viambot added safe to test This pull request is marked safe to test from a trusted zone and removed safe to test This pull request is marked safe to test from a trusted zone labels Jan 8, 2025
@cheukt cheukt requested a review from benjirewis January 8, 2025 18:39
@cheukt
Copy link
Member

cheukt commented Jan 8, 2025

also adding @benjirewis

@viambot viambot added safe to test This pull request is marked safe to test from a trusted zone and removed safe to test This pull request is marked safe to test from a trusted zone labels Jan 8, 2025
@viambot viambot added safe to test This pull request is marked safe to test from a trusted zone and removed safe to test This pull request is marked safe to test from a trusted zone labels Jan 13, 2025
Copy link
Member

@cheukt cheukt left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

these tests look great! thanks for adding them, some small comments and a general request to add documentation in places where you're doing something unexpected

@@ -102,6 +106,247 @@ func TestSyntheticModule(t *testing.T) {
})
}

func TestFirstRun(t *testing.T) {
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
func TestFirstRun(t *testing.T) {
func TestRegistryModuleFirstRun(t *testing.T) {

this test is specifically for registry modules, right?

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

yep


tmp := t.TempDir()
exePath := filepath.Join(tmp, "whatever.sh")
m.ExePath = exePath
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

can you also add a case where the exepath is one level deep and the meta json is in the same directory as exepath? and also if meta.json is in tmp

ctx := context.Background()
localPackagesDir := ""
dataDir := ""
env := map[string]string{"VIAM_MODULE_ROOT": tmp}
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

leave a comment explaining why this is set

test.That(t, err.Error(), test.ShouldNotContainSubstring, topLevelMetaJSONFilepath)

// meta.json not found; top level module directory and unpacked module directories searched
env["VIAM_MODULE_ROOT"] = topLevelDir
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

leave a comment as to why we're setting this var here

@viambot viambot added safe to test This pull request is marked safe to test from a trusted zone and removed safe to test This pull request is marked safe to test from a trusted zone labels Jan 15, 2025
@viambot viambot added safe to test This pull request is marked safe to test from a trusted zone and removed safe to test This pull request is marked safe to test from a trusted zone labels Jan 15, 2025
@viambot viambot added safe to test This pull request is marked safe to test from a trusted zone and removed safe to test This pull request is marked safe to test from a trusted zone labels Jan 15, 2025
@viambot viambot added safe to test This pull request is marked safe to test from a trusted zone and removed safe to test This pull request is marked safe to test from a trusted zone labels Jan 15, 2025
@viambot viambot added safe to test This pull request is marked safe to test from a trusted zone and removed safe to test This pull request is marked safe to test from a trusted zone labels Jan 15, 2025
@viambot viambot added safe to test This pull request is marked safe to test from a trusted zone and removed safe to test This pull request is marked safe to test from a trusted zone labels Jan 15, 2025
@bashar-515 bashar-515 requested a review from cheukt January 15, 2025 22:00
Copy link
Member

@cheukt cheukt left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

LGTM

Copy link
Member

@benjirewis benjirewis left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Nice! Just some nits.

config/module.go Outdated
// getJSONManifest returns a loaded meta.json from one of two sources (in order of precedence):
// 1. if there is a meta.json in the exe dir, use that, except in local non-tarball case.
// 2. if this is a local tarball and there's a meta.json next to the tarball, use that.
// TODO(RSDK-9498): write test(s)
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Remove?


var registryTarballErr error

localNonTarball := m.Type == ModuleTypeLocal && !m.NeedsSyntheticPackage()
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

[nit] Realize you didn't write this, but I think the m.Type == ModuleTypeLocal is redundant with the NeedSyntheticPackage?

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think it actually is necessary to strictly define the case. The only way I've been able to wrap my head around it is by writing out the boolean expressions defining the case and using an online boolean expression simplifier (lol) to, well, simplify them. As its written, case 2 is entered when the following expression evaluates to true.
A) case 2 = !localNonTarball = !m.Type == ModuleTypeLocal && !m.NeedsSyntheticPackage() = !(L(!(LT))) which simplifies to !L+T. So, the case is supposed to be registry or tarball. Now if we do the same thing without m.Type == ModuleTypeLocal:
B) case 2 = !localNonTarball = !!m.NeedsSyntheticPackage() = !(!(LT)) = LT, which is only modules that are both local and tarballs.

@@ -102,6 +107,296 @@ func TestSyntheticModule(t *testing.T) {
})
}

func TestRegistryModuleFirstRun(t *testing.T) {
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Great tests thank you!

@viambot viambot added safe to test This pull request is marked safe to test from a trusted zone and removed safe to test This pull request is marked safe to test from a trusted zone labels Jan 16, 2025
@bashar-515 bashar-515 merged commit 59841c8 into viamrobotics:main Jan 16, 2025
17 checks passed
@bashar-515 bashar-515 deleted the RSDK-9498 branch January 16, 2025 21:30
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
safe to test This pull request is marked safe to test from a trusted zone
Projects
None yet
Development

Successfully merging this pull request may close these issues.

4 participants