Documentation ¶
Overview ¶
Package plugin provides data structures and functions that enable a plugin to read data from an API and stream it into Postgres tables by way of Steampipe's [foreign data wrapper] (FDW).
Index ¶
- Constants
- func DiagsToError(prefix string, diags hcl.Diagnostics) error
- func DiagsToWarnings(diags hcl.Diagnostics) []string
- func GetFreeMemInterval() int64
- func GetMatrixItem(ctx context.Context) map[string]interface{}deprecated
- func GetMaxMemoryBytes() int64
- func IsCancelled(ctx context.Context) bool
- func Logger(ctx context.Context) hclog.Logger
- func Memoize[T any](f HydrateFunc, opts ...MemoizeOption) func(ctx context.Context, d *QueryData, h *HydrateData) (T, error)
- func RetryHydrate(ctx context.Context, d *QueryData, hydrateData *HydrateData, ...) (hydrateResult interface{}, err error)
- func Serve(opts *ServeOpts)
- func ValidateSchemaMode(m string) error
- type AggregationMode
- type Column
- type ConfigParsedHook
- type Connection
- type ConnectionConfigInstanceFunc
- type ConnectionConfigSchema
- type ConnectionData
- type CreatePlugin
- type DefaultConcurrencyConfig
- type ErrorPredicate
- type ErrorPredicateWithContext
- type ForceImport
- type GetConfig
- type HydrateConfig
- type HydrateData
- type HydrateDependenciesdeprecated
- type HydrateFunc
- type IgnoreConfig
- type KeyColumn
- type KeyColumnEqualsQualMap
- type KeyColumnQualMap
- func (m KeyColumnQualMap) GetListQualValues() quals.QualSlice
- func (m KeyColumnQualMap) GetUnsatisfiedKeyColumns(columns KeyColumnSlice) KeyColumnSlice
- func (m KeyColumnQualMap) String() string
- func (m KeyColumnQualMap) ToEqualsQualValueMap() map[string]*proto.QualValue
- func (m KeyColumnQualMap) ToProtoQualMap() map[string]*proto.Quals
- func (m KeyColumnQualMap) ToQualMap() map[string]quals.QualSlice
- type KeyColumnQuals
- type KeyColumnSlice
- func (k KeyColumnSlice) AllEquals() bool
- func (k KeyColumnSlice) Find(name string) *KeyColumn
- func (k KeyColumnSlice) IsAnyOf() bool
- func (k KeyColumnSlice) String() string
- func (k KeyColumnSlice) StringSlice() []string
- func (k KeyColumnSlice) ToProtobuf() []*proto.KeyColumn
- func (k KeyColumnSlice) Validate() []string
- type ListConfig
- type MatrixItemFunc
- type MatrixItemMapFunc
- type MemoizeConfiguration
- type MemoizeOption
- type NewPluginOptions
- type Plugin
- func (p *Plugin) ClearConnectionCache(ctx context.Context, connectionName string)
- func (p *Plugin) ClearQueryCache(ctx context.Context, connectionName string)
- func (p *Plugin) ConnectionSchemaChanged(connection *Connection) error
- func (p *Plugin) EstablishMessageStream(stream proto.WrapperPlugin_EstablishMessageStreamServer) error
- func (p *Plugin) Execute(req *proto.ExecuteRequest, stream proto.WrapperPlugin_ExecuteServer) (err error)
- func (p *Plugin) GetSchema(connectionName string) (*grpc.PluginSchema, error)
- func (p *Plugin) SetAllConnectionConfigs(configs []*proto.ConnectionConfig, maxCacheSizeMb int) (failedConnections map[string]error, err error)
- func (p *Plugin) SetConnectionConfig(connectionName, connectionConfigString string) (err error)
- func (p *Plugin) UpdateConnectionConfigs(added []*proto.ConnectionConfig, deleted []*proto.ConnectionConfig, ...) (failedConnections map[string]error, err error)
- type PluginFunc
- type QueryColumn
- type QueryContext
- type QueryData
- type RetryConfig
- type ServeOpts
- type Table
- type TableCacheOptions
- type TableMapData
- type TableMapFunc
Constants ¶
const ( // Require values Required = "required" Optional = "optional" AnyOf = "any_of" )
const ( SchemaModeStatic = "static" SchemaModeDynamic = "dynamic" )
Variables ¶
This section is empty.
Functions ¶
func DiagsToError ¶
DiagsToError converts tfdiags diags into an error
func DiagsToWarnings ¶ added in v5.1.0
func DiagsToWarnings(diags hcl.Diagnostics) []string
DiagsToWarnings converts warning diags into a list of warning strings
func GetFreeMemInterval ¶
func GetFreeMemInterval() int64
func GetMatrixItem
deprecated
Deprecated: Please use plugin.Table.GetMatrixItemFunc instead.
func GetMaxMemoryBytes ¶
func GetMaxMemoryBytes() int64
func IsCancelled ¶
IsCancelled returns whether the context has been cancelled.
To use:
for _, i := range items { d.StreamListItem(ctx, i) if plugin.IsCancelled(ctx) { return nil, nil } }
Plugin examples:
func Logger ¶
Logger retrieves the hclog.Logger from the context.
Log at trace level:
plugin.Logger(ctx).Trace("Code execution starts here")
Log at error level with single data:
plugin.Logger(ctx).Error("hackernews_item.itemList", "query_error", err)
Log at info level with single data:
plugin.Logger(ctx).Info("listGreeting", "number", i)
Log at warn level with multiple data:
plugin.Logger(ctx).Warn("getDomain", "invalid_name", err, "query_response", resp)
func Memoize ¶
func Memoize[T any](f HydrateFunc, opts ...MemoizeOption) func(ctx context.Context, d *QueryData, h *HydrateData) (T, error)
Memoize is a generic function which wraps the given hydrate function with Memoization, and also casts the return value to the desired type Note: the function return from this is NOT a HydrateFunc, which returns 'any', but rather a function returning 'T' - this means it could not be used anywhere expecting a hydrate func, e.g. in a hydrate config etc.
func RetryHydrate ¶
func RetryHydrate(ctx context.Context, d *QueryData, hydrateData *HydrateData, hydrateFunc HydrateFunc, retryConfig *RetryConfig) (hydrateResult interface{}, err error)
RetryHydrate function invokes the hydrate function with retryable errors and retries the function until the maximum attempts before throwing error
func Serve ¶
func Serve(opts *ServeOpts)
Serve creates and starts the GRPC server which serves the plugin,
passing callback functions to implement each of the plugin interface functions:
SetConnectionConfig
SetAllConnectionConfigs
UpdateConnectionConfigs
GetSchema
Execute
It is called from the main function of the plugin.
func ValidateSchemaMode ¶ added in v5.2.0
Types ¶
type AggregationMode ¶
type AggregationMode string
const ( AggregationModeAggregate AggregationMode = "aggregate" AggregationModeNone AggregationMode = "none" )
type Column ¶
type Column struct { // column name Name string // column type Type proto.ColumnType // column description Description string // explicitly specify the function which populates this data // - this is only needed if any of the default hydrate functions will NOT return this column Hydrate HydrateFunc // the default column value Default interface{} // a list of transforms to generate the column value Transform *transform.ColumnTransforms }
Column defines a column of a table.
A column may be populated by a List or Get call. It may alternatively define its own HydrateFunc that makes an additional API call for each row.
A column may transform the data it receives using one or more transform functions.
To define a column populated by a List or Get call:
func itemCols() []*plugin.Column { return []*plugin.Column{ {Name: "id", Type: proto.ColumnType_INT, Description: "The item's unique id."}, } }
To define a column populated by a HydrateFunc:
Columns: awsColumns([]*plugin.Column{ { Name: "permissions_boundary_arn", Description: "The ARN of the policy used to set the permissions boundary for the user.", Type: proto.ColumnType_STRING, Hydrate: getAwsIamUserData, }, }
To define columns that transform the data:
Columns: awsColumns([]*plugin.Column{ { Name: "mfa_enabled", Description: "The MFA status of the user.", Type: proto.ColumnType_BOOL, Hydrate: getAwsIamUserMfaDevices, Transform: transform.From(handleEmptyUserMfaStatus), }, { Name: "login_profile", Description: "Contains the user name and password create date for a user.", Type: proto.ColumnType_JSON, Hydrate: getAwsIamUserLoginProfile, Transform: transform.FromValue(), }, ... }
Examples:
type ConfigParsedHook ¶
ConfigParsedHook hook function called after connection config is parsed - it provides the opportunity to validate and transform the config
type Connection ¶
type Connection struct { Name string // the connection config // NOTE: we always pass and store connection config BY VALUE Config any }
Connection is a struct which is used to store connection config.
The connection config is parsed and stored as plugin.Plugin.Connection.
The connection may be retrieved by the plugin by calling: plugin.QueryData.Connection
Plugin examples:
type ConnectionConfigInstanceFunc ¶
type ConnectionConfigInstanceFunc func() any
ConnectionConfigInstanceFunc is a function type which returns 'any'.
It is used to implement plugin.ConnectionConfigSchema.NewInstance.
type ConnectionConfigSchema ¶
type ConnectionConfigSchema struct { Schema map[string]*schema.Attribute // function which returns an instance of a connection config struct NewInstance ConnectionConfigInstanceFunc // function which validates/transforms the parsed config OnConfigParsed ConfigParsedHook }
ConnectionConfigSchema is a struct that defines custom arguments in the plugin spc file that are passed to the plugin as plugin.Connection.Config.
A plugin that uses custom connection config must set plugin.Plugin.ConnectionConfigSchema.
Usage:
p := &plugin.Plugin{ Name: "steampipe-plugin-hackernews", ConnectionConfigSchema: &plugin.ConnectionConfigSchema{ NewInstance: ConfigInstance, Schema: ConfigSchema, }, ... } var ConfigSchema = map[string]*schema.Attribute{ "max_items": { Type: schema.TypeInt, }, } func ConfigInstance() any { return &hackernewsConfig{} }
Plugin examples:
func (*ConnectionConfigSchema) Validate ¶
func (c *ConnectionConfigSchema) Validate() []string
Validate validates the connection config
type ConnectionData ¶
type ConnectionData struct { // map of all the tables in the connection schema, keyed by the table name TableMap map[string]*Table Connection *Connection // the connection schema Schema *grpc.PluginSchema // connection specific filewatcher to watch for the changes in files (if needed) Watcher *filewatcher.FileWatcher Plugin *Plugin // map of aggregated tables, keyed by connection name // NOTE: only set for aggregator connections AggregatedTablesByConnection map[string]map[string]*Table // contains filtered or unexported fields }
ConnectionData is the data stored by the plugin which is connection dependent.
func NewConnectionData ¶ added in v5.2.0
func NewConnectionData(c *Connection, p *Plugin, config *proto.ConnectionConfig) *ConnectionData
func (*ConnectionData) GetConnectionTempDir ¶
func (d *ConnectionData) GetConnectionTempDir(pluginTempDir string) string
GetConnectionTempDir appends the connection name to the plugin temporary directory path
type DefaultConcurrencyConfig ¶
type DefaultConcurrencyConfig struct { // sets how many HydrateFunc calls can run concurrently in total TotalMaxConcurrency int // sets the default for how many calls to each HydrateFunc can run concurrently DefaultMaxConcurrency int }
DefaultConcurrencyConfig sets the default maximum number of concurrent HydrateFunc calls.
Limit total concurrent hydrate calls:
DefaultConcurrency: &plugin.DefaultConcurrencyConfig{ TotalMaxConcurrency: 500, }
Limit concurrent hydrate calls to any single HydrateFunc which does not have a HydrateConfig:
DefaultConcurrency: &plugin.DefaultConcurrencyConfig{ DefaultMaxConcurrency: 100, }
Do both:
DefaultConcurrency: &plugin.DefaultConcurrencyConfig{ TotalMaxConcurrency: 500, DefaultMaxConcurrency: 200, }
Plugin examples:
type ErrorPredicate ¶
ErrorPredicate is a function type which accepts error as an input and returns a boolean value.
type ErrorPredicateWithContext ¶
ErrorPredicateWithContext is a function type which accepts context, query data, hydrate data and error as an input and returns a boolean value.
Plugin examples:
type ForceImport ¶
type ForceImport string
ForceImport is a mechanism to ensure godoc can reference all required packages
type GetConfig ¶
type GetConfig struct { // key or keys which are used to uniquely identify rows - used to determine whether a query is a 'get' call KeyColumns KeyColumnSlice // the hydrate function which is called first when performing a 'get' call. // if this returns 'not found', no further hydrate functions are called Hydrate HydrateFunc // a function which will return whenther to ignore a given error // deprecated - use IgnoreConfig ShouldIgnoreError ErrorPredicate IgnoreConfig *IgnoreConfig RetryConfig *RetryConfig // max concurrency - this applies when the get function is ALSO used as a column hydrate function MaxConcurrency int }
A GetConfig defines how to get a single row of a table:
The key_columns that uniquely identify a row.
The error_handling behaviour.
How many concurrent HydrateFunc calls to allow, when the get call is used as a column hydrate func: plugin.GetConfig.MaxConcurrency.
A GetConfig with KeyColumns:
Get: &plugin.GetConfig{ KeyColumns: plugin.SingleColumn("id"), Hydrate: getItem, }
A GetConfig with IgnoreConfig:
Get: &plugin.GetConfig{ KeyColumns: plugin.SingleColumn("id"), Hydrate: getItem, IgnoreConfig: &plugin.IgnoreConfig{ShouldIgnoreErrorFunc: shouldIgnoreError}, }
A GetConfig with RetryConfig:
Get: &plugin.GetConfig{ KeyColumns: plugin.SingleColumn("id"), Hydrate: getItem, RetryConfig: &plugin.RetryConfig{ ShouldRetryErrorFunc: shouldRetryError, }, }
A GetConfig with all fields specified:
Get: &plugin.GetConfig{ KeyColumns: plugin.SingleColumn("id"), Hydrate: getItem, RetryConfig: &plugin.RetryConfig{ ShouldRetryErrorFunc: shouldRetryError, }, IgnoreConfig: &plugin.IgnoreConfig{ShouldIgnoreErrorFunc: shouldIgnoreError}, MaxConcurrency: 50, }
Plugin examples:
type HydrateConfig ¶
type HydrateConfig struct { Func HydrateFunc MaxConcurrency int RetryConfig *RetryConfig IgnoreConfig *IgnoreConfig // deprecated - use IgnoreConfig ShouldIgnoreError ErrorPredicate Depends []HydrateFunc }
HydrateConfig defines how to run a HydrateFunc:
Which errors to ignore: plugin.HydrateConfig.IgnoreConfig.
Which errors to retry: plugin.HydrateConfig.RetryConfig.
How many concurrent calls to allow: plugin.HydrateConfig.MaxConcurrency.
Which hydrate calls must complete before this HydrateFunc can start: plugin.HydrateConfig.Depends.
It's not valid to have a HydrateConfig for a HydrateFunc that is specified in a GetConfig.
A HydrateConfig with IgnoreConfig:
HydrateConfig: []plugin.HydrateConfig{ { Func: getRetentionPeriod, IgnoreConfig: &plugin.IgnoreConfig{ShouldIgnoreErrorFunc: shouldIgnoreError}, }
A HydrateConfig with MaxConcurrency:
HydrateConfig: []plugin.HydrateConfig{ { Func: getRetentionPeriod, MaxConcurrency: 50, IgnoreConfig: &plugin.IgnoreConfig{ShouldIgnoreErrorFunc: shouldIgnoreError}, }
A HydrateConfig with all fields specified:
HydrateConfig: []plugin.HydrateConfig{ { Func: getRetentionPeriod, MaxConcurrency: 50, IgnoreConfig: &plugin.IgnoreConfig{ShouldIgnoreErrorFunc: shouldIgnoreError}, RetryConfig: &plugin.RetryConfig{ ShouldRetryErrorFunc: shouldRetryError, }
Steampipe parallelizes hydrate functions as much as possible. Sometimes, however, one hydrate function requires the output from another.
return &plugin.Table{ Name: "hydrate_columns_dependency", List: &plugin.ListConfig{ Hydrate: hydrateList, }, HydrateConfig: []plugin.HydrateConfig{ { Func: hydrate2, Depends: []plugin.HydrateFunc{hydrate1}, }, }, Columns: []*plugin.Column{ {Name: "id", Type: proto.ColumnType_INT}, {Name: "hydrate_column_1", Type: proto.ColumnType_STRING, Hydrate: hydrate1}, {Name: "hydrate_column_2", Type: proto.ColumnType_STRING, Hydrate: hydrate2}, }, }
Here, hydrate function hydrate2 is dependent on hydrate1. This means hydrate2 will not execute until hydrate1 has completed and the results are available. hydrate2 can refer to the results from hydrate1 as follows:
func hydrate2(ctx context.Context, d *plugin.QueryData, h *plugin.HydrateData) (interface{}, error) { // NOTE: in this case we know the output of hydrate1 is map[string]interface{} so we cast it accordingly. // the data should be cast to th appropriate type hydrate1Results := h.HydrateResults["hydrate1"].(map[string]interface{}) ..... }
Note that:
- Multiple dependencies are supported.
- Circular dependencies will be detected and cause a validation failure.
- The Get and List hydrate functions ***CANNOT*** have dependencies.
Examples:
func (*HydrateConfig) String ¶
func (c *HydrateConfig) String() interface{}
func (*HydrateConfig) Validate ¶
func (c *HydrateConfig) Validate(table *Table) []string
type HydrateData ¶
type HydrateData struct { // if there was a parent-child list call, store the parent list item ParentItem interface{} Item interface{} HydrateResults map[string]interface{} }
HydrateData contains the input data passed to every hydrate function
type HydrateDependencies
deprecated
type HydrateDependencies struct { Func HydrateFunc Depends []HydrateFunc }
Deprecated: Use HydrateConfig instead.
type HydrateFunc ¶
type HydrateFunc func(context.Context, *QueryData, *HydrateData) (interface{}, error)
HydrateFunc is a function that gathers data to build table rows. Typically this would make an API call and return the raw API output.
List and Get are special hydrate functions.
List returns data for all rows. Almost all tables will have a List function.
Get returns data for a single row. In order to filter as cheaply as possible a Get function should be implemented if the API supports fetching single items by key.
A column may require data not returned by the List or Get calls and an additional API call will be required. A HydrateFunc that wraps this API call can be specified in the Column definition.
You could do this the hard way by looping through the List API results and enriching each item by making an additional API call. However the SDK does all this for you.
func WrapHydrate ¶
func WrapHydrate(hydrateFunc HydrateFunc, ignoreConfig *IgnoreConfig) HydrateFunc
WrapHydrate is a higher order function which returns a HydrateFunc that handles Ignorable errors.
func (HydrateFunc) Memoize ¶ added in v5.2.0
func (hydrate HydrateFunc) Memoize(opts ...MemoizeOption) HydrateFunc
Memoize ensures the HydrateFunc results are saved in the connection.ConnectionCache.
Use it to reduce the number of API calls if the HydrateFunc is used by multiple tables.
Usage ¶
{ Name: "account", Type: proto.ColumnType_STRING, Hydrate: plugin.HydrateFunc(getCommonColumns).Memoize(), Description: "The Snowflake account ID.", Transform: transform.FromCamel(), }
func (HydrateFunc) WithCache ¶
func (hydrate HydrateFunc) WithCache(args ...HydrateFunc) HydrateFunc
WithCache ensures the HydrateFunc results are saved in the connection.ConnectionCache.
Use it to reduce the number of API calls if the HydrateFunc is used by multiple tables.
Usage ¶
{ Name: "account", Type: proto.ColumnType_STRING, Hydrate: plugin.HydrateFunc(getCommonColumns).WithCache(), Description: "The Snowflake account ID.", Transform: transform.FromCamel(), }
Plugin examples:
// deprecated: use Memoize
type IgnoreConfig ¶
type IgnoreConfig struct { ShouldIgnoreErrorFunc ErrorPredicateWithContext // deprecated, used ShouldIgnoreErrorFunc ShouldIgnoreError ErrorPredicate }
IgnoreConfig defines errors to ignore. When that happens, an empty row is returned.
If a HydrateFunc has specific errors that should not block query execution, set plugin.GetConfig.IgnoreConfig, plugin.ListConfig.IgnoreConfig or plugin.HydrateConfig.IgnoreConfig.
For errors common to many HydrateFuncs, you can define a default IgnoreConfig by setting plugin.DefaultGetConfig.
Ignore errors from a HydrateFunc that has a GetConfig:
Get: &plugin.GetConfig{ IgnoreConfig: &plugin.IgnoreConfig{ ShouldIgnoreErrorFunc: isIgnorableErrorPredicate([]string{"Request_ResourceNotFound", "Invalid object identifier"}), }, ... },
Ignore errors from a HydrateFunc that has a ListConfig:
List: &plugin.ListConfig{ IgnoreConfig: &plugin.IgnoreConfig{ ShouldIgnoreErrorFunc: isIgnorableErrorPredicate([]string{"Request_UnsupportedQuery"}), }, ... },
Ignore errors from a HydrateFunc that has a HydrateConfig:
HydrateConfig: []plugin.HydrateConfig{ IgnoreConfig: &plugin.IgnoreConfig{ ShouldIgnoreErrorFunc: isIgnorableErrorPredicate([]string{"Request_UnsupportedQuery"}), }, ... },
Ignore errors that may occur in many HydrateFuncs:
DefaultIgnoreConfig: &plugin.DefaultIgnoreConfig{ IgnoreConfig: &plugin.IgnoreConfig{ ShouldIgnoreErrorFunc: isIgnorableErrorPredicate([]string{"Request_ResourceNotFound"}), }, ... },
Plugin examples:
func (*IgnoreConfig) DefaultTo ¶
func (c *IgnoreConfig) DefaultTo(other *IgnoreConfig)
func (*IgnoreConfig) String ¶
func (c *IgnoreConfig) String() interface{}
type KeyColumn ¶
KeyColumn is a struct representing the definition of a column used to filter and Get and List calls.
At least one key column must be defined for a Get call. They are optional for List calls.
Operators ¶
This property specifies the accepted operators (from a possible set: "=", "<>", "<", "<=", ">", ">=")
Require ¶
This property determines whether the column is required or optional. Possible values:
"required"
The key column must be provided as a query qualifier (i.e. in a where clause in the query).
"optional"
The key column is optional but if provided it will be used to filter the results.
"any_of"
Any one of the given columns must be provided.
CacheMatch ¶
This property determines the logic used by the query results cache to determine whether a cached value matches a given query. Possible values:
"subset" [default value]
A cached item is considered a match (i.e. a cache hit) if the qual for the query is a subset of the quals for the cached item.
For example, is the cached qual is "val < 100", and the query qual is "val < 50", this would be considered a qual subset so would lead to a cache match.
"exact"
A cached item is considered a match ONLY if the qual for the cached item is the same as as the qual for the query.
This is used for columns which are only populated if the qual for that column is passed. A common pattern is to provide a "filter" column, which is populated using the qual value provided. This filter value is used when making the API call the fetch the data. If no filter qual is provided, then the filter column returned by the plugin is empty.
This breaks the subset logic as if there is a cached data with no qual for the filter column, this cached data would contain null values for the filter column. This data would be considered a superset of the data returned from a query which provides a filter qual, which is incorrect as the data returned if a filter qual is passed would include a non null filter column.
The solution is to set CacheMatch="exact".
Plugin examples:
func (*KeyColumn) InitialiseOperators ¶
func (k *KeyColumn) InitialiseOperators()
InitialiseOperators adds a default '=' operator is no operators are set, and converts "!=" to "<>".
func (*KeyColumn) SingleEqualsQual ¶
SingleEqualsQual returns whether this key column has a single = operator.
func (*KeyColumn) ToProtobuf ¶
ToProtobuf converts the KeyColumn to a protobuf object.
type KeyColumnEqualsQualMap ¶
KeyColumnEqualsQualMap is a map of column name to qual value. It is used to represent a set of columns which have a single 'equals' qual.
func (KeyColumnEqualsQualMap) GetListQualValues ¶
func (m KeyColumnEqualsQualMap) GetListQualValues() map[string]*proto.QualValueList
GetListQualValues returns a map of all qual values with a List value
func (KeyColumnEqualsQualMap) String ¶
func (m KeyColumnEqualsQualMap) String() string
type KeyColumnQualMap ¶
type KeyColumnQualMap map[string]*KeyColumnQuals
KeyColumnQualMap is a map of KeyColumnQuals keyed by column name
func NewKeyColumnQualValueMap ¶
func NewKeyColumnQualValueMap(qualMap map[string]*proto.Quals, keyColumns KeyColumnSlice) KeyColumnQualMap
NewKeyColumnQualValueMap creates a KeyColumnQualMap from a qual map and a KeyColumnSlice
func (KeyColumnQualMap) GetListQualValues ¶
func (m KeyColumnQualMap) GetListQualValues() quals.QualSlice
GetListQualValues returns a slice of any quals we have which have a list value
func (KeyColumnQualMap) GetUnsatisfiedKeyColumns ¶
func (m KeyColumnQualMap) GetUnsatisfiedKeyColumns(columns KeyColumnSlice) KeyColumnSlice
func (KeyColumnQualMap) String ¶
func (m KeyColumnQualMap) String() string
func (KeyColumnQualMap) ToEqualsQualValueMap ¶
func (m KeyColumnQualMap) ToEqualsQualValueMap() map[string]*proto.QualValue
ToEqualsQualValueMap converts a KeyColumnQualMap to a column-qual value map, including only the
func (KeyColumnQualMap) ToProtoQualMap ¶
func (m KeyColumnQualMap) ToProtoQualMap() map[string]*proto.Quals
ToQualMap converts the map into a map of column to *proto.Quals used for cache indexes
type KeyColumnQuals ¶
KeyColumnQuals defines all qualifiers for a column.
Use it in a table definition, by way of the plugin.QueryData object.
The query writer must specify the qualifiers, in a WHERE or a JOIN clause, in order to limit the number of API calls that Steampipe makes to satisfy the query.
func listUser(ctx context.Context, d *plugin.QueryData, h *plugin.HydrateData) (interface{}, error) { var item User var id string if h.Item != nil { user := h.Item.(*User) id = user.ID } else { quals := d.KeyColumnQuals id = quals["id"].GetStringValue() } ... }
Examples:
func (KeyColumnQuals) SatisfiesKeyColumn ¶
func (k KeyColumnQuals) SatisfiesKeyColumn(keyColumn *KeyColumn) bool
func (KeyColumnQuals) SingleEqualsQual ¶
func (k KeyColumnQuals) SingleEqualsQual() bool
type KeyColumnSlice ¶
type KeyColumnSlice []*KeyColumn
KeyColumnSlice is an array of KeyColumn
func AllColumns ¶
func AllColumns(columns []string) KeyColumnSlice
AllColumns creates a KeyColumnSlice based on a slice of column names, each with a single equals operator and Require=Required
func AnyColumn ¶
func AnyColumn(columns []string) KeyColumnSlice
AnyColumn Columns creates a KeyColumnSlice based on a slice of column names, each with a single equals operator and Require=AnyOf
func NewEqualsKeyColumnSlice ¶
func NewEqualsKeyColumnSlice(columns []string, require string) KeyColumnSlice
NewEqualsKeyColumnSlice creates a KeyColumnSlice from a list of column names, each with a single equals operator
func OptionalColumns ¶
func OptionalColumns(columns []string) KeyColumnSlice
OptionalColumns Columns creates a KeyColumnSlice based on a slice of column names, with a single equals operator and Require=Optional
func SingleColumn ¶
func SingleColumn(column string) KeyColumnSlice
SingleColumn creates a KeyColumnSlice based on a column name The created slice has a single KeyColumn using a single equals operator and Require=Required
func (KeyColumnSlice) AllEquals ¶
func (k KeyColumnSlice) AllEquals() bool
AllEquals returns whether all KeyColumns only use equals operators
func (KeyColumnSlice) Find ¶
func (k KeyColumnSlice) Find(name string) *KeyColumn
Find looks for a key column with the given name and returns it if found
func (KeyColumnSlice) IsAnyOf ¶
func (k KeyColumnSlice) IsAnyOf() bool
IsAnyOf returns whether all key columns have Require == AnyOf
func (KeyColumnSlice) String ¶
func (k KeyColumnSlice) String() string
func (KeyColumnSlice) StringSlice ¶
func (k KeyColumnSlice) StringSlice() []string
StringSlice converts a KeyColumnSlice to a slice of strings
func (KeyColumnSlice) ToProtobuf ¶
func (k KeyColumnSlice) ToProtobuf() []*proto.KeyColumn
ToProtobuf converts the KeyColumnSlice to a slice of proto.KeyColumn
func (KeyColumnSlice) Validate ¶
func (k KeyColumnSlice) Validate() []string
Validate validates all child columns
type ListConfig ¶
type ListConfig struct { KeyColumns KeyColumnSlice // the list function, this should stream the list results back using the QueryData object and return nil Hydrate HydrateFunc // the parent list function - if we list items with a parent-child relationship, this will list the parent items ParentHydrate HydrateFunc // deprecated - use IgnoreConfig ShouldIgnoreError ErrorPredicate IgnoreConfig *IgnoreConfig RetryConfig *RetryConfig }
ListConfig defines how to return all rows in the table:
The HydrateFunc to use.
The key_columns that may be used to optimize the fetch.
The error_handling behaviour.
To define a table's List function:
func tableHackernewsItem(ctx context.Context) *plugin.Table { return &plugin.Table{ Name: "hackernews_item", Description: "This table includes the most recent items posted to Hacker News.", List: &plugin.ListConfig{ Hydrate: itemList, }, ... } }
Examples:
func (*ListConfig) Validate ¶
func (c *ListConfig) Validate(table *Table) []string
type MatrixItemFunc ¶
type MatrixItemFunc func(context.Context, *Connection) []map[string]interface{}
Deprecated
type MatrixItemMapFunc ¶
MatrixItemMapFunc is a callback function which may be implemented by the plugin to provide a map of matrix_items to execute the query with.
matrix_items are a powerful way of executing the same query multiple times in parallel for a set of parameters.
Plugin examples:
- Declaration of MatrixItemMapFunc and its implementation.
type MemoizeConfiguration ¶ added in v5.2.0
type MemoizeConfiguration struct { GetCacheKeyFunc HydrateFunc Ttl time.Duration }
type MemoizeOption ¶ added in v5.2.0
type MemoizeOption = func(config *MemoizeConfiguration)
func MemoizeCacheKeyFunction ¶
func MemoizeCacheKeyFunction(getCacheKeyFunc HydrateFunc) MemoizeOption
MemoizeCacheKeyFunction sets the function used to build the cache key
func MemoizeTtl ¶
func MemoizeTtl(ttl time.Duration) MemoizeOption
MemoizeTtl sets the function used to build the cache key
type NewPluginOptions ¶
type Plugin ¶
type Plugin struct { Name string Logger hclog.Logger // TableMap is a map of all the tables in the plugin, keyed by the table name // NOTE: it must be NULL for plugins with dynamic schema TableMap map[string]*Table TableMapFunc TableMapFunc DefaultTransform *transform.ColumnTransforms DefaultConcurrency *DefaultConcurrencyConfig DefaultRetryConfig *RetryConfig DefaultIgnoreConfig *IgnoreConfig // deprecated - use DefaultRetryConfig and DefaultIgnoreConfig DefaultGetConfig *GetConfig // deprecated - use DefaultIgnoreConfig DefaultShouldIgnoreError ErrorPredicate // every table must implement these columns RequiredColumns []*Column ConnectionConfigSchema *ConnectionConfigSchema // ConnectionConfigChangedFunc is a callback function which is called from UpdateConnectionConfigs // when any connection configs have changed ConnectionConfigChangedFunc func(ctx context.Context, p *Plugin, old, new *Connection) error // map of connection data (schema, config, connection cache) // keyed by connection name ConnectionMap map[string]*ConnectionData // is this a static or dynamic schema SchemaMode string // callback function which is called when any watched source file(s) gets changed WatchedFileChangedFunc func(ctx context.Context, p *Plugin, connection *Connection, events []fsnotify.Event) // contains filtered or unexported fields }
func (*Plugin) ClearConnectionCache ¶
ClearConnectionCache clears the connection cache for the given connection.
func (*Plugin) ClearQueryCache ¶
ClearQueryCache clears the query cache for the given connection.
func (*Plugin) ConnectionSchemaChanged ¶
func (p *Plugin) ConnectionSchemaChanged(connection *Connection) error
ConnectionSchemaChanged sends a message to the plugin-manager that the schema of this plugin has changed
This should be called from the plugin implementation of plugin.Plugin.WatchedFileChangedFunc if a change in watched source files has changed the plugin schema.
func (*Plugin) EstablishMessageStream ¶
func (p *Plugin) EstablishMessageStream(stream proto.WrapperPlugin_EstablishMessageStreamServer) error
EstablishMessageStream establishes a streaming message connection between the plugin and the plugin manager This is used if the plugin has a dynamic schema and uses file watching
This is the handler function for the EstablishMessageStream grpc function
func (*Plugin) Execute ¶
func (p *Plugin) Execute(req *proto.ExecuteRequest, stream proto.WrapperPlugin_ExecuteServer) (err error)
Execute starts a query and streams the results using the given GRPC stream.
This is the handler function for the Execute GRPC function.
func (*Plugin) GetSchema ¶
func (p *Plugin) GetSchema(connectionName string) (*grpc.PluginSchema, error)
GetSchema returns the grpc.PluginSchema. Note: the connection config must be set before calling this function.
This is the handler function for the GetSchema grpc function
func (*Plugin) SetAllConnectionConfigs ¶
func (p *Plugin) SetAllConnectionConfigs(configs []*proto.ConnectionConfig, maxCacheSizeMb int) (failedConnections map[string]error, err error)
SetAllConnectionConfigs sets the connection config for a list of connections.
This is the handler function for the SetAllConnectionConfigs GRPC function.
func (*Plugin) SetConnectionConfig ¶
SetConnectionConfig sets the connection config for the given connection. (for legacy plugins) This is the handler function for the SetConnectionConfig GRPC function.
func (*Plugin) UpdateConnectionConfigs ¶
func (p *Plugin) UpdateConnectionConfigs(added []*proto.ConnectionConfig, deleted []*proto.ConnectionConfig, changed []*proto.ConnectionConfig) (failedConnections map[string]error, err error)
UpdateConnectionConfigs handles added, changed and deleted connections:
Added connections are inserted into plugin.Plugin.ConnectionMap.
Deleted connections are removed from ConnectionMap.
For updated connections, ConnectionMap is updated and plugin.Plugin.ConnectionConfigChangedFunc is called.
This is the handler function for the UpdateConnectionConfigs GRPC function.
type PluginFunc ¶
type QueryColumn ¶
type QueryColumn struct { *Column // contains filtered or unexported fields }
QueryColumn is struct storing column name and resolved hydrate name this is used in the query data when the hydrate function has been resolved
func NewQueryColumn ¶
func NewQueryColumn(column *Column, hydrateName string) *QueryColumn
type QueryContext ¶
type QueryContext struct { Columns []string UnsafeQuals map[string]*proto.Quals Limit *int64 CacheEnabled bool CacheTTL int64 }
QueryContext contains important query properties:
The columns requested.
All quals specified (not just key column quals).
The limit.
Cache properties.
func NewQueryContext ¶
func NewQueryContext(p *proto.QueryContext, limit *proto.NullableInt, cacheEnabled bool, cacheTTL int64) *QueryContext
NewQueryContext maps from a proto.QueryContext to a plugin.QueryContext.
func (*QueryContext) GetLimit ¶
func (q *QueryContext) GetLimit() int64
GetLimit converts plugin.QueryContext.Limit from a *int64 to an int64 (where -1 means no limit).
type QueryData ¶
type QueryData struct { // The table this query is associated with Table *Table // a map of key columns which have a _single equals qual_ // this is intended for use by Get calls only // key: column name, value: QualValue // (this will also be populated for a list call if list key columns are specified - // however this usage is deprecated and provided for legacy reasons only) EqualsQuals KeyColumnEqualsQualMap // a map of all KeyColumnQuals which were specified in the query. ([key_columns] Quals KeyColumnQualMap // is this a 'get' or a 'list' call FetchType fetchType // query context data passed from postgres - this includes the requested columns and the quals QueryContext *QueryContext // connection details - the connection name and any config declared in the connection config file Connection *Connection // Matrix is an array of parameter maps (MatrixItems) // the list/get calls with be executed for each element of this array Matrix []map[string]interface{} // object to handle caching of connection specific data // deprecated use ConnectionCache ConnectionManager *connection_manager.Manager ConnectionCache *connection_manager.ConnectionCache // streaming funcs StreamListItem func(context.Context, ...interface{}) // deprecated - plugins should no longer call StreamLeafListItem directly and should just call StreamListItem // event for the child list of a parent child list call StreamLeafListItem func(context.Context, ...interface{}) // contains filtered or unexported fields }
QueryData is passed to all HydrateFunc calls. It contains all required information about the executing query:
The table (Table).
A map of all equals key column quals (KeyColumnEqualsQualMap).
A map of all key column quals (KeyColumnQualMap). See key_columns.
Is it a list or a get call?
Context data passed from postgres (QueryContext).
The steampipe connection (Connection).
A cache which can be used to store connection specific data (connection.ConnectionCache).
The function which is used to stream rows of data (connection.StreamListItem).
Plugin examples:
func (*QueryData) EqualsQualString ¶
EqualsQualString looks for the specified key column quals and if it exists, return the value as a string
func (*QueryData) GetSourceFiles ¶
GetSourceFiles accept a source path downloads files if necessary, and returns a list of local file paths
func (*QueryData) RowsRemaining ¶
RowsRemaining returns how many rows are required to complete the query
- if no limit has been parsed from the query, this will return math.MaxInt32 (meaning an unknown number of rows remain)
- if there is a limit, it will return the number of rows required to reach this limit
- if the context has been cancelled, it will return zero
func (*QueryData) ShallowCopy ¶
ShallowCopy creates a shallow copy of the QueryData, i.e. most pointer properties are copied this is used to pass different quals to multiple list/get calls, when an 'in' clause is specified
type RetryConfig ¶
type RetryConfig struct { ShouldRetryErrorFunc ErrorPredicateWithContext // deprecated use ShouldRetryErrorFunc ShouldRetryError ErrorPredicate // Maximum number of retry operation to be performed. Default set to 10. MaxAttempts int64 // Algorithm for the backoff. Supported values: Fibonacci, Exponential, and Constant. Default set to Fibonacci. BackoffAlgorithm string // Starting interval. Default set to 100ms. RetryInterval int64 // Set a maximum on the duration (in ms) returned from the next backoff. CappedDuration int64 // Sets a maximum on the total amount of time (in ms) a backoff should execute. MaxDuration int64 }
RetryConfig retries HydrateFunc errors.
If a HydrateFunc returns an error in the first attempt but resolves itself in a future attempt, for instance API rate limit or throttling errors, set plugin.GetConfig.RetryConfig, plugin.ListConfig.RetryConfig or plugin.HydrateConfig.RetryConfig.
For errors common to many HydrateFuncs, you can define a default RetryConfig by setting plugin.DefaultGetConfig.
Retry errors from a HydrateFunc that has a GetConfig:
Get: &plugin.GetConfig{ RetryConfig: &plugin.RetryConfig{ ShouldRetryError: shouldRetryError, }, ... },
Retry errors from a HydrateFunc that has a ListConfig:
List: &plugin.ListConfig{ RetryConfig: &plugin.RetryConfig{ ShouldRetryError: shouldRetryError, }, ... },
Retry errors from a HydrateFunc that has a HydrateConfig:
HydrateConfig: []plugin.HydrateConfig{ RetryConfig: &plugin.RetryConfig{ ShouldRetryError: shouldRetryError, }, ... },
Retry errors that may occur in many HydrateFuncs:
DefaultIgnoreConfig: &plugin.DefaultIgnoreConfig{ RetryConfig: &plugin.RetryConfig{ ShouldRetryError: shouldRetryError, }, ... },
func (*RetryConfig) DefaultTo ¶
func (c *RetryConfig) DefaultTo(other *RetryConfig)
func (*RetryConfig) GetListRetryConfig ¶
func (c *RetryConfig) GetListRetryConfig() *RetryConfig
GetListRetryConfig wraps the ShouldRetry function with an additional check of the rows streamed (as we cannot retry errors in the list hydrate function after streaming has started)
func (*RetryConfig) String ¶
func (c *RetryConfig) String() interface{}
type ServeOpts ¶
type ServeOpts struct { PluginName string PluginFunc PluginFunc }
ServeOpts are the configurations to serve a plugin.
type Table ¶
type Table struct { Name string // table description Description string // column definitions Columns []*Column // the function used to list table rows List *ListConfig // the function used to efficiently retrieve a row by id Get *GetConfig // deprecated // the function used when retrieving data for multiple 'matrix items', e.g. regions GetMatrixItem MatrixItemFunc GetMatrixItemFunc MatrixItemMapFunc // default transform applied to all columns DefaultTransform *transform.ColumnTransforms // function controlling default error handling behaviour DefaultIgnoreConfig *IgnoreConfig DefaultRetryConfig *RetryConfig // the parent plugin object Plugin *Plugin // Deprecated: used HydrateConfig HydrateDependencies []HydrateDependencies // Config for any required hydrate functions, including dependencies between hydrate functions, // error handling and concurrency behaviour HydrateConfig []HydrateConfig // cache options - allows disabling of cache for this table Cache *TableCacheOptions // specify whether to include this table in an aggregate connection Aggregation AggregationMode // deprecated - use DefaultIgnoreConfig DefaultShouldIgnoreError ErrorPredicate // contains filtered or unexported fields }
Table defines the properties of a plugin table:
The columns that are returned: plugin.Table.Columns.
How to fetch all rows in the table: plugin.Table.List.
How to fetch a single row by key: plugin.Table.Get.
Additional configuration for a column hydrate function: plugin.Table.HydrateConfig.
Function used to retrieve data for multiple matrix items: plugin.Table.GetMatrixItemFunc.
The table default error_handling behaviour.
func (*Table) GetSchema ¶
func (t *Table) GetSchema() (*proto.TableSchema, error)
GetSchema returns the proto.TableSchema, which defines the columns returned by the table.
Note: an additional '_ctx' column is added to all table schemas. This contains Steampipe specific data. (Currently this is populated with the connection name.)
func (*Table) ValidateColumnsExist ¶
func (t *Table) ValidateColumnsExist(keyColumns KeyColumnSlice) []string
type TableCacheOptions ¶
type TableCacheOptions struct {
Enabled bool
}
TableCacheOptions provides a mechanism to disable caching for a specific table.
It is useful in cases where the table returns a huge volume of data cheaply.
Use TableCacheOptions to override the .cache off property of the CLI.
type TableMapData ¶
type TableMapData struct { Connection *Connection ConectionCache *connection.ConnectionCache }
type TableMapFunc ¶
TableMapFunc is callback function which can be used to populate plugin.Plugin.TableMap and allows the connection config to be used in the table creation (connection config is not available at plugin creation time).
This callback function should be implemented by the plugin writer for dynamic plugins.
Plugin examples:
Source Files ¶
- column.go
- concurrency.go
- connection_config_schema.go
- connection_config_schema_validate.go
- connection_data.go
- context.go
- diags.go
- doc.go
- env.go
- funcs.go
- get_config.go
- hydrate_cache.go
- hydrate_call.go
- hydrate_config.go
- hydrate_data.go
- hydrate_dependencies.go
- hydrate_error.go
- ignore_error_config.go
- key_column.go
- key_column_qual.go
- key_column_qual_map.go
- key_column_qual_value_map.go
- key_column_slice.go
- key_column_slice_create.go
- list_config.go
- memoize.go
- memoize_configuration.go
- plugin.go
- plugin_aggregator.go
- plugin_connection_config.go
- plugin_validate.go
- query_context.go
- query_data.go
- query_data_cache.go
- query_data_getter.go
- query_status.go
- required_hydrate_calls.go
- retry_config.go
- row_data.go
- schema_mode.go
- serve.go
- table.go
- table_column.go
- table_fetch.go
- table_schema.go
- table_validate.go
Directories ¶
Path | Synopsis |
---|---|
Package context_key provides keys used to retrieve items from the context
|
Package context_key provides keys used to retrieve items from the context |
Package quals is the SDK representation of a SQL query qualifier, i.e.
|
Package quals is the SDK representation of a SQL query qualifier, i.e. |
Package schema provides types used to define the plugin.ConnectionConfigSchema
|
Package schema provides types used to define the plugin.ConnectionConfigSchema |
Package transform defines functions that modify plugin.Column values.
|
Package transform defines functions that modify plugin.Column values. |