2022-03-23 00:56:33 +00:00
|
|
|
package tables
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
2022-06-21 03:37:07 +00:00
|
|
|
"github.com/aws/aws-sdk-go-v2/feature/dynamodb/expression"
|
2022-03-30 10:55:16 +00:00
|
|
|
"strings"
|
2022-03-24 21:17:52 +00:00
|
|
|
|
2022-03-23 00:56:33 +00:00
|
|
|
"github.com/lmika/awstools/internal/dynamo-browse/models"
|
|
|
|
"github.com/pkg/errors"
|
|
|
|
)
|
|
|
|
|
|
|
|
type Service struct {
|
|
|
|
provider TableProvider
|
|
|
|
}
|
|
|
|
|
|
|
|
func NewService(provider TableProvider) *Service {
|
|
|
|
return &Service{
|
|
|
|
provider: provider,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-03-27 23:19:38 +00:00
|
|
|
func (s *Service) ListTables(ctx context.Context) ([]string, error) {
|
|
|
|
return s.provider.ListTables(ctx)
|
|
|
|
}
|
|
|
|
|
2022-03-24 21:13:43 +00:00
|
|
|
func (s *Service) Describe(ctx context.Context, table string) (*models.TableInfo, error) {
|
|
|
|
return s.provider.DescribeTable(ctx, table)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *Service) Scan(ctx context.Context, tableInfo *models.TableInfo) (*models.ResultSet, error) {
|
2022-06-21 03:37:07 +00:00
|
|
|
return s.doScan(ctx, tableInfo, nil)
|
|
|
|
}
|
|
|
|
|
2022-06-22 01:57:12 +00:00
|
|
|
func (s *Service) doScan(ctx context.Context, tableInfo *models.TableInfo, expr models.Queryable) (*models.ResultSet, error) {
|
|
|
|
var filterExpr *expression.Expression
|
|
|
|
|
|
|
|
if expr != nil {
|
|
|
|
plan, err := expr.Plan(tableInfo)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
// TEMP
|
|
|
|
if plan.CanQuery {
|
|
|
|
return nil, errors.Errorf("queries not yet supported")
|
|
|
|
}
|
|
|
|
|
|
|
|
filterExpr = &plan.Expression
|
|
|
|
}
|
|
|
|
|
2022-06-21 03:37:07 +00:00
|
|
|
results, err := s.provider.ScanItems(ctx, tableInfo.Name, filterExpr, 1000)
|
2022-03-23 00:56:33 +00:00
|
|
|
if err != nil {
|
2022-03-24 21:13:43 +00:00
|
|
|
return nil, errors.Wrapf(err, "unable to scan table %v", tableInfo.Name)
|
2022-03-23 00:56:33 +00:00
|
|
|
}
|
|
|
|
|
2022-03-23 11:02:46 +00:00
|
|
|
// Get the columns
|
2022-06-27 06:05:59 +00:00
|
|
|
//seenColumns := make(map[string]int)
|
|
|
|
//seenColumns[tableInfo.Keys.PartitionKey] = 0
|
|
|
|
//if tableInfo.Keys.SortKey != "" {
|
|
|
|
// seenColumns[tableInfo.Keys.SortKey] = 1
|
|
|
|
//}
|
|
|
|
//
|
|
|
|
//for _, definedAttribute := range tableInfo.DefinedAttributes {
|
|
|
|
// if _, seen := seenColumns[definedAttribute]; !seen {
|
|
|
|
// seenColumns[definedAttribute] = len(seenColumns)
|
|
|
|
// }
|
|
|
|
//}
|
|
|
|
//
|
|
|
|
//otherColsRank := len(seenColumns)
|
|
|
|
//for _, result := range results {
|
|
|
|
// for k := range result {
|
|
|
|
// if _, isSeen := seenColumns[k]; !isSeen {
|
|
|
|
// seenColumns[k] = otherColsRank
|
|
|
|
// }
|
|
|
|
// }
|
|
|
|
//}
|
|
|
|
//
|
|
|
|
//columns := make([]string, 0, len(seenColumns))
|
|
|
|
//for k := range seenColumns {
|
|
|
|
// columns = append(columns, k)
|
|
|
|
//}
|
|
|
|
//sort.Slice(columns, func(i, j int) bool {
|
|
|
|
// if seenColumns[columns[i]] == seenColumns[columns[j]] {
|
|
|
|
// return columns[i] < columns[j]
|
|
|
|
// }
|
|
|
|
// return seenColumns[columns[i]] < seenColumns[columns[j]]
|
|
|
|
//})
|
2022-03-23 00:56:33 +00:00
|
|
|
|
2022-03-24 21:13:43 +00:00
|
|
|
models.Sort(results, tableInfo)
|
2022-03-23 11:02:46 +00:00
|
|
|
|
2022-03-30 10:55:16 +00:00
|
|
|
resultSet := &models.ResultSet{
|
2022-03-24 21:17:52 +00:00
|
|
|
TableInfo: tableInfo,
|
2022-06-22 01:57:12 +00:00
|
|
|
Query: expr,
|
2022-06-27 06:05:59 +00:00
|
|
|
//Columns: columns,
|
2022-03-30 10:55:16 +00:00
|
|
|
}
|
|
|
|
resultSet.SetItems(results)
|
2022-06-27 06:05:59 +00:00
|
|
|
resultSet.RefreshColumns()
|
2022-03-30 10:55:16 +00:00
|
|
|
|
|
|
|
return resultSet, nil
|
2022-03-23 00:56:33 +00:00
|
|
|
}
|
2022-03-23 04:40:31 +00:00
|
|
|
|
2022-03-24 21:13:43 +00:00
|
|
|
func (s *Service) Put(ctx context.Context, tableInfo *models.TableInfo, item models.Item) error {
|
|
|
|
return s.provider.PutItem(ctx, tableInfo.Name, item)
|
2022-03-24 01:54:32 +00:00
|
|
|
}
|
|
|
|
|
2022-05-26 00:17:21 +00:00
|
|
|
func (s *Service) PutItemAt(ctx context.Context, resultSet *models.ResultSet, index int) error {
|
|
|
|
item := resultSet.Items()[index]
|
|
|
|
if err := s.provider.PutItem(ctx, resultSet.TableInfo.Name, item); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
resultSet.SetDirty(index, false)
|
|
|
|
resultSet.SetNew(index, false)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2022-03-30 10:04:30 +00:00
|
|
|
func (s *Service) Delete(ctx context.Context, tableInfo *models.TableInfo, items []models.Item) error {
|
|
|
|
for _, item := range items {
|
|
|
|
if err := s.provider.DeleteItem(ctx, tableInfo.Name, item.KeyValue(tableInfo)); err != nil {
|
|
|
|
return errors.Wrapf(err, "cannot delete item")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
2022-03-23 04:40:31 +00:00
|
|
|
}
|
2022-03-30 10:55:16 +00:00
|
|
|
|
2022-06-22 01:57:12 +00:00
|
|
|
func (s *Service) ScanOrQuery(ctx context.Context, tableInfo *models.TableInfo, expr models.Queryable) (*models.ResultSet, error) {
|
|
|
|
return s.doScan(ctx, tableInfo, expr)
|
2022-06-21 03:37:07 +00:00
|
|
|
}
|
|
|
|
|
2022-03-30 10:55:16 +00:00
|
|
|
// TODO: move into a new service
|
|
|
|
func (s *Service) Filter(resultSet *models.ResultSet, filter string) *models.ResultSet {
|
|
|
|
for i, item := range resultSet.Items() {
|
|
|
|
if filter == "" {
|
|
|
|
resultSet.SetHidden(i, false)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
var shouldHide = true
|
|
|
|
for k := range item {
|
|
|
|
str, ok := item.AttributeValueAsString(k)
|
|
|
|
if !ok {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
if strings.Contains(str, filter) {
|
|
|
|
shouldHide = false
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
resultSet.SetHidden(i, shouldHide)
|
|
|
|
}
|
|
|
|
|
|
|
|
return resultSet
|
|
|
|
}
|