diff --git a/docs/index.md b/docs/index.md
index d92a5109..901209c6 100644
--- a/docs/index.md
+++ b/docs/index.md
@@ -56,9 +56,9 @@ steampipe plugin install gcp
| Item | Description |
| ----------- | ------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------ |
-| Credentials | When running locally, you must configure your [Application Default Credentials](https://cloud.google.com/sdk/gcloud/reference/auth/application-default). If you are running in Cloud Shell or Cloud Code, [the tool uses the credentials you provided when you logged in, and manages any authorizations required](https://cloud.google.com/docs/authentication/provide-credentials-adc#cloud-based-dev). |
-| Permissions | Assign the `Viewer` role to your user or service account. You may also need additional permissions related to IAM policies, like `pubsub.subscriptions.getIamPolicy`, `pubsub.topics.getIamPolicy`, `storage.buckets.getIamPolicy`, since these are not included in the `Viewer` role. You can grant these by creating a custom role in your project. |
-| Radius | Each connection represents a single GCP project. |
+| Credentials | When running locally, you must configure your [Application Default Credentials](https://cloud.google.com/sdk/gcloud/reference/auth/application-default). If you are running in Cloud Shell or Cloud Code, [the tool uses the credentials you provided when you logged in, and manages any authorizations required](https://cloud.google.com/docs/authentication/provide-credentials-adc#cloud-based-dev). |
+| Permissions | Assign the `Viewer` role to your user or service account. You may also need additional permissions related to IAM policies, like `pubsub.subscriptions.getIamPolicy`, `pubsub.topics.getIamPolicy`, `storage.buckets.getIamPolicy`, since these are not included in the `Viewer` role. You can grant these by creating a custom role in your project. |
+| Radius | Each connection represents a single GCP project, except for some tables like `gcp_organization` and `gcp_organization_project` which return all resources the credentials attached to the connection have access to. |
| Resolution | 1. Credentials from the JSON file specified by the `credentials` parameter in your steampipe config.
2. Credentials from the JSON file specified by the `GOOGLE_APPLICATION_CREDENTIALS` environment variable.
3. Credentials from the default JSON file location (~/.config/gcloud/application_default_credentials.json).
4. Credentials from [the metadata server](https://cloud.google.com/docs/authentication/application-default-credentials#attached-sa) |
### Configuration
diff --git a/docs/tables/gcp_organization_project.md b/docs/tables/gcp_organization_project.md
new file mode 100644
index 00000000..77e1ae67
--- /dev/null
+++ b/docs/tables/gcp_organization_project.md
@@ -0,0 +1,83 @@
+---
+title: "Steampipe Table: gcp_organization_project - Query Google Cloud Platform Projects using SQL"
+description: "Allows users to query Projects in Google Cloud Platform, specifically providing details about the project's ID, name, labels, and lifecycle state."
+---
+
+# Table: gcp_organization_project - Query Google Cloud Platform Projects using SQL
+
+**Note: this table is a variant of the `gcp_project` table which does not filter on the GCP project attached to connection, and thus, will return all projects that the credentials used by the connection have access to. Using this table in aggregator connections can produce unexpected duplicate results.**
+
+A Google Cloud Platform Project acts as an organizational unit within GCP where resources are allocated. It is used to group resources that belong to the same logical application or business unit. Each project is linked to a billing account and can have users, roles, and permissions assigned to it.
+
+## Table Usage Guide
+
+The `gcp_organization_project` table provides insights into Projects within Google Cloud Platform. As a DevOps engineer, explore project-specific details through this table, including ID, name, labels, and lifecycle state. Utilize it to uncover information about projects, such as their associated resources, user roles, permissions, and billing details.
+
+## Examples
+
+### Basic info
+Explore which Google Cloud Platform projects are active, by looking at their lifecycle state and creation time. This can help you manage resources effectively and keep track of ongoing projects.
+
+```sql+postgres
+select
+ name,
+ project_id,
+ project_number,
+ lifecycle_state,
+ create_time
+from
+ gcp_organization_project;
+```
+
+```sql+sqlite
+select
+ name,
+ project_id,
+ project_number,
+ lifecycle_state,
+ create_time
+from
+ gcp_organization_project;
+```
+
+### Get access approval settings for all projects
+Explore the access approval settings across your various projects. This can help you understand and manage permissions and approvals more effectively.
+
+```sql+postgres
+select
+ name,
+ jsonb_pretty(access_approval_settings) as access_approval_settings
+from
+ gcp_organization_project;
+```
+
+```sql+sqlite
+select
+ name,
+ access_approval_settings
+from
+ gcp_organization_project;
+```
+
+### Get parent and organization ID for all projects
+Get the organization ID across your various projects.
+
+```sql+postgres
+select
+ project_id,
+ parent ->> 'id' as parent_id,
+ parent ->> 'type' as parent_type,
+ case when jsonb_array_length(ancestors) > 1 then ancestors -> -1 -> 'resourceId' ->> 'id' else null end as organization_id
+from
+ gcp_project;
+```
+
+```sql+sqlite
+select
+ project_id,
+ parent ->> 'id' as parent_id,
+ parent ->> 'type' as parent_type,
+ case when json_array_length(ancestors) > 1 then ancestors -> -1 -> 'resourceId' ->> 'id' else null end as organization_id
+from
+ gcp_project;
+```
diff --git a/docs/tables/gcp_project.md b/docs/tables/gcp_project.md
index 12e1d07e..0ce02e40 100644
--- a/docs/tables/gcp_project.md
+++ b/docs/tables/gcp_project.md
@@ -5,6 +5,8 @@ description: "Allows users to query Projects in Google Cloud Platform, specifica
# Table: gcp_project - Query Google Cloud Platform Projects using SQL
+**Note: this table is a variant of the `gcp_organization_project` table which filters on the GCP project attached to connection, and thus, will only ever return details about that specific project.**
+
A Google Cloud Platform Project acts as an organizational unit within GCP where resources are allocated. It is used to group resources that belong to the same logical application or business unit. Each project is linked to a billing account and can have users, roles, and permissions assigned to it.
## Table Usage Guide
diff --git a/gcp/plugin.go b/gcp/plugin.go
index bff5226c..2e8d7e3f 100644
--- a/gcp/plugin.go
+++ b/gcp/plugin.go
@@ -124,6 +124,7 @@ func Plugin(ctx context.Context) *plugin.Plugin {
"gcp_monitoring_group": tableGcpMonitoringGroup(ctx),
"gcp_monitoring_notification_channel": tableGcpMonitoringNotificationChannel(ctx),
"gcp_organization": tableGcpOrganization(ctx),
+ "gcp_organization_project": tableGcpOrganizationProject(ctx),
"gcp_project": tableGcpProject(ctx),
"gcp_project_organization_policy": tableGcpProjectOrganizationPolicy(ctx),
"gcp_project_service": tableGcpProjectService(ctx),
diff --git a/gcp/table_gcp_organization_project.go b/gcp/table_gcp_organization_project.go
new file mode 100644
index 00000000..0ed76d67
--- /dev/null
+++ b/gcp/table_gcp_organization_project.go
@@ -0,0 +1,142 @@
+package gcp
+
+import (
+ "context"
+
+ "github.com/turbot/go-kit/types"
+ "github.com/turbot/steampipe-plugin-sdk/v5/grpc/proto"
+ "github.com/turbot/steampipe-plugin-sdk/v5/plugin"
+ "github.com/turbot/steampipe-plugin-sdk/v5/plugin/transform"
+ "google.golang.org/api/cloudresourcemanager/v1"
+)
+
+//// TABLE DEFINITION
+
+func tableGcpOrganizationProject(_ context.Context) *plugin.Table {
+ return &plugin.Table{
+ Name: "gcp_organization_project",
+ Description: "GCP Organization Project",
+ List: &plugin.ListConfig{
+ Hydrate: listGCPOrganizationProjects,
+ },
+ Columns: []*plugin.Column{
+ {
+ Name: "name",
+ Description: "The name of the project.",
+ Type: proto.ColumnType_STRING,
+ },
+ {
+ Name: "project_id",
+ Description: "An unique, user-assigned ID of the Project.",
+ Type: proto.ColumnType_STRING,
+ },
+ {
+ Name: "self_link",
+ Description: "Server-defined URL for the resource.",
+ Type: proto.ColumnType_STRING,
+ Transform: transform.From(projectSelfLink),
+ },
+ {
+ Name: "project_number",
+ Description: "The number uniquely identifying the project.",
+ Type: proto.ColumnType_INT,
+ },
+ {
+ Name: "lifecycle_state",
+ Description: "Specifies the project lifecycle state.",
+ Type: proto.ColumnType_STRING,
+ },
+ {
+ Name: "create_time",
+ Description: "Creation time of the project.",
+ Type: proto.ColumnType_TIMESTAMP,
+ },
+ {
+ Name: "parent",
+ Description: "An optional reference to a parent Resource.",
+ Type: proto.ColumnType_JSON,
+ },
+ {
+ Name: "labels",
+ Description: "A list of labels attached to this project.",
+ Type: proto.ColumnType_JSON,
+ },
+ {
+ Name: "access_approval_settings",
+ Description: "The access approval settings associated with this project.",
+ Type: proto.ColumnType_JSON,
+ Hydrate: getProjectAccessApprovalSettings,
+ Transform: transform.FromValue(),
+ },
+ {
+ Name: "ancestors",
+ Description: "The ancestors of the project in the resource hierarchy, from bottom to top.",
+ Type: proto.ColumnType_JSON,
+ Hydrate: getProjectAncestors,
+ Transform: transform.FromValue(),
+ },
+
+ // Steampipe standard columns
+ {
+ Name: "title",
+ Description: ColumnDescriptionTitle,
+ Type: proto.ColumnType_STRING,
+ Transform: transform.FromField("Name"),
+ },
+ {
+ Name: "tags",
+ Description: ColumnDescriptionTags,
+ Type: proto.ColumnType_JSON,
+ Transform: transform.FromField("Labels"),
+ },
+ {
+ Name: "akas",
+ Description: ColumnDescriptionAkas,
+ Type: proto.ColumnType_JSON,
+ Hydrate: getProjectAka,
+ Transform: transform.FromValue(),
+ },
+ },
+ }
+}
+
+//// LIST FUNCTION
+
+func listGCPOrganizationProjects(ctx context.Context, d *plugin.QueryData, h *plugin.HydrateData) (interface{}, error) {
+ // Create Service Connection
+ service, err := CloudResourceManagerService(ctx, d)
+ if err != nil {
+ plugin.Logger(ctx).Error("gcp_organization_project.listGCPOrganizationProjects", "service_err", err)
+ return nil, err
+ }
+
+ // Max limit is not documented
+ pageSize := types.Int64(500)
+ limit := d.QueryContext.Limit
+ if d.QueryContext.Limit != nil {
+ if *limit < *pageSize {
+ pageSize = limit
+ }
+ }
+
+ // List projects
+ resp := service.Projects.List().PageSize(*pageSize)
+ if err := resp.Pages(ctx, func(page *cloudresourcemanager.ListProjectsResponse) error {
+ for _, project := range page.Projects {
+ d.StreamListItem(ctx, project)
+
+ // Check if context has been cancelled or if the limit has been hit (if specified)
+ // if there is a limit, it will return the number of rows required to reach this limit
+ if d.RowsRemaining(ctx) == 0 {
+ page.NextPageToken = ""
+ return nil
+ }
+ }
+ return nil
+ }); err != nil {
+ plugin.Logger(ctx).Error("gcp_organization_project.listGCPOrganizationProjects", "api_err", err)
+ return nil, err
+ }
+
+ return nil, nil
+}
diff --git a/gcp/table_gcp_project.go b/gcp/table_gcp_project.go
index 18a2bbc4..a9d14672 100644
--- a/gcp/table_gcp_project.go
+++ b/gcp/table_gcp_project.go
@@ -106,6 +106,7 @@ func listGCPProjects(ctx context.Context, d *plugin.QueryData, h *plugin.Hydrate
// Create Service Connection
service, err := CloudResourceManagerService(ctx, d)
if err != nil {
+ plugin.Logger(ctx).Error("gcp_project.listGCPProjects", "service_err", err)
return nil, err
}
@@ -120,6 +121,7 @@ func listGCPProjects(ctx context.Context, d *plugin.QueryData, h *plugin.Hydrate
resp, err := service.Projects.List().Filter("id=" + project).Do()
if err != nil {
+ plugin.Logger(ctx).Error("gcp_project.listGCPProjects", "api_err", err)
return nil, err
}
@@ -151,14 +153,9 @@ func getProjectAccessApprovalSettings(ctx context.Context, d *plugin.QueryData,
}
// Get project details
+ projectId := h.Item.(*cloudresourcemanager.Project).ProjectId
- projectId, err := getProject(ctx, d, h)
- if err != nil {
- return nil, err
- }
- project := projectId.(string)
-
- resp, err := service.Projects.GetAccessApprovalSettings("projects/" + project + "/accessApprovalSettings").Do()
+ resp, err := service.Projects.GetAccessApprovalSettings("projects/" + projectId + "/accessApprovalSettings").Do()
if err != nil {
if strings.Contains(err.Error(), "404") {
return nil, nil
@@ -178,13 +175,9 @@ func getProjectAncestors(ctx context.Context, d *plugin.QueryData, h *plugin.Hyd
}
// Get project details
- projectId, err := getProject(ctx, d, h)
- if err != nil {
- return nil, err
- }
- project := projectId.(string)
+ projectId := h.Item.(*cloudresourcemanager.Project).ProjectId
- resp, err := service.Projects.GetAncestry(project, &cloudresourcemanager.GetAncestryRequest{}).Do()
+ resp, err := service.Projects.GetAncestry(projectId, &cloudresourcemanager.GetAncestryRequest{}).Do()
if err != nil {
if strings.Contains(err.Error(), "404") {
return nil, nil