From fee2d22d68109b86bd62c6a8efe186a1d5ce4170 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Fran=C3=A7ois=20Proulx?= Date: Fri, 21 Nov 2025 13:04:12 -0500 Subject: [PATCH 1/2] Add Google Gemini AI plugin Adds shell plugin for Google Gemini AI API, providing support for: - API key provisioning via GEMINI_API_KEY environment variable - Environment variable configuration - Google AI Studio integration - Automatic credential import from existing environment The plugin enables 1Password CLI integration with the gemini CLI tool, allowing secure management of Gemini API credentials. --- plugins/gemini/api_key.go | 40 +++++++++++++++++++++++++++++++++ plugins/gemini/api_key_test.go | 41 ++++++++++++++++++++++++++++++++++ plugins/gemini/gemini.go | 24 ++++++++++++++++++++ plugins/gemini/plugin.go | 22 ++++++++++++++++++ 4 files changed, 127 insertions(+) create mode 100644 plugins/gemini/api_key.go create mode 100644 plugins/gemini/api_key_test.go create mode 100644 plugins/gemini/gemini.go create mode 100644 plugins/gemini/plugin.go diff --git a/plugins/gemini/api_key.go b/plugins/gemini/api_key.go new file mode 100644 index 00000000..f5100e25 --- /dev/null +++ b/plugins/gemini/api_key.go @@ -0,0 +1,40 @@ +package gemini + +import ( + "github.com/1Password/shell-plugins/sdk" + "github.com/1Password/shell-plugins/sdk/importer" + "github.com/1Password/shell-plugins/sdk/provision" + "github.com/1Password/shell-plugins/sdk/schema" + "github.com/1Password/shell-plugins/sdk/schema/credname" + "github.com/1Password/shell-plugins/sdk/schema/fieldname" +) + +func APIKey() schema.CredentialType { + return schema.CredentialType{ + Name: credname.APIKey, + DocsURL: sdk.URL("https://ai.google.dev/gemini-api/docs/quickstart"), + ManagementURL: sdk.URL("https://aistudio.google.com/app/apikey"), + Fields: []schema.CredentialField{ + { + Name: fieldname.APIKey, + MarkdownDescription: "API Key used to authenticate to Google Gemini CLI.", + Secret: true, + Composition: &schema.ValueComposition{ + Length: 39, + Charset: schema.Charset{ + Uppercase: true, + Lowercase: true, + Digits: true, + }, + }, + }, + }, + DefaultProvisioner: provision.EnvVars(defaultEnvVarMapping), + Importer: importer.TryAll( + importer.TryEnvVarPair(defaultEnvVarMapping), + )} +} + +var defaultEnvVarMapping = map[string]sdk.FieldName{ + "GEMINI_API_KEY": fieldname.APIKey, // TODO: Check if this is correct +} diff --git a/plugins/gemini/api_key_test.go b/plugins/gemini/api_key_test.go new file mode 100644 index 00000000..dcd592be --- /dev/null +++ b/plugins/gemini/api_key_test.go @@ -0,0 +1,41 @@ +package gemini + +import ( + "testing" + + "github.com/1Password/shell-plugins/sdk" + "github.com/1Password/shell-plugins/sdk/plugintest" + "github.com/1Password/shell-plugins/sdk/schema/fieldname" +) + +func TestAPIKeyProvisioner(t *testing.T) { + plugintest.TestProvisioner(t, APIKey().DefaultProvisioner, map[string]plugintest.ProvisionCase{ + "default": { + ItemFields: map[sdk.FieldName]string{ + fieldname.APIKey: "1y1RvqUVfm6eTlAUfzFfxcEo1EP9dWMdEXAMPLE", + }, + ExpectedOutput: sdk.ProvisionOutput{ + Environment: map[string]string{ + "GEMINI_API_KEY": "1y1RvqUVfm6eTlAUfzFfxcEo1EP9dWMdEXAMPLE", + }, + }, + }, + }) +} + +func TestAPIKeyImporter(t *testing.T) { + plugintest.TestImporter(t, APIKey().Importer, map[string]plugintest.ImportCase{ + "environment": { + Environment: map[string]string{ + "GEMINI_API_KEY": "1y1RvqUVfm6eTlAUfzFfxcEo1EP9dWMdEXAMPLE", + }, + ExpectedCandidates: []sdk.ImportCandidate{ + { + Fields: map[sdk.FieldName]string{ + fieldname.APIKey: "1y1RvqUVfm6eTlAUfzFfxcEo1EP9dWMdEXAMPLE", + }, + }, + }, + }, + }) +} diff --git a/plugins/gemini/gemini.go b/plugins/gemini/gemini.go new file mode 100644 index 00000000..fafa8de4 --- /dev/null +++ b/plugins/gemini/gemini.go @@ -0,0 +1,24 @@ +package gemini + +import ( + "github.com/1Password/shell-plugins/sdk" + "github.com/1Password/shell-plugins/sdk/needsauth" + "github.com/1Password/shell-plugins/sdk/schema" + "github.com/1Password/shell-plugins/sdk/schema/credname" +) + +func GoogleGeminiCLI() schema.Executable { + return schema.Executable{ + Name: "Google Gemini CLI", + Runs: []string{"gemini"}, + DocsURL: sdk.URL("https://geminicli.com/"), + NeedsAuth: needsauth.IfAll( + needsauth.NotForHelpOrVersion(), + ), + Uses: []schema.CredentialUsage{ + { + Name: credname.APIKey, + }, + }, + } +} diff --git a/plugins/gemini/plugin.go b/plugins/gemini/plugin.go new file mode 100644 index 00000000..e408996e --- /dev/null +++ b/plugins/gemini/plugin.go @@ -0,0 +1,22 @@ +package gemini + +import ( + "github.com/1Password/shell-plugins/sdk" + "github.com/1Password/shell-plugins/sdk/schema" +) + +func New() schema.Plugin { + return schema.Plugin{ + Name: "gemini", + Platform: schema.PlatformInfo{ + Name: "Google Gemini CLI", + Homepage: sdk.URL("https://geminicli.com/"), + }, + Credentials: []schema.CredentialType{ + APIKey(), + }, + Executables: []schema.Executable{ + GoogleGeminiCLI(), + }, + } +} From 1ff117c5567e7ee8c56dcb8c1cfca3fc44158cca Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Fran=C3=A7ois=20Proulx?= <76956526+fproulx-boostsecurity@users.noreply.github.com> Date: Tue, 25 Nov 2025 10:13:05 -0500 Subject: [PATCH 2/2] Update api_key.go --- plugins/gemini/api_key.go | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/plugins/gemini/api_key.go b/plugins/gemini/api_key.go index f5100e25..b79723f2 100644 --- a/plugins/gemini/api_key.go +++ b/plugins/gemini/api_key.go @@ -36,5 +36,5 @@ func APIKey() schema.CredentialType { } var defaultEnvVarMapping = map[string]sdk.FieldName{ - "GEMINI_API_KEY": fieldname.APIKey, // TODO: Check if this is correct + "GEMINI_API_KEY": fieldname.APIKey, }