| // Copyright 2018 The LUCI Authors. |
| // |
| // Licensed under the Apache License, Version 2.0 (the "License"); |
| // you may not use this file except in compliance with the License. |
| // You may obtain a copy of the License at |
| // |
| // http://www.apache.org/licenses/LICENSE-2.0 |
| // |
| // Unless required by applicable law or agreed to in writing, software |
| // distributed under the License is distributed on an "AS IS" BASIS, |
| // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| // See the License for the specific language governing permissions and |
| // limitations under the License. |
| |
| package git |
| |
| import ( |
| "context" |
| "fmt" |
| |
| "google.golang.org/grpc/codes" |
| "google.golang.org/grpc/status" |
| "google.golang.org/protobuf/proto" |
| |
| gerritpb "go.chromium.org/luci/common/proto/gerrit" |
| "go.chromium.org/luci/gae/service/memcache" |
| "go.chromium.org/luci/milo/common" |
| ) |
| |
| // errGRPCNotFound is what gRPC API would have returned for NotFound error. |
| var errGRPCNotFound = status.Errorf(codes.NotFound, "not found") |
| |
| // CLEmail implements Client interface. |
| func (p *implementation) CLEmail(c context.Context, host string, changeNumber int64) (email string, err error) { |
| defer func() { err = common.TagGRPC(c, err) }() |
| changeInfo, err := p.clEmailAndProjectNoACLs(c, host, changeNumber) |
| if err != nil { |
| return |
| } |
| allowed, err := p.acls.IsAllowed(c, host, changeInfo.Project) |
| switch { |
| case err != nil: |
| return |
| case allowed: |
| email = changeInfo.GetOwner().GetEmail() |
| default: |
| err = errGRPCNotFound |
| } |
| return |
| } |
| |
| // clEmailAndProjectNoACLs fetches and caches change owner email and project. |
| // |
| // Gerrit change owner and project are deemed immutable. |
| // Caveat: technically only owner's account id is immutable. Owner's email |
| // associated with this account id may change, but this is rare. |
| func (p *implementation) clEmailAndProjectNoACLs(c context.Context, host string, changeNumber int64) (*gerritpb.ChangeInfo, error) { |
| cache := memcache.NewItem(c, fmt.Sprintf("gerrit-change-owner/%s/%d", host, changeNumber)) |
| if err := memcache.Get(c, cache); err == nil { |
| val := &gerritpb.ChangeInfo{} |
| if err = proto.Unmarshal(cache.Value(), val); err == nil && val.Project != "" { |
| return val, nil |
| } |
| } |
| |
| client, err := p.gerritClient(c, host) |
| if err != nil { |
| return nil, err |
| } |
| changeInfo, err := client.GetChange(c, &gerritpb.GetChangeRequest{ |
| Number: changeNumber, |
| Options: []gerritpb.QueryOption{ |
| gerritpb.QueryOption_DETAILED_ACCOUNTS, |
| gerritpb.QueryOption_SKIP_MERGEABLE, |
| }, |
| }) |
| // We can't cache outcome of not found CL because |
| // * Milo may not at first have access to a CL, say while CL was hidden or |
| // because of bad ACLs. |
| // * Gerrit is known to return 404 flakes. |
| if err != nil { |
| return nil, err |
| } |
| // Cache and return only email and project. |
| ret := &gerritpb.ChangeInfo{ |
| Project: changeInfo.Project, |
| Owner: &gerritpb.AccountInfo{Email: changeInfo.GetOwner().GetEmail()}, |
| } |
| if blob, err := proto.Marshal(ret); err == nil { |
| cache.SetValue(blob) |
| memcache.Set(c, cache) |
| } |
| return ret, nil |
| } |