// CrossReferences implements part of the xrefs Service interface. func (g *GraphStoreService) CrossReferences(ctx context.Context, req *xpb.CrossReferencesRequest) (*xpb.CrossReferencesReply, error) { // TODO(zarko): Callgraph integration. if len(req.Ticket) == 0 { return nil, errors.New("no cross-references requested") } requestedPageSize := int(req.PageSize) if requestedPageSize == 0 { requestedPageSize = defaultXRefPageSize } eReply, err := g.Edges(ctx, &gpb.EdgesRequest{ Ticket: req.Ticket, PageSize: int32(requestedPageSize), PageToken: req.PageToken, }) if err != nil { return nil, fmt.Errorf("error getting edges for cross-references: %v", err) } reply := &xpb.CrossReferencesReply{ CrossReferences: make(map[string]*xpb.CrossReferencesReply_CrossReferenceSet), NextPageToken: eReply.NextPageToken, } var allRelatedNodes stringset.Set if len(req.Filter) > 0 { reply.Nodes = make(map[string]*cpb.NodeInfo) } // Cache parent files across all anchors files := make(map[string]*fileNode) var totalXRefs int for { for source, es := range eReply.EdgeSets { xr, ok := reply.CrossReferences[source] if !ok { xr = &xpb.CrossReferencesReply_CrossReferenceSet{Ticket: source} } var count int for kind, grp := range es.Groups { switch { // TODO(schroeder): handle declarations case xrefs.IsDefKind(req.DefinitionKind, kind, false): anchors, err := completeAnchors(ctx, g, req.AnchorText, files, kind, edgeTickets(grp.Edge)) if err != nil { return nil, fmt.Errorf("error resolving definition anchors: %v", err) } count += len(anchors) xr.Definition = append(xr.Definition, anchors...) case xrefs.IsRefKind(req.ReferenceKind, kind): anchors, err := completeAnchors(ctx, g, req.AnchorText, files, kind, edgeTickets(grp.Edge)) if err != nil { return nil, fmt.Errorf("error resolving reference anchors: %v", err) } count += len(anchors) xr.Reference = append(xr.Reference, anchors...) case xrefs.IsDocKind(req.DocumentationKind, kind): anchors, err := completeAnchors(ctx, g, req.AnchorText, files, kind, edgeTickets(grp.Edge)) if err != nil { return nil, fmt.Errorf("error resolving documentation anchors: %v", err) } count += len(anchors) xr.Documentation = append(xr.Documentation, anchors...) case !allRelatedNodes.Empty() && !edges.IsAnchorEdge(kind): count += len(grp.Edge) for _, edge := range grp.Edge { xr.RelatedNode = append(xr.RelatedNode, &xpb.CrossReferencesReply_RelatedNode{ Ticket: edge.TargetTicket, RelationKind: kind, Ordinal: edge.Ordinal, }) allRelatedNodes.Add(edge.TargetTicket) } } } if count > 0 { reply.CrossReferences[xr.Ticket] = xr totalXRefs += count } } if reply.NextPageToken == "" || totalXRefs > 0 { break } // We need to return at least 1 xref, if there are any log.Println("Extra CrossReferences Edges call: ", reply.NextPageToken) eReply, err = g.Edges(ctx, &gpb.EdgesRequest{ Ticket: req.Ticket, PageSize: int32(requestedPageSize), PageToken: reply.NextPageToken, }) if err != nil { return nil, fmt.Errorf("error getting edges for cross-references: %v", err) } reply.NextPageToken = eReply.NextPageToken } if !allRelatedNodes.Empty() { nReply, err := g.Nodes(ctx, &gpb.NodesRequest{ Ticket: allRelatedNodes.Elements(), Filter: req.Filter, }) if err != nil { return nil, fmt.Errorf("error retrieving related nodes: %v", err) } for ticket, n := range nReply.Nodes { reply.Nodes[ticket] = n } } return reply, nil }
func (d *DB) edges(ctx context.Context, req *gpb.EdgesRequest, edgeFilter func(kind string) bool) (*gpb.EdgesReply, error) { tickets, err := xrefs.FixTickets(req.Ticket) if err != nil { return nil, err } pageSize := int(req.PageSize) if pageSize <= 0 { pageSize = defaultPageSize } else if pageSize > maxPageSize { pageSize = maxPageSize } var pageOffset int if req.PageToken != "" { rec, err := base64.StdEncoding.DecodeString(req.PageToken) if err != nil { return nil, fmt.Errorf("invalid page_token: %q", req.PageToken) } var t ipb.PageToken if err := proto.Unmarshal(rec, &t); err != nil || t.Index < 0 { return nil, fmt.Errorf("invalid page_token: %q", req.PageToken) } pageOffset = int(t.Index) } // Select only the edges from the given source tickets setQ, args := sqlSetQuery(1, tickets) query := fmt.Sprintf(` SELECT * FROM AllEdges WHERE source IN %s`, setQ) // Filter by edges kinds, if given if len(req.Kind) > 0 { kSetQ, kArgs := sqlSetQuery(1+len(args), req.Kind) query += fmt.Sprintf(` AND kind IN %s`, kSetQ) args = append(args, kArgs...) } // Scan edge sets/groups in order; necessary for CrossReferences query += " ORDER BY source, kind, ordinal" // Seek to the requested page offset (req.PageToken.Index). We don't use // LIMIT here because we don't yet know how many edges will be filtered by // edgeFilter. query += fmt.Sprintf(" OFFSET $%d", len(args)+1) args = append(args, pageOffset) rs, err := d.Query(query, args...) if err != nil { return nil, fmt.Errorf("error querying for edges: %v", err) } defer closeRows(rs) var scanned int // edges := map { source -> kind -> target -> ordinal set } edges := make(map[string]map[string]map[string]map[int32]struct{}, len(tickets)) for count := 0; count < pageSize && rs.Next(); scanned++ { var source, kind, target string var ordinal int if err := rs.Scan(&source, &kind, &target, &ordinal); err != nil { return nil, fmt.Errorf("edges scan error: %v", err) } if edgeFilter != nil && !edgeFilter(kind) { continue } count++ groups, ok := edges[source] if !ok { groups = make(map[string]map[string]map[int32]struct{}) edges[source] = groups } targets, ok := groups[kind] if !ok { targets = make(map[string]map[int32]struct{}) groups[kind] = targets } ordinals, ok := targets[target] if !ok { ordinals = make(map[int32]struct{}) targets[target] = ordinals } ordinals[int32(ordinal)] = struct{}{} } reply := &gpb.EdgesReply{EdgeSets: make(map[string]*gpb.EdgeSet, len(edges))} var nodeTickets stringset.Set for src, groups := range edges { gs := make(map[string]*gpb.EdgeSet_Group, len(groups)) nodeTickets.Add(src) for kind, targets := range groups { edges := make([]*gpb.EdgeSet_Group_Edge, 0, len(targets)) for ticket, ordinals := range targets { for ordinal := range ordinals { edges = append(edges, &gpb.EdgeSet_Group_Edge{ TargetTicket: ticket, Ordinal: ordinal, }) } nodeTickets.Add(ticket) } sort.Sort(xrefs.ByOrdinal(edges)) gs[kind] = &gpb.EdgeSet_Group{ Edge: edges, } } reply.EdgeSets[src] = &gpb.EdgeSet{ Groups: gs, } } // If there is another row, there is a NextPageToken. if rs.Next() { rec, err := proto.Marshal(&ipb.PageToken{Index: int32(pageOffset + scanned)}) if err != nil { return nil, fmt.Errorf("internal error: error marshalling page token: %v", err) } reply.NextPageToken = base64.StdEncoding.EncodeToString(rec) } // TODO(schroederc): faster node lookups if len(req.Filter) > 0 && !nodeTickets.Empty() { nodes, err := d.Nodes(ctx, &gpb.NodesRequest{ Ticket: nodeTickets.Elements(), Filter: req.Filter, }) if err != nil { return nil, fmt.Errorf("error filtering nodes:%v", err) } reply.Nodes = nodes.Nodes } return reply, nil }