-
Notifications
You must be signed in to change notification settings - Fork 6.7k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
[AOT] Refactor Logger function to improve performance and mark managedCommon as AOT compatible #36327
base: main
Are you sure you want to change the base?
[AOT] Refactor Logger function to improve performance and mark managedCommon as AOT compatible #36327
Changes from 3 commits
6acfda1
f648b7a
0c03d2c
8c41509
925f44c
e86fbef
e0a2758
5ce827e
69244d0
226c375
2c15ebb
fbf0a46
fcc3bbf
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -15,15 +15,14 @@ namespace ManagedCommon | |
{ | ||
public static class Logger | ||
{ | ||
private static readonly Assembly Assembly = Assembly.GetExecutingAssembly(); | ||
private static readonly string Version = FileVersionInfo.GetVersionInfo(Assembly.Location).ProductVersion; | ||
|
||
private static readonly string Error = "Error"; | ||
private static readonly string Warning = "Warning"; | ||
private static readonly string Info = "Info"; | ||
private static readonly string Debug = "Debug"; | ||
private static readonly string TraceFlag = "Trace"; | ||
|
||
private static readonly string Version = GetVersion(); | ||
|
||
/// <summary> | ||
/// Initializes the logger and sets the path for logging. | ||
/// </summary> | ||
|
@@ -53,18 +52,37 @@ public static void InitializeLogger(string applicationLogPath, bool isLocalLow = | |
Trace.AutoFlush = true; | ||
} | ||
|
||
public static string GetVersion() | ||
{ | ||
string applicationName = "PowerToys.exe"; | ||
try | ||
{ | ||
var versionInfo = FileVersionInfo.GetVersionInfo(Path.Combine(System.AppContext.BaseDirectory, applicationName)); | ||
if (versionInfo != null) | ||
{ | ||
return versionInfo.ProductVersion; | ||
} | ||
} | ||
catch (Exception) | ||
{ | ||
return "0.0.0.1"; | ||
} | ||
|
||
return "0.0.0.1"; | ||
} | ||
|
||
[MethodImpl(MethodImplOptions.NoInlining)] | ||
public static void LogError(string message) | ||
public static void LogError(string message, [System.Runtime.CompilerServices.CallerMemberName] string memberName = "", [System.Runtime.CompilerServices.CallerFilePath] string sourceFilePath = "", [System.Runtime.CompilerServices.CallerLineNumber] int sourceLineNumber = 0) | ||
{ | ||
Log(message, Error); | ||
Log(message, Error, memberName, sourceFilePath, sourceLineNumber); | ||
} | ||
|
||
[MethodImpl(MethodImplOptions.NoInlining)] | ||
public static void LogError(string message, Exception ex) | ||
public static void LogError(string message, Exception ex, [System.Runtime.CompilerServices.CallerMemberName] string memberName = "", [System.Runtime.CompilerServices.CallerFilePath] string sourceFilePath = "", [System.Runtime.CompilerServices.CallerLineNumber] int sourceLineNumber = 0) | ||
{ | ||
if (ex == null) | ||
{ | ||
Log(message, Error); | ||
Log(message, Error, memberName, sourceFilePath, sourceLineNumber); | ||
} | ||
else | ||
{ | ||
|
@@ -83,38 +101,38 @@ public static void LogError(string message, Exception ex) | |
"Stack trace: " + Environment.NewLine + | ||
ex.StackTrace; | ||
|
||
Log(exMessage, Error); | ||
Log(exMessage, Error, memberName, sourceFilePath, sourceLineNumber); | ||
} | ||
} | ||
|
||
[MethodImpl(MethodImplOptions.NoInlining)] | ||
public static void LogWarning(string message) | ||
public static void LogWarning(string message, [System.Runtime.CompilerServices.CallerMemberName] string memberName = "", [System.Runtime.CompilerServices.CallerFilePath] string sourceFilePath = "", [System.Runtime.CompilerServices.CallerLineNumber] int sourceLineNumber = 0) | ||
{ | ||
Log(message, Warning); | ||
Log(message, Warning, memberName, sourceFilePath, sourceLineNumber); | ||
} | ||
|
||
[MethodImpl(MethodImplOptions.NoInlining)] | ||
public static void LogInfo(string message) | ||
public static void LogInfo(string message, [System.Runtime.CompilerServices.CallerMemberName] string memberName = "", [System.Runtime.CompilerServices.CallerFilePath] string sourceFilePath = "", [System.Runtime.CompilerServices.CallerLineNumber] int sourceLineNumber = 0) | ||
{ | ||
Log(message, Info); | ||
Log(message, Info, memberName, sourceFilePath, sourceLineNumber); | ||
} | ||
|
||
[MethodImpl(MethodImplOptions.NoInlining)] | ||
public static void LogDebug(string message) | ||
public static void LogDebug(string message, [System.Runtime.CompilerServices.CallerMemberName] string memberName = "", [System.Runtime.CompilerServices.CallerFilePath] string sourceFilePath = "", [System.Runtime.CompilerServices.CallerLineNumber] int sourceLineNumber = 0) | ||
{ | ||
Log(message, Debug); | ||
Log(message, Debug, memberName, sourceFilePath, sourceLineNumber); | ||
} | ||
|
||
[MethodImpl(MethodImplOptions.NoInlining)] | ||
public static void LogTrace() | ||
public static void LogTrace([System.Runtime.CompilerServices.CallerMemberName] string memberName = "", [System.Runtime.CompilerServices.CallerFilePath] string sourceFilePath = "", [System.Runtime.CompilerServices.CallerLineNumber] int sourceLineNumber = 0) | ||
{ | ||
Log(string.Empty, TraceFlag); | ||
Log(string.Empty, TraceFlag, memberName, sourceFilePath, sourceLineNumber); | ||
} | ||
|
||
[MethodImpl(MethodImplOptions.NoInlining)] | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. May be we don't need this one anymore. |
||
private static void Log(string message, string type) | ||
private static void Log(string message, string type, string memberName, string sourceFilePath, int sourceLineNumber) | ||
{ | ||
Trace.WriteLine("[" + DateTime.Now.TimeOfDay + "] [" + type + "] " + GetCallerInfo()); | ||
Trace.WriteLine("[" + DateTime.Now.TimeOfDay + "] [" + type + "] " + GetCallerInfo(memberName, sourceFilePath, sourceLineNumber)); | ||
Trace.Indent(); | ||
if (message != string.Empty) | ||
{ | ||
|
@@ -124,49 +142,11 @@ private static void Log(string message, string type) | |
Trace.Unindent(); | ||
} | ||
|
||
[MethodImpl(MethodImplOptions.NoInlining)] | ||
private static string GetCallerInfo() | ||
{ | ||
StackTrace stackTrace = new(); | ||
|
||
var callerMethod = GetCallerMethod(stackTrace); | ||
|
||
return $"{callerMethod?.DeclaringType?.Name}::{callerMethod.Name}"; | ||
} | ||
|
||
private static MethodBase GetCallerMethod(StackTrace stackTrace) | ||
private static string GetCallerInfo(string memberName, string sourceFilePath, int sourceLineNumber) | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. For sourceLineNumber. Last but not least, do we know how to view PowerToys log? If we can parrse the log and show a summary of before after, it will give confidence of this changes. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. There's currently no automated parsing. The logs are saved on %LOCALAPPDATA%\Microsoft\PowerToys\ and then inside a folder with the module name. |
||
{ | ||
const int topFrame = 3; | ||
|
||
var topMethod = stackTrace.GetFrame(topFrame)?.GetMethod(); | ||
|
||
try | ||
{ | ||
if (topMethod?.Name == nameof(IAsyncStateMachine.MoveNext) && typeof(IAsyncStateMachine).IsAssignableFrom(topMethod?.DeclaringType)) | ||
{ | ||
// Async method; return actual method as determined by heuristic: | ||
// "Nearest method on stack to async state-machine's MoveNext() in same namespace but in a different type". | ||
// There are tighter ways of determining the actual method, but this is good enough and probably faster. | ||
for (int deepFrame = topFrame + 1; deepFrame < stackTrace.FrameCount; deepFrame++) | ||
{ | ||
var deepMethod = stackTrace.GetFrame(deepFrame)?.GetMethod(); | ||
|
||
if (deepMethod?.DeclaringType != topMethod?.DeclaringType && deepMethod?.DeclaringType?.Namespace == topMethod?.DeclaringType?.Namespace) | ||
{ | ||
return deepMethod; | ||
} | ||
} | ||
} | ||
} | ||
catch (Exception) | ||
{ | ||
// Ignore exceptions in Release. The code above won't throw, but if it does, we don't want to crash the app. | ||
#if DEBUG | ||
throw; | ||
#endif | ||
} | ||
string fileName = Path.GetFileName(sourceFilePath); | ||
moooyo marked this conversation as resolved.
Show resolved
Hide resolved
|
||
|
||
return topMethod; | ||
return $"{fileName}::{memberName}::{sourceLineNumber}"; | ||
} | ||
} | ||
} |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,13 @@ | ||
// Copyright (c) Microsoft Corporation | ||
// The Microsoft Corporation licenses this file to you under the MIT license. | ||
// See the LICENSE file in the project root for more information. | ||
|
||
using System.Text.Json.Serialization; | ||
using static ManagedCommon.LanguageHelper; | ||
|
||
namespace ManagedCommon; | ||
|
||
[JsonSerializable(typeof(OutGoingLanguageSettings))] | ||
internal sealed partial class ManagedCommonJsonSerializerContext : JsonSerializerContext | ||
{ | ||
} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Can we just ask the caller to pass in its own class type?
From Github copilot, the following code should be AOT compatible:
var versionInfo = FileVersionInfo.GetVersionInfo(typeof(Program).Assembly.Location);
Console.WriteLine($"Version: {versionInfo.FileVersion}
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
@moooyo , I agree with @yeelam-gordon here that this doesn't make much sense. This is looking for PowerToys.exe when we should get the own version of the module that calls the Logger. Also, this needs to work for things logged in WinUI3 applications as well, which reside in a different folder in runtime, so I'm not sure the current state of the code would work for them.