<!-- PR description--> This PR moves the `api` module from `src/internal/connector/discovery` to `src/pkg/services/m365/api` so that the Client can be reused. #### Does this PR need a docs update or release note? - [ ] ✅ Yes, it's included - [ ] 🕐 Yes, but in a later PR - [x] ⛔ No #### Type of change <!--- Please check the type of change your PR introduces: ---> - [x] 🌻 Feature - [ ] 🐛 Bugfix - [ ] 🗺️ Documentation - [ ] 🤖 Supportability/Tests - [ ] 💻 CI/Deployment - [ ] 🧹 Tech Debt/Cleanup #### Issue(s) <!-- Can reference multiple issues. Use one of the following "magic words" - "closes, fixes" to auto-close the Github issue. --> * #ALC-2214 #### Test Plan <!-- How will this be tested prior to merging.--> - [ ] 💪 Manual - [x] ⚡ Unit test - [ ] 💚 E2E --------- Co-authored-by: aviator-app[bot] <48659329+aviator-app[bot]@users.noreply.github.com>
206 lines
5.7 KiB
Go
206 lines
5.7 KiB
Go
package api
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"net/url"
|
|
"regexp"
|
|
"strings"
|
|
|
|
"github.com/alcionai/clues"
|
|
msgraphgocore "github.com/microsoftgraph/msgraph-sdk-go-core"
|
|
"github.com/microsoftgraph/msgraph-sdk-go/models"
|
|
"github.com/pkg/errors"
|
|
|
|
"github.com/alcionai/corso/src/internal/common/ptr"
|
|
"github.com/alcionai/corso/src/internal/connector/graph"
|
|
"github.com/alcionai/corso/src/internal/connector/graph/betasdk/sites"
|
|
"github.com/alcionai/corso/src/pkg/fault"
|
|
)
|
|
|
|
// ---------------------------------------------------------------------------
|
|
// controller
|
|
// ---------------------------------------------------------------------------
|
|
|
|
func (c Client) Sites() Sites {
|
|
return Sites{c}
|
|
}
|
|
|
|
// Sites is an interface-compliant provider of the client.
|
|
type Sites struct {
|
|
Client
|
|
}
|
|
|
|
// ---------------------------------------------------------------------------
|
|
// methods
|
|
// ---------------------------------------------------------------------------
|
|
|
|
// GetAll retrieves all sites.
|
|
func (c Sites) GetAll(ctx context.Context, errs *fault.Bus) ([]models.Siteable, error) {
|
|
service, err := c.Service()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
resp, err := service.Client().Sites().Get(ctx, nil)
|
|
if err != nil {
|
|
return nil, graph.Wrap(ctx, err, "getting all sites")
|
|
}
|
|
|
|
iter, err := msgraphgocore.NewPageIterator(
|
|
resp,
|
|
service.Adapter(),
|
|
models.CreateSiteCollectionResponseFromDiscriminatorValue)
|
|
if err != nil {
|
|
return nil, graph.Wrap(ctx, err, "creating sites iterator")
|
|
}
|
|
|
|
var (
|
|
us = make([]models.Siteable, 0)
|
|
el = errs.Local()
|
|
)
|
|
|
|
iterator := func(item any) bool {
|
|
if el.Failure() != nil {
|
|
return false
|
|
}
|
|
|
|
s, err := validateSite(item)
|
|
if errors.Is(err, errKnownSkippableCase) {
|
|
// safe to no-op
|
|
return true
|
|
}
|
|
|
|
if err != nil {
|
|
el.AddRecoverable(graph.Wrap(ctx, err, "validating site"))
|
|
return true
|
|
}
|
|
|
|
us = append(us, s)
|
|
|
|
return true
|
|
}
|
|
|
|
if err := iter.Iterate(ctx, iterator); err != nil {
|
|
return nil, graph.Wrap(ctx, err, "enumerating sites")
|
|
}
|
|
|
|
return us, el.Failure()
|
|
}
|
|
|
|
const uuidRE = "[0-9a-fA-F]{8}-[0-9a-fA-F]{4}-[0-9a-fA-F]{4}-[0-9a-fA-F]{4}-[0-9a-fA-F]{12}"
|
|
|
|
// matches a site ID, with or without a doman name. Ex, either one of:
|
|
// 10rqc2.sharepoint.com,deadbeef-0000-0000-0000-000000000000,beefdead-0000-0000-0000-000000000000
|
|
// deadbeef-0000-0000-0000-000000000000,beefdead-0000-0000-0000-000000000000
|
|
var siteIDRE = regexp.MustCompile(`(.+,)?` + uuidRE + "," + uuidRE)
|
|
|
|
const webURLGetTemplate = "https://graph.microsoft.com/v1.0/sites/%s:/%s"
|
|
|
|
// GetByID looks up the site matching the given identifier. The identifier can be either a
|
|
// canonical site id or a webURL. Assumes the webURL is complete and well formed;
|
|
// eg: https://10rqc2.sharepoint.com/sites/Example
|
|
func (c Sites) GetByID(ctx context.Context, identifier string) (models.Siteable, error) {
|
|
var (
|
|
resp models.Siteable
|
|
err error
|
|
)
|
|
|
|
ctx = clues.Add(ctx, "given_site_id", identifier)
|
|
|
|
if siteIDRE.MatchString(identifier) {
|
|
resp, err = c.stable.Client().SitesById(identifier).Get(ctx, nil)
|
|
if err != nil {
|
|
return nil, graph.Wrap(ctx, err, "getting site by id")
|
|
}
|
|
|
|
return resp, err
|
|
}
|
|
|
|
// if the id is not a standard sharepoint ID, assume it's a url.
|
|
// if it has a leading slash, assume it's only a path. If it doesn't,
|
|
// ensure it has a prefix https://
|
|
if !strings.HasPrefix(identifier, "/") {
|
|
identifier = strings.TrimPrefix(identifier, "https://")
|
|
identifier = strings.TrimPrefix(identifier, "http://")
|
|
identifier = "https://" + identifier
|
|
}
|
|
|
|
u, err := url.Parse(identifier)
|
|
if err != nil {
|
|
return nil, clues.Wrap(err, "site is not parseable as a url")
|
|
}
|
|
|
|
// don't construct a path with double leading slashes
|
|
path := strings.TrimPrefix(u.Path, "/")
|
|
rawURL := fmt.Sprintf(webURLGetTemplate, u.Host, path)
|
|
|
|
resp, err = sites.
|
|
NewItemSitesSiteItemRequestBuilder(rawURL, c.stable.Adapter()).
|
|
Get(ctx, nil)
|
|
if err != nil {
|
|
return nil, graph.Wrap(ctx, err, "getting site by weburl")
|
|
}
|
|
|
|
return resp, err
|
|
}
|
|
|
|
// GetIDAndName looks up the site matching the given ID, and returns
|
|
// its canonical ID and the webURL as the name. Accepts an ID or a
|
|
// WebURL as an ID.
|
|
func (c Sites) GetIDAndName(ctx context.Context, siteID string) (string, string, error) {
|
|
s, err := c.GetByID(ctx, siteID)
|
|
if err != nil {
|
|
return "", "", err
|
|
}
|
|
|
|
return ptr.Val(s.GetId()), ptr.Val(s.GetWebUrl()), nil
|
|
}
|
|
|
|
// ---------------------------------------------------------------------------
|
|
// helpers
|
|
// ---------------------------------------------------------------------------
|
|
|
|
var errKnownSkippableCase = clues.New("case is known and skippable")
|
|
|
|
const personalSitePath = "sharepoint.com/personal/"
|
|
|
|
// validateSite ensures the item is a Siteable, and contains the necessary
|
|
// identifiers that we handle with all users.
|
|
// returns the item as a Siteable model.
|
|
func validateSite(item any) (models.Siteable, error) {
|
|
m, ok := item.(models.Siteable)
|
|
if !ok {
|
|
return nil, clues.New(fmt.Sprintf("unexpected model: %T", item))
|
|
}
|
|
|
|
id := ptr.Val(m.GetId())
|
|
if len(id) == 0 {
|
|
return nil, clues.New("missing ID")
|
|
}
|
|
|
|
wURL := ptr.Val(m.GetWebUrl())
|
|
if len(wURL) == 0 {
|
|
return nil, clues.New("missing webURL").With("site_id", id) // TODO: pii
|
|
}
|
|
|
|
// personal (ie: oneDrive) sites have to be filtered out server-side.
|
|
if strings.Contains(wURL, personalSitePath) {
|
|
return nil, clues.Stack(errKnownSkippableCase).
|
|
With("site_id", id, "site_web_url", wURL) // TODO: pii
|
|
}
|
|
|
|
name := ptr.Val(m.GetDisplayName())
|
|
if len(name) == 0 {
|
|
// the built-in site at "https://{tenant-domain}/search" never has a name.
|
|
if strings.HasSuffix(wURL, "/search") {
|
|
return nil, clues.Stack(errKnownSkippableCase).
|
|
With("site_id", id, "site_web_url", wURL) // TODO: pii
|
|
}
|
|
|
|
return nil, clues.New("missing site display name").With("site_id", id)
|
|
}
|
|
|
|
return m, nil
|
|
}
|