This repository has been archived by the owner on Dec 23, 2024. It is now read-only.
-
-
Notifications
You must be signed in to change notification settings - Fork 10
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Merge pull request #120 from Myzel394/feat-0.15.0-sort-relays
Feat 0.15.0 sort relays
- Loading branch information
Showing
9 changed files
with
342 additions
and
36 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,185 @@ | ||
import 'dart:convert'; | ||
|
||
import 'package:collection/collection.dart'; | ||
import 'package:flutter_animate/flutter_animate.dart'; | ||
import 'package:locus/utils/access-deeply-nested-key.dart'; | ||
import 'package:locus/utils/nostr_fetcher/BasicNostrFetchSocket.dart'; | ||
import 'package:locus/utils/nostr_fetcher/NostrSocket.dart'; | ||
import 'package:nostr/nostr.dart'; | ||
|
||
const MIN_LENGTH = 5000; | ||
|
||
class RelaysMetaFetcher extends BasicNostrFetchSocket { | ||
List<RelayMeta> meta = []; | ||
|
||
RelaysMetaFetcher({ | ||
required super.relay, | ||
super.timeout, | ||
}); | ||
|
||
@override | ||
void onEndOfStream() { | ||
closeConnection(); | ||
} | ||
|
||
@override | ||
void onNostrEvent(final Message message) { | ||
// Relay URL, canWrite and canRead are in message.tags | ||
// Latencies are saved in content, separated per region | ||
// with the following schema: | ||
// [ | ||
// [<connection speed>], | ||
// [<read speed>], | ||
// [<write speed>], | ||
// ] | ||
final event = message.message as Event; | ||
|
||
final relayMeta = RelayMeta.fromFetchedContent( | ||
canWrite: event.tags[1][1] == "true", | ||
canRead: event.tags[2][1] == "true", | ||
relay: event.tags[0][1], | ||
content: jsonDecode(event.content), | ||
worldRegion: "eu-west", | ||
); | ||
|
||
meta.add(relayMeta); | ||
} | ||
|
||
@override | ||
void onError(error) { | ||
closeConnection(); | ||
} | ||
} | ||
|
||
class RelayMeta { | ||
final String relay; | ||
final bool canWrite; | ||
final bool canRead; | ||
final String contactInfo; | ||
final String description; | ||
final String name; | ||
|
||
final List<int> connectionLatencies; | ||
final List<int> readLatencies; | ||
final List<int> writeLatencies; | ||
|
||
final int maxMessageLength; | ||
final int maxContentLength; | ||
|
||
final int minPowDifficulty; | ||
final bool requiresPayment; | ||
|
||
const RelayMeta({ | ||
required this.relay, | ||
required this.canWrite, | ||
required this.canRead, | ||
required this.contactInfo, | ||
required this.description, | ||
required this.name, | ||
required this.connectionLatencies, | ||
required this.readLatencies, | ||
required this.writeLatencies, | ||
required this.maxMessageLength, | ||
required this.maxContentLength, | ||
required this.minPowDifficulty, | ||
required this.requiresPayment, | ||
}); | ||
|
||
factory RelayMeta.fromFetchedContent({ | ||
required final Map<String, dynamic> content, | ||
required final String relay, | ||
required final bool canRead, | ||
required final bool canWrite, | ||
required final String worldRegion, | ||
}) => | ||
RelayMeta( | ||
relay: relay, | ||
canRead: canRead, | ||
canWrite: canWrite, | ||
name: adnk<dynamic>(content, "info.name") ?? relay, | ||
contactInfo: adnk<dynamic>(content, "info.contact") ?? "", | ||
description: adnk<dynamic>(content, "info.description") ?? "", | ||
connectionLatencies: List<int?>.from( | ||
adnk<dynamic>(content, "latency.$worldRegion.0") ?? []) | ||
.where((value) => value != null) | ||
.toList() | ||
.cast<int>(), | ||
readLatencies: | ||
List<int?>.from(adnk<dynamic>(content, "latency.$worldRegion.1") ?? []) | ||
.where((value) => value != null) | ||
.toList() | ||
.cast<int>(), | ||
writeLatencies: | ||
List<int?>.from(adnk<dynamic>(content, "latency.$worldRegion.2") ?? []) | ||
.where((value) => value != null) | ||
.toList() | ||
.cast<int>(), | ||
maxContentLength: | ||
adnk<dynamic>(content, "info.limitations.max_content_length") ?? | ||
MIN_LENGTH, | ||
maxMessageLength: | ||
adnk<dynamic>(content, "info.limitations.max_message_length") ?? | ||
MIN_LENGTH, | ||
requiresPayment: | ||
adnk<dynamic>(content, "info.limitations.payment_required") ?? | ||
false, | ||
minPowDifficulty: | ||
adnk<dynamic>(content, "info.limitations.min_pow_difficulty") ?? | ||
0); | ||
|
||
bool get isSuitable => | ||
canWrite && | ||
canRead && | ||
!requiresPayment && | ||
minPowDifficulty == 0 && | ||
maxContentLength >= MIN_LENGTH; | ||
|
||
// Calculate average latency, we use the average as we want extreme highs | ||
// to be taken into account. | ||
double get score { | ||
if (connectionLatencies.isEmpty || | ||
readLatencies.isEmpty || | ||
writeLatencies.isEmpty) { | ||
// If there is no data available, we don't know if the relay is fully intact | ||
return double.infinity; | ||
} | ||
|
||
// Each latency has it's own factor to give each of them a different weight | ||
// Lower latency = better - Because of this | ||
// a factor closer to 0 resembles a HIGHER weight | ||
// We prioritize read latency as we want to be able to provide a fast app | ||
// Lower score = better | ||
return (connectionLatencies.average * 0.9 + | ||
readLatencies.average * 0.5 + | ||
writeLatencies.average) + | ||
(maxContentLength - MIN_LENGTH) * 0.0001; | ||
} | ||
} | ||
|
||
// Values taken from https://github.com/dskvr/nostr-watch/blob/develop/src/components/relays/jobs/LoadSeed.vue#L91 | ||
final REQUEST_DATA = NostrSocket.createNostrRequestData( | ||
kinds: [30304], | ||
limit: 1000, | ||
from: DateTime.now().subtract(2.hours), | ||
authors: ["b3b0d247f66bf40c4c9f4ce721abfe1fd3b7529fbc1ea5e64d5f0f8df3a4b6e6"], | ||
); | ||
|
||
Future<Map<String, List<RelayMeta>>> fetchRelaysMeta() async { | ||
final fetcher = RelaysMetaFetcher( | ||
relay: "wss://history.nostr.watch", | ||
); | ||
await fetcher.connect(); | ||
fetcher.addData( | ||
Request( | ||
generate64RandomHexChars(), | ||
[ | ||
REQUEST_DATA, | ||
], | ||
).serialize(), | ||
); | ||
await fetcher.onComplete; | ||
|
||
return { | ||
"meta": fetcher.meta, | ||
}; | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,19 @@ | ||
T? accessDeeplyNestedKey<T>(final Map<String, dynamic> obj, final String path) { | ||
dynamic result = obj; | ||
|
||
for (final subPath in path.split(".")) { | ||
if (result is List) { | ||
final index = int.tryParse(subPath)!; | ||
|
||
result = result[index]; | ||
} else if (result.containsKey(subPath)) { | ||
result = result[subPath]; | ||
} else { | ||
return null; | ||
} | ||
} | ||
|
||
return result as T; | ||
} | ||
|
||
const adnk = accessDeeplyNestedKey; |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,4 +1,5 @@ | ||
import 'dart:async'; | ||
import 'dart:convert'; | ||
import 'dart:io'; | ||
|
||
import 'package:flutter_logs/flutter_logs.dart'; | ||
|
Oops, something went wrong.