commit 217621daac5cc18075c90d6a5d876923ff5ebe50 Author: Henry-Hiles Date: Wed Jun 18 10:32:03 2025 -0400 initial commit diff --git a/.envrc b/.envrc new file mode 100644 index 0000000..8392d15 --- /dev/null +++ b/.envrc @@ -0,0 +1 @@ +use flake \ No newline at end of file diff --git a/.gitignore b/.gitignore new file mode 100644 index 0000000..aa6922d --- /dev/null +++ b/.gitignore @@ -0,0 +1,2 @@ +.dart_tool/ +.direnv \ No newline at end of file diff --git a/.vscode/launch.json b/.vscode/launch.json new file mode 100644 index 0000000..6d7a358 --- /dev/null +++ b/.vscode/launch.json @@ -0,0 +1,24 @@ +{ + // Use IntelliSense to learn about possible attributes. + // Hover to view descriptions of existing attributes. + // For more information, visit: https://go.microsoft.com/fwlink/?linkid=830387 + "version": "0.2.0", + "configurations": [ + { + "name": "matrix-oauth2oidc", + "program": "bin/matrixgate.dart", + "args": [ + "--homeserver", + "foo", + "--jwtSecretFile", + "foo", + "--issuer", + "http://localhost:8080", + "--authorizeEndpoint", + "https://federated.nexus/login" + ], + "request": "launch", + "type": "dart" + } + ] +} diff --git a/CHANGELOG.md b/CHANGELOG.md new file mode 100644 index 0000000..effe43c --- /dev/null +++ b/CHANGELOG.md @@ -0,0 +1,3 @@ +## 1.0.0 + +- Initial version. diff --git a/JUSTFILE b/JUSTFILE new file mode 100644 index 0000000..c2af145 --- /dev/null +++ b/JUSTFILE @@ -0,0 +1,8 @@ +watch: + dart run build_runner watch + +build: + dart run build_runner build + +test: + oauth2c http://localhost:8080 --client-id yourclientid --redirect-url http://localhost/callback --scopes openid --grant-type authorization_code --auth-method none --response-mode query \ No newline at end of file diff --git a/README.md b/README.md new file mode 100644 index 0000000..fbe6920 --- /dev/null +++ b/README.md @@ -0,0 +1 @@ +An attempt to make an OIDC provider that authenticates with matrix OAuth. diff --git a/analysis_options.yaml b/analysis_options.yaml new file mode 100644 index 0000000..dee8927 --- /dev/null +++ b/analysis_options.yaml @@ -0,0 +1,30 @@ +# This file configures the static analysis results for your project (errors, +# warnings, and lints). +# +# This enables the 'recommended' set of lints from `package:lints`. +# This set helps identify many issues that may lead to problems when running +# or consuming Dart code, and enforces writing Dart using a single, idiomatic +# style and format. +# +# If you want a smaller set of lints you can change this to specify +# 'package:lints/core.yaml'. These are just the most critical lints +# (the recommended set includes the core lints). +# The core lints are also what is used by pub.dev for scoring packages. + +include: package:lints/recommended.yaml + +# Uncomment the following section to specify additional rules. + +# linter: +# rules: +# - camel_case_types + +# analyzer: +# exclude: +# - path/to/excluded/files/** + +# For more information about the core and recommended set of lints, see +# https://dart.dev/go/core-lints + +# For additional information about configuring this file, see +# https://dart.dev/guides/language/analysis-options diff --git a/bin/matrixgate.dart b/bin/matrixgate.dart new file mode 100644 index 0000000..469967d --- /dev/null +++ b/bin/matrixgate.dart @@ -0,0 +1,58 @@ +import "dart:io"; +import "package:args/args.dart"; +import "package:matrixgate/controllers/settings_controller.dart"; +import "package:matrixgate/helpers/api_helper.dart"; +import "package:riverpod/riverpod.dart"; +import "package:shelf/shelf.dart"; +import "package:shelf/shelf_io.dart"; +import "package:shelf_router/shelf_router.dart"; + +void main(List argsRaw) async { + final parser = ArgParser() + ..addOption("socket", abbr: "s") + ..addOption("address", abbr: "a", defaultsTo: "127.0.0.1") + ..addOption("port", abbr: "p", defaultsTo: "8080") + ..addOption("issuer", abbr: "i", mandatory: true) + ..addOption("homeserver", abbr: "h", mandatory: true) + ..addOption("jwtSecretFile", abbr: "j", mandatory: true) + ..addOption("authorizeEndpoint", abbr: "e", mandatory: true); + + final container = ProviderContainer(); + + container + .read(SettingsController.provider.notifier) + .set(parser.parse(argsRaw)); + + final apiHelper = container.read(ApiHelper.provider); + + final handler = const Pipeline() + .addMiddleware(logRequests()) + .addHandler( + (Router() + ..get( + "/.well-known/openid-configuration", + apiHelper.openidConfiguration, + ) + ..get("/userInfo", apiHelper.userinfoHandler) + ..get("/jwks.json", apiHelper.jwks) + ..post("/login", apiHelper.handleLogin) + ..post("/token", apiHelper.tokenHandler)) + .call, + ); + + final args = container.read(SettingsController.provider)!; + final server = HttpServer.listenOn( + await ServerSocket.bind( + InternetAddress( + args.socket ?? args.address, + type: args.socket == null + ? InternetAddressType.any + : InternetAddressType.unix, + ), + int.parse(args.port), + ), + ); + + serveRequests(server, handler); + print("OIDC provider listening at ${server.address.address}:${server.port}"); +} diff --git a/flake.lock b/flake.lock new file mode 100644 index 0000000..4929fe3 --- /dev/null +++ b/flake.lock @@ -0,0 +1,61 @@ +{ + "nodes": { + "flake-parts": { + "inputs": { + "nixpkgs-lib": "nixpkgs-lib" + }, + "locked": { + "lastModified": 1749398372, + "narHash": "sha256-tYBdgS56eXYaWVW3fsnPQ/nFlgWi/Z2Ymhyu21zVM98=", + "owner": "hercules-ci", + "repo": "flake-parts", + "rev": "9305fe4e5c2a6fcf5ba6a3ff155720fbe4076569", + "type": "github" + }, + "original": { + "owner": "hercules-ci", + "repo": "flake-parts", + "type": "github" + } + }, + "nixpkgs": { + "locked": { + "lastModified": 1750134718, + "narHash": "sha256-v263g4GbxXv87hMXMCpjkIxd/viIF7p3JpJrwgKdNiI=", + "owner": "nixos", + "repo": "nixpkgs", + "rev": "9e83b64f727c88a7711a2c463a7b16eedb69a84c", + "type": "github" + }, + "original": { + "owner": "nixos", + "ref": "nixos-unstable", + "repo": "nixpkgs", + "type": "github" + } + }, + "nixpkgs-lib": { + "locked": { + "lastModified": 1748740939, + "narHash": "sha256-rQaysilft1aVMwF14xIdGS3sj1yHlI6oKQNBRTF40cc=", + "owner": "nix-community", + "repo": "nixpkgs.lib", + "rev": "656a64127e9d791a334452c6b6606d17539476e2", + "type": "github" + }, + "original": { + "owner": "nix-community", + "repo": "nixpkgs.lib", + "type": "github" + } + }, + "root": { + "inputs": { + "flake-parts": "flake-parts", + "nixpkgs": "nixpkgs" + } + } + }, + "root": "root", + "version": 7 +} diff --git a/flake.nix b/flake.nix new file mode 100644 index 0000000..ae04dfa --- /dev/null +++ b/flake.nix @@ -0,0 +1,27 @@ +{ + inputs = { + flake-parts.url = "github:hercules-ci/flake-parts"; + nixpkgs.url = "github:nixos/nixpkgs/nixos-unstable"; + }; + + outputs = { + flake-parts, + nixpkgs, + self, + ... + } @ inputs: + flake-parts.lib.mkFlake {inherit inputs;} { + systems = ["x86_64-linux" "aarch64-linux"]; + perSystem = { + pkgs, + system, + ... + }: { + _module.args.pkgs = import nixpkgs {inherit system;}; + + devShells.default = pkgs.mkShell { + buildInputs = with pkgs; [just dart oauth2c watchexec]; + }; + }; + }; +} diff --git a/lib/controllers/auth_code_controller.dart b/lib/controllers/auth_code_controller.dart new file mode 100644 index 0000000..d404b3e --- /dev/null +++ b/lib/controllers/auth_code_controller.dart @@ -0,0 +1,17 @@ +import "package:fast_immutable_collections/fast_immutable_collections.dart"; +import "package:matrixgate/models/matrix_user.dart"; +import "package:riverpod/riverpod.dart"; + +class AuthCodeController extends Notifier> { + @override + IMap build() => const IMap.empty(); + + void set(String name, MatrixUser user) => + state = state.update(name, (_) => user, ifAbsent: () => user); + void remove(String name) => state = state.remove(name); + + static final provider = + NotifierProvider>( + AuthCodeController.new, + ); +} diff --git a/lib/controllers/settings_controller.dart b/lib/controllers/settings_controller.dart new file mode 100644 index 0000000..54ff1d2 --- /dev/null +++ b/lib/controllers/settings_controller.dart @@ -0,0 +1,16 @@ +import "package:args/args.dart"; +import "package:matrixgate/models/settings.dart"; +import "package:riverpod/riverpod.dart"; + +class SettingsController extends Notifier { + @override + Settings? build() => null; + + void set(ArgResults args) => state = Settings.fromJson({ + for (final opt in args.options) opt: args.option(opt), + }); + + static final provider = NotifierProvider( + SettingsController.new, + ); +} diff --git a/lib/helpers/api_helper.dart b/lib/helpers/api_helper.dart new file mode 100644 index 0000000..591259d --- /dev/null +++ b/lib/helpers/api_helper.dart @@ -0,0 +1,159 @@ +import "dart:io"; +import "dart:convert"; +import "package:dart_jsonwebtoken/dart_jsonwebtoken.dart"; +import "package:matrixgate/controllers/auth_code_controller.dart"; +import "package:matrixgate/controllers/settings_controller.dart"; +import "package:shelf/shelf.dart"; +import "package:http/http.dart" as http; +import "package:matrixgate/models/matrix_user.dart"; +import "package:riverpod/riverpod.dart"; + +class ApiHelper { + final Ref ref; + ApiHelper(this.ref); + + Future handleLogin(Request request) async { + final body = await request.readAsString(); + final data = Uri.splitQueryString(body); + final settings = ref.read(SettingsController.provider)!; + + final username = data["username"]; + final password = data["password"]; + final redirectUri = data["redirect_uri"]; + final state = data["state"] ?? ""; + + final loginRes = await http.post( + Uri.https(settings.homeserver, "_matrix/client/v3/login"), + body: json.encode({ + "type": "m.login.password", + "identifier": {"type": "m.id.user", "user": username}, + "password": password, + }), + ); + + if (loginRes.statusCode != 200) { + return Response.forbidden("Login failed"); + } + + final loginData = json.decode(loginRes.body); + final userId = loginData["user_id"]; + final accessToken = loginData["access_token"]; + + final openidRes = await http.post( + Uri.https( + settings.homeserver, + "_matrix/client/v3/user/$userId/openid/request", + ), + headers: {"Authorization": "Bearer $accessToken"}, + ); + + if (openidRes.statusCode != 200) { + return Response.forbidden( + "OpenID request failed, status code ${openidRes.statusCode}", + ); + } + + final openidToken = json.decode(openidRes.body)["access_token"]; + + final code = base64Url.encode( + List.generate(16, (_) => DateTime.now().millisecond % 256), + ); + ref + .read(AuthCodeController.provider.notifier) + .set(code, MatrixUser(userId: userId, matrixToken: openidToken)); + + return Response.found("$redirectUri?code=$code&state=$state"); + } + + Future tokenHandler(Request request) async { + final settings = ref.read(SettingsController.provider)!; + final body = Uri.splitQueryString(await request.readAsString()); + final code = body["code"]; + + final codes = ref.read(AuthCodeController.provider); + + if (code == null || !codes.containsKey(code)) { + return Response(400, body: "Invalid code"); + } + + final user = codes[code]!; + ref.read(AuthCodeController.provider.notifier).remove(code); + + final jwt = JWT( + { + "exp": + DateTime.now().add(Duration(minutes: 10)).millisecondsSinceEpoch ~/ + 1000, + "iat": DateTime.now().millisecondsSinceEpoch ~/ 1000, + }, + subject: user.userId, + issuer: ref.read(SettingsController.provider)!.issuer, + audience: Audience([body["client_id"]!]), + ); + + final token = jwt.sign( + SecretKey( + await File.fromUri(Uri.file(settings.jwtSecretFile)).readAsString(), + ), + algorithm: JWTAlgorithm.HS256, + ); + + return Response.ok( + json.encode({ + "id_token": token, + "token_type": "Bearer", + "expires_in": 600, + }), + headers: {"Content-Type": "application/json"}, + ); + } + + Future userinfoHandler(Request request) async { + final auth = request.headers["authorization"]; + if (auth == null || !auth.startsWith("Bearer ")) { + return Response.forbidden( + json.encode({"error": "missing_token"}), + headers: {"content-type": "application/json"}, + ); + } + + final token = auth.substring(7); + final matrixResp = await http.get( + Uri.https( + ref.read(SettingsController.provider)!.homeserver, + "_matrix/federation/v1/openid/userinfo", + ), + headers: {"Authorization": "Bearer $token"}, + ); + + if (matrixResp.statusCode != 200) { + return Response.forbidden( + json.encode({"error": "invalid_token"}), + headers: {"content-type": "application/json"}, + ); + } + + return Response.ok(matrixResp.body); + } + + Response jwks(_) => Response.ok(json.encode({"keys": []})); + + Response openidConfiguration(_) { + final settings = ref.read(SettingsController.provider)!; + return Response.ok( + json.encode({ + "issuer": settings.issuer, + "authorization_endpoint": settings.authorizeEndpoint, + "token_endpoint": "${settings.issuer}/token", + "userinfo_endpoint": "${settings.issuer}/userInfo", + "jwks_uri": "${settings.issuer}/jwks.json", + "response_types_supported": ["code"], + "subject_types_supported": ["public"], + "id_token_signing_alg_values_supported": ["HS256"], + }), + headers: {"Content-Type": "application/json"}, + ); + } + + static final provider = Provider(ApiHelper.new); +} diff --git a/lib/models/matrix_user.dart b/lib/models/matrix_user.dart new file mode 100644 index 0000000..805fb3f --- /dev/null +++ b/lib/models/matrix_user.dart @@ -0,0 +1,15 @@ +import "package:freezed_annotation/freezed_annotation.dart"; + +part "matrix_user.freezed.dart"; +part "matrix_user.g.dart"; + +@freezed +abstract class MatrixUser with _$MatrixUser { + const factory MatrixUser({ + required String userId, + required String matrixToken, + }) = _MatrixUser; + + factory MatrixUser.fromJson(Map json) => + _$MatrixUserFromJson(json); +} diff --git a/lib/models/matrix_user.freezed.dart b/lib/models/matrix_user.freezed.dart new file mode 100644 index 0000000..0dd35c7 --- /dev/null +++ b/lib/models/matrix_user.freezed.dart @@ -0,0 +1,151 @@ +// dart format width=80 +// coverage:ignore-file +// GENERATED CODE - DO NOT MODIFY BY HAND +// ignore_for_file: type=lint +// ignore_for_file: unused_element, deprecated_member_use, deprecated_member_use_from_same_package, use_function_type_syntax_for_parameters, unnecessary_const, avoid_init_to_null, invalid_override_different_default_values_named, prefer_expression_function_bodies, annotate_overrides, invalid_annotation_target, unnecessary_question_mark + +part of 'matrix_user.dart'; + +// ************************************************************************** +// FreezedGenerator +// ************************************************************************** + +// dart format off +T _$identity(T value) => value; + +/// @nodoc +mixin _$MatrixUser { + + String get userId; String get matrixToken; +/// Create a copy of MatrixUser +/// with the given fields replaced by the non-null parameter values. +@JsonKey(includeFromJson: false, includeToJson: false) +@pragma('vm:prefer-inline') +$MatrixUserCopyWith get copyWith => _$MatrixUserCopyWithImpl(this as MatrixUser, _$identity); + + /// Serializes this MatrixUser to a JSON map. + Map toJson(); + + +@override +bool operator ==(Object other) { + return identical(this, other) || (other.runtimeType == runtimeType&&other is MatrixUser&&(identical(other.userId, userId) || other.userId == userId)&&(identical(other.matrixToken, matrixToken) || other.matrixToken == matrixToken)); +} + +@JsonKey(includeFromJson: false, includeToJson: false) +@override +int get hashCode => Object.hash(runtimeType,userId,matrixToken); + +@override +String toString() { + return 'MatrixUser(userId: $userId, matrixToken: $matrixToken)'; +} + + +} + +/// @nodoc +abstract mixin class $MatrixUserCopyWith<$Res> { + factory $MatrixUserCopyWith(MatrixUser value, $Res Function(MatrixUser) _then) = _$MatrixUserCopyWithImpl; +@useResult +$Res call({ + String userId, String matrixToken +}); + + + + +} +/// @nodoc +class _$MatrixUserCopyWithImpl<$Res> + implements $MatrixUserCopyWith<$Res> { + _$MatrixUserCopyWithImpl(this._self, this._then); + + final MatrixUser _self; + final $Res Function(MatrixUser) _then; + +/// Create a copy of MatrixUser +/// with the given fields replaced by the non-null parameter values. +@pragma('vm:prefer-inline') @override $Res call({Object? userId = null,Object? matrixToken = null,}) { + return _then(_self.copyWith( +userId: null == userId ? _self.userId : userId // ignore: cast_nullable_to_non_nullable +as String,matrixToken: null == matrixToken ? _self.matrixToken : matrixToken // ignore: cast_nullable_to_non_nullable +as String, + )); +} + +} + + +/// @nodoc +@JsonSerializable() + +class _MatrixUser implements MatrixUser { + const _MatrixUser({required this.userId, required this.matrixToken}); + factory _MatrixUser.fromJson(Map json) => _$MatrixUserFromJson(json); + +@override final String userId; +@override final String matrixToken; + +/// Create a copy of MatrixUser +/// with the given fields replaced by the non-null parameter values. +@override @JsonKey(includeFromJson: false, includeToJson: false) +@pragma('vm:prefer-inline') +_$MatrixUserCopyWith<_MatrixUser> get copyWith => __$MatrixUserCopyWithImpl<_MatrixUser>(this, _$identity); + +@override +Map toJson() { + return _$MatrixUserToJson(this, ); +} + +@override +bool operator ==(Object other) { + return identical(this, other) || (other.runtimeType == runtimeType&&other is _MatrixUser&&(identical(other.userId, userId) || other.userId == userId)&&(identical(other.matrixToken, matrixToken) || other.matrixToken == matrixToken)); +} + +@JsonKey(includeFromJson: false, includeToJson: false) +@override +int get hashCode => Object.hash(runtimeType,userId,matrixToken); + +@override +String toString() { + return 'MatrixUser(userId: $userId, matrixToken: $matrixToken)'; +} + + +} + +/// @nodoc +abstract mixin class _$MatrixUserCopyWith<$Res> implements $MatrixUserCopyWith<$Res> { + factory _$MatrixUserCopyWith(_MatrixUser value, $Res Function(_MatrixUser) _then) = __$MatrixUserCopyWithImpl; +@override @useResult +$Res call({ + String userId, String matrixToken +}); + + + + +} +/// @nodoc +class __$MatrixUserCopyWithImpl<$Res> + implements _$MatrixUserCopyWith<$Res> { + __$MatrixUserCopyWithImpl(this._self, this._then); + + final _MatrixUser _self; + final $Res Function(_MatrixUser) _then; + +/// Create a copy of MatrixUser +/// with the given fields replaced by the non-null parameter values. +@override @pragma('vm:prefer-inline') $Res call({Object? userId = null,Object? matrixToken = null,}) { + return _then(_MatrixUser( +userId: null == userId ? _self.userId : userId // ignore: cast_nullable_to_non_nullable +as String,matrixToken: null == matrixToken ? _self.matrixToken : matrixToken // ignore: cast_nullable_to_non_nullable +as String, + )); +} + + +} + +// dart format on diff --git a/lib/models/matrix_user.g.dart b/lib/models/matrix_user.g.dart new file mode 100644 index 0000000..e3aa600 --- /dev/null +++ b/lib/models/matrix_user.g.dart @@ -0,0 +1,18 @@ +// GENERATED CODE - DO NOT MODIFY BY HAND + +part of 'matrix_user.dart'; + +// ************************************************************************** +// JsonSerializableGenerator +// ************************************************************************** + +_MatrixUser _$MatrixUserFromJson(Map json) => _MatrixUser( + userId: json['userId'] as String, + matrixToken: json['matrixToken'] as String, +); + +Map _$MatrixUserToJson(_MatrixUser instance) => + { + 'userId': instance.userId, + 'matrixToken': instance.matrixToken, + }; diff --git a/lib/models/settings.dart b/lib/models/settings.dart new file mode 100644 index 0000000..3fa8820 --- /dev/null +++ b/lib/models/settings.dart @@ -0,0 +1,20 @@ +import "package:freezed_annotation/freezed_annotation.dart"; + +part "settings.freezed.dart"; +part "settings.g.dart"; + +@freezed +abstract class Settings with _$Settings { + const factory Settings({ + required String? socket, + required String address, + required String port, + required String homeserver, + required String issuer, + required String jwtSecretFile, + required String authorizeEndpoint, + }) = _Settings; + + factory Settings.fromJson(Map json) => + _$SettingsFromJson(json); +} diff --git a/lib/models/settings.freezed.dart b/lib/models/settings.freezed.dart new file mode 100644 index 0000000..81dc77c --- /dev/null +++ b/lib/models/settings.freezed.dart @@ -0,0 +1,166 @@ +// dart format width=80 +// coverage:ignore-file +// GENERATED CODE - DO NOT MODIFY BY HAND +// ignore_for_file: type=lint +// ignore_for_file: unused_element, deprecated_member_use, deprecated_member_use_from_same_package, use_function_type_syntax_for_parameters, unnecessary_const, avoid_init_to_null, invalid_override_different_default_values_named, prefer_expression_function_bodies, annotate_overrides, invalid_annotation_target, unnecessary_question_mark + +part of 'settings.dart'; + +// ************************************************************************** +// FreezedGenerator +// ************************************************************************** + +// dart format off +T _$identity(T value) => value; + +/// @nodoc +mixin _$Settings { + + String? get socket; String get address; String get port; String get homeserver; String get issuer; String get jwtSecretFile; String get authorizeEndpoint; +/// Create a copy of Settings +/// with the given fields replaced by the non-null parameter values. +@JsonKey(includeFromJson: false, includeToJson: false) +@pragma('vm:prefer-inline') +$SettingsCopyWith get copyWith => _$SettingsCopyWithImpl(this as Settings, _$identity); + + /// Serializes this Settings to a JSON map. + Map toJson(); + + +@override +bool operator ==(Object other) { + return identical(this, other) || (other.runtimeType == runtimeType&&other is Settings&&(identical(other.socket, socket) || other.socket == socket)&&(identical(other.address, address) || other.address == address)&&(identical(other.port, port) || other.port == port)&&(identical(other.homeserver, homeserver) || other.homeserver == homeserver)&&(identical(other.issuer, issuer) || other.issuer == issuer)&&(identical(other.jwtSecretFile, jwtSecretFile) || other.jwtSecretFile == jwtSecretFile)&&(identical(other.authorizeEndpoint, authorizeEndpoint) || other.authorizeEndpoint == authorizeEndpoint)); +} + +@JsonKey(includeFromJson: false, includeToJson: false) +@override +int get hashCode => Object.hash(runtimeType,socket,address,port,homeserver,issuer,jwtSecretFile,authorizeEndpoint); + +@override +String toString() { + return 'Settings(socket: $socket, address: $address, port: $port, homeserver: $homeserver, issuer: $issuer, jwtSecretFile: $jwtSecretFile, authorizeEndpoint: $authorizeEndpoint)'; +} + + +} + +/// @nodoc +abstract mixin class $SettingsCopyWith<$Res> { + factory $SettingsCopyWith(Settings value, $Res Function(Settings) _then) = _$SettingsCopyWithImpl; +@useResult +$Res call({ + String? socket, String address, String port, String homeserver, String issuer, String jwtSecretFile, String authorizeEndpoint +}); + + + + +} +/// @nodoc +class _$SettingsCopyWithImpl<$Res> + implements $SettingsCopyWith<$Res> { + _$SettingsCopyWithImpl(this._self, this._then); + + final Settings _self; + final $Res Function(Settings) _then; + +/// Create a copy of Settings +/// with the given fields replaced by the non-null parameter values. +@pragma('vm:prefer-inline') @override $Res call({Object? socket = freezed,Object? address = null,Object? port = null,Object? homeserver = null,Object? issuer = null,Object? jwtSecretFile = null,Object? authorizeEndpoint = null,}) { + return _then(_self.copyWith( +socket: freezed == socket ? _self.socket : socket // ignore: cast_nullable_to_non_nullable +as String?,address: null == address ? _self.address : address // ignore: cast_nullable_to_non_nullable +as String,port: null == port ? _self.port : port // ignore: cast_nullable_to_non_nullable +as String,homeserver: null == homeserver ? _self.homeserver : homeserver // ignore: cast_nullable_to_non_nullable +as String,issuer: null == issuer ? _self.issuer : issuer // ignore: cast_nullable_to_non_nullable +as String,jwtSecretFile: null == jwtSecretFile ? _self.jwtSecretFile : jwtSecretFile // ignore: cast_nullable_to_non_nullable +as String,authorizeEndpoint: null == authorizeEndpoint ? _self.authorizeEndpoint : authorizeEndpoint // ignore: cast_nullable_to_non_nullable +as String, + )); +} + +} + + +/// @nodoc +@JsonSerializable() + +class _Settings implements Settings { + const _Settings({required this.socket, required this.address, required this.port, required this.homeserver, required this.issuer, required this.jwtSecretFile, required this.authorizeEndpoint}); + factory _Settings.fromJson(Map json) => _$SettingsFromJson(json); + +@override final String? socket; +@override final String address; +@override final String port; +@override final String homeserver; +@override final String issuer; +@override final String jwtSecretFile; +@override final String authorizeEndpoint; + +/// Create a copy of Settings +/// with the given fields replaced by the non-null parameter values. +@override @JsonKey(includeFromJson: false, includeToJson: false) +@pragma('vm:prefer-inline') +_$SettingsCopyWith<_Settings> get copyWith => __$SettingsCopyWithImpl<_Settings>(this, _$identity); + +@override +Map toJson() { + return _$SettingsToJson(this, ); +} + +@override +bool operator ==(Object other) { + return identical(this, other) || (other.runtimeType == runtimeType&&other is _Settings&&(identical(other.socket, socket) || other.socket == socket)&&(identical(other.address, address) || other.address == address)&&(identical(other.port, port) || other.port == port)&&(identical(other.homeserver, homeserver) || other.homeserver == homeserver)&&(identical(other.issuer, issuer) || other.issuer == issuer)&&(identical(other.jwtSecretFile, jwtSecretFile) || other.jwtSecretFile == jwtSecretFile)&&(identical(other.authorizeEndpoint, authorizeEndpoint) || other.authorizeEndpoint == authorizeEndpoint)); +} + +@JsonKey(includeFromJson: false, includeToJson: false) +@override +int get hashCode => Object.hash(runtimeType,socket,address,port,homeserver,issuer,jwtSecretFile,authorizeEndpoint); + +@override +String toString() { + return 'Settings(socket: $socket, address: $address, port: $port, homeserver: $homeserver, issuer: $issuer, jwtSecretFile: $jwtSecretFile, authorizeEndpoint: $authorizeEndpoint)'; +} + + +} + +/// @nodoc +abstract mixin class _$SettingsCopyWith<$Res> implements $SettingsCopyWith<$Res> { + factory _$SettingsCopyWith(_Settings value, $Res Function(_Settings) _then) = __$SettingsCopyWithImpl; +@override @useResult +$Res call({ + String? socket, String address, String port, String homeserver, String issuer, String jwtSecretFile, String authorizeEndpoint +}); + + + + +} +/// @nodoc +class __$SettingsCopyWithImpl<$Res> + implements _$SettingsCopyWith<$Res> { + __$SettingsCopyWithImpl(this._self, this._then); + + final _Settings _self; + final $Res Function(_Settings) _then; + +/// Create a copy of Settings +/// with the given fields replaced by the non-null parameter values. +@override @pragma('vm:prefer-inline') $Res call({Object? socket = freezed,Object? address = null,Object? port = null,Object? homeserver = null,Object? issuer = null,Object? jwtSecretFile = null,Object? authorizeEndpoint = null,}) { + return _then(_Settings( +socket: freezed == socket ? _self.socket : socket // ignore: cast_nullable_to_non_nullable +as String?,address: null == address ? _self.address : address // ignore: cast_nullable_to_non_nullable +as String,port: null == port ? _self.port : port // ignore: cast_nullable_to_non_nullable +as String,homeserver: null == homeserver ? _self.homeserver : homeserver // ignore: cast_nullable_to_non_nullable +as String,issuer: null == issuer ? _self.issuer : issuer // ignore: cast_nullable_to_non_nullable +as String,jwtSecretFile: null == jwtSecretFile ? _self.jwtSecretFile : jwtSecretFile // ignore: cast_nullable_to_non_nullable +as String,authorizeEndpoint: null == authorizeEndpoint ? _self.authorizeEndpoint : authorizeEndpoint // ignore: cast_nullable_to_non_nullable +as String, + )); +} + + +} + +// dart format on diff --git a/lib/models/settings.g.dart b/lib/models/settings.g.dart new file mode 100644 index 0000000..c5f7f94 --- /dev/null +++ b/lib/models/settings.g.dart @@ -0,0 +1,27 @@ +// GENERATED CODE - DO NOT MODIFY BY HAND + +part of 'settings.dart'; + +// ************************************************************************** +// JsonSerializableGenerator +// ************************************************************************** + +_Settings _$SettingsFromJson(Map json) => _Settings( + socket: json['socket'] as String?, + address: json['address'] as String, + port: json['port'] as String, + homeserver: json['homeserver'] as String, + issuer: json['issuer'] as String, + jwtSecretFile: json['jwtSecretFile'] as String, + authorizeEndpoint: json['authorizeEndpoint'] as String, +); + +Map _$SettingsToJson(_Settings instance) => { + 'socket': instance.socket, + 'address': instance.address, + 'port': instance.port, + 'homeserver': instance.homeserver, + 'issuer': instance.issuer, + 'jwtSecretFile': instance.jwtSecretFile, + 'authorizeEndpoint': instance.authorizeEndpoint, +}; diff --git a/pubspec.lock b/pubspec.lock new file mode 100644 index 0000000..84ee519 --- /dev/null +++ b/pubspec.lock @@ -0,0 +1,573 @@ +# Generated by pub +# See https://dart.dev/tools/pub/glossary#lockfile +packages: + _fe_analyzer_shared: + dependency: transitive + description: + name: _fe_analyzer_shared + sha256: e55636ed79578b9abca5fecf9437947798f5ef7456308b5cb85720b793eac92f + url: "https://pub.dev" + source: hosted + version: "82.0.0" + adaptive_number: + dependency: transitive + description: + name: adaptive_number + sha256: "3a567544e9b5c9c803006f51140ad544aedc79604fd4f3f2c1380003f97c1d77" + url: "https://pub.dev" + source: hosted + version: "1.0.0" + analyzer: + dependency: transitive + description: + name: analyzer + sha256: "904ae5bb474d32c38fb9482e2d925d5454cda04ddd0e55d2e6826bc72f6ba8c0" + url: "https://pub.dev" + source: hosted + version: "7.4.5" + args: + dependency: "direct main" + description: + name: args + sha256: d0481093c50b1da8910eb0bb301626d4d8eb7284aa739614d2b394ee09e3ea04 + url: "https://pub.dev" + source: hosted + version: "2.7.0" + async: + dependency: transitive + description: + name: async + sha256: "758e6d74e971c3e5aceb4110bfd6698efc7f501675bcfe0c775459a8140750eb" + url: "https://pub.dev" + source: hosted + version: "2.13.0" + boolean_selector: + dependency: transitive + description: + name: boolean_selector + sha256: "8aab1771e1243a5063b8b0ff68042d67334e3feab9e95b9490f9a6ebf73b42ea" + url: "https://pub.dev" + source: hosted + version: "2.1.2" + build: + dependency: transitive + description: + name: build + sha256: "486337d40a48d75049f2a01efceefc1412e3a6d6342d39624753e7d5a4e70016" + url: "https://pub.dev" + source: hosted + version: "2.5.1" + build_config: + dependency: transitive + description: + name: build_config + sha256: "4ae2de3e1e67ea270081eaee972e1bd8f027d459f249e0f1186730784c2e7e33" + url: "https://pub.dev" + source: hosted + version: "1.1.2" + build_daemon: + dependency: transitive + description: + name: build_daemon + sha256: "8e928697a82be082206edb0b9c99c5a4ad6bc31c9e9b8b2f291ae65cd4a25daa" + url: "https://pub.dev" + source: hosted + version: "4.0.4" + build_resolvers: + dependency: transitive + description: + name: build_resolvers + sha256: abe6e4b5b36ce2bf01aec8f2a59424d1ecfc3cb340e7145a64359adc7233a0d1 + url: "https://pub.dev" + source: hosted + version: "2.5.1" + build_runner: + dependency: "direct dev" + description: + name: build_runner + sha256: "398ec7898b9b60be126067835a8202240b26dc54aa34d91d0198a539dcd5942e" + url: "https://pub.dev" + source: hosted + version: "2.5.1" + build_runner_core: + dependency: transitive + description: + name: build_runner_core + sha256: "9821dbf604ed74a6dabeaba0c33ac58190332ea238862a62cdf25fc8eba226b8" + url: "https://pub.dev" + source: hosted + version: "9.0.1" + built_collection: + dependency: transitive + description: + name: built_collection + sha256: "376e3dd27b51ea877c28d525560790aee2e6fbb5f20e2f85d5081027d94e2100" + url: "https://pub.dev" + source: hosted + version: "5.1.1" + built_value: + dependency: transitive + description: + name: built_value + sha256: "082001b5c3dc495d4a42f1d5789990505df20d8547d42507c29050af6933ee27" + url: "https://pub.dev" + source: hosted + version: "8.10.1" + checked_yaml: + dependency: transitive + description: + name: checked_yaml + sha256: "959525d3162f249993882720d52b7e0c833978df229be20702b33d48d91de70f" + url: "https://pub.dev" + source: hosted + version: "2.0.4" + clock: + dependency: transitive + description: + name: clock + sha256: fddb70d9b5277016c77a80201021d40a2247104d9f4aa7bab7157b7e3f05b84b + url: "https://pub.dev" + source: hosted + version: "1.1.2" + code_builder: + dependency: transitive + description: + name: code_builder + sha256: "0ec10bf4a89e4c613960bf1e8b42c64127021740fb21640c29c909826a5eea3e" + url: "https://pub.dev" + source: hosted + version: "4.10.1" + collection: + dependency: transitive + description: + name: collection + sha256: "2f5709ae4d3d59dd8f7cd309b4e023046b57d8a6c82130785d2b0e5868084e76" + url: "https://pub.dev" + source: hosted + version: "1.19.1" + convert: + dependency: transitive + description: + name: convert + sha256: b30acd5944035672bc15c6b7a8b47d773e41e2f17de064350988c5d02adb1c68 + url: "https://pub.dev" + source: hosted + version: "3.1.2" + crypto: + dependency: transitive + description: + name: crypto + sha256: "1e445881f28f22d6140f181e07737b22f1e099a5e1ff94b0af2f9e4a463f4855" + url: "https://pub.dev" + source: hosted + version: "3.0.6" + dart_jsonwebtoken: + dependency: "direct main" + description: + name: dart_jsonwebtoken + sha256: "21ce9f8a8712f741e8d6876a9c82c0f8a257fe928c4378a91d8527b92a3fd413" + url: "https://pub.dev" + source: hosted + version: "3.2.0" + dart_style: + dependency: transitive + description: + name: dart_style + sha256: "5b236382b47ee411741447c1f1e111459c941ea1b3f2b540dde54c210a3662af" + url: "https://pub.dev" + source: hosted + version: "3.1.0" + ed25519_edwards: + dependency: transitive + description: + name: ed25519_edwards + sha256: "6ce0112d131327ec6d42beede1e5dfd526069b18ad45dcf654f15074ad9276cd" + url: "https://pub.dev" + source: hosted + version: "0.3.1" + fast_immutable_collections: + dependency: "direct main" + description: + name: fast_immutable_collections + sha256: d1aa3d7788fab06cce7f303f4969c7a16a10c865e1bd2478291a8ebcbee084e5 + url: "https://pub.dev" + source: hosted + version: "11.0.4" + file: + dependency: transitive + description: + name: file + sha256: a3b4f84adafef897088c160faf7dfffb7696046cb13ae90b508c2cbc95d3b8d4 + url: "https://pub.dev" + source: hosted + version: "7.0.1" + fixnum: + dependency: transitive + description: + name: fixnum + sha256: b6dc7065e46c974bc7c5f143080a6764ec7a4be6da1285ececdc37be96de53be + url: "https://pub.dev" + source: hosted + version: "1.1.1" + freezed: + dependency: "direct dev" + description: + name: freezed + sha256: "6022db4c7bfa626841b2a10f34dd1e1b68e8f8f9650db6112dcdeeca45ca793c" + url: "https://pub.dev" + source: hosted + version: "3.0.6" + freezed_annotation: + dependency: "direct main" + description: + name: freezed_annotation + sha256: c87ff004c8aa6af2d531668b46a4ea379f7191dc6dfa066acd53d506da6e044b + url: "https://pub.dev" + source: hosted + version: "3.0.0" + frontend_server_client: + dependency: transitive + description: + name: frontend_server_client + sha256: f64a0333a82f30b0cca061bc3d143813a486dc086b574bfb233b7c1372427694 + url: "https://pub.dev" + source: hosted + version: "4.0.0" + glob: + dependency: transitive + description: + name: glob + sha256: c3f1ee72c96f8f78935e18aa8cecced9ab132419e8625dc187e1c2408efc20de + url: "https://pub.dev" + source: hosted + version: "2.1.3" + graphs: + dependency: transitive + description: + name: graphs + sha256: "741bbf84165310a68ff28fe9e727332eef1407342fca52759cb21ad8177bb8d0" + url: "https://pub.dev" + source: hosted + version: "2.3.2" + http: + dependency: "direct main" + description: + name: http + sha256: "2c11f3f94c687ee9bad77c171151672986360b2b001d109814ee7140b2cf261b" + url: "https://pub.dev" + source: hosted + version: "1.4.0" + http_methods: + dependency: transitive + description: + name: http_methods + sha256: "6bccce8f1ec7b5d701e7921dca35e202d425b57e317ba1a37f2638590e29e566" + url: "https://pub.dev" + source: hosted + version: "1.1.1" + http_multi_server: + dependency: transitive + description: + name: http_multi_server + sha256: aa6199f908078bb1c5efb8d8638d4ae191aac11b311132c3ef48ce352fb52ef8 + url: "https://pub.dev" + source: hosted + version: "3.2.2" + http_parser: + dependency: transitive + description: + name: http_parser + sha256: "178d74305e7866013777bab2c3d8726205dc5a4dd935297175b19a23a2e66571" + url: "https://pub.dev" + source: hosted + version: "4.1.2" + io: + dependency: transitive + description: + name: io + sha256: dfd5a80599cf0165756e3181807ed3e77daf6dd4137caaad72d0b7931597650b + url: "https://pub.dev" + source: hosted + version: "1.0.5" + js: + dependency: transitive + description: + name: js + sha256: "53385261521cc4a0c4658fd0ad07a7d14591cf8fc33abbceae306ddb974888dc" + url: "https://pub.dev" + source: hosted + version: "0.7.2" + json_annotation: + dependency: "direct main" + description: + name: json_annotation + sha256: "1ce844379ca14835a50d2f019a3099f419082cfdd231cd86a142af94dd5c6bb1" + url: "https://pub.dev" + source: hosted + version: "4.9.0" + json_serializable: + dependency: "direct dev" + description: + name: json_serializable + sha256: c50ef5fc083d5b5e12eef489503ba3bf5ccc899e487d691584699b4bdefeea8c + url: "https://pub.dev" + source: hosted + version: "6.9.5" + lints: + dependency: "direct dev" + description: + name: lints + sha256: a5e2b223cb7c9c8efdc663ef484fdd95bb243bff242ef5b13e26883547fce9a0 + url: "https://pub.dev" + source: hosted + version: "6.0.0" + logging: + dependency: transitive + description: + name: logging + sha256: c8245ada5f1717ed44271ed1c26b8ce85ca3228fd2ffdb75468ab01979309d61 + url: "https://pub.dev" + source: hosted + version: "1.3.0" + matcher: + dependency: transitive + description: + name: matcher + sha256: dc58c723c3c24bf8d3e2d3ad3f2f9d7bd9cf43ec6feaa64181775e60190153f2 + url: "https://pub.dev" + source: hosted + version: "0.12.17" + meta: + dependency: transitive + description: + name: meta + sha256: "23f08335362185a5ea2ad3a4e597f1375e78bce8a040df5c600c8d3552ef2394" + url: "https://pub.dev" + source: hosted + version: "1.17.0" + mime: + dependency: transitive + description: + name: mime + sha256: "41a20518f0cb1256669420fdba0cd90d21561e560ac240f26ef8322e45bb7ed6" + url: "https://pub.dev" + source: hosted + version: "2.0.0" + package_config: + dependency: transitive + description: + name: package_config + sha256: f096c55ebb7deb7e384101542bfba8c52696c1b56fca2eb62827989ef2353bbc + url: "https://pub.dev" + source: hosted + version: "2.2.0" + path: + dependency: transitive + description: + name: path + sha256: "75cca69d1490965be98c73ceaea117e8a04dd21217b37b292c9ddbec0d955bc5" + url: "https://pub.dev" + source: hosted + version: "1.9.1" + pointycastle: + dependency: transitive + description: + name: pointycastle + sha256: "92aa3841d083cc4b0f4709b5c74fd6409a3e6ba833ffc7dc6a8fee096366acf5" + url: "https://pub.dev" + source: hosted + version: "4.0.0" + pool: + dependency: transitive + description: + name: pool + sha256: "20fe868b6314b322ea036ba325e6fc0711a22948856475e2c2b6306e8ab39c2a" + url: "https://pub.dev" + source: hosted + version: "1.5.1" + pub_semver: + dependency: transitive + description: + name: pub_semver + sha256: "5bfcf68ca79ef689f8990d1160781b4bad40a3bd5e5218ad4076ddb7f4081585" + url: "https://pub.dev" + source: hosted + version: "2.2.0" + pubspec_parse: + dependency: transitive + description: + name: pubspec_parse + sha256: "0560ba233314abbed0a48a2956f7f022cce7c3e1e73df540277da7544cad4082" + url: "https://pub.dev" + source: hosted + version: "1.5.0" + riverpod: + dependency: "direct main" + description: + name: riverpod + sha256: "59062512288d3056b2321804332a13ffdd1bf16df70dcc8e506e411280a72959" + url: "https://pub.dev" + source: hosted + version: "2.6.1" + shelf: + dependency: "direct main" + description: + name: shelf + sha256: e7dd780a7ffb623c57850b33f43309312fc863fb6aa3d276a754bb299839ef12 + url: "https://pub.dev" + source: hosted + version: "1.4.2" + shelf_router: + dependency: "direct main" + description: + name: shelf_router + sha256: f5e5d492440a7fb165fe1e2e1a623f31f734d3370900070b2b1e0d0428d59864 + url: "https://pub.dev" + source: hosted + version: "1.1.4" + shelf_web_socket: + dependency: transitive + description: + name: shelf_web_socket + sha256: "3632775c8e90d6c9712f883e633716432a27758216dfb61bd86a8321c0580925" + url: "https://pub.dev" + source: hosted + version: "3.0.0" + source_gen: + dependency: transitive + description: + name: source_gen + sha256: "35c8150ece9e8c8d263337a265153c3329667640850b9304861faea59fc98f6b" + url: "https://pub.dev" + source: hosted + version: "2.0.0" + source_helper: + dependency: transitive + description: + name: source_helper + sha256: "86d247119aedce8e63f4751bd9626fc9613255935558447569ad42f9f5b48b3c" + url: "https://pub.dev" + source: hosted + version: "1.3.5" + source_span: + dependency: transitive + description: + name: source_span + sha256: "254ee5351d6cb365c859e20ee823c3bb479bf4a293c22d17a9f1bf144ce86f7c" + url: "https://pub.dev" + source: hosted + version: "1.10.1" + stack_trace: + dependency: transitive + description: + name: stack_trace + sha256: "8b27215b45d22309b5cddda1aa2b19bdfec9df0e765f2de506401c071d38d1b1" + url: "https://pub.dev" + source: hosted + version: "1.12.1" + state_notifier: + dependency: transitive + description: + name: state_notifier + sha256: b8677376aa54f2d7c58280d5a007f9e8774f1968d1fb1c096adcb4792fba29bb + url: "https://pub.dev" + source: hosted + version: "1.0.0" + stream_channel: + dependency: transitive + description: + name: stream_channel + sha256: "969e04c80b8bcdf826f8f16579c7b14d780458bd97f56d107d3950fdbeef059d" + url: "https://pub.dev" + source: hosted + version: "2.1.4" + stream_transform: + dependency: transitive + description: + name: stream_transform + sha256: ad47125e588cfd37a9a7f86c7d6356dde8dfe89d071d293f80ca9e9273a33871 + url: "https://pub.dev" + source: hosted + version: "2.1.1" + string_scanner: + dependency: transitive + description: + name: string_scanner + sha256: "921cd31725b72fe181906c6a94d987c78e3b98c2e205b397ea399d4054872b43" + url: "https://pub.dev" + source: hosted + version: "1.4.1" + term_glyph: + dependency: transitive + description: + name: term_glyph + sha256: "7f554798625ea768a7518313e58f83891c7f5024f88e46e7182a4558850a4b8e" + url: "https://pub.dev" + source: hosted + version: "1.2.2" + test_api: + dependency: transitive + description: + name: test_api + sha256: "522f00f556e73044315fa4585ec3270f1808a4b186c936e612cab0b565ff1e00" + url: "https://pub.dev" + source: hosted + version: "0.7.6" + timing: + dependency: transitive + description: + name: timing + sha256: "62ee18aca144e4a9f29d212f5a4c6a053be252b895ab14b5821996cff4ed90fe" + url: "https://pub.dev" + source: hosted + version: "1.0.2" + typed_data: + dependency: transitive + description: + name: typed_data + sha256: f9049c039ebfeb4cf7a7104a675823cd72dba8297f264b6637062516699fa006 + url: "https://pub.dev" + source: hosted + version: "1.4.0" + watcher: + dependency: transitive + description: + name: watcher + sha256: "0b7fd4a0bbc4b92641dbf20adfd7e3fd1398fe17102d94b674234563e110088a" + url: "https://pub.dev" + source: hosted + version: "1.1.2" + web: + dependency: transitive + description: + name: web + sha256: "868d88a33d8a87b18ffc05f9f030ba328ffefba92d6c127917a2ba740f9cfe4a" + url: "https://pub.dev" + source: hosted + version: "1.1.1" + web_socket: + dependency: transitive + description: + name: web_socket + sha256: "34d64019aa8e36bf9842ac014bb5d2f5586ca73df5e4d9bf5c936975cae6982c" + url: "https://pub.dev" + source: hosted + version: "1.0.1" + web_socket_channel: + dependency: transitive + description: + name: web_socket_channel + sha256: d645757fb0f4773d602444000a8131ff5d48c9e47adfe9772652dd1a4f2d45c8 + url: "https://pub.dev" + source: hosted + version: "3.0.3" + yaml: + dependency: transitive + description: + name: yaml + sha256: b9da305ac7c39faa3f030eccd175340f968459dae4af175130b3fc47e40d76ce + url: "https://pub.dev" + source: hosted + version: "3.1.3" +sdks: + dart: ">=3.8.0 <4.0.0" diff --git a/pubspec.yaml b/pubspec.yaml new file mode 100644 index 0000000..7db5d1a --- /dev/null +++ b/pubspec.yaml @@ -0,0 +1,24 @@ +name: matrixgate +description: A minimal OpenID Connect provider backed by Matrix OpenID tokens. +version: 1.0.0 +publish_to: none + +environment: + sdk: ^3.8.0 + +dependencies: + http: ^1.4.0 + shelf: ^1.4.0 + riverpod: ^2.4.0 + dart_jsonwebtoken: ^3.2.0 + freezed_annotation: ^3.0.0 + json_annotation: ^4.9.0 + shelf_router: ^1.1.4 + args: ^2.7.0 + fast_immutable_collections: ^11.0.4 + +dev_dependencies: + build_runner: ^2.4.6 + freezed: ^3.0.6 + json_serializable: ^6.7.1 + lints: ^6.0.0 \ No newline at end of file