diff --git a/analysis_options.yaml b/analysis_options.yaml index 8a15d68f..646bccec 100644 --- a/analysis_options.yaml +++ b/analysis_options.yaml @@ -2,3 +2,4 @@ include: package:very_good_analysis/analysis_options.5.1.0.yaml analyzer: exclude: - "**/version.dart" + - "bricks/**/__brick__" diff --git a/lib/src/pub_license/pub_license.dart b/lib/src/pub_license/pub_license.dart new file mode 100644 index 00000000..7a0cf3bd --- /dev/null +++ b/lib/src/pub_license/pub_license.dart @@ -0,0 +1,137 @@ +/// Enables checking a package's license from pub.dev. +/// +/// This library is intented to be used by Very Good CLI to help extracting +/// license information. The existance of this library is likely to be +/// ephemeral. It may be obsolete once [pub.dev](https://pub.dev/) exposes +/// stable license information in their official API; you may track the +/// progress [here](https://github.com/dart-lang/pub-dev/issues/4717). +library pub_license; + +import 'package:html/dom.dart' as html_dom; +import 'package:html/parser.dart' as html_parser; +import 'package:http/http.dart' as http; +import 'package:meta/meta.dart'; + +/// The pub.dev [Uri] used to retrieve the license of a package. +Uri _pubPackageLicenseUri(String packageName) => + Uri.parse('https://pub.dev/packages/$packageName/license'); + +/// {@template pub_license_exception} +/// An exception thrown by [PubLicense]. +/// {@endtemplate} +class PubLicenseException implements Exception { + /// {@macro pub_license_exception} + const PubLicenseException(String message) + : message = '[pub_license] $message'; + + /// The exception message. + final String message; +} + +/// The function signature for parsing HTML documents. +@visibleForTesting +typedef HtmlDocumentParse = html_dom.Document Function( + dynamic input, { + String? encoding, + bool generateSpans, + String? sourceUrl, +}); + +/// {@template pub_license} +/// Enables checking pub.dev's hosted packages license. +/// {@endtemplate} +class PubLicense { + /// {@macro pub_license} + PubLicense({ + @visibleForTesting http.Client? client, + @visibleForTesting HtmlDocumentParse? parse, + }) : _client = client ?? http.Client(), + _parse = parse ?? html_parser.parse; + + final http.Client _client; + + final html_dom.Document Function( + dynamic input, { + String? encoding, + bool generateSpans, + String? sourceUrl, + }) _parse; + + /// Retrieves the license of a package. + /// + /// Some packages may have multiple licenses, hence a [Set] is returned. + /// + /// It may throw a [PubLicenseException] if: + /// * The response from pub.dev is not successful. + /// * The response body cannot be parsed. + Future> getLicense(String packageName) async { + final response = await _client.get(_pubPackageLicenseUri(packageName)); + + if (response.statusCode != 200) { + throw PubLicenseException( + '''Failed to retrieve the license of the package, received status code: ${response.statusCode}''', + ); + } + + late final html_dom.Document document; + try { + document = _parse(response.body); + } on html_parser.ParseError catch (e) { + throw PubLicenseException( + 'Failed to parse the response body, received error: $e', + ); + } catch (e) { + throw PubLicenseException( + '''An unknown error occurred when trying to parse the response body, received error: $e''', + ); + } + + return _scrapeLicense(document); + } +} + +/// Scrapes the license from the pub.dev's package license page. +/// +/// The expected HTML structure is: +/// ```html +/// +/// ``` +/// +/// It may throw a [PubLicenseException] if: +/// * The detail info box is not found. +/// * The license header is not found. +Set _scrapeLicense(html_dom.Document document) { + final detailInfoBox = document.querySelector('.detail-info-box'); + if (detailInfoBox == null) { + throw const PubLicenseException( + '''Failed to scrape license because `.detail-info-box` was not found.''', + ); + } + + String? rawLicenseText; + for (var i = 0; i < detailInfoBox.children.length; i++) { + final child = detailInfoBox.children[i]; + + final headerText = child.text.trim().toLowerCase(); + if (headerText == 'license') { + rawLicenseText = detailInfoBox.children[i + 1].text.trim(); + break; + } + } + if (rawLicenseText == null) { + throw const PubLicenseException( + '''Failed to scrape license because the license header was not found.''', + ); + } + + final licenseText = rawLicenseText.split('(').first.trim(); + return licenseText.split(',').map((e) => e.trim()).toSet(); +} diff --git a/pubspec.yaml b/pubspec.yaml index e0790c8a..5029fd53 100644 --- a/pubspec.yaml +++ b/pubspec.yaml @@ -15,6 +15,8 @@ dependencies: cli_completion: ^0.4.0 collection: ^1.17.1 glob: ^2.0.2 + html: ^0.15.4 # This dependency is temporary and should be removed once pub_license is obsolete. + http: ^1.1.0 # This dependency is temporary and should be removed once pub_license is obsolete. lcov_parser: ^0.1.2 mason: 0.1.0-dev.51 mason_logger: ^0.2.2 diff --git a/test/src/pub_license/fixtures/generate_pub_license_fixtures.dart b/test/src/pub_license/fixtures/generate_pub_license_fixtures.dart new file mode 100644 index 00000000..476b70fe --- /dev/null +++ b/test/src/pub_license/fixtures/generate_pub_license_fixtures.dart @@ -0,0 +1,74 @@ +/// A small script used to generate the fixture for the pub_license test. +/// +/// Fixtures are simply a temporary snapshot of an HTML response from pub.dev. +/// The generated fixtures allow testing pub_license scraping logic without +/// making a request to pub.dev every time the test is run. +/// +/// To run this script, use the following command: +/// ```bash +/// dart test/src/pub_license/fixtures/generate_pub_license_fixtures.dart +/// ``` +/// +/// Or simply use the "Run" CodeLens from VSCode's Dart extension if running +/// from VSCode. +library generate_pub_license_fixtures; + +// ignore_for_file: avoid_print + +import 'dart:io'; + +import 'package:http/http.dart' as http; +import 'package:path/path.dart' as path; + +/// [Uri] used to test the case where a package has a single license. +final _singleLicenseUri = Uri.parse( + 'https://pub.dev/packages/very_good_cli/license', +); + +/// [Uri] used to test the case where a package has multiple licenses. +final _multipleLicenseUri = Uri.parse( + 'https://pub.dev/packages/just_audio/license', +); + +/// [Uri] used to test the case where a package has no license. +final _noLicenseUri = Uri.parse( + 'https://pub.dev/packages/music_control_notification/license', +); + +Future main() async { + final fixtureUris = { + 'singleLicense': _singleLicenseUri, + 'multipleLicense': _multipleLicenseUri, + 'noLicense': _noLicenseUri, + }; + + final httpClient = http.Client(); + + for (final entry in fixtureUris.entries) { + final name = entry.key; + final uri = entry.value; + + final response = await httpClient.get(uri); + + if (response.statusCode != 200) { + print( + '''Failed to generate a fixture for $name, received status code: ${response.statusCode}''', + ); + continue; + } + + final fixturePath = path.joinAll([ + Directory.current.path, + 'test', + 'src', + 'pub_license', + 'fixtures', + '$name.html', + ]); + File(fixturePath) + ..createSync(recursive: true) + ..writeAsStringSync(response.body); + + print('Fixture generated at $fixturePath'); + } +} diff --git a/test/src/pub_license/fixtures/multipleLicense.html b/test/src/pub_license/fixtures/multipleLicense.html new file mode 100644 index 00000000..1b3902bb --- /dev/null +++ b/test/src/pub_license/fixtures/multipleLicense.html @@ -0,0 +1,231 @@ + +just_audio | Flutter Package
large Flutter Favorite logosmall Flutter Favorite logo

just_audio 0.9.35 icon indicating copy to clipboard operation
just_audio: ^0.9.35 copied to clipboard

A feature-rich audio player for Flutter. Loop, clip and concatenate any sound from any source (asset/file/URL/stream) in a variety of audio formats with gapless playback.

License

MIT License
+
+Copyright (c) 2019-2020 Ryan Heise and the project contributors.
+
+Permission is hereby granted, free of charge, to any person obtaining a copy
+of this software and associated documentation files (the "Software"), to deal
+in the Software without restriction, including without limitation the rights
+to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
+copies of the Software, and to permit persons to whom the Software is
+furnished to do so, subject to the following conditions:
+
+The above copyright notice and this permission notice shall be included in all
+copies or substantial portions of the Software.
+
+THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
+IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
+FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
+AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
+LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
+OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
+SOFTWARE.
+
+==============================================================================
+
+This software includes the ExoPlayer library which is licensed under the Apache
+License, Version 2.0.
+
+
+                                 Apache License
+                           Version 2.0, January 2004
+                        http://www.apache.org/licenses/
+
+   TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION
+
+   1. Definitions.
+
+      "License" shall mean the terms and conditions for use, reproduction,
+      and distribution as defined by Sections 1 through 9 of this document.
+
+      "Licensor" shall mean the copyright owner or entity authorized by
+      the copyright owner that is granting the License.
+
+      "Legal Entity" shall mean the union of the acting entity and all
+      other entities that control, are controlled by, or are under common
+      control with that entity. For the purposes of this definition,
+      "control" means (i) the power, direct or indirect, to cause the
+      direction or management of such entity, whether by contract or
+      otherwise, or (ii) ownership of fifty percent (50%) or more of the
+      outstanding shares, or (iii) beneficial ownership of such entity.
+
+      "You" (or "Your") shall mean an individual or Legal Entity
+      exercising permissions granted by this License.
+
+      "Source" form shall mean the preferred form for making modifications,
+      including but not limited to software source code, documentation
+      source, and configuration files.
+
+      "Object" form shall mean any form resulting from mechanical
+      transformation or translation of a Source form, including but
+      not limited to compiled object code, generated documentation,
+      and conversions to other media types.
+
+      "Work" shall mean the work of authorship, whether in Source or
+      Object form, made available under the License, as indicated by a
+      copyright notice that is included in or attached to the work
+      (an example is provided in the Appendix below).
+
+      "Derivative Works" shall mean any work, whether in Source or Object
+      form, that is based on (or derived from) the Work and for which the
+      editorial revisions, annotations, elaborations, or other modifications
+      represent, as a whole, an original work of authorship. For the purposes
+      of this License, Derivative Works shall not include works that remain
+      separable from, or merely link (or bind by name) to the interfaces of,
+      the Work and Derivative Works thereof.
+
+      "Contribution" shall mean any work of authorship, including
+      the original version of the Work and any modifications or additions
+      to that Work or Derivative Works thereof, that is intentionally
+      submitted to Licensor for inclusion in the Work by the copyright owner
+      or by an individual or Legal Entity authorized to submit on behalf of
+      the copyright owner. For the purposes of this definition, "submitted"
+      means any form of electronic, verbal, or written communication sent
+      to the Licensor or its representatives, including but not limited to
+      communication on electronic mailing lists, source code control systems,
+      and issue tracking systems that are managed by, or on behalf of, the
+      Licensor for the purpose of discussing and improving the Work, but
+      excluding communication that is conspicuously marked or otherwise
+      designated in writing by the copyright owner as "Not a Contribution."
+
+      "Contributor" shall mean Licensor and any individual or Legal Entity
+      on behalf of whom a Contribution has been received by Licensor and
+      subsequently incorporated within the Work.
+
+   2. Grant of Copyright License. Subject to the terms and conditions of
+      this License, each Contributor hereby grants to You a perpetual,
+      worldwide, non-exclusive, no-charge, royalty-free, irrevocable
+      copyright license to reproduce, prepare Derivative Works of,
+      publicly display, publicly perform, sublicense, and distribute the
+      Work and such Derivative Works in Source or Object form.
+
+   3. Grant of Patent License. Subject to the terms and conditions of
+      this License, each Contributor hereby grants to You a perpetual,
+      worldwide, non-exclusive, no-charge, royalty-free, irrevocable
+      (except as stated in this section) patent license to make, have made,
+      use, offer to sell, sell, import, and otherwise transfer the Work,
+      where such license applies only to those patent claims licensable
+      by such Contributor that are necessarily infringed by their
+      Contribution(s) alone or by combination of their Contribution(s)
+      with the Work to which such Contribution(s) was submitted. If You
+      institute patent litigation against any entity (including a
+      cross-claim or counterclaim in a lawsuit) alleging that the Work
+      or a Contribution incorporated within the Work constitutes direct
+      or contributory patent infringement, then any patent licenses
+      granted to You under this License for that Work shall terminate
+      as of the date such litigation is filed.
+
+   4. Redistribution. You may reproduce and distribute copies of the
+      Work or Derivative Works thereof in any medium, with or without
+      modifications, and in Source or Object form, provided that You
+      meet the following conditions:
+
+      (a) You must give any other recipients of the Work or
+          Derivative Works a copy of this License; and
+
+      (b) You must cause any modified files to carry prominent notices
+          stating that You changed the files; and
+
+      (c) You must retain, in the Source form of any Derivative Works
+          that You distribute, all copyright, patent, trademark, and
+          attribution notices from the Source form of the Work,
+          excluding those notices that do not pertain to any part of
+          the Derivative Works; and
+
+      (d) If the Work includes a "NOTICE" text file as part of its
+          distribution, then any Derivative Works that You distribute must
+          include a readable copy of the attribution notices contained
+          within such NOTICE file, excluding those notices that do not
+          pertain to any part of the Derivative Works, in at least one
+          of the following places: within a NOTICE text file distributed
+          as part of the Derivative Works; within the Source form or
+          documentation, if provided along with the Derivative Works; or,
+          within a display generated by the Derivative Works, if and
+          wherever such third-party notices normally appear. The contents
+          of the NOTICE file are for informational purposes only and
+          do not modify the License. You may add Your own attribution
+          notices within Derivative Works that You distribute, alongside
+          or as an addendum to the NOTICE text from the Work, provided
+          that such additional attribution notices cannot be construed
+          as modifying the License.
+
+      You may add Your own copyright statement to Your modifications and
+      may provide additional or different license terms and conditions
+      for use, reproduction, or distribution of Your modifications, or
+      for any such Derivative Works as a whole, provided Your use,
+      reproduction, and distribution of the Work otherwise complies with
+      the conditions stated in this License.
+
+   5. Submission of Contributions. Unless You explicitly state otherwise,
+      any Contribution intentionally submitted for inclusion in the Work
+      by You to the Licensor shall be under the terms and conditions of
+      this License, without any additional terms or conditions.
+      Notwithstanding the above, nothing herein shall supersede or modify
+      the terms of any separate license agreement you may have executed
+      with Licensor regarding such Contributions.
+
+   6. Trademarks. This License does not grant permission to use the trade
+      names, trademarks, service marks, or product names of the Licensor,
+      except as required for reasonable and customary use in describing the
+      origin of the Work and reproducing the content of the NOTICE file.
+
+   7. Disclaimer of Warranty. Unless required by applicable law or
+      agreed to in writing, Licensor provides the Work (and each
+      Contributor provides its Contributions) on an "AS IS" BASIS,
+      WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or
+      implied, including, without limitation, any warranties or conditions
+      of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A
+      PARTICULAR PURPOSE. You are solely responsible for determining the
+      appropriateness of using or redistributing the Work and assume any
+      risks associated with Your exercise of permissions under this License.
+
+   8. Limitation of Liability. In no event and under no legal theory,
+      whether in tort (including negligence), contract, or otherwise,
+      unless required by applicable law (such as deliberate and grossly
+      negligent acts) or agreed to in writing, shall any Contributor be
+      liable to You for damages, including any direct, indirect, special,
+      incidental, or consequential damages of any character arising as a
+      result of this License or out of the use or inability to use the
+      Work (including but not limited to damages for loss of goodwill,
+      work stoppage, computer failure or malfunction, or any and all
+      other commercial damages or losses), even if such Contributor
+      has been advised of the possibility of such damages.
+
+   9. Accepting Warranty or Additional Liability. While redistributing
+      the Work or Derivative Works thereof, You may choose to offer,
+      and charge a fee for, acceptance of support, warranty, indemnity,
+      or other liability obligations and/or rights consistent with this
+      License. However, in accepting such obligations, You may act only
+      on Your own behalf and on Your sole responsibility, not on behalf
+      of any other Contributor, and only if You agree to indemnify,
+      defend, and hold each Contributor harmless for any liability
+      incurred by, or claims asserted against, such Contributor by reason
+      of your accepting any such warranty or additional liability.
+
+   END OF TERMS AND CONDITIONS
+
+   APPENDIX: How to apply the Apache License to your work.
+
+      To apply the Apache License to your work, attach the following
+      boilerplate notice, with the fields enclosed by brackets "[]"
+      replaced with your own identifying information. (Don't include
+      the brackets!)  The text should be enclosed in the appropriate
+      comment syntax for the file format. We also recommend that a
+      file or class name and description of purpose be included on the
+      same "printed page" as the copyright notice for easier
+      identification within third-party archives.
+
+   Copyright [yyyy] [name of copyright owner]
+
+   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.
+
\ No newline at end of file diff --git a/test/src/pub_license/fixtures/noLicense.html b/test/src/pub_license/fixtures/noLicense.html new file mode 100644 index 00000000..bd818452 --- /dev/null +++ b/test/src/pub_license/fixtures/noLicense.html @@ -0,0 +1,3 @@ + +music_control_notification | Flutter Package

music_control_notification 0.0.1+1 icon indicating copy to clipboard operation
music_control_notification: ^0.0.1+1 copied to clipboard

Android通知栏音乐控制

License

TODO: Add your license here.
+
\ No newline at end of file diff --git a/test/src/pub_license/fixtures/singleLicense.html b/test/src/pub_license/fixtures/singleLicense.html new file mode 100644 index 00000000..f64e3a38 --- /dev/null +++ b/test/src/pub_license/fixtures/singleLicense.html @@ -0,0 +1,22 @@ + +very_good_cli | Dart Package

very_good_cli 0.16.0 icon indicating copy to clipboard operation
very_good_cli: ^0.16.0 copied to clipboard

A Very Good Command-Line Interface for Dart created by Very Good Ventures.

License

MIT License
+
+Copyright (c) 2021 Very Good Ventures
+
+Permission is hereby granted, free of charge, to any person obtaining a copy
+of this software and associated documentation files (the "Software"), to deal
+in the Software without restriction, including without limitation the rights
+to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
+copies of the Software, and to permit persons to whom the Software is
+furnished to do so, subject to the following conditions:
+
+The above copyright notice and this permission notice shall be included in all
+copies or substantial portions of the Software.
+
+THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
+IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
+FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
+AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
+LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
+OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
+SOFTWARE.
\ No newline at end of file diff --git a/test/src/pub_license/pub_license_test.dart b/test/src/pub_license/pub_license_test.dart new file mode 100644 index 00000000..9b346e18 --- /dev/null +++ b/test/src/pub_license/pub_license_test.dart @@ -0,0 +1,223 @@ +import 'dart:io'; + +import 'package:html/dom.dart' as html_dom; +import 'package:html/parser.dart' as html_parser; +import 'package:http/http.dart' as http; +import 'package:mocktail/mocktail.dart'; +import 'package:path/path.dart' as path; +import 'package:test/test.dart'; +import 'package:very_good_cli/src/pub_license/pub_license.dart'; + +class _MockClient extends Mock implements http.Client {} + +class _MockResponse extends Mock implements http.Response {} + +class _MockParseError extends Mock implements html_parser.ParseError { + @override + String toString({dynamic color}) => super.toString(); +} + +class _MockDocument extends Mock implements html_dom.Document {} + +class _MockElement extends Mock implements html_dom.Element {} + +void main() { + group('PubLicense', () { + late http.Client client; + late http.Response response; + + setUpAll(() { + registerFallbackValue(Uri.parse('https://vgv.dev/')); + }); + + setUp(() { + response = _MockResponse(); + when(() => response.statusCode).thenReturn(200); + when(() => response.body).thenReturn(''); + + client = _MockClient(); + when(() => client.get(any())).thenAnswer((_) async => response); + }); + + test('can be instantiated', () { + expect(PubLicense(), isA()); + }); + + group('getLicense', () { + group('returns as expected', () { + String fixturePath(String name) => path.joinAll([ + Directory.current.path, + 'test', + 'src', + 'pub_license', + 'fixtures', + '$name.html', + ]); + + test('when parsing a single license fixture', () async { + final fixture = File(fixturePath('singleLicense')).readAsStringSync(); + when(() => response.body).thenReturn(fixture); + + final pubLicense = PubLicense(client: client); + final license = await pubLicense.getLicense('very_good_cli'); + + expect(license.length, equals(1)); + expect(license.first, equals('MIT')); + }); + + test('when parsing a multiple license fixture', () async { + final fixture = + File(fixturePath('multipleLicense')).readAsStringSync(); + when(() => response.body).thenReturn(fixture); + + final pubLicense = PubLicense(client: client); + final license = await pubLicense.getLicense('just_audio'); + + expect(license.length, equals(2)); + expect(license.first, equals('Apache-2.0')); + expect(license.last, equals('MIT')); + }); + + test('when parsing a no license fixture', () async { + final fixture = File(fixturePath('noLicense')).readAsStringSync(); + when(() => response.body).thenReturn(fixture); + + final pubLicense = PubLicense(client: client); + final license = await pubLicense.getLicense( + 'music_control_notification', + ); + + expect(license.length, equals(1)); + expect(license.first, equals('unknown')); + }); + }); + + group('throws a PubLicenseException', () { + test('when statusCode is not 200', () async { + when(() => response.statusCode).thenReturn(404); + + final pubLicense = PubLicense(client: client); + + final errorMessage = + '''[pub_license] Failed to retrieve the license of the package, received status code: ${response.statusCode}'''; + await expectLater( + () => pubLicense.getLicense('very_good_cli'), + throwsA( + isA().having( + (exception) => exception.message, + 'message', + equals(errorMessage), + ), + ), + ); + }); + + group('when parsing fails', () { + test('with a ParseError', () async { + final parseError = _MockParseError(); + final pubLicense = PubLicense( + client: client, + parse: (input, {encoding, generateSpans = true, sourceUrl}) => + throw parseError, + ); + + final errorMessage = + '''[pub_license] Failed to parse the response body, received error: $parseError'''; + await expectLater( + () => pubLicense.getLicense('very_good_cli'), + throwsA( + isA().having( + (exception) => exception.message, + 'message', + equals(errorMessage), + ), + ), + ); + }); + + test('with an unexpected error', () async { + const error = 'unexpected error'; + final pubLicense = PubLicense( + client: client, + parse: (input, {encoding, generateSpans = true, sourceUrl}) => + // ignore: only_throw_errors + throw error, + ); + + const errorMessage = + '''[pub_license] An unknown error occurred when trying to parse the response body, received error: $error'''; + await expectLater( + () => pubLicense.getLicense('very_good_cli'), + throwsA( + isA().having( + (exception) => exception.message, + 'message', + equals(errorMessage), + ), + ), + ); + }); + }); + + group('when scraping fails', () { + late html_dom.Document document; + late html_dom.Element element; + + setUp(() { + document = _MockDocument(); + element = _MockElement(); + }); + + test('due to missing `.detail-info-box`', () async { + when(() => document.querySelector('.detail-info-box')) + .thenReturn(null); + + final pubLicense = PubLicense( + client: client, + parse: (input, {encoding, generateSpans = true, sourceUrl}) => + document, + ); + + const errorMessage = + '''[pub_license] Failed to scrape license because `.detail-info-box` was not found.'''; + await expectLater( + () => pubLicense.getLicense('very_good_cli'), + throwsA( + isA().having( + (exception) => exception.message, + 'message', + equals(errorMessage), + ), + ), + ); + }); + + test('due to missing license header', () async { + when(() => document.querySelector('.detail-info-box')) + .thenReturn(element); + when(() => element.children).thenReturn([]); + + final pubLicense = PubLicense( + client: client, + parse: (input, {encoding, generateSpans = true, sourceUrl}) => + document, + ); + + const errorMessage = + '''[pub_license] Failed to scrape license because the license header was not found.'''; + await expectLater( + () => pubLicense.getLicense('very_good_cli'), + throwsA( + isA().having( + (exception) => exception.message, + 'message', + equals(errorMessage), + ), + ), + ); + }); + }); + }); + }); + }); +}