mirror of
https://github.com/hanskokx/arcane_implementations.git
synced 2026-05-14 02:19:05 +02:00
@@ -0,0 +1,98 @@
|
||||
import "dart:convert";
|
||||
import "dart:io" show Platform;
|
||||
|
||||
import "package:arcane_framework/arcane_framework.dart";
|
||||
import "package:arcane_helper_utils/arcane_helper_utils.dart";
|
||||
import "package:flutter/foundation.dart";
|
||||
import "package:logger/logger.dart" as l;
|
||||
|
||||
class DebugConsole implements LoggingInterface {
|
||||
static final DebugConsole _instance = DebugConsole._internal();
|
||||
static DebugConsole get I => _instance;
|
||||
|
||||
final bool _initialized = true;
|
||||
|
||||
@override
|
||||
bool get initialized => I._initialized;
|
||||
|
||||
DebugConsole._internal();
|
||||
|
||||
@visibleForTesting
|
||||
void setMocked() => _mocked = true;
|
||||
bool _mocked = false;
|
||||
|
||||
@override
|
||||
void log(
|
||||
String message, {
|
||||
Map<String, dynamic>? metadata,
|
||||
Level? level,
|
||||
StackTrace? stackTrace,
|
||||
}) {
|
||||
if (Feature.logging.disabled) return;
|
||||
if (Feature.debugConsoleLogging.disabled) return;
|
||||
if (!kDebugMode) return;
|
||||
if (!initialized) init();
|
||||
|
||||
const Level cutoff = AppConfig.debugLoggingThreshold;
|
||||
|
||||
if ((level?.value ?? Level.debug.value) < cutoff.value) return;
|
||||
|
||||
final l.Level logLevel = l.Level.values
|
||||
.firstWhere((value) => value.name == (level ?? Level.debug).name);
|
||||
|
||||
final Map<String, dynamic> localMetadata = metadata ?? {};
|
||||
|
||||
final String? module = localMetadata["module"] as String?;
|
||||
final String? method = localMetadata["method"] as String?;
|
||||
|
||||
const JsonEncoder encoder = JsonEncoder.withIndent(" ");
|
||||
final String? prettyprint =
|
||||
(localMetadata.isNotEmpty) ? encoder.convert(localMetadata) : null;
|
||||
|
||||
final l.Logger logger = l.Logger(
|
||||
level: logLevel,
|
||||
printer: l.PrettyPrinter(
|
||||
methodCount: 2,
|
||||
errorMethodCount: kDebugMode &&
|
||||
!(level == Level.error ||
|
||||
level == Level.warning ||
|
||||
level == Level.trace ||
|
||||
level == Level.fatal)
|
||||
? 4
|
||||
: 8,
|
||||
stackTraceBeginIndex: 1,
|
||||
lineLength: 120,
|
||||
colors: !Platform.isIOS,
|
||||
printEmojis: kDebugMode,
|
||||
dateTimeFormat: l.DateTimeFormat.none,
|
||||
),
|
||||
);
|
||||
|
||||
// Print the message to the debug console
|
||||
String prefix = "";
|
||||
if (module != null) prefix += "[$module]";
|
||||
if (method != null) prefix += "[$method]";
|
||||
if (prefix.isNotEmpty) prefix += " ";
|
||||
message = "$prefix$message";
|
||||
|
||||
if (prettyprint.isNotNullOrEmpty) message += "\n\n$prettyprint";
|
||||
|
||||
localMetadata.removeWhere((key, value) => key == "module");
|
||||
localMetadata.removeWhere((key, value) => key == "method");
|
||||
localMetadata.removeWhere((key, value) => key == "filenameAndLineNumber");
|
||||
|
||||
logger.log(
|
||||
logLevel,
|
||||
message,
|
||||
error: localMetadata["error"] ?? "",
|
||||
stackTrace: stackTrace,
|
||||
);
|
||||
}
|
||||
|
||||
@override
|
||||
Future<LoggingInterface?> init() async {
|
||||
if (_mocked) return null;
|
||||
|
||||
return I;
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,24 @@
|
||||
import "package:arcane_framework/arcane_framework.dart";
|
||||
import "package:flutter/foundation.dart";
|
||||
|
||||
class DebugPrint implements LoggingInterface {
|
||||
DebugPrint._internal();
|
||||
static final DebugPrint _instance = DebugPrint._internal();
|
||||
static DebugPrint get I => _instance;
|
||||
|
||||
@override
|
||||
bool get initialized => true;
|
||||
|
||||
@override
|
||||
void log(
|
||||
String message, {
|
||||
Map<String, dynamic>? metadata,
|
||||
Level? level = Level.debug,
|
||||
StackTrace? stackTrace,
|
||||
}) {
|
||||
debugPrint("[${level!.name}] $message ($metadata)");
|
||||
}
|
||||
|
||||
@override
|
||||
Future<LoggingInterface?> init() async => I;
|
||||
}
|
||||
@@ -0,0 +1,115 @@
|
||||
import "dart:io" show Platform;
|
||||
|
||||
import "package:arcane_framework/arcane_framework.dart";
|
||||
import "package:arcane_helper_utils/arcane_helper_utils.dart";
|
||||
import "package:flutter/foundation.dart";
|
||||
import "package:newrelic_mobile/config.dart";
|
||||
import "package:newrelic_mobile/newrelic_mobile.dart";
|
||||
|
||||
class NewRelic implements LoggingInterface {
|
||||
static final NewRelic _instance = NewRelic._internal();
|
||||
static NewRelic get I => _instance;
|
||||
|
||||
bool _initialized = false;
|
||||
@override
|
||||
bool get initialized => I._initialized;
|
||||
|
||||
NewRelic._internal();
|
||||
|
||||
@visibleForTesting
|
||||
void setMocked() => _mocked = true;
|
||||
bool _mocked = false;
|
||||
|
||||
@override
|
||||
void log(
|
||||
String message, {
|
||||
Map<String, dynamic>? metadata,
|
||||
Level? level,
|
||||
StackTrace? stackTrace,
|
||||
}) {
|
||||
if (Feature.logging.disabled) return;
|
||||
if (Feature.newRelic.disabled) return;
|
||||
if (!initialized) return;
|
||||
|
||||
final Map<String, dynamic> metadataToSend = metadata ?? {};
|
||||
|
||||
// New Relic strips this out, anyway, so let's not cause additional logs.
|
||||
metadataToSend.removeWhere((key, _) => key == "timestamp");
|
||||
|
||||
// Add the logging level to the metadata
|
||||
metadataToSend.putIfAbsent(
|
||||
"level",
|
||||
() => (level?.name ?? "debug").capitalize,
|
||||
);
|
||||
|
||||
NewrelicMobile.instance.recordCustomEvent(
|
||||
"App",
|
||||
eventName: message,
|
||||
eventAttributes: metadataToSend,
|
||||
);
|
||||
|
||||
if (stackTrace != null) {
|
||||
NewrelicMobile.instance.recordError(message, stackTrace);
|
||||
}
|
||||
}
|
||||
|
||||
@override
|
||||
Future<NewRelic?> init() async {
|
||||
if (_mocked) return null;
|
||||
if (Feature.newRelic.disabled) return null;
|
||||
if (initialized) return I;
|
||||
|
||||
final String appToken = AppEnv.valueOf(
|
||||
Platform.isAndroid
|
||||
? EnvVar.newRelicAppTokenAndroid
|
||||
: EnvVar.newRelicAppTokenIos,
|
||||
);
|
||||
|
||||
final Config config = Config(
|
||||
accessToken: appToken,
|
||||
|
||||
//Android Specific
|
||||
// Optional: Enable or disable collection of event data.
|
||||
analyticsEventEnabled: true,
|
||||
// Optional: Enable or disable reporting successful HTTP requests to the MobileRequest event type.
|
||||
networkErrorRequestEnabled: true,
|
||||
// Optional: Enable or disable reporting network and HTTP request errors to the MobileRequestError event type.
|
||||
networkRequestEnabled: true,
|
||||
// Optional: Enable or disable crash reporting.
|
||||
crashReportingEnabled: true,
|
||||
// Optional: Enable or disable interaction tracing. Trace instrumentation still occurs, but no traces are harvested. This will disable default and custom interactions.
|
||||
interactionTracingEnabled: true,
|
||||
// Optional: Enable or disable capture of HTTP response bodies for HTTP error traces and MobileRequestError events.
|
||||
httpResponseBodyCaptureEnabled: true,
|
||||
// Optional: Enable or disable agent logging.
|
||||
loggingEnabled: true,
|
||||
// iOS specific
|
||||
// Optional: Enable or disable automatic instrumentation of WebViews
|
||||
webViewInstrumentation: false,
|
||||
//Optional: Enable or disable Print Statements as Analytics Events
|
||||
printStatementAsEventsEnabled: false,
|
||||
// Optional: Enable or disable automatic instrumentation of HTTP Request
|
||||
httpInstrumentationEnabled: true,
|
||||
// Optional: Enable or disable reporting data using different endpoints for US government clients
|
||||
fedRampEnabled: false,
|
||||
// Optional: Enable or disable offline data storage when no internet connection is available.
|
||||
offlineStorageEnabled: true,
|
||||
// iOS Specific
|
||||
// Optional: Enable or disable background reporting functionality.
|
||||
backgroundReportingEnabled: false,
|
||||
// iOS Specific
|
||||
// Optional: Enable or disable to use our new, more stable, event system for iOS agent.
|
||||
newEventSystemEnabled: true,
|
||||
|
||||
// Optional: Enable or disable distributed tracing.
|
||||
distributedTracingEnabled: true,
|
||||
);
|
||||
|
||||
await NewrelicMobile.instance.startAgent(config);
|
||||
|
||||
// Initialization complete.
|
||||
I._initialized = true;
|
||||
|
||||
return I;
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user