From 0150c6a6383ca6dc167edfe184df9948ddc2431d Mon Sep 17 00:00:00 2001 From: gkc Date: Mon, 7 Oct 2024 11:00:44 +0100 Subject: [PATCH] fix: allow leading underscore in device name to preserve full backwards compatibility re device name validation. --- .../lib/src/common/default_args.dart | 2 +- .../lib/src/common/validation_utils.dart | 7 ++-- .../test/sshnp/models/sshnp_params_test.dart | 34 +++++++++++++++---- .../test/sshnpd/sshnpd_params_test.dart | 10 +++--- 4 files changed, 37 insertions(+), 16 deletions(-) diff --git a/packages/dart/noports_core/lib/src/common/default_args.dart b/packages/dart/noports_core/lib/src/common/default_args.dart index e45f7342a..bceae692d 100644 --- a/packages/dart/noports_core/lib/src/common/default_args.dart +++ b/packages/dart/noports_core/lib/src/common/default_args.dart @@ -54,5 +54,5 @@ class DefaultSshnpdArgs { static const String deviceGroupName = '__none__'; static const String sshPublicKeyPermissions = ""; static const Duration policyHeartbeatFrequency = Duration(minutes: 5); - static const String permitOpen ='localhost:22,localhost:3389'; + static const String permitOpen = 'localhost:22,localhost:3389'; } diff --git a/packages/dart/noports_core/lib/src/common/validation_utils.dart b/packages/dart/noports_core/lib/src/common/validation_utils.dart index b8062358c..6c4a5756e 100644 --- a/packages/dart/noports_core/lib/src/common/validation_utils.dart +++ b/packages/dart/noports_core/lib/src/common/validation_utils.dart @@ -8,10 +8,11 @@ import 'package:noports_core/src/common/file_system_utils.dart'; import 'package:noports_core/src/common/io_types.dart'; import 'package:path/path.dart' as path; -const String sshnpDeviceNameRegex = r'[a-z0-9][a-z0-9_\-]{1,35}'; +const String sshnpDeviceNameRegex = r'[a-z0-9_][a-z0-9_\-]{1,35}'; const String invalidDeviceNameMsg = 'Device name must be alphanumeric' - ' snake case, max length 36. First char must be a-z or 0-9.'; -const String deviceNameFormatHelp = 'Alphanumeric snake case, max length 36. First char must be a-z or 0-9.'; + ' snake case, max length 36. First char must be _, a-z, or 0-9.'; +const String deviceNameFormatHelp = 'Alphanumeric snake case, max length 36.' + ' First char must be _, a-z, or 0-9.'; const String invalidSshKeyPermissionsMsg = 'Detected newline characters in the ssh public key permissions which malforms the authorized_keys file.'; diff --git a/packages/dart/noports_core/test/sshnp/models/sshnp_params_test.dart b/packages/dart/noports_core/test/sshnp/models/sshnp_params_test.dart index c8ac12e2f..ccb4b75f6 100644 --- a/packages/dart/noports_core/test/sshnp/models/sshnp_params_test.dart +++ b/packages/dart/noports_core/test/sshnp/models/sshnp_params_test.dart @@ -130,13 +130,33 @@ void main() { throwsA(TypeMatcher())); }); test('SshnpParams.device invalid must start with a-z or 0-9', () { - expect( - () => SshnpParams( - clientAtSign: '', - sshnpdAtSign: '', - srvdAtSign: '', - device: '_abcde-12345-abcde_12345_abcde_12345'), - throwsA(TypeMatcher())); + final l = [ + '-abcde', + '#abcde', + ' abcde', + '@abcde', + '£abcde', + '\$abcde', + '^abcde', + ]; + for (final s in l) { + expect( + () => SshnpParams( + clientAtSign: '', + sshnpdAtSign: '', + srvdAtSign: '', + device: s), + throwsA(TypeMatcher())); + } + }); + test('SshnpParams.device may start with underscore', () { + String deviceName = '_my-device-name_12345'; + final params = SshnpParams( + clientAtSign: '', + sshnpdAtSign: '', + srvdAtSign: '', + device: deviceName); + expect(params.device, equals(deviceName)); }); test('SshnpParams.device test pure snake case', () { String deviceName = 'my_device_name_12345'; diff --git a/packages/dart/noports_core/test/sshnpd/sshnpd_params_test.dart b/packages/dart/noports_core/test/sshnpd/sshnpd_params_test.dart index dabc885bd..54bc18f69 100644 --- a/packages/dart/noports_core/test/sshnpd/sshnpd_params_test.dart +++ b/packages/dart/noports_core/test/sshnpd/sshnpd_params_test.dart @@ -4,10 +4,10 @@ import 'package:test/test.dart'; void main() { group('test sshnpd params defaults', () { - test('require at least one of managers and policyManager options', () async { + test('require at least one of managers and policyManager options', + () async { List args = '-a @daemon'.split(' '); - await expectLater( - () => SshnpdParams.fromArgs(args), + await expectLater(() => SshnpdParams.fromArgs(args), throwsA(TypeMatcher())); }); test('just managers option supplied', () async { @@ -26,7 +26,7 @@ void main() { final p = await SshnpdParams.fromArgs(args); expect(p.deviceAtsign, '@daemon'); expect(p.policyManagerAtsign, '@policy'); - expect(p.managerAtsigns, ['@bob','@chuck']); + expect(p.managerAtsigns, ['@bob', '@chuck']); }); test('test permitOpen default without policyManager', () async { List args = '-a @daemon -m @bob'.split(' '); @@ -57,4 +57,4 @@ void main() { // local-sshd-port // sshpublickey-permissions }); -} \ No newline at end of file +}