This commit is contained in:
nsnail 2022-12-02 17:58:45 +08:00
parent 507b0aec5b
commit c09644c9c1
24 changed files with 384 additions and 443 deletions

View File

@ -2,14 +2,13 @@ namespace Dot;
public class DirOption : IOption
{
[Option('f', "filter", HelpText = nameof(Strings.FileSearchPattern), Default = "*.*"
, ResourceType = typeof(Strings))]
[Option('f', "filter", HelpText = nameof(Str.FileSearchPattern), Default = "*.*", ResourceType = typeof(Str))]
public string Filter { get; set; }
[Value(0, HelpText = nameof(Strings.FolderPath), Default = ".", ResourceType = typeof(Strings))]
[Value(0, HelpText = nameof(Str.FolderPath), Default = ".", ResourceType = typeof(Str))]
public string Path { get; set; }
[Option('r', "readonly", HelpText = nameof(Strings.ReadOnly), Default = false, ResourceType = typeof(Strings))]
[Option('r', "readonly", HelpText = nameof(Str.ReadOnly), Default = false, ResourceType = typeof(Str))]
public bool ReadOnly { get; set; }
}

View File

@ -7,11 +7,12 @@ public sealed class Main : Tool<Option>
public Main(Option opt) : base(opt) { }
public override void Run()
public override Task Run()
{
var guid = System.Guid.NewGuid().ToString();
if (Opt.Upper) guid = guid.ToUpper();
ClipboardService.SetText(guid);
Console.WriteLine(Strings.Copied, guid);
Console.WriteLine(Str.Copied, guid);
return Task.CompletedTask;
}
}

View File

@ -1,8 +1,8 @@
namespace Dot.Guid;
[Verb("guid", HelpText = nameof(Strings.GuidTool), ResourceType = typeof(Strings))]
[Verb("guid", HelpText = nameof(Str.GuidTool), ResourceType = typeof(Str))]
public class Option : IOption
{
[Option('u', "upper", HelpText = nameof(Strings.UseUppercase), Default = false, ResourceType = typeof(Strings))]
[Option('u', "upper", HelpText = nameof(Str.UseUppercase), Default = false, ResourceType = typeof(Str))]
public bool Upper { get; set; } //normal options here
}

View File

@ -2,5 +2,5 @@ namespace Dot;
public interface ITool
{
void Run();
Task Run();
}

View File

@ -21,14 +21,14 @@ namespace Dot.Lang {
[global::System.CodeDom.Compiler.GeneratedCodeAttribute("System.Resources.Tools.StronglyTypedResourceBuilder", "4.0.0.0")]
[global::System.Diagnostics.DebuggerNonUserCodeAttribute()]
[global::System.Runtime.CompilerServices.CompilerGeneratedAttribute()]
public class Strings {
public class Str {
private static global::System.Resources.ResourceManager resourceMan;
private static global::System.Globalization.CultureInfo resourceCulture;
[global::System.Diagnostics.CodeAnalysis.SuppressMessageAttribute("Microsoft.Performance", "CA1811:AvoidUncalledPrivateCode")]
internal Strings() {
internal Str() {
}
/// <summary>
@ -38,7 +38,7 @@ namespace Dot.Lang {
public static global::System.Resources.ResourceManager ResourceManager {
get {
if (object.ReferenceEquals(resourceMan, null)) {
global::System.Resources.ResourceManager temp = new global::System.Resources.ResourceManager("Dot.Lang.Strings", typeof(Strings).Assembly);
global::System.Resources.ResourceManager temp = new global::System.Resources.ResourceManager("Dot.Lang.Str", typeof(Str).Assembly);
resourceMan = temp;
}
return resourceMan;
@ -123,92 +123,29 @@ namespace Dot.Lang {
}
/// <summary>
/// Looks up a localized string similar to 连接NTP服务器: {0} &lt;{1}&gt; ... .
/// Looks up a localized string similar to Local clock offset.
/// </summary>
public static string Main_GetUtc_ {
public static string LocalClockOffset {
get {
return ResourceManager.GetString("Main_GetUtc_", resourceCulture);
return ResourceManager.GetString("LocalClockOffset", resourceCulture);
}
}
/// <summary>
/// Looks up a localized string similar to {0} ms.
/// Looks up a localized string similar to {0}, 本机时钟偏移: {1} ms.
/// </summary>
public static string Main_GetUtc__0_us {
public static string LocalTimeOffset {
get {
return ResourceManager.GetString("Main_GetUtc__0_us", resourceCulture);
return ResourceManager.GetString("LocalTimeOffset", resourceCulture);
}
}
/// <summary>
/// Looks up a localized string similar to 发送请求... .
/// Looks up a localized string similar to 本机时间已同步.
/// </summary>
public static string Main_GetUtc_sdf {
public static string LocalTimeSyncDone {
get {
return ResourceManager.GetString("Main_GetUtc_sdf", resourceCulture);
}
}
/// <summary>
/// Looks up a localized string similar to 接收数据... .
/// </summary>
public static string Main_GetUtc_接收请求___ {
get {
return ResourceManager.GetString("Main_GetUtc_接收请求___", resourceCulture);
}
}
/// <summary>
/// Looks up a localized string similar to NTP服务器时间: {0}.
/// </summary>
public static string Main_Run_NTP服务器时间___0_ {
get {
return ResourceManager.GetString("Main_Run_NTP服务器时间___0_", resourceCulture);
}
}
/// <summary>
/// Looks up a localized string similar to 按&quot;y&quot;同步系统时钟, 其它按键退出程序.
/// </summary>
public static string Main_Run_SyncClock {
get {
return ResourceManager.GetString("Main_Run_SyncClock", resourceCulture);
}
}
/// <summary>
/// Looks up a localized string similar to 同步完成, 开始复检....
/// </summary>
public static string Main_Run_SyncDone {
get {
return ResourceManager.GetString("Main_Run_SyncDone", resourceCulture);
}
}
/// <summary>
/// Looks up a localized string similar to 本地时钟快: {0} ms.
/// </summary>
public static string Main_Run_时差___0__ms {
get {
return ResourceManager.GetString("Main_Run_时差___0__ms", resourceCulture);
}
}
/// <summary>
/// Looks up a localized string similar to 本地时钟慢: {0} ms.
/// </summary>
public static string Main_Run_时差___1__ms {
get {
return ResourceManager.GetString("Main_Run_时差___1__ms", resourceCulture);
}
}
/// <summary>
/// Looks up a localized string similar to 本地时间: {0}.
/// </summary>
public static string Main_Run_本地时间___0_ {
get {
return ResourceManager.GetString("Main_Run_本地时间___0_", resourceCulture);
return ResourceManager.GetString("LocalTimeSyncDone", resourceCulture);
}
}
@ -222,20 +159,29 @@ namespace Dot.Lang {
}
/// <summary>
/// Looks up a localized string similar to 所有服务均不可用.
/// Looks up a localized string similar to {0} 通信中....
/// </summary>
public static string NoService {
public static string NtpCalling {
get {
return ResourceManager.GetString("NoService", resourceCulture);
return ResourceManager.GetString("NtpCalling", resourceCulture);
}
}
/// <summary>
/// Looks up a localized string similar to OK.
/// Looks up a localized string similar to 成功 {0}/{1} , 本机时钟偏移平均值: {2} ms.
/// </summary>
public static string OK {
public static string NtpReceiveDone {
get {
return ResourceManager.GetString("OK", resourceCulture);
return ResourceManager.GetString("NtpReceiveDone", resourceCulture);
}
}
/// <summary>
/// Looks up a localized string similar to {0}/{1} 个 NTP 服务器.
/// </summary>
public static string NtpServerCount {
get {
return ResourceManager.GetString("NtpServerCount", resourceCulture);
}
}
@ -320,6 +266,15 @@ namespace Dot.Lang {
}
}
/// <summary>
/// Looks up a localized string similar to Server.
/// </summary>
public static string Server {
get {
return ResourceManager.GetString("Server", resourceCulture);
}
}
/// <summary>
/// Looks up a localized string similar to 已读取:{0}/{1},处理:{2},跳过:{3}.
/// </summary>
@ -329,6 +284,24 @@ namespace Dot.Lang {
}
}
/// <summary>
/// Looks up a localized string similar to Status.
/// </summary>
public static string Status {
get {
return ResourceManager.GetString("Status", resourceCulture);
}
}
/// <summary>
/// Looks up a localized string similar to 同步本机时间.
/// </summary>
public static string SyncToLocalTime {
get {
return ResourceManager.GetString("SyncToLocalTime", resourceCulture);
}
}
/// <summary>
/// Looks up a localized string similar to 要处理的文本(默认取取剪贴板值).
/// </summary>
@ -339,11 +312,11 @@ namespace Dot.Lang {
}
/// <summary>
/// Looks up a localized string similar to 超时.
/// Looks up a localized string similar to 连接NTP服务器超时时间 (毫秒).
/// </summary>
public static string Timeout {
public static string TimeoutMillSecs {
get {
return ResourceManager.GetString("Timeout", resourceCulture);
return ResourceManager.GetString("TimeoutMillSecs", resourceCulture);
}
}

View File

@ -83,4 +83,34 @@
<data name="NoFileToBeProcessed" xml:space="preserve">
<value>No documents to be processed</value>
</data>
<data name="TimeoutMillSecs" xml:space="preserve">
<value>Timeout for connecting to the NTP server (milliseconds)</value>
</data>
<data name="SyncToLocalTime" xml:space="preserve">
<value>Synchronize local time</value>
</data>
<data name="NtpReceiveDone" xml:space="preserve">
<value>Success {0}/{1}, the average value of the clock offset of the machine:{2}ms</value>
</data>
<data name="NtpServerCount" xml:space="preserve">
<value>{0}/{1} NTP servers</value>
</data>
<data name="NtpCalling" xml:space="preserve">
<value>{0} In communication...</value>
</data>
<data name="LocalTimeOffset" xml:space="preserve">
<value>{0}, local clock offset: {1} ms</value>
</data>
<data name="LocalTimeSyncDone" xml:space="preserve">
<value>Local time has been synchronized</value>
</data>
<data name="Server" xml:space="preserve">
<value>Server</value>
</data>
<data name="Status" xml:space="preserve">
<value>Status</value>
</data>
<data name="LocalClockOffset" xml:space="preserve">
<value>Local clock offset</value>
</data>
</root>

View File

@ -51,6 +51,13 @@
<data name="FileSearchPattern" xml:space="preserve">
<value>文件通配符</value>
</data>
<data name="TimeoutMillSecs" xml:space="preserve">
<value>连接NTP服务器超时时间 (毫秒)</value>
</data>
<data name="SyncToLocalTime" xml:space="preserve">
<value>同步本机时间</value>
</data>
<data name="FolderPath" xml:space="preserve">
<value>要处理的目录路径</value>
</data>
@ -91,43 +98,28 @@
<data name="NoFileToBeProcessed" xml:space="preserve">
<value>没有需要处理的文件</value>
</data>
<data name="Main_GetUtc_" xml:space="preserve">
<value>连接NTP服务器: {0} &lt;{1}&gt; ... </value>
<data name="NtpReceiveDone" xml:space="preserve">
<value>成功 {0}/{1} , 本机时钟偏移平均值: {2} ms</value>
</data>
<data name="Main_GetUtc__0_us" xml:space="preserve">
<value>{0} ms</value>
<data name="NtpServerCount" xml:space="preserve">
<value>{0}/{1} 个 NTP 服务器</value>
</data>
<data name="Main_GetUtc_sdf" xml:space="preserve">
<value>发送请求... </value>
<data name="NtpCalling" xml:space="preserve">
<value>{0} 通信中...</value>
</data>
<data name="Main_GetUtc_接收请求___" xml:space="preserve">
<value>接收数据... </value>
<data name="LocalTimeOffset" xml:space="preserve">
<value>{0}, 本机时钟偏移: {1} ms</value>
</data>
<data name="Timeout" xml:space="preserve">
<value>超时</value>
<data name="LocalTimeSyncDone" xml:space="preserve">
<value>本机时间已同步</value>
</data>
<data name="OK" xml:space="preserve">
<value>OK</value>
<data name="Server" xml:space="preserve">
<value>Server</value>
</data>
<data name="NoService" xml:space="preserve">
<value>所有服务均不可用</value>
<data name="Status" xml:space="preserve">
<value>Status</value>
</data>
<data name="Main_Run_NTP服务器时间___0_" xml:space="preserve">
<value>NTP服务器时间: {0}</value>
</data>
<data name="Main_Run_本地时间___0_" xml:space="preserve">
<value>本地时间: {0}</value>
</data>
<data name="Main_Run_时差___0__ms" xml:space="preserve">
<value>本地时钟快: {0} ms</value>
</data>
<data name="Main_Run_时差___1__ms" xml:space="preserve">
<value>本地时钟慢: {0} ms</value>
</data>
<data name="Main_Run_SyncClock" xml:space="preserve">
<value>按"y"同步系统时钟, 其它按键退出程序</value>
</data>
<data name="Main_Run_SyncDone" xml:space="preserve">
<value>同步完成, 开始复检...</value>
<data name="LocalClockOffset" xml:space="preserve">
<value>Local clock offset</value>
</data>
</root>

View File

@ -10,10 +10,10 @@ Type[] LoadVerbs()
}
void Run(object args)
async Task Run(object args)
{
var tool = ToolsFactory.Create(args as IOption);
tool.Run();
await tool.Run();
}
@ -22,7 +22,7 @@ void Run(object args)
var types = LoadVerbs();
try {
Parser.Default.ParseArguments(args, types).WithParsed(Run);
await Parser.Default.ParseArguments(args, types).WithParsedAsync(Run);
}
catch (ArgumentException ex) {
Console.Error.WriteLine(ex.Message);

View File

@ -15,7 +15,7 @@ public sealed class Main : Tool<Option>
public Main(Option opt) : base(opt) { }
public override void Run()
public override Task Run()
{
unsafe {
var pSource = stackalloc char[_charTable.Sum(x => x.Length)];
@ -47,7 +47,9 @@ public sealed class Main : Tool<Option>
var result = new string(pDest, 0, Opt.Length);
ClipboardService.SetText(result);
Console.WriteLine(Strings.Copied, result);
Console.WriteLine(Str.Copied, result);
}
return Task.CompletedTask;
}
}

View File

@ -1,6 +1,6 @@
namespace Dot.Pwd;
[Verb("pwd", HelpText = nameof(Strings.RandomPasswordGenerator), ResourceType = typeof(Strings))]
[Verb("pwd", HelpText = nameof(Str.RandomPasswordGenerator), ResourceType = typeof(Str))]
public class Option : IOption
{
[Flags]
@ -12,10 +12,10 @@ public class Option : IOption
, SpecialCharacter = 8
}
[Value(1, Required = true, HelpText = nameof(Strings.PwdLength), ResourceType = typeof(Strings))]
[Value(1, Required = true, HelpText = nameof(Str.PwdLength), ResourceType = typeof(Str))]
public int Length { get; set; }
[Value(0, Required = true, HelpText = nameof(Strings.PwdGenerateTypes), ResourceType = typeof(Strings))]
[Value(0, Required = true, HelpText = nameof(Str.PwdGenerateTypes), ResourceType = typeof(Str))]
public GenerateTypes Type { get; set; }
}

View File

@ -1,66 +0,0 @@
using System.Text.RegularExpressions;
using NSExt.Extensions;
using TextCopy;
namespace Dot.Random;
public sealed partial class Main : Tool<Option>
{
private readonly char[][] _charTable = {
"0123456789".ToCharArray() //
, "abcdefghijklmnopqrstuvwxyz".ToCharArray()
, "ABCDEFGHIJKLMNOPQRSTUVWXYZ".ToCharArray()
, "!\"#$%&'()*+,-./:;<=>?@[\\]^_`{|}~".ToCharArray()
};
public Main(Option opt) : base(opt) { }
[GeneratedRegex("[a-z]")]
private static partial Regex RegexLowerCaseLetter();
[GeneratedRegex("\\d")]
private static partial Regex RegexNumber();
[GeneratedRegex("[^\\da-zA-Z]")]
private static partial Regex RegexSpecialCharacter();
[GeneratedRegex("[A-Z]")]
private static partial Regex RegexUpperCaseLetter();
public override void Run()
{
unsafe {
var pSource = stackalloc char[_charTable.Sum(x => x.Length)];
var pDest = stackalloc char[Opt.Length];
var sourceLen = 0;
if (Opt.Type.HasFlag(Option.GenerateTypes.Number))
foreach (var c in _charTable[0])
*(pSource + sourceLen++) = c;
if (Opt.Type.HasFlag(Option.GenerateTypes.LowerCaseLetter))
foreach (var c in _charTable[1])
*(pSource + sourceLen++) = c;
if (Opt.Type.HasFlag(Option.GenerateTypes.UpperCaseLetter))
foreach (var c in _charTable[2])
*(pSource + sourceLen++) = c;
if (Opt.Type.HasFlag(Option.GenerateTypes.SpecialCharacter))
foreach (var c in _charTable[3])
*(pSource + sourceLen++) = c;
var randScope = new[] { 0, sourceLen };
for (var i = 0; i != Opt.Length; ++i) //
*(pDest + i) = *(pSource + randScope.Rand());
var result = new string(pDest, 0, Opt.Length);
ClipboardService.SetText(result);
Console.WriteLine($"已复制到剪贴板:{result}");
}
}
}

View File

@ -1,21 +0,0 @@
namespace Dot.Random;
[Verb("rand", HelpText = "随机数生成器")]
public class Option : IOption
{
[Flags]
public enum GenerateTypes
{
Number = 1
, LowerCaseLetter = 2
, UpperCaseLetter = 4
, SpecialCharacter = 8
}
[Value(1, MetaName = "长度", Required = true, HelpText = "随机数字长度")]
public int Length { get; set; }
[Value(0, MetaName = "生成类型", Required = true, HelpText = "BitSet 1[0-9]2[a-z]4[A-Z]8[ascii.0x21-0x2F]")]
public GenerateTypes Type { get; set; }
}

View File

@ -27,13 +27,13 @@ public sealed class Main : Tool<Option>, IDisposable
_disposed = true;
}
private void FileHandle(string file)
private async ValueTask FileHandle(string file, CancellationToken _)
{
_step2Bar.Tick();
ShowMessage(1, 0, 0);
var spacesCnt = 0;
using var fsrw = OpenFileStream(file, FileMode.Open, FileAccess.ReadWrite);
await using var fsrw = OpenFileStream(file, FileMode.Open, FileAccess.ReadWrite);
if (Opt.ReadOnly) { //测试,只读模式
ShowMessage(0, 1, 0);
@ -77,7 +77,7 @@ public sealed class Main : Tool<Option>, IDisposable
_procedCnt += procedCnt;
_replaceCnt += removeCnt;
_breakCnt += breakCnt;
_step2Bar.Message = string.Format(Strings.ShowMessageTemp, _procedCnt, _totalCnt, _replaceCnt, _breakCnt);
_step2Bar.Message = string.Format(Str.ShowMessageTemp, _procedCnt, _totalCnt, _replaceCnt, _breakCnt);
}
}
@ -89,25 +89,25 @@ public sealed class Main : Tool<Option>, IDisposable
/// <inheritdoc />
[SuppressMessage("ReSharper", "PossibleMultipleEnumeration")]
public override void Run()
public override async Task Run()
{
if (!Directory.Exists(Opt.Path))
throw new ArgumentException(nameof(Opt.Path), string.Format(Strings.PathNotFound, Opt.Path));
throw new ArgumentException(nameof(Opt.Path), string.Format(Str.PathNotFound, Opt.Path));
using var step1Bar = new IndeterminateProgressBar(Strings.SearchingFile, DefaultProgressBarOptions);
using var step1Bar = new IndeterminateProgressBar(Str.SearchingFile, DefaultProgressBarOptions);
var fileList = EnumerateFiles(Opt.Path, Opt.Filter);
_totalCnt = fileList.Count();
step1Bar.Message = string.Format(Strings.SearchingFileOK, _totalCnt);
step1Bar.Message = string.Format(Str.SearchingFileOK, _totalCnt);
step1Bar.Finished();
if (_totalCnt == 0) return;
_step2Bar = step1Bar.Spawn(_totalCnt, string.Empty, DefaultProgressBarOptions);
Parallel.ForEach(fileList, FileHandle);
await Parallel.ForEachAsync(fileList, FileHandle);
}
}

View File

@ -1,4 +1,4 @@
namespace Dot.RmBlank;
[Verb("rm-blank", HelpText = nameof(Strings.RemoveTrailingWhiteSpaces), ResourceType = typeof(Strings))]
[Verb("rm-blank", HelpText = nameof(Str.RemoveTrailingWhiteSpaces), ResourceType = typeof(Str))]
public class Option : DirOption { }

View File

@ -11,6 +11,7 @@ public sealed class Main : Tool<Option>, IDisposable
private ChildProgressBar _step2Bar;
private int _totalCnt;
private int _trimCnt;
private readonly byte[] _utf8Bom = { 0xef, 0xbb, 0xbf };
public Main(Option opt) : base(opt) { }
@ -19,6 +20,17 @@ public sealed class Main : Tool<Option>, IDisposable
Dispose(false);
}
private bool CreateTempFile(Stream fsr, string tmpFile)
{
Span<byte> buffer = stackalloc byte[_utf8Bom.Length];
var readLen = fsr.Read(buffer);
if (readLen != _utf8Bom.Length || !buffer.SequenceEqual(_utf8Bom)) return false;
using var fsw = OpenFileStream(tmpFile, FileMode.OpenOrCreate, FileAccess.Write);
int data;
while ((data = fsr.ReadByte()) != -1) fsw.WriteByte((byte)data);
return true;
}
private void Dispose(bool disposing)
{
if (_disposed) return;
@ -26,50 +38,14 @@ public sealed class Main : Tool<Option>, IDisposable
_disposed = true;
}
private void ShowMessage(int procedCnt, int replaceCnt, int breakCnt)
private async ValueTask FileHandle(string file, CancellationToken _)
{
lock (_lockObj) {
_procedCnt += procedCnt;
_trimCnt += replaceCnt;
_breakCnt += breakCnt;
_step2Bar.Message = string.Format(Strings.ShowMessageTemp, _procedCnt, _totalCnt, _trimCnt, _breakCnt);
}
}
public void Dispose()
{
Dispose(true);
GC.SuppressFinalize(this);
}
[SuppressMessage("ReSharper", "PossibleMultipleEnumeration")]
public override void Run()
{
if (!Directory.Exists(Opt.Path))
throw new ArgumentException(nameof(Opt.Path), string.Format(Strings.PathNotFound, Opt.Path));
var utf8Bom = new byte[] { 0xef, 0xbb, 0xbf };
using var step1Bar = new IndeterminateProgressBar(Strings.SearchingFile, DefaultProgressBarOptions);
var fileList = EnumerateFiles(Opt.Path, Opt.Filter);
_totalCnt = fileList.Count();
step1Bar.Message = string.Format(Strings.SearchingFileOK, _totalCnt);
step1Bar.Finished();
if (_totalCnt == 0) return;
_step2Bar = step1Bar.Spawn(_totalCnt, string.Empty, DefaultProgressBarOptions);
Parallel.ForEach(fileList, file => {
_step2Bar.Tick();
ShowMessage(1, 0, 0);
var tmpFile = $"{file}.tmp";
var isReplaced = false;
using (var fsr = OpenFileStream(file, FileMode.Open, FileAccess.Read, FileShare.ReadWrite)) {
bool isReplaced;
await using (var fsr = OpenFileStream(file, FileMode.Open, FileAccess.Read, FileShare.ReadWrite)) {
if (Opt.ReadOnly) { //测试,只读模式
ShowMessage(0, 1, 0);
return;
@ -81,14 +57,7 @@ public sealed class Main : Tool<Option>, IDisposable
}
Span<byte> buffer = stackalloc byte[utf8Bom.Length];
var readLen = fsr.Read(buffer);
if (readLen == utf8Bom.Length && buffer.SequenceEqual(utf8Bom)) {
using var fsw = OpenFileStream(tmpFile, FileMode.OpenOrCreate, FileAccess.Write);
int data;
while ((data = fsr.ReadByte()) != -1) fsw.WriteByte((byte)data);
isReplaced = true;
}
isReplaced = CreateTempFile(fsr, tmpFile);
}
if (isReplaced) {
@ -98,6 +67,44 @@ public sealed class Main : Tool<Option>, IDisposable
else {
ShowMessage(0, 0, 1);
}
});
}
private void ShowMessage(int procedCnt, int replaceCnt, int breakCnt)
{
lock (_lockObj) {
_procedCnt += procedCnt;
_trimCnt += replaceCnt;
_breakCnt += breakCnt;
_step2Bar.Message = string.Format(Str.ShowMessageTemp, _procedCnt, _totalCnt, _trimCnt, _breakCnt);
}
}
public void Dispose()
{
Dispose(true);
GC.SuppressFinalize(this);
}
[SuppressMessage("ReSharper", "PossibleMultipleEnumeration")]
public override async Task Run()
{
if (!Directory.Exists(Opt.Path))
throw new ArgumentException(nameof(Opt.Path), string.Format(Str.PathNotFound, Opt.Path));
using var step1Bar = new IndeterminateProgressBar(Str.SearchingFile, DefaultProgressBarOptions);
var fileList = EnumerateFiles(Opt.Path, Opt.Filter);
_totalCnt = fileList.Count();
step1Bar.Message = string.Format(Str.SearchingFileOK, _totalCnt);
step1Bar.Finished();
if (_totalCnt == 0) return;
_step2Bar = step1Bar.Spawn(_totalCnt, string.Empty, DefaultProgressBarOptions);
await Parallel.ForEachAsync(fileList, FileHandle);
}
}

View File

@ -1,4 +1,4 @@
namespace Dot.RmBom;
[Verb("rm-bom", HelpText = nameof(Strings.TrimUtf8Bom), ResourceType = typeof(Strings))]
[Verb("rm-bom", HelpText = nameof(Str.TrimUtf8Bom), ResourceType = typeof(Str))]
public class Option : DirOption { }

View File

@ -25,11 +25,7 @@ public sealed class Main : Tool<Option>
public ReadOnlySpan<char> UrlEncode;
}
public Main(Option opt) : base(opt)
{
if (Opt.Text.NullOrEmpty()) Opt.Text = ClipboardService.GetText();
if (Opt.Text.NullOrEmpty()) throw new ArgumentException(Strings.InputTextIsEmpty);
}
public Main(Option opt) : base(opt) { }
private static Output BuildOutput(string text, Encoding enc)
{
@ -68,6 +64,22 @@ public sealed class Main : Tool<Option>
return ret;
}
private static void ParseAndShow(string text)
{
Encoding.RegisterProvider(CodePagesEncodingProvider.Instance);
var ansi = BuildOutput(text, Encoding.GetEncoding("gbk"));
var utf8 = BuildOutput(text, Encoding.UTF8);
var unicodeLittleEndian = BuildOutput(text, Encoding.Unicode);
var unicodeBigEndian = BuildOutput(text, Encoding.BigEndianUnicode);
PrintOutput(ansi);
PrintOutput(utf8);
PrintOutput(unicodeLittleEndian);
PrintOutput(unicodeBigEndian);
}
private static void PrintOutput(Output o)
{
var outputTemp = $"""
@ -89,20 +101,14 @@ html-decode: {o.HtmlDecode}
Console.WriteLine(outputTemp);
}
public override void Run()
public override async Task Run()
{
Encoding.RegisterProvider(CodePagesEncodingProvider.Instance);
var ansi = BuildOutput(Opt.Text, Encoding.GetEncoding("gbk"));
var utf8 = BuildOutput(Opt.Text, Encoding.UTF8);
var unicodeLittleEndian = BuildOutput(Opt.Text, Encoding.Unicode);
var unicodeBigEndian = BuildOutput(Opt.Text, Encoding.BigEndianUnicode);
if (Opt.Text.NullOrEmpty()) Opt.Text = await ClipboardService.GetTextAsync();
if (Opt.Text.NullOrEmpty()) throw new ArgumentException(Str.InputTextIsEmpty);
PrintOutput(ansi);
PrintOutput(utf8);
PrintOutput(unicodeLittleEndian);
PrintOutput(unicodeBigEndian);
Console.Write(Strings.PressAnyKey);
ParseAndShow(Opt.Text);
Console.Write(Str.PressAnyKey);
Console.ReadKey();
}
}

View File

@ -1,8 +1,8 @@
namespace Dot.Text;
[Verb("text", HelpText = nameof(Strings.HelpForText), ResourceType = typeof(Strings))]
[Verb("text", HelpText = nameof(Str.HelpForText), ResourceType = typeof(Str))]
public class Option : IOption
{
[Value(0, HelpText = nameof(Strings.TextTobeProcessed), ResourceType = typeof(Strings))]
[Value(0, HelpText = nameof(Str.TextTobeProcessed), ResourceType = typeof(Str))]
public string Text { get; set; }
}

View File

@ -1,3 +1,4 @@
using System.Diagnostics.CodeAnalysis;
using System.Net.Sockets;
using System.Runtime.InteropServices;
@ -5,8 +6,16 @@ namespace Dot.Time;
public sealed class Main : Tool<Option>
{
private enum ServerStatues : byte
{
Ready
, Connecting
, Succeed
, Failed
}
[StructLayout(LayoutKind.Explicit)]
public ref struct _SYSTEMTIME
private ref struct Systemtime
{
[FieldOffset(6)] public ushort wDay;
[FieldOffset(4)] public ushort wDayOfWeek;
@ -18,163 +27,168 @@ public sealed class Main : Tool<Option>
[FieldOffset(0)] public ushort wYear;
}
private readonly Dictionary<string, string> _serverList = new() {
{ "ntp.ntsc.ac.cn", "国家授时中心 NTP 服务器" }
, { "cn.ntp.org.cn", "中国 NTP 快速授时服务" }
, { "edu.ntp.org.cn", "中国 NTP 快速授时服务" }
, { "cn.pool.ntp.org", "国际 NTP 快速授时服务" }
, { "time.pool.aliyun.com", "阿里云公共 NTP 服务器" }
, { "time1.aliyun.com", "阿里云公共 NTP 服务器" }
, { "time2.aliyun.com", "阿里云公共 NTP 服务器" }
, { "time3.aliyun.com", "阿里云公共 NTP 服务器" }
, { "time4.aliyun.com", "阿里云公共 NTP 服务器" }
, { "time5.aliyun.com", "阿里云公共 NTP 服务器" }
, { "time6.aliyun.com", "阿里云公共 NTP 服务器" }
, { "time7.aliyun.com", "阿里云公共 NTP 服务器" }
, { "time1.cloud.tencent.com", "腾讯云公共 NTP 服务器" }
, { "time2.cloud.tencent.com", "腾讯云公共 NTP 服务器" }
, { "time3.cloud.tencent.com", "腾讯云公共 NTP 服务器" }
, { "time4.cloud.tencent.com", "腾讯云公共 NTP 服务器" }
, { "time5.cloud.tencent.com", "腾讯云公共 NTP 服务器" }
, { "ntp.sjtu.edu.cn", "教育网(高校自建)" }
, { "ntp.neu.edu.cn", "教育网(高校自建)" }
, { "ntp.bupt.edu.cn", "教育网(高校自建)" }
, { "ntp.shu.edu.cn", "教育网(高校自建)" }
, { "pool.ntp.org", "国际 NTP 快速授时服务" }
, { "0.pool.ntp.org", "国际 NTP 快速授时服务" }
, { "1.pool.ntp.org", "国际 NTP 快速授时服务" }
, { "2.pool.ntp.org", "国际 NTP 快速授时服务" }
, { "3.pool.ntp.org", "国际 NTP 快速授时服务" }
, { "asia.pool.ntp.org", "国际 NTP 快速授时服务" }
, { "time1.google.com", "谷歌公共 NTP 服务器" }
, { "time2.google.com", "谷歌公共 NTP 服务器" }
, { "time3.google.com", "谷歌公共 NTP 服务器" }
, { "time4.google.com", "谷歌公共 NTP 服务器" }
, { "time.apple.com", "苹果公司公共 NTP 服务器" }
, { "time1.apple.com", "苹果公司公共 NTP 服务器" }
, { "time2.apple.com", "苹果公司公共 NTP 服务器" }
, { "time3.apple.com", "苹果公司公共 NTP 服务器" }
, { "time4.apple.com", "苹果公司公共 NTP 服务器" }
, { "time5.apple.com", "苹果公司公共 NTP 服务器" }
, { "time6.apple.com", "苹果公司公共 NTP 服务器" }
, { "time7.apple.com", "苹果公司公共 NTP 服务器" }
, { "time.windows.com", "微软 Windows NTP 服务器" }
, { "time.nist.gov", "美国标准技术研究院 NTP 服务器" }
, { "time-nw.nist.gov", "美国标准技术研究院 NTP 服务器" }
, { "time-a.nist.gov", "美国标准技术研究院 NTP 服务器" }
, { "time-b.nist.gov", "美国标准技术研究院 NTP 服务器" }
, { "stdtime.gov.hk", "香港天文台公共 NTP 服务器" }
private const int _MAX_DEGREE_OF_PARALLELISM = 10;
private const int _NTP_PORT = 123;
private readonly char[] _loading = { '-', '\\', '|', '/' };
private int _procedCnt;
private readonly int _serverCnt;
private readonly Dictionary<string, Server> _serverDictionary;
private readonly string[] _serverDomains = {
"ntp.ntsc.ac.cn", "cn.ntp.org.cn", "edu.ntp.org.cn"
, "cn.pool.ntp.org", "time.pool.aliyun.com", "time1.aliyun.com"
, "time2.aliyun.com", "time3.aliyun.com", "time4.aliyun.com"
, "time5.aliyun.com", "time6.aliyun.com", "time7.aliyun.com"
, "time1.cloud.tencent.com", "time2.cloud.tencent.com"
, "time3.cloud.tencent.com", "time4.cloud.tencent.com"
, "time5.cloud.tencent.com", "ntp.sjtu.edu.cn", "ntp.neu.edu.cn"
, "ntp.bupt.edu.cn", "ntp.shu.edu.cn", "pool.ntp.org", "0.pool.ntp.org"
, "1.pool.ntp.org", "2.pool.ntp.org", "3.pool.ntp.org"
, "asia.pool.ntp.org", "time1.google.com", "time2.google.com"
, "time3.google.com", "time4.google.com", "time.apple.com"
, "time1.apple.com", "time2.apple.com", "time3.apple.com"
, "time4.apple.com", "time5.apple.com", "time6.apple.com"
, "time7.apple.com", "time.windows.com", "time.nist.gov"
, "time-nw.nist.gov", "time-a.nist.gov", "time-b.nist.gov"
, "stdtime.gov.hk"
};
public Main(Option opt) : base(opt) { }
private int _successCnt;
private TimeSpan GetNtpOffset()
public Main(Option opt) : base(opt)
{
_serverCnt = _serverDomains.Length;
_serverDictionary = _serverDomains.ToDictionary(x => x, _ => new Server { Status = ServerStatues.Ready });
}
private TimeSpan GetNtpOffset(string server)
{
Span<byte> ntpData = stackalloc byte[48];
TimeSpan ts;
ntpData[0] = 0x1B;
foreach (var server in _serverList) {
Console.Write(Strings.Main_GetUtc_, server.Key, server.Value);
using var socket
= new Socket(AddressFamily.InterNetwork, SocketType.Dgram, ProtocolType.Udp) { ReceiveTimeout = 3000 };
= new Socket(AddressFamily.InterNetwork, SocketType.Dgram, ProtocolType.Udp) {
ReceiveTimeout = Opt.Timeout
};
try {
socket.Connect(_serverList.First().Key, 123);
Console.WriteLine(Strings.OK);
Console.Write(Strings.Main_GetUtc_sdf);
socket.Connect(server, _NTP_PORT);
socket.Send(ntpData);
Console.WriteLine(Strings.OK);
Console.Write(Strings.Main_GetUtc_接收请求___);
var timeStart = DateTime.Now;
var timeBefore = DateTime.Now;
socket.Receive(ntpData);
ts = DateTime.Now - timeStart;
Console.WriteLine(Strings.Main_GetUtc__0_us, ts.TotalMilliseconds);
}
catch (Exception) {
Console.WriteLine(Strings.Timeout);
continue;
}
finally {
socket.Close();
}
var transferTime = DateTime.Now - timeBefore;
var intPart = ((ulong)ntpData[40] << 24) //
| ((ulong)ntpData[41] << 16) //
| ((ulong)ntpData[42] << 8) //
| ntpData[43];
var fractPart = ((ulong)ntpData[44] << 24) //
| ((ulong)ntpData[45] << 16) //
| ((ulong)ntpData[46] << 8) //
| ntpData[47];
var from1900Ms = intPart * 1000 + fractPart * 1000 / 0x100000000L;
var onlineTime = new DateTime(1900, 1, 1).AddMilliseconds((long)from1900Ms) + ts;
var onlineTime = new DateTime(1900, 1, 1).AddMilliseconds((long)from1900Ms) + transferTime / 2;
return DateTime.UtcNow - onlineTime;
}
catch (Exception) {
return TimeSpan.Zero;
}
finally {
socket.Close();
}
}
throw new TimeoutException();
private async void Printing()
{
const string outputTemp = "{0,-30}\t{1}\t{2,20}\t{3,20}";
var rolling = 0;
Console.Clear();
while (true) {
await Task.Delay(100);
Console.SetCursorPosition(0, 0);
var row = //
_serverDictionary.Select(x //
=> string.Format(outputTemp, x.Key
, x.Value.Status == ServerStatues.Connecting
? _loading[++rolling % 4]
: ' ', x.Value.Status
, x.Value.Offset == TimeSpan.Zero
? string.Empty
: x.Value.Offset));
Console.WriteLine(outputTemp, Str.Server, ' ', Str.Status, Str.LocalClockOffset);
Console.WriteLine(string.Join(Environment.NewLine, row));
if (_procedCnt == _serverCnt) break;
}
}
[DllImport("Kernel32.dll")]
public static extern unsafe void GetLocalTime(_SYSTEMTIME* st);
private static extern void SetLocalTime(Systemtime st);
public override void Run()
private static void SetSysteTime(DateTime time)
{
while (true) {
TimeSpan offset;
try {
offset = GetNtpOffset();
}
catch (TimeoutException) {
Console.Error.WriteLine(Strings.NoService);
return;
}
using var tokenSource = new CancellationTokenSource();
var token = tokenSource.Token;
Task.Run(async () => {
for (;;) {
if (token.IsCancellationRequested) return;
Console.WriteLine(Strings.Main_Run_NTP服务器时间___0_, (DateTime.Now + offset).ToString("O"));
Console.WriteLine(Strings.Main_Run_本地时间___0_, DateTime.Now.ToString("O"));
Console.WriteLine(offset > TimeSpan.Zero
? string.Format(Strings.Main_Run_时差___0__ms, offset.TotalMilliseconds)
: string.Format(Strings.Main_Run_时差___1__ms, -offset.TotalMilliseconds));
Console.WriteLine(Strings.Main_Run_SyncClock);
await Task.Delay(1000, token);
Console.Clear();
}
}, token);
if (Console.ReadKey().Key == ConsoleKey.Y) {
var ntpTime = DateTime.Now - offset;
var f = new _SYSTEMTIME {
wDay = (ushort)ntpTime.Day
, wDayOfWeek = (ushort)ntpTime.DayOfWeek
, wHour = (ushort)ntpTime.Hour
, wMilliseconds = (ushort)ntpTime.Millisecond
, wMinute = (ushort)ntpTime.Minute
, wMonth = (ushort)ntpTime.Month
, wSecond = (ushort)ntpTime.Second
, wYear = (ushort)ntpTime.Year
var timeToSet = new Systemtime {
wDay = (ushort)time.Day
, wDayOfWeek = (ushort)time.DayOfWeek
, wHour = (ushort)time.Hour
, wMilliseconds = (ushort)time.Millisecond
, wMinute = (ushort)time.Minute
, wMonth = (ushort)time.Month
, wSecond = (ushort)time.Second
, wYear = (ushort)time.Year
};
SetLocalTime(f);
Console.WriteLine(Strings.Main_Run_SyncDone);
tokenSource.Cancel();
continue;
SetLocalTime(timeToSet);
}
break;
[SuppressMessage("ReSharper", "AccessToDisposedClosure")]
public override async Task Run()
{
var tPrinting = Task.Run(Printing);
await Parallel.ForEachAsync(_serverDictionary
, new ParallelOptions { MaxDegreeOfParallelism = _MAX_DEGREE_OF_PARALLELISM }
, (server, _) => {
server.Value.Status = ServerStatues.Connecting;
var offset = GetNtpOffset(server.Key);
Interlocked.Increment(ref _procedCnt);
if (offset == TimeSpan.Zero) {
server.Value.Status = ServerStatues.Failed;
}
else {
server.Value.Status = ServerStatues.Succeed;
Interlocked.Increment(ref _successCnt);
server.Value.Offset = offset;
}
[DllImportAttribute("Kernel32.dll")]
public static extern void SetLocalTime(_SYSTEMTIME st);
return ValueTask.CompletedTask;
});
tPrinting.Wait();
var avgOffset = TimeSpan.FromTicks((long)_serverDictionary //
.Where(x => x.Value.Status == ServerStatues.Succeed)
.Average(x => x.Value.Offset.Ticks));
Console.WriteLine(Str.NtpReceiveDone, _successCnt, _serverCnt, avgOffset.TotalMilliseconds);
if (!Opt.Sync) return;
Console.WriteLine();
SetSysteTime(DateTime.Now - avgOffset);
Console.WriteLine(Str.LocalTimeSyncDone);
}
private record Server
{
public TimeSpan Offset;
public ServerStatues Status;
}
}

View File

@ -1,8 +1,11 @@
namespace Dot.Time;
[Verb("time", HelpText = nameof(Strings.HelpForText), ResourceType = typeof(Strings))]
[Verb("time", HelpText = nameof(Str.HelpForText), ResourceType = typeof(Str))]
public class Option : IOption
{
[Value(0, HelpText = nameof(Strings.TextTobeProcessed), ResourceType = typeof(Strings))]
public string Text { get; set; }
[Option('s', "sync", HelpText = nameof(Str.SyncToLocalTime), Default = false, ResourceType = typeof(Str))]
public bool Sync { get; set; }
[Option('t', "timeout", HelpText = nameof(Str.TimeoutMillSecs), Default = 3000, ResourceType = typeof(Str))]
public int Timeout { get; set; }
}

View File

@ -26,7 +26,7 @@ public sealed class Main : Tool<Option>, IDisposable
_disposed = true;
}
private void FileHandle(string file)
private async ValueTask FileHandle(string file, CancellationToken _)
{
_step2Bar.Tick();
ShowMessage(1, 0, 0);
@ -36,7 +36,7 @@ public sealed class Main : Tool<Option>, IDisposable
var isBin = false;
int data;
using (var fsr = OpenFileStream(file, FileMode.Open, FileAccess.Read, FileShare.ReadWrite)) {
await using (var fsr = OpenFileStream(file, FileMode.Open, FileAccess.Read, FileShare.ReadWrite)) {
if (Opt.ReadOnly) { //测试,只读模式
ShowMessage(0, 1, 0);
return;
@ -48,7 +48,7 @@ public sealed class Main : Tool<Option>, IDisposable
}
using var fsw = OpenFileStream(tmpFile, FileMode.OpenOrCreate, FileAccess.Write);
await using var fsw = OpenFileStream(tmpFile, FileMode.OpenOrCreate, FileAccess.Write);
while ((data = fsr.ReadByte()) != -1) {
switch (data) {
@ -92,7 +92,7 @@ public sealed class Main : Tool<Option>, IDisposable
_procedCnt += procedCnt;
_replaceCnt += replaceCnt;
_breakCnt += breakCnt;
_step2Bar.Message = string.Format(Strings.ShowMessageTemp, _procedCnt, _totalCnt, _replaceCnt, _breakCnt);
_step2Bar.Message = string.Format(Str.ShowMessageTemp, _procedCnt, _totalCnt, _replaceCnt, _breakCnt);
}
}
@ -103,25 +103,25 @@ public sealed class Main : Tool<Option>, IDisposable
}
[SuppressMessage("ReSharper", "PossibleMultipleEnumeration")]
public override void Run()
public override async Task Run()
{
if (!Directory.Exists(Opt.Path))
throw new ArgumentException(nameof(Opt.Path), string.Format(Strings.PathNotFound, Opt.Path));
throw new ArgumentException(nameof(Opt.Path), string.Format(Str.PathNotFound, Opt.Path));
using var step1Bar = new IndeterminateProgressBar(Strings.SearchingFile, DefaultProgressBarOptions);
using var step1Bar = new IndeterminateProgressBar(Str.SearchingFile, DefaultProgressBarOptions);
var fileList = EnumerateFiles(Opt.Path, Opt.Filter);
_totalCnt = fileList.Count();
step1Bar.Message = string.Format(Strings.SearchingFileOK, _totalCnt);
step1Bar.Message = string.Format(Str.SearchingFileOK, _totalCnt);
step1Bar.Finished();
if (_totalCnt == 0) return;
_step2Bar = step1Bar.Spawn(_totalCnt, string.Empty, DefaultProgressBarOptions);
Parallel.ForEach(fileList, FileHandle);
await Parallel.ForEachAsync(fileList, FileHandle);
}
}

View File

@ -1,4 +1,4 @@
namespace Dot.ToLf;
[Verb("tolf", HelpText = nameof(Strings.ConvertEndOfLineToLF), ResourceType = typeof(Strings))]
[Verb("tolf", HelpText = nameof(Str.ConvertEndOfLineToLF), ResourceType = typeof(Str))]
public class Option : DirOption { }

View File

@ -13,7 +13,7 @@ public abstract class Tool<TOption> : ITool
, ProgressCharacter = '\u2500'
};
protected TOption Opt { get; set; }
protected TOption Opt { get; }
protected Tool(TOption opt)
{
@ -61,9 +61,10 @@ public abstract class Tool<TOption> : ITool
}
}
catch (IOException) { }
return fsr;
}
public abstract void Run();
public abstract Task Run();
}

View File

@ -23,24 +23,24 @@
</PropertyGroup>
<ItemGroup>
<PackageReference Include="CommandLineParser" Version="2.9.1" />
<PackageReference Include="NSExt" Version="1.0.6" />
<PackageReference Include="ShellProgressBar" Version="5.2.0" />
<PackageReference Include="TextCopy" Version="6.2.0" />
<PackageReference Include="CommandLineParser" Version="2.9.1"/>
<PackageReference Include="NSExt" Version="1.0.6"/>
<PackageReference Include="ShellProgressBar" Version="5.2.0"/>
<PackageReference Include="TextCopy" Version="6.2.0"/>
</ItemGroup>
<ItemGroup>
<EmbeddedResource Update="Lang\Strings.resx">
<EmbeddedResource Update="Lang\Str.resx">
<Generator>PublicResXFileCodeGenerator</Generator>
<LastGenOutput>Strings.Designer.cs</LastGenOutput>
<LastGenOutput>Str.Designer.cs</LastGenOutput>
</EmbeddedResource>
</ItemGroup>
<ItemGroup>
<Compile Update="Lang\Strings.Designer.cs">
<Compile Update="Lang\Str.Designer.cs">
<DesignTime>True</DesignTime>
<AutoGen>True</AutoGen>
<DependentUpon>Strings.resx</DependentUpon>
<DependentUpon>Str.resx</DependentUpon>
</Compile>
</ItemGroup>