blob: 80305ee181e34f8f9260d679e9aacd25b2aacceb [file] [log] [blame]
// Copyright (c) 2023, the Dart project authors. Please see the AUTHORS file
// for details. All rights reserved. Use of this source code is governed by a
// BSD-style license that can be found in the LICENSE file.
import 'dart:convert';
import 'package:clock/clock.dart';
import 'package:file/file.dart';
import 'package:path/path.dart' as p;
import 'constants.dart';
import 'error_handler.dart';
import 'event.dart';
import 'initializer.dart';
class Session {
final Directory homeDirectory;
final FileSystem fs;
final File sessionFile;
final ErrorHandler _errorHandler;
late int _sessionId;
late int _lastPing;
Session({
required this.homeDirectory,
required this.fs,
required ErrorHandler errorHandler,
}) : sessionFile = fs.file(p.join(
homeDirectory.path, kDartToolDirectoryName, kSessionFileName)),
_errorHandler = errorHandler;
/// This will use the data parsed from the
/// session json file in the dart-tool directory
/// to get the session id if the last ping was within
/// [kSessionDurationMinutes].
///
/// If time since last ping exceeds the duration, then the file
/// will be updated with a new session id and that will be returned.
///
/// Note, the file will always be updated when calling this method
/// because the last ping variable will always need to be persisted.
int getSessionId() {
_refreshSessionData();
final now = clock.now();
// Convert the epoch time from the last ping into datetime and check if we
// are within the kSessionDurationMinutes.
final lastPingDateTime = DateTime.fromMillisecondsSinceEpoch(_lastPing);
if (now.difference(lastPingDateTime).inMinutes > kSessionDurationMinutes) {
// In this case, we will need to change both the session id
// and the last ping value
_sessionId = now.millisecondsSinceEpoch;
}
// Update the last ping to reflect session activity continuing
_lastPing = now.millisecondsSinceEpoch;
// Rewrite the session object back to the file to persist
// for future events
sessionFile.writeAsStringSync(toJson());
return _sessionId;
}
/// Preps the [Session] class with the data found in the session json file.
///
/// We must check if telemetry is enabled to refresh the session data
/// because the refresh method will write to the session file and for
/// users that have opted out, we have to leave the session file empty
/// per the privacy document
void initialize(bool telemetryEnabled) {
if (telemetryEnabled) _refreshSessionData();
}
/// Return a json formatted representation of the class.
String toJson() => jsonEncode(<String, int>{
'session_id': _sessionId,
'last_ping': _lastPing,
});
/// This will go to the session file within the dart-tool
/// directory and fetch the latest data from the json to update
/// the class's variables. If the json file is malformed, a new
/// session file will be recreated.
///
/// This allows the session data in this class to always be up
/// to date incase another tool is also calling this package and
/// making updates to the session file.
void _refreshSessionData() {
/// Using a nested function here to reduce verbosity
void parseContents() {
final sessionFileContents = sessionFile.readAsStringSync();
final sessionObj =
jsonDecode(sessionFileContents) as Map<String, Object?>;
_sessionId = sessionObj['session_id'] as int;
_lastPing = sessionObj['last_ping'] as int;
}
try {
parseContents();
} on FormatException catch (err) {
Initializer.createSessionFile(sessionFile: sessionFile);
_errorHandler.log(Event.analyticsException(
workflow: 'Session._refreshSessionData',
error: err.runtimeType.toString(),
description: 'message: ${err.message}\nsource: ${err.source}',
));
// Fallback to setting the session id as the current time
final now = clock.now();
_sessionId = now.millisecondsSinceEpoch;
_lastPing = now.millisecondsSinceEpoch;
} on FileSystemException catch (err) {
Initializer.createSessionFile(sessionFile: sessionFile);
_errorHandler.log(Event.analyticsException(
workflow: 'Session._refreshSessionData',
error: err.runtimeType.toString(),
description: err.osError?.toString(),
));
// Fallback to setting the session id as the current time
final now = clock.now();
_sessionId = now.millisecondsSinceEpoch;
_lastPing = now.millisecondsSinceEpoch;
}
}
}