From 84be1e80ddb2b467f603c28d3513906eafa9daba Mon Sep 17 00:00:00 2001 From: keyonghan <54558023+keyonghan@users.noreply.github.com> Date: Wed, 13 Mar 2024 09:34:25 -0700 Subject: [PATCH] Query recent commits from firestore (#3572) Part of https://github.com/flutter/flutter/issues/142951. This is to prepare switching get-build-status queries to firestore. This PR: 1) add function queryRecentCommits 2) add query option: `orders` and `limit`. --- app_dart/lib/src/service/firestore.dart | 55 +++++++++- app_dart/test/service/firestore_test.dart | 18 ++++ app_dart/test/src/utilities/mocks.mocks.dart | 106 +++++++++++++------ 3 files changed, 143 insertions(+), 36 deletions(-) diff --git a/app_dart/lib/src/service/firestore.dart b/app_dart/lib/src/service/firestore.dart index 73efa69ab..8285257e0 100644 --- a/app_dart/lib/src/service/firestore.dart +++ b/app_dart/lib/src/service/firestore.dart @@ -9,6 +9,7 @@ import 'package:github/github.dart'; import 'package:googleapis/firestore/v1.dart'; import 'package:http/http.dart'; +import '../model/firestore/commit.dart'; import '../model/firestore/github_gold_status.dart'; import '../model/firestore/task.dart'; import 'access_client_provider.dart'; @@ -18,12 +19,17 @@ const String kDatabase = 'projects/${Config.flutterGcpProjectId}/databases/${Con const String kDocumentParent = '$kDatabase/documents'; const String kFieldFilterOpEqual = 'EQUAL'; const String kCompositeFilterOpAnd = 'AND'; +const String kQueryOrderDescending = 'DESCENDING'; const int kFilterStringSpaceSplitElements = 2; const int kFilterStringSpaceSplitOpIndex = 1; const Map kRelationMapping = { '=': 'EQUAL', + '<': 'LESS_THAN', + '<=': 'LESS_THAN_OR_EQUAL', + '>': 'GREATER_THAN', + '>=': 'GREATER_THAN_OR_EQUAL', }; class FirestoreService { @@ -77,6 +83,31 @@ class FirestoreService { return databasesDocumentsResource.commit(commitRequest, kDatabase); } + /// Queries for recent commits. + /// + /// The [limit] argument specifies the maximum number of commits to retrieve. + /// + /// The returned commits will be ordered by most recent [Commit.timestamp]. + Future> queryRecentCommits({ + int limit = 100, + int? timestamp, + String? branch, + required RepositorySlug slug, + }) async { + timestamp ??= DateTime.now().millisecondsSinceEpoch; + branch ??= Config.defaultBranch(slug); + final Map filterMap = { + '$kCommitBranchField =': branch, + '$kCommitRepositoryPathField =': slug.fullName, + '$kCommitCreateTimestampField <': timestamp, + }; + final Map orderMap = { + kCommitCreateTimestampField: kQueryOrderDescending, + }; + final List documents = await query(kCommitCollectionId, filterMap, orderMap: orderMap, limit: limit); + return documents.map((Document document) => Commit.fromDocument(commitDocument: document)).toList(); + } + /// Returns all tasks running against the speificed [commitSha]. Future> queryCommitTasks(String commitSha) async { final Map filterMap = { @@ -161,6 +192,17 @@ class FirestoreService { ); } + List? generateOrders(Map? orderMap) { + if (orderMap == null || orderMap.isEmpty) { + return null; + } + final List orders = []; + orderMap.forEach((field, direction) { + orders.add(Order(field: FieldReference(fieldPath: field), direction: direction)); + }); + return orders; + } + /// Wrapper to simplify Firestore query. /// /// The [filterMap] follows format: @@ -175,6 +217,8 @@ class FirestoreService { Future> query( String collectionId, Map filterMap, { + int? limit, + Map? orderMap, String compositeFilterOp = kCompositeFilterOpAnd, }) async { final ProjectsDatabasesDocumentsResource databasesDocumentsResource = await documentResource(); @@ -182,8 +226,15 @@ class FirestoreService { CollectionSelector(collectionId: collectionId), ]; final Filter filter = generateFilter(filterMap, compositeFilterOp); - final RunQueryRequest runQueryRequest = - RunQueryRequest(structuredQuery: StructuredQuery(from: from, where: filter)); + final List? orders = generateOrders(orderMap); + final RunQueryRequest runQueryRequest = RunQueryRequest( + structuredQuery: StructuredQuery( + from: from, + where: filter, + orderBy: orders, + limit: limit, + ), + ); final List runQueryResponseElements = await databasesDocumentsResource.runQuery(runQueryRequest, kDocumentParent); return documentsFromQueryResponse(runQueryResponseElements); diff --git a/app_dart/test/service/firestore_test.dart b/app_dart/test/service/firestore_test.dart index 6c13472a3..5b1911082 100644 --- a/app_dart/test/service/firestore_test.dart +++ b/app_dart/test/service/firestore_test.dart @@ -98,4 +98,22 @@ void main() { expect(documents[0].name, githubGoldStatus.name); }); }); + + group('generateOrders', () { + final FirestoreService firestoreService = FirestoreService(AccessClientProvider()); + Map? orderMap; + test('when there is no orderMap', () async { + final List? orders = firestoreService.generateOrders(orderMap); + expect(orders, isNull); + }); + + test('when there is non-null orderMap', () async { + orderMap = {'createTimestamp': kQueryOrderDescending}; + final List? orders = firestoreService.generateOrders(orderMap); + expect(orders!.length, 1); + final Order resultOrder = orders[0]; + expect(resultOrder.direction, kQueryOrderDescending); + expect(resultOrder.field!.fieldPath, 'createTimestamp'); + }); + }); } diff --git a/app_dart/test/src/utilities/mocks.mocks.dart b/app_dart/test/src/utilities/mocks.mocks.dart index df2822c94..0760f3ca9 100644 --- a/app_dart/test/src/utilities/mocks.mocks.dart +++ b/app_dart/test/src/utilities/mocks.mocks.dart @@ -18,12 +18,13 @@ import 'package:cocoon_service/src/model/appengine/github_gold_status_update.dar import 'package:cocoon_service/src/model/appengine/key_helper.dart' as _i12; import 'package:cocoon_service/src/model/appengine/stage.dart' as _i35; import 'package:cocoon_service/src/model/appengine/task.dart' as _i34; -import 'package:cocoon_service/src/model/ci_yaml/target.dart' as _i40; +import 'package:cocoon_service/src/model/ci_yaml/target.dart' as _i41; +import 'package:cocoon_service/src/model/firestore/commit.dart' as _i38; import 'package:cocoon_service/src/model/firestore/github_gold_status.dart' as _i22; -import 'package:cocoon_service/src/model/firestore/task.dart' as _i38; -import 'package:cocoon_service/src/model/github/checks.dart' as _i41; +import 'package:cocoon_service/src/model/firestore/task.dart' as _i39; +import 'package:cocoon_service/src/model/github/checks.dart' as _i42; import 'package:cocoon_service/src/model/luci/buildbucket.dart' as _i8; -import 'package:cocoon_service/src/model/luci/push_message.dart' as _i39; +import 'package:cocoon_service/src/model/luci/push_message.dart' as _i40; import 'package:cocoon_service/src/service/access_client_provider.dart' as _i5; import 'package:cocoon_service/src/service/access_token_provider.dart' as _i27; import 'package:cocoon_service/src/service/bigquery.dart' as _i16; @@ -43,11 +44,11 @@ import 'package:http/http.dart' as _i2; import 'package:mockito/mockito.dart' as _i1; import 'package:mockito/src/dummies.dart' as _i29; import 'package:neat_cache/neat_cache.dart' as _i26; -import 'package:process/src/interface/process_manager.dart' as _i42; +import 'package:process/src/interface/process_manager.dart' as _i43; import 'package:retry/retry.dart' as _i28; import '../../service/cache_service_test.dart' as _i36; -import 'mocks.dart' as _i43; +import 'mocks.dart' as _i44; // ignore_for_file: type=lint // ignore_for_file: avoid_redundant_argument_values @@ -2624,13 +2625,34 @@ class MockFirestoreService extends _i1.Mock implements _i15.FirestoreService { ) as _i20.Future<_i21.CommitResponse>); @override - _i20.Future> queryCommitTasks(String? commitSha) => (super.noSuchMethod( + _i20.Future> queryRecentCommits({ + int? limit = 100, + int? timestamp, + String? branch, + required _i13.RepositorySlug? slug, + }) => + (super.noSuchMethod( + Invocation.method( + #queryRecentCommits, + [], + { + #limit: limit, + #timestamp: timestamp, + #branch: branch, + #slug: slug, + }, + ), + returnValue: _i20.Future>.value(<_i38.Commit>[]), + ) as _i20.Future>); + + @override + _i20.Future> queryCommitTasks(String? commitSha) => (super.noSuchMethod( Invocation.method( #queryCommitTasks, [commitSha], ), - returnValue: _i20.Future>.value(<_i38.Task>[]), - ) as _i20.Future>); + returnValue: _i20.Future>.value(<_i39.Task>[]), + ) as _i20.Future>); @override _i20.Future<_i22.GithubGoldStatus> queryLastGoldStatus( @@ -2701,6 +2723,8 @@ class MockFirestoreService extends _i1.Mock implements _i15.FirestoreService { _i20.Future> query( String? collectionId, Map? filterMap, { + int? limit, + Map? orderMap, String? compositeFilterOp = r'AND', }) => (super.noSuchMethod( @@ -2710,10 +2734,24 @@ class MockFirestoreService extends _i1.Mock implements _i15.FirestoreService { collectionId, filterMap, ], - {#compositeFilterOp: compositeFilterOp}, + { + #limit: limit, + #orderMap: orderMap, + #compositeFilterOp: compositeFilterOp, + }, ), returnValue: _i20.Future>.value(<_i21.Document>[]), ) as _i20.Future>); + + @override + List<_i21.Document> documentsFromQueryResponse(List<_i21.RunQueryResponseElement>? runQueryResponseElements) => + (super.noSuchMethod( + Invocation.method( + #documentsFromQueryResponse, + [runQueryResponseElements], + ), + returnValue: <_i21.Document>[], + ) as List<_i21.Document>); } /// A class which mocks [IssuesService]. @@ -3491,7 +3529,7 @@ class MockGithubChecksService extends _i1.Mock implements _i15.GithubChecksServi @override _i20.Future updateCheckStatus( - _i39.BuildPushMessage? buildPushMessage, + _i40.BuildPushMessage? buildPushMessage, _i15.LuciBuildService? luciBuildService, _i13.RepositorySlug? slug, { bool? rescheduled = false, @@ -3510,7 +3548,7 @@ class MockGithubChecksService extends _i1.Mock implements _i15.GithubChecksServi ) as _i20.Future); @override - bool taskFailed(_i39.BuildPushMessage? buildPushMessage) => (super.noSuchMethod( + bool taskFailed(_i40.BuildPushMessage? buildPushMessage) => (super.noSuchMethod( Invocation.method( #taskFailed, [buildPushMessage], @@ -3519,7 +3557,7 @@ class MockGithubChecksService extends _i1.Mock implements _i15.GithubChecksServi ) as bool); @override - int currentAttempt(_i39.Build? build) => (super.noSuchMethod( + int currentAttempt(_i40.Build? build) => (super.noSuchMethod( Invocation.method( #currentAttempt, [build], @@ -3543,7 +3581,7 @@ class MockGithubChecksService extends _i1.Mock implements _i15.GithubChecksServi ) as String); @override - _i13.CheckRunConclusion conclusionForResult(_i39.Result? result) => (super.noSuchMethod( + _i13.CheckRunConclusion conclusionForResult(_i40.Result? result) => (super.noSuchMethod( Invocation.method( #conclusionForResult, [result], @@ -3558,7 +3596,7 @@ class MockGithubChecksService extends _i1.Mock implements _i15.GithubChecksServi ) as _i13.CheckRunConclusion); @override - _i13.CheckRunStatus statusForResult(_i39.Status? status) => (super.noSuchMethod( + _i13.CheckRunStatus statusForResult(_i40.Status? status) => (super.noSuchMethod( Invocation.method( #statusForResult, [status], @@ -6502,8 +6540,8 @@ class MockLuciBuildService extends _i1.Mock implements _i15.LuciBuildService { ) as _i20.Future>); @override - _i20.Future> scheduleTryBuilds({ - required List<_i40.Target>? targets, + _i20.Future> scheduleTryBuilds({ + required List<_i41.Target>? targets, required _i13.PullRequest? pullRequest, _i31.CheckSuiteEvent? checkSuiteEvent, }) => @@ -6517,8 +6555,8 @@ class MockLuciBuildService extends _i1.Mock implements _i15.LuciBuildService { #checkSuiteEvent: checkSuiteEvent, }, ), - returnValue: _i20.Future>.value(<_i40.Target>[]), - ) as _i20.Future>); + returnValue: _i20.Future>.value(<_i41.Target>[]), + ) as _i20.Future>); @override _i20.Future cancelBuilds( @@ -6540,7 +6578,7 @@ class MockLuciBuildService extends _i1.Mock implements _i15.LuciBuildService { @override _i20.Future> failedBuilds( _i13.PullRequest? pullRequest, - List<_i40.Target>? targets, + List<_i41.Target>? targets, ) => (super.noSuchMethod( Invocation.method( @@ -6556,7 +6594,7 @@ class MockLuciBuildService extends _i1.Mock implements _i15.LuciBuildService { @override _i20.Future<_i8.Build> rescheduleBuild({ required String? builderName, - required _i39.BuildPushMessage? buildPushMessage, + required _i40.BuildPushMessage? buildPushMessage, required int? rescheduleAttempt, }) => (super.noSuchMethod( @@ -6584,7 +6622,7 @@ class MockLuciBuildService extends _i1.Mock implements _i15.LuciBuildService { ) as _i20.Future<_i8.Build>); @override - _i20.Future<_i8.Build> reschedulePresubmitBuildUsingCheckRunEvent(_i41.CheckRunEvent? checkRunEvent) => + _i20.Future<_i8.Build> reschedulePresubmitBuildUsingCheckRunEvent(_i42.CheckRunEvent? checkRunEvent) => (super.noSuchMethod( Invocation.method( #reschedulePresubmitBuildUsingCheckRunEvent, @@ -6614,10 +6652,10 @@ class MockLuciBuildService extends _i1.Mock implements _i15.LuciBuildService { @override _i20.Future<_i8.Build> reschedulePostsubmitBuildUsingCheckRunEvent( - _i41.CheckRunEvent? checkRunEvent, { + _i42.CheckRunEvent? checkRunEvent, { required _i33.Commit? commit, required _i34.Task? task, - required _i40.Target? target, + required _i41.Target? target, }) => (super.noSuchMethod( Invocation.method( @@ -6682,9 +6720,9 @@ class MockLuciBuildService extends _i1.Mock implements _i15.LuciBuildService { ) as _i20.Future>); @override - _i20.Future>> schedulePostsubmitBuilds({ + _i20.Future>> schedulePostsubmitBuilds({ required _i33.Commit? commit, - required List<_i15.Tuple<_i40.Target, _i34.Task, int>>? toBeScheduled, + required List<_i15.Tuple<_i41.Target, _i34.Task, int>>? toBeScheduled, }) => (super.noSuchMethod( Invocation.method( @@ -6695,14 +6733,14 @@ class MockLuciBuildService extends _i1.Mock implements _i15.LuciBuildService { #toBeScheduled: toBeScheduled, }, ), - returnValue: _i20.Future>>.value( - <_i15.Tuple<_i40.Target, _i34.Task, int>>[]), - ) as _i20.Future>>); + returnValue: _i20.Future>>.value( + <_i15.Tuple<_i41.Target, _i34.Task, int>>[]), + ) as _i20.Future>>); @override _i20.Future createPostsubmitCheckRun( _i33.Commit? commit, - _i40.Target? target, + _i41.Target? target, Map? rawUserData, ) => (super.noSuchMethod( @@ -6721,13 +6759,13 @@ class MockLuciBuildService extends _i1.Mock implements _i15.LuciBuildService { @override _i20.Future checkRerunBuilder({ required _i33.Commit? commit, - required _i40.Target? target, + required _i41.Target? target, required _i34.Task? task, required _i9.DatastoreService? datastore, _i15.FirestoreService? firestoreService, Map>? tags, bool? ignoreChecks = false, - _i38.Task? taskDocument, + _i39.Task? taskDocument, }) => (super.noSuchMethod( Invocation.method( @@ -6751,7 +6789,7 @@ class MockLuciBuildService extends _i1.Mock implements _i15.LuciBuildService { /// A class which mocks [ProcessManager]. /// /// See the documentation for Mockito's code generation for more information. -class MockProcessManager extends _i1.Mock implements _i42.ProcessManager { +class MockProcessManager extends _i1.Mock implements _i43.ProcessManager { MockProcessManager() { _i1.throwOnMissingStub(this); } @@ -9482,7 +9520,7 @@ class MockGitHub extends _i1.Mock implements _i13.GitHub { #preview: preview, }, ), - returnValue: _i43.postJsonShim( + returnValue: _i44.postJsonShim( path, statusCode: statusCode, fail: fail,