Add a script to post-process docs. (#112228)
This commit is contained in:
parent
e167162830
commit
8fbd87065a
155
dev/bots/post_process_docs.dart
Normal file
155
dev/bots/post_process_docs.dart
Normal file
@ -0,0 +1,155 @@
|
|||||||
|
// Copyright 2014 The Flutter Authors. All rights reserved.
|
||||||
|
// Use of this source code is governed by a BSD-style license that can be
|
||||||
|
// found in the LICENSE file.
|
||||||
|
|
||||||
|
import 'dart:convert';
|
||||||
|
import 'dart:io';
|
||||||
|
import 'package:intl/intl.dart';
|
||||||
|
import 'package:meta/meta.dart';
|
||||||
|
|
||||||
|
import 'package:path/path.dart' as path;
|
||||||
|
import 'package:platform/platform.dart' as platform;
|
||||||
|
|
||||||
|
import 'package:process/process.dart';
|
||||||
|
|
||||||
|
const String kDocsRoot = 'dev/docs';
|
||||||
|
const String kPublishRoot = '$kDocsRoot/doc';
|
||||||
|
|
||||||
|
class CommandException implements Exception {}
|
||||||
|
|
||||||
|
Future<void> main() async {
|
||||||
|
await postProcess();
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Post-processes an APIs documentation zip file to modify the footer and version
|
||||||
|
/// strings for commits promoted to either beta or stable channels.
|
||||||
|
Future<void> postProcess() async {
|
||||||
|
final String revision = await gitRevision(fullLength: true);
|
||||||
|
print('Docs revision being processed: $revision');
|
||||||
|
final Directory tmpFolder = Directory.systemTemp.createTempSync();
|
||||||
|
final String zipDestination = path.join(tmpFolder.path, 'api_docs.zip');
|
||||||
|
|
||||||
|
if (!Platform.environment.containsKey('SDK_CHECKOUT_PATH')) {
|
||||||
|
print('SDK_CHECKOUT_PATH env variable is required for this script');
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
final String checkoutPath = Platform.environment['SDK_CHECKOUT_PATH']!;
|
||||||
|
final String docsPath = path.join(checkoutPath, 'dev', 'docs');
|
||||||
|
await runProcessWithValidations(
|
||||||
|
<String>[
|
||||||
|
'curl',
|
||||||
|
'-L',
|
||||||
|
'https://storage.googleapis.com/flutter_infra_release/flutter/$revision/api_docs.zip',
|
||||||
|
'--output',
|
||||||
|
zipDestination,
|
||||||
|
'--fail',
|
||||||
|
],
|
||||||
|
docsPath,
|
||||||
|
);
|
||||||
|
|
||||||
|
// Unzip to docs folder.
|
||||||
|
await runProcessWithValidations(
|
||||||
|
<String>[
|
||||||
|
'unzip',
|
||||||
|
'-o',
|
||||||
|
zipDestination,
|
||||||
|
],
|
||||||
|
docsPath,
|
||||||
|
);
|
||||||
|
|
||||||
|
// Generate versions file.
|
||||||
|
await runProcessWithValidations(
|
||||||
|
<String>['flutter', '--version'],
|
||||||
|
docsPath,
|
||||||
|
);
|
||||||
|
final File versionFile = File('version');
|
||||||
|
final String version = versionFile.readAsStringSync();
|
||||||
|
// Recreate footer
|
||||||
|
final String publishPath = path.join(docsPath, 'doc', 'api', 'footer.js');
|
||||||
|
final File footerFile = File(publishPath)..createSync(recursive: true);
|
||||||
|
createFooter(footerFile, version);
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Gets the git revision of the current checkout. [fullLength] if true will return
|
||||||
|
/// the full commit hash, if false it will return the first 10 characters only.
|
||||||
|
Future<String> gitRevision({
|
||||||
|
bool fullLength = false,
|
||||||
|
@visibleForTesting platform.Platform platform = const platform.LocalPlatform(),
|
||||||
|
@visibleForTesting ProcessManager processManager = const LocalProcessManager(),
|
||||||
|
}) async {
|
||||||
|
const int kGitRevisionLength = 10;
|
||||||
|
|
||||||
|
final ProcessResult gitResult = processManager.runSync(<String>['git', 'rev-parse', 'HEAD']);
|
||||||
|
if (gitResult.exitCode != 0) {
|
||||||
|
throw 'git rev-parse exit with non-zero exit code: ${gitResult.exitCode}';
|
||||||
|
}
|
||||||
|
final String gitRevision = (gitResult.stdout as String).trim();
|
||||||
|
if (fullLength) {
|
||||||
|
return gitRevision;
|
||||||
|
}
|
||||||
|
return gitRevision.length > kGitRevisionLength ? gitRevision.substring(0, kGitRevisionLength) : gitRevision;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Wrapper function to run a subprocess checking exit code and printing stderr and stdout.
|
||||||
|
/// [executable] is a string with the script/binary to execute, [args] is the list of flags/arguments
|
||||||
|
/// and [workingDirectory] is as string to the working directory where the subprocess will be run.
|
||||||
|
Future<void> runProcessWithValidations(
|
||||||
|
List<String> command,
|
||||||
|
String workingDirectory, {
|
||||||
|
@visibleForTesting ProcessManager processManager = const LocalProcessManager(),
|
||||||
|
}) async {
|
||||||
|
final ProcessResult result =
|
||||||
|
processManager.runSync(command, stdoutEncoding: utf8, workingDirectory: workingDirectory);
|
||||||
|
if (result.exitCode == 0) {
|
||||||
|
print('Stdout: ${result.stdout}');
|
||||||
|
} else {
|
||||||
|
print('StdErr: ${result.stderr}');
|
||||||
|
throw CommandException();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Get the name of the release branch.
|
||||||
|
///
|
||||||
|
/// On LUCI builds, the git HEAD is detached, so first check for the env
|
||||||
|
/// variable "LUCI_BRANCH"; if it is not set, fall back to calling git.
|
||||||
|
Future<String> getBranchName({
|
||||||
|
@visibleForTesting platform.Platform platform = const platform.LocalPlatform(),
|
||||||
|
@visibleForTesting ProcessManager processManager = const LocalProcessManager(),
|
||||||
|
}) async {
|
||||||
|
final RegExp gitBranchRegexp = RegExp(r'^## (.*)');
|
||||||
|
final String? luciBranch = platform.environment['LUCI_BRANCH'];
|
||||||
|
if (luciBranch != null && luciBranch.trim().isNotEmpty) {
|
||||||
|
return luciBranch.trim();
|
||||||
|
}
|
||||||
|
final ProcessResult gitResult = processManager.runSync(<String>['git', 'status', '-b', '--porcelain']);
|
||||||
|
if (gitResult.exitCode != 0) {
|
||||||
|
throw 'git status exit with non-zero exit code: ${gitResult.exitCode}';
|
||||||
|
}
|
||||||
|
final RegExpMatch? gitBranchMatch = gitBranchRegexp.firstMatch((gitResult.stdout as String).trim().split('\n').first);
|
||||||
|
return gitBranchMatch == null ? '' : gitBranchMatch.group(1)!.split('...').first;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Updates the footer of the api documentation with the correct branch and versions.
|
||||||
|
/// [footerPath] is the path to the location of the footer js file and [version] is a
|
||||||
|
/// string with the version calculated by the flutter tool.
|
||||||
|
Future<void> createFooter(File footerFile, String version,
|
||||||
|
{@visibleForTesting String? timestampParam,
|
||||||
|
@visibleForTesting String? branchParam,
|
||||||
|
@visibleForTesting String? revisionParam}) async {
|
||||||
|
final String timestamp = timestampParam ?? DateFormat('yyyy-MM-dd HH:mm').format(DateTime.now());
|
||||||
|
final String gitBranch = branchParam ?? await getBranchName();
|
||||||
|
final String revision = revisionParam ?? await gitRevision();
|
||||||
|
final String gitBranchOut = gitBranch.isEmpty ? '' : '• $gitBranch';
|
||||||
|
footerFile.writeAsStringSync('''
|
||||||
|
(function() {
|
||||||
|
var span = document.querySelector('footer>span');
|
||||||
|
if (span) {
|
||||||
|
span.innerText = 'Flutter $version • $timestamp • $revision $gitBranchOut';
|
||||||
|
}
|
||||||
|
var sourceLink = document.querySelector('a.source-link');
|
||||||
|
if (sourceLink) {
|
||||||
|
sourceLink.href = sourceLink.href.replace('/master/', '/$revision/');
|
||||||
|
}
|
||||||
|
})();
|
||||||
|
''');
|
||||||
|
}
|
@ -7,6 +7,7 @@ environment:
|
|||||||
dependencies:
|
dependencies:
|
||||||
args: 2.3.1
|
args: 2.3.1
|
||||||
crypto: 3.0.2
|
crypto: 3.0.2
|
||||||
|
intl: 0.17.0
|
||||||
flutter_devicelab:
|
flutter_devicelab:
|
||||||
path: ../devicelab
|
path: ../devicelab
|
||||||
http_parser: 4.0.1
|
http_parser: 4.0.1
|
||||||
@ -23,6 +24,7 @@ dependencies:
|
|||||||
async: 2.9.0 # THIS LINE IS AUTOGENERATED - TO UPDATE USE "flutter update-packages --force-upgrade"
|
async: 2.9.0 # THIS LINE IS AUTOGENERATED - TO UPDATE USE "flutter update-packages --force-upgrade"
|
||||||
boolean_selector: 2.1.0 # THIS LINE IS AUTOGENERATED - TO UPDATE USE "flutter update-packages --force-upgrade"
|
boolean_selector: 2.1.0 # THIS LINE IS AUTOGENERATED - TO UPDATE USE "flutter update-packages --force-upgrade"
|
||||||
checked_yaml: 2.0.1 # THIS LINE IS AUTOGENERATED - TO UPDATE USE "flutter update-packages --force-upgrade"
|
checked_yaml: 2.0.1 # THIS LINE IS AUTOGENERATED - TO UPDATE USE "flutter update-packages --force-upgrade"
|
||||||
|
clock: 1.1.1 # THIS LINE IS AUTOGENERATED - TO UPDATE USE "flutter update-packages --force-upgrade"
|
||||||
collection: 1.16.0 # THIS LINE IS AUTOGENERATED - TO UPDATE USE "flutter update-packages --force-upgrade"
|
collection: 1.16.0 # THIS LINE IS AUTOGENERATED - TO UPDATE USE "flutter update-packages --force-upgrade"
|
||||||
convert: 3.0.2 # THIS LINE IS AUTOGENERATED - TO UPDATE USE "flutter update-packages --force-upgrade"
|
convert: 3.0.2 # THIS LINE IS AUTOGENERATED - TO UPDATE USE "flutter update-packages --force-upgrade"
|
||||||
coverage: 1.6.1 # THIS LINE IS AUTOGENERATED - TO UPDATE USE "flutter update-packages --force-upgrade"
|
coverage: 1.6.1 # THIS LINE IS AUTOGENERATED - TO UPDATE USE "flutter update-packages --force-upgrade"
|
||||||
@ -69,4 +71,4 @@ dependencies:
|
|||||||
dev_dependencies:
|
dev_dependencies:
|
||||||
test_api: 0.4.14
|
test_api: 0.4.14
|
||||||
|
|
||||||
# PUBSPEC CHECKSUM: 09b7
|
# PUBSPEC CHECKSUM: 7a48
|
||||||
|
165
dev/bots/test/post_process_docs_test.dart
Normal file
165
dev/bots/test/post_process_docs_test.dart
Normal file
@ -0,0 +1,165 @@
|
|||||||
|
// Copyright 2014 The Flutter Authors. All rights reserved.
|
||||||
|
// Use of this source code is governed by a BSD-style license that can be
|
||||||
|
// found in the LICENSE file.
|
||||||
|
|
||||||
|
import 'dart:io';
|
||||||
|
|
||||||
|
import 'package:file/memory.dart';
|
||||||
|
import 'package:platform/platform.dart';
|
||||||
|
|
||||||
|
import '../../../packages/flutter_tools/test/src/fake_process_manager.dart';
|
||||||
|
import '../post_process_docs.dart';
|
||||||
|
import 'common.dart';
|
||||||
|
|
||||||
|
void main() async {
|
||||||
|
group('getBranch', () {
|
||||||
|
const String branchName = 'stable';
|
||||||
|
test('getBranchName does not call git if env LUCI_BRANCH provided', () async {
|
||||||
|
final Platform platform = FakePlatform(
|
||||||
|
environment: <String, String>{
|
||||||
|
'LUCI_BRANCH': branchName,
|
||||||
|
},
|
||||||
|
);
|
||||||
|
final ProcessManager processManager = FakeProcessManager.empty();
|
||||||
|
final String calculatedBranchName = await getBranchName(
|
||||||
|
platform: platform,
|
||||||
|
processManager: processManager,
|
||||||
|
);
|
||||||
|
expect(calculatedBranchName, branchName);
|
||||||
|
});
|
||||||
|
|
||||||
|
test('getBranchName calls git if env LUCI_BRANCH not provided', () async {
|
||||||
|
final Platform platform = FakePlatform(
|
||||||
|
environment: <String, String>{},
|
||||||
|
);
|
||||||
|
|
||||||
|
final ProcessManager processManager = FakeProcessManager.list(
|
||||||
|
<FakeCommand>[
|
||||||
|
const FakeCommand(
|
||||||
|
command: <String>['git', 'status', '-b', '--porcelain'],
|
||||||
|
stdout: '## $branchName',
|
||||||
|
),
|
||||||
|
],
|
||||||
|
);
|
||||||
|
|
||||||
|
final String calculatedBranchName = await getBranchName(platform: platform, processManager: processManager);
|
||||||
|
expect(
|
||||||
|
calculatedBranchName,
|
||||||
|
branchName,
|
||||||
|
);
|
||||||
|
expect(processManager, hasNoRemainingExpectations);
|
||||||
|
});
|
||||||
|
test('getBranchName calls git if env LUCI_BRANCH is empty', () async {
|
||||||
|
final Platform platform = FakePlatform(
|
||||||
|
environment: <String, String>{
|
||||||
|
'LUCI_BRANCH': '',
|
||||||
|
},
|
||||||
|
);
|
||||||
|
|
||||||
|
final ProcessManager processManager = FakeProcessManager.list(
|
||||||
|
<FakeCommand>[
|
||||||
|
const FakeCommand(
|
||||||
|
command: <String>['git', 'status', '-b', '--porcelain'],
|
||||||
|
stdout: '## $branchName',
|
||||||
|
),
|
||||||
|
],
|
||||||
|
);
|
||||||
|
final String calculatedBranchName = await getBranchName(
|
||||||
|
platform: platform,
|
||||||
|
processManager: processManager,
|
||||||
|
);
|
||||||
|
expect(
|
||||||
|
calculatedBranchName,
|
||||||
|
branchName,
|
||||||
|
);
|
||||||
|
expect(processManager, hasNoRemainingExpectations);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
group('gitRevision', () {
|
||||||
|
test('Return short format', () async {
|
||||||
|
const String commitHash = 'e65f01793938e13cac2d321b9fcdc7939f9b2ea6';
|
||||||
|
final ProcessManager processManager = FakeProcessManager.list(
|
||||||
|
<FakeCommand>[
|
||||||
|
const FakeCommand(
|
||||||
|
command: <String>['git', 'rev-parse', 'HEAD'],
|
||||||
|
stdout: commitHash,
|
||||||
|
),
|
||||||
|
],
|
||||||
|
);
|
||||||
|
final String revision = await gitRevision(processManager: processManager);
|
||||||
|
expect(processManager, hasNoRemainingExpectations);
|
||||||
|
expect(revision, commitHash.substring(0, 10));
|
||||||
|
});
|
||||||
|
|
||||||
|
test('Return full length', () async {
|
||||||
|
const String commitHash = 'e65f01793938e13cac2d321b9fcdc7939f9b2ea6';
|
||||||
|
final ProcessManager processManager = FakeProcessManager.list(
|
||||||
|
<FakeCommand>[
|
||||||
|
const FakeCommand(
|
||||||
|
command: <String>['git', 'rev-parse', 'HEAD'],
|
||||||
|
stdout: commitHash,
|
||||||
|
),
|
||||||
|
],
|
||||||
|
);
|
||||||
|
final String revision = await gitRevision(fullLength: true, processManager: processManager);
|
||||||
|
expect(processManager, hasNoRemainingExpectations);
|
||||||
|
expect(revision, commitHash);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
group('runProcessWithValidation', () {
|
||||||
|
test('With no error', () async {
|
||||||
|
const List<String> command = <String>['git', 'rev-parse', 'HEAD'];
|
||||||
|
final ProcessManager processManager = FakeProcessManager.list(
|
||||||
|
<FakeCommand>[
|
||||||
|
const FakeCommand(
|
||||||
|
command: command,
|
||||||
|
),
|
||||||
|
],
|
||||||
|
);
|
||||||
|
await runProcessWithValidations(command, '', processManager: processManager);
|
||||||
|
expect(processManager, hasNoRemainingExpectations);
|
||||||
|
});
|
||||||
|
|
||||||
|
test('With error', () async {
|
||||||
|
const List<String> command = <String>['git', 'rev-parse', 'HEAD'];
|
||||||
|
final ProcessManager processManager = FakeProcessManager.list(
|
||||||
|
<FakeCommand>[
|
||||||
|
const FakeCommand(
|
||||||
|
command: command,
|
||||||
|
exitCode: 1,
|
||||||
|
),
|
||||||
|
],
|
||||||
|
);
|
||||||
|
try {
|
||||||
|
await runProcessWithValidations(command, '', processManager: processManager);
|
||||||
|
throw Exception('Exception was not thrown');
|
||||||
|
} on CommandException catch (e) {
|
||||||
|
expect(e, isA<Exception>());
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
group('generateFooter', () {
|
||||||
|
test('generated correctly', () async {
|
||||||
|
const String expectedContent = '''
|
||||||
|
(function() {
|
||||||
|
var span = document.querySelector('footer>span');
|
||||||
|
if (span) {
|
||||||
|
span.innerText = 'Flutter 3.0.0 • 2022-09-22 14:09 • abcdef • stable';
|
||||||
|
}
|
||||||
|
var sourceLink = document.querySelector('a.source-link');
|
||||||
|
if (sourceLink) {
|
||||||
|
sourceLink.href = sourceLink.href.replace('/master/', '/abcdef/');
|
||||||
|
}
|
||||||
|
})();
|
||||||
|
''';
|
||||||
|
final MemoryFileSystem fs = MemoryFileSystem();
|
||||||
|
final File footerFile = fs.file('/a/b/c/footer.js')..createSync(recursive: true);
|
||||||
|
await createFooter(footerFile, '3.0.0', timestampParam: '2022-09-22 14:09', branchParam: 'stable', revisionParam: 'abcdef');
|
||||||
|
final String content = await footerFile.readAsString();
|
||||||
|
expect(content, expectedContent);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
Loading…
x
Reference in New Issue
Block a user