Compare commits

16 Commits

Author SHA1 Message Date
7e9c87de44 feat: convert png and jpeg to webp to reduce usage. 2025-03-24 22:39:10 +08:00
fda4c01c22 feat: compress command. 2025-03-24 22:38:18 +08:00
2b9c374e8c feat: beatify the blog content.
All checks were successful
Build blog docker image / Build-Blog-Image (push) Successful in 3m25s
Remove some unused font files and upgrade tailwindcss to v4.0.0.
2025-03-22 17:34:43 +08:00
4df3b98e6d blog: msbuild-generate-files
All checks were successful
Build blog docker image / Build-Blog-Image (push) Successful in 1m41s
blog: fix title in hpc-2025-cpu-architecture
2025-03-20 22:35:19 +08:00
c293d2f6d7 blog: update mlir-standalone
All checks were successful
Build blog docker image / Build-Blog-Image (push) Successful in 1m42s
2025-03-20 20:51:49 +08:00
132261831b blog: mlir-standalone
All checks were successful
Build blog docker image / Build-Blog-Image (push) Successful in 2m1s
2025-03-19 20:59:34 +08:00
043376c6d3 fix: build action.
All checks were successful
Build blog docker image / Build-Blog-Image (push) Successful in 1m43s
2025-03-14 00:41:50 +08:00
4682dacc79 fix: build actions.
Some checks failed
Build blog docker image / Build-Blog-Image (push) Failing after 1m33s
2025-03-14 00:29:33 +08:00
383dd41695 fix: build actions.
Some checks failed
Build blog docker image / Build-Blog-Image (push) Failing after 1m37s
2025-03-14 00:23:43 +08:00
7f3221fde9 chore: fix build action and migrate to slnx.
Some checks failed
Build blog docker image / Build-Blog-Image (push) Failing after 3m15s
2025-03-14 00:16:30 +08:00
05ea729950 blog: hpc cpu architecture
Some checks failed
Build blog docker image / Build-Blog-Image (push) Failing after 5m36s
2025-03-14 00:06:18 +08:00
dcad453eb1 blog: hpc-2025-intro
All checks were successful
Build blog docker image / Build-Blog-Image (push) Successful in 2m4s
2025-03-08 00:32:09 +08:00
dec2bc937a chore: remove unused steps in build actions.
All checks were successful
Build blog docker image / Build-Blog-Image (push) Successful in 1m48s
2025-03-07 20:21:17 +08:00
b9c44408ad fix: update build action to use podman
All checks were successful
Build blog docker image / Build-Blog-Image (push) Successful in 2m7s
2025-03-07 16:52:59 +08:00
e1c5362cf5 blog: adjust tags for some essays
All checks were successful
Build blog docker image / Build-Blog-Image (push) Successful in 1m12s
2025-01-25 14:36:12 +08:00
bdcfed5506 fix: failed to handle images in draft. 2025-01-25 14:27:20 +08:00
290 changed files with 1440 additions and 1614 deletions

View File

@@ -7,7 +7,7 @@ jobs:
Build-Blog-Image: Build-Blog-Image:
runs-on: archlinux runs-on: archlinux
steps: steps:
- uses: https://git.rrricardo.top/actions/checkout@v4 - uses: https://mirrors.rrricardo.top/actions/checkout.git@v4
name: Check out code name: Check out code
with: with:
lfs: true lfs: true
@@ -18,12 +18,16 @@ jobs:
- name: Build docker image - name: Build docker image
run: | run: |
cd YaeBlog cd YaeBlog
docker build . -t registry.cn-beijing.aliyuncs.com/jackfiled/blog:latest podman build . -t registry.cn-beijing.aliyuncs.com/jackfiled/blog:latest
- name: Workaround to make sure podman login succeed
run: |
mkdir /root/.docker
- name: Login aliyun docker registry - name: Login aliyun docker registry
uses: https://git.rrricardo.top/actions/login-action@v3 uses: https://mirrors.rrricardo.top/actions/podman-login.git@v1
with: with:
registry: registry.cn-beijing.aliyuncs.com registry: registry.cn-beijing.aliyuncs.com
username: 初冬的朝阳 username: 初冬的朝阳
password: ${{ secrets.ALIYUN_PASSWORD }} password: ${{ secrets.ALIYUN_PASSWORD }}
auth_file_path: /etc/containers/auth.json
- name: Push docker image - name: Push docker image
run: docker push registry.cn-beijing.aliyuncs.com/jackfiled/blog:latest run: podman push registry.cn-beijing.aliyuncs.com/jackfiled/blog:latest

2
.gitignore vendored
View File

@@ -484,4 +484,4 @@ $RECYCLE.BIN/
*.swp *.swp
# Tailwind auto-generated stylesheet # Tailwind auto-generated stylesheet
output.css *.g.css

View File

@@ -1,41 +0,0 @@

Microsoft Visual Studio Solution File, Format Version 12.00
# Visual Studio Version 17
VisualStudioVersion = 17.0.31903.59
MinimumVisualStudioVersion = 10.0.40219.1
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "YaeBlog", "YaeBlog\YaeBlog.csproj", "{20438EFD-8DDE-43AF-92E2-76495C29233C}"
EndProject
Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = ".gitea", ".gitea", "{9B5AAA29-37D8-454A-8D8F-3E6B6BCF38E6}"
EndProject
Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "workflows", "workflows", "{ADBC3DA8-F65C-4B5D-A97A-DC351F8E6592}"
ProjectSection(SolutionItems) = preProject
.gitea\workflows\build.yaml = .gitea\workflows\build.yaml
EndProjectSection
EndProject
Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "Solution Items", "Solution Items", "{753B998C-1B9E-498F-B949-845CE86C4075}"
ProjectSection(SolutionItems) = preProject
.editorconfig = .editorconfig
.gitattributes = .gitattributes
.gitignore = .gitignore
README.md = README.md
LICENSE = LICENSE
EndProjectSection
EndProject
Global
GlobalSection(SolutionConfigurationPlatforms) = preSolution
Debug|Any CPU = Debug|Any CPU
Release|Any CPU = Release|Any CPU
EndGlobalSection
GlobalSection(ProjectConfigurationPlatforms) = postSolution
{20438EFD-8DDE-43AF-92E2-76495C29233C}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
{20438EFD-8DDE-43AF-92E2-76495C29233C}.Debug|Any CPU.Build.0 = Debug|Any CPU
{20438EFD-8DDE-43AF-92E2-76495C29233C}.Release|Any CPU.ActiveCfg = Release|Any CPU
{20438EFD-8DDE-43AF-92E2-76495C29233C}.Release|Any CPU.Build.0 = Release|Any CPU
EndGlobalSection
GlobalSection(SolutionProperties) = preSolution
HideSolutionNode = FALSE
EndGlobalSection
GlobalSection(NestedProjects) = preSolution
{ADBC3DA8-F65C-4B5D-A97A-DC351F8E6592} = {9B5AAA29-37D8-454A-8D8F-3E6B6BCF38E6}
EndGlobalSection
EndGlobal

14
YaeBlog.slnx Normal file
View File

@@ -0,0 +1,14 @@
<Solution>
<Folder Name="/.gitea/" />
<Folder Name="/.gitea/workflows/">
<File Path=".gitea/workflows/build.yaml" />
</Folder>
<Folder Name="/Solution Items/">
<File Path=".editorconfig" />
<File Path=".gitattributes" />
<File Path=".gitignore" />
<File Path="LICENSE" />
<File Path="README.md" />
</Folder>
<Project Path="YaeBlog/YaeBlog.csproj" />
</Solution>

View File

@@ -7,6 +7,4 @@ public interface IEssayScanService
public Task<BlogContents> ScanContents(); public Task<BlogContents> ScanContents();
public Task SaveBlogContent(BlogContent content, bool isDraft = true); public Task SaveBlogContent(BlogContent content, bool isDraft = true);
public Task<ImageScanResult> ScanImages();
} }

View File

@@ -0,0 +1,21 @@
using System.CommandLine.Binding;
using YaeBlog.Abstraction;
using YaeBlog.Services;
namespace YaeBlog.Commands.Binders;
public sealed class ImageCompressServiceBinder : BinderBase<ImageCompressService>
{
protected override ImageCompressService GetBoundValue(BindingContext bindingContext)
{
bindingContext.AddService(provider =>
{
IEssayScanService essayScanService = provider.GetRequiredService<IEssayScanService>();
ILogger<ImageCompressService> logger = provider.GetRequiredService<ILogger<ImageCompressService>>();
return new ImageCompressService(essayScanService, logger);
});
return bindingContext.GetRequiredService<ImageCompressService>();
}
}

View File

@@ -1,4 +1,6 @@
using System.CommandLine; using System.CommandLine;
using Microsoft.Extensions.Options;
using YaeBlog.Abstraction;
using YaeBlog.Commands.Binders; using YaeBlog.Commands.Binders;
using YaeBlog.Components; using YaeBlog.Components;
using YaeBlog.Extensions; using YaeBlog.Extensions;
@@ -19,6 +21,7 @@ public sealed class YaeBlogCommand
AddNewCommand(_rootCommand); AddNewCommand(_rootCommand);
AddPublishCommand(_rootCommand); AddPublishCommand(_rootCommand);
AddScanCommand(_rootCommand); AddScanCommand(_rootCommand);
AddCompressCommand(_rootCommand);
} }
public Task<int> RunAsync(string[] args) public Task<int> RunAsync(string[] args)
@@ -94,22 +97,20 @@ public sealed class YaeBlogCommand
Argument<string> filenameArgument = new(name: "blog name", description: "The created blog filename."); Argument<string> filenameArgument = new(name: "blog name", description: "The created blog filename.");
newCommand.AddArgument(filenameArgument); newCommand.AddArgument(filenameArgument);
newCommand.SetHandler(async (file, _, _, essayScanService) => newCommand.SetHandler(async (file, blogOption, _, essayScanService) =>
{ {
BlogContents contents = await essayScanService.ScanContents(); BlogContents contents = await essayScanService.ScanContents();
if (contents.Posts.Any(content => content.FileName == file)) if (contents.Posts.Any(content => content.BlogName == file))
{ {
Console.WriteLine("There exists the same title blog in posts."); Console.WriteLine("There exists the same title blog in posts.");
return; return;
} }
await essayScanService.SaveBlogContent(new BlogContent await essayScanService.SaveBlogContent(new BlogContent(
{ new FileInfo(Path.Combine(blogOption.Value.Root, "drafts", file + ".md")),
FileName = file, new MarkdownMetadata { Title = file, Date = DateTime.Now },
FileContent = string.Empty, string.Empty, true, [], []));
Metadata = new MarkdownMetadata { Title = file, Date = DateTime.Now }
});
Console.WriteLine($"Created new blog '{file}."); Console.WriteLine($"Created new blog '{file}.");
}, filenameArgument, new BlogOptionsBinder(), new LoggerBinder<EssayScanService>(), }, filenameArgument, new BlogOptionsBinder(), new LoggerBinder<EssayScanService>(),
@@ -126,15 +127,15 @@ public sealed class YaeBlogCommand
BlogContents contents = await essyScanService.ScanContents(); BlogContents contents = await essyScanService.ScanContents();
Console.WriteLine($"All {contents.Posts.Count} Posts:"); Console.WriteLine($"All {contents.Posts.Count} Posts:");
foreach (BlogContent content in contents.Posts.OrderBy(x => x.FileName)) foreach (BlogContent content in contents.Posts.OrderBy(x => x.BlogName))
{ {
Console.WriteLine($" - {content.FileName}"); Console.WriteLine($" - {content.BlogName}");
} }
Console.WriteLine($"All {contents.Drafts.Count} Drafts:"); Console.WriteLine($"All {contents.Drafts.Count} Drafts:");
foreach (BlogContent content in contents.Drafts.OrderBy(x => x.FileName)) foreach (BlogContent content in contents.Drafts.OrderBy(x => x.BlogName))
{ {
Console.WriteLine($" - {content.FileName}"); Console.WriteLine($" - {content.BlogName}");
} }
}, new BlogOptionsBinder(), new LoggerBinder<EssayScanService>(), new EssayScanServiceBinder()); }, new BlogOptionsBinder(), new LoggerBinder<EssayScanService>(), new EssayScanServiceBinder());
} }
@@ -150,32 +151,39 @@ public sealed class YaeBlogCommand
command.SetHandler(async (_, _, essayScanService, removeOptionValue) => command.SetHandler(async (_, _, essayScanService, removeOptionValue) =>
{ {
ImageScanResult result = await essayScanService.ScanImages(); BlogContents contents = await essayScanService.ScanContents();
List<BlogImageInfo> unusedImages = (from content in contents
from image in content.Images
where image is { IsUsed: false }
select image).ToList();
if (result.UnusedImages.Count != 0) if (unusedImages.Count != 0)
{ {
Console.WriteLine("Found unused images:"); Console.WriteLine("Found unused images:");
Console.WriteLine("HINT: use '--rm' to remove unused images."); Console.WriteLine("HINT: use '--rm' to remove unused images.");
} }
foreach (FileInfo image in result.UnusedImages) foreach (BlogImageInfo image in unusedImages)
{ {
Console.WriteLine($" - {image.FullName}"); Console.WriteLine($" - {image.File.FullName}");
} }
if (removeOptionValue) if (removeOptionValue)
{ {
foreach (FileInfo image in result.UnusedImages) foreach (BlogImageInfo image in unusedImages)
{ {
image.Delete(); image.File.Delete();
} }
} }
Console.WriteLine("Used not existed images:"); Console.WriteLine("Used not existed images:");
foreach (FileInfo image in result.NotFoundImages) foreach (BlogContent content in contents)
{ {
Console.WriteLine($" - {image.FullName}"); foreach (FileInfo file in content.NotfoundImages)
{
Console.WriteLine($"- {file.Name} in {content.BlogName}");
}
} }
}, new BlogOptionsBinder(), new LoggerBinder<EssayScanService>(), new EssayScanServiceBinder(), removeOption); }, new BlogOptionsBinder(), new LoggerBinder<EssayScanService>(), new EssayScanServiceBinder(), removeOption);
} }
@@ -193,7 +201,7 @@ public sealed class YaeBlogCommand
BlogContents contents = await essayScanService.ScanContents(); BlogContents contents = await essayScanService.ScanContents();
BlogContent? content = (from blog in contents.Drafts BlogContent? content = (from blog in contents.Drafts
where blog.FileName == filename where blog.BlogName == filename
select blog).FirstOrDefault(); select blog).FirstOrDefault();
if (content is null) if (content is null)
@@ -202,14 +210,17 @@ public sealed class YaeBlogCommand
return; return;
} }
// 设置发布的时间
content.Metadata.Date = DateTime.Now;
// 将选中的博客文件复制到posts // 将选中的博客文件复制到posts
await essayScanService.SaveBlogContent(content, isDraft: false); await essayScanService.SaveBlogContent(content, isDraft: false);
// 复制图片文件夹 // 复制图片文件夹
DirectoryInfo sourceImageDirectory = DirectoryInfo sourceImageDirectory =
new(Path.Combine(blogOptions.Value.Root, "drafts", content.FileName)); new(Path.Combine(blogOptions.Value.Root, "drafts", content.BlogName));
DirectoryInfo targetImageDirectory = DirectoryInfo targetImageDirectory =
new(Path.Combine(blogOptions.Value.Root, "posts", content.FileName)); new(Path.Combine(blogOptions.Value.Root, "posts", content.BlogName));
if (sourceImageDirectory.Exists) if (sourceImageDirectory.Exists)
{ {
@@ -223,9 +234,30 @@ public sealed class YaeBlogCommand
} }
// 删除原始的文件 // 删除原始的文件
FileInfo sourceBlogFile = new(Path.Combine(blogOptions.Value.Root, "drafts", content.FileName + ".md")); FileInfo sourceBlogFile = new(Path.Combine(blogOptions.Value.Root, "drafts", content.BlogName + ".md"));
sourceBlogFile.Delete(); sourceBlogFile.Delete();
}, new BlogOptionsBinder(), }, new BlogOptionsBinder(),
new LoggerBinder<EssayScanService>(), new EssayScanServiceBinder(), filenameArgument); new LoggerBinder<EssayScanService>(), new EssayScanServiceBinder(), filenameArgument);
} }
private static void AddCompressCommand(RootCommand rootCommand)
{
Command command = new("compress", "Compress png/jpeg image to webp image to reduce size.");
rootCommand.Add(command);
Option<bool> dryRunOption = new("--dry-run", description: "Dry run the compression task but not write.",
getDefaultValue: () => false);
command.AddOption(dryRunOption);
command.SetHandler(ImageCommandHandler,
new BlogOptionsBinder(), new LoggerBinder<EssayScanService>(), new LoggerBinder<ImageCompressService>(),
new EssayScanServiceBinder(), new ImageCompressServiceBinder(), dryRunOption);
}
private static async Task ImageCommandHandler(IOptions<BlogOptions> _, ILogger<EssayScanService> _1,
ILogger<ImageCompressService> _2,
IEssayScanService _3, ImageCompressService imageCompressService, bool dryRun)
{
await imageCompressService.Compress(dryRun);
}
} }

View File

@@ -8,7 +8,7 @@
<link rel="stylesheet" href="YaeBlog.styles.css"/> <link rel="stylesheet" href="YaeBlog.styles.css"/>
<link rel="icon" href="images/favicon.ico"/> <link rel="icon" href="images/favicon.ico"/>
<link rel="stylesheet" href="globals.css"/> <link rel="stylesheet" href="globals.css"/>
<link rel="stylesheet" href="output.css"/> <link rel="stylesheet" href="tailwind.g.css"/>
<HeadOutlet/> <HeadOutlet/>
</head> </head>

View File

@@ -0,0 +1,18 @@
using AngleSharp.Dom;
namespace YaeBlog.Extensions;
public static class AngleSharpExtensions
{
public static IEnumerable<IElement> EnumerateParentElements(this IElement element)
{
IElement? e = element.ParentElement;
while (e is not null)
{
IElement c = e;
e = e.ParentElement;
yield return c;
}
}
}

View File

@@ -1,12 +1,20 @@
namespace YaeBlog.Models; namespace YaeBlog.Models;
public class BlogContent /// <summary>
/// 单个博客文件的所有数据和元数据
/// </summary>
/// <param name="BlogFile">博客文件</param>
/// <param name="Metadata">文件中的MD元数据</param>
/// <param name="Content">文件内容</param>
/// <param name="IsDraft">是否为草稿</param>
/// <param name="Images">博客中使用的文件</param>
public record BlogContent(
FileInfo BlogFile,
MarkdownMetadata Metadata,
string Content,
bool IsDraft,
List<BlogImageInfo> Images,
List<FileInfo> NotfoundImages)
{ {
public required string FileName { get; init; } public string BlogName => BlogFile.Name.Split('.')[0];
public required MarkdownMetadata Metadata { get; init; }
public required string FileContent { get; set; }
public bool IsDraft { get; set; } = false;
} }

View File

@@ -1,10 +1,15 @@
using System.Collections.Concurrent; using System.Collections;
using System.Collections.Concurrent;
namespace YaeBlog.Models; namespace YaeBlog.Models;
public sealed class BlogContents(ConcurrentBag<BlogContent> drafts, ConcurrentBag<BlogContent> posts) public record BlogContents(ConcurrentBag<BlogContent> Drafts, ConcurrentBag<BlogContent> Posts)
: IEnumerable<BlogContent>
{ {
public ConcurrentBag<BlogContent> Drafts { get; } = drafts; IEnumerator<BlogContent> IEnumerable<BlogContent>.GetEnumerator()
{
public ConcurrentBag<BlogContent> Posts { get; } = posts; return Posts.Concat(Drafts).GetEnumerator();
}
public IEnumerator GetEnumerator() => ((IEnumerable<BlogContent>)this).GetEnumerator();
} }

View File

@@ -0,0 +1,44 @@
using System.Text;
namespace YaeBlog.Models;
public record BlogImageInfo(FileInfo File, long Width, long Height, string MineType, byte[] Content, bool IsUsed)
: IComparable<BlogImageInfo>
{
public int Size => Content.Length;
public override string ToString()
{
StringBuilder builder = new();
builder.AppendLine($"Blog image {File.Name}:");
builder.AppendLine($"\tWidth: {Width}; Height: {Height}");
builder.AppendLine($"\tSize: {FormatSize()}");
builder.AppendLine($"\tImage Format: {MineType}");
return builder.ToString();
}
public int CompareTo(BlogImageInfo? other)
{
if (other is null)
{
return -1;
}
return other.Size.CompareTo(Size);
}
private string FormatSize()
{
double size = Size;
if (size / 1024 > 3)
{
size /= 1024;
return size / 1024 > 3 ? $"{size / 1024}MB" : $"{size}KB";
}
return $"{size}B";
}
}

View File

@@ -1,3 +0,0 @@
namespace YaeBlog.Models;
public record struct ImageScanResult(List<FileInfo> UnusedImages, List<FileInfo> NotFoundImages);

View File

@@ -1,6 +1,7 @@
using AngleSharp; using AngleSharp;
using AngleSharp.Dom; using AngleSharp.Dom;
using YaeBlog.Abstraction; using YaeBlog.Abstraction;
using YaeBlog.Extensions;
using YaeBlog.Models; using YaeBlog.Models;
namespace YaeBlog.Processors; namespace YaeBlog.Processors;
@@ -20,20 +21,21 @@ public sealed class EssayStylesPostRenderProcessor : IPostRenderProcessor
ApplyGlobalCssStyles(document); ApplyGlobalCssStyles(document);
BeatifyTable(document); BeatifyTable(document);
BeatifyList(document);
BeatifyInlineCode(document);
return essay.WithNewHtmlContent(document.DocumentElement.OuterHtml); return essay.WithNewHtmlContent(document.DocumentElement.OuterHtml);
} }
private readonly Dictionary<string, string> _globalCssStyles = new() private readonly Dictionary<string, string> _globalCssStyles = new()
{ {
{ "pre", "p-4 bg-slate-300 rounded-sm overflow-x-auto" }, { "pre", "p-4 bg-gray-100 rounded-sm overflow-x-auto" },
{ "h2", "text-3xl font-bold py-4" }, { "h2", "text-3xl font-bold py-4" },
{ "h3", "text-2xl font-bold py-3" }, { "h3", "text-2xl font-bold py-3" },
{ "h4", "text-xl font-bold py-2" }, { "h4", "text-xl font-bold py-2" },
{ "h5", "text-lg font-bold py-1" }, { "h5", "text-lg font-bold py-1" },
{ "p", "p-2" }, { "p", "p-2" },
{ "img", "w-11/12 block mx-auto my-2 rounded-md shadow-md" }, { "img", "w-11/12 block mx-auto my-2 rounded-md shadow-md" },
{ "ul", "list-disc pl-2" }
}; };
private void ApplyGlobalCssStyles(IDocument document) private void ApplyGlobalCssStyles(IDocument document)
@@ -99,4 +101,45 @@ public sealed class EssayStylesPostRenderProcessor : IPostRenderProcessor
} }
} }
} }
private static void BeatifyList(IDocument document)
{
foreach (IElement ulElement in from e in document.All
where e.LocalName == "ul"
select e)
{
// 首先给<ul>元素添加样式
ulElement.ClassList.Add("list-disc ml-10");
foreach (IElement liElement in from e in ulElement.Children
where e.LocalName == "li"
select e)
{
// 修改<li>元素中的<p>元素样式
// 默认的p-2间距有点太宽了
foreach (IElement pElement in from e in liElement.Children
where e.LocalName == "p"
select e)
{
pElement.ClassList.Remove("p-2");
pElement.ClassList.Add("p-1");
}
}
}
}
private static void BeatifyInlineCode(IDocument document)
{
// 选择不在<pre>元素内的<code>元素
// 即行内代码
IEnumerable<IElement> inlineCodes = from e in document.All
where e.LocalName == "code" && e.EnumerateParentElements().All(p => p.LocalName != "pre")
select e;
foreach (IElement e in inlineCodes)
{
e.ClassList.Add("bg-gray-100 inline p-1 rounded-xs");
}
}
} }

View File

@@ -7,7 +7,8 @@ using YaeBlog.Models;
namespace YaeBlog.Processors; namespace YaeBlog.Processors;
public class ImagePostRenderProcessor(ILogger<ImagePostRenderProcessor> logger, public class ImagePostRenderProcessor(
ILogger<ImagePostRenderProcessor> logger,
IOptions<BlogOptions> options) IOptions<BlogOptions> options)
: IPostRenderProcessor : IPostRenderProcessor
{ {
@@ -29,22 +30,27 @@ public class ImagePostRenderProcessor(ILogger<ImagePostRenderProcessor> logger,
if (attr is not null) if (attr is not null)
{ {
logger.LogDebug("Found image link: '{}'", attr.Value); logger.LogDebug("Found image link: '{}'", attr.Value);
attr.Value = GenerateImageLink(attr.Value, essay.FileName); attr.Value = GenerateImageLink(attr.Value, essay.FileName, essay.IsDraft);
} }
} }
return essay.WithNewHtmlContent(html.DocumentElement.OuterHtml); return essay.WithNewHtmlContent(html.DocumentElement.OuterHtml);
} }
public string Name => nameof(ImagePostRenderProcessor); public string Name => nameof(ImagePostRenderProcessor);
private string GenerateImageLink(string filename, string essayFilename) private string GenerateImageLink(string filename, string essayFilename, bool isDraft)
{ {
// 如果图片路径中没有包含文件名
// 则添加文件名
if (!filename.Contains(essayFilename)) if (!filename.Contains(essayFilename))
{ {
filename = Path.Combine(essayFilename, filename); filename = Path.Combine(essayFilename, filename);
} }
filename = Path.Combine(_options.Root, "posts", filename); filename = isDraft
? Path.Combine(_options.Root, "drafts", filename)
: Path.Combine(_options.Root, "posts", filename);
if (!Path.Exists(filename)) if (!Path.Exists(filename))
{ {

View File

@@ -1,5 +1,7 @@
using System.Collections.Concurrent; using System.Collections.Concurrent;
using System.Text.RegularExpressions; using System.Text.RegularExpressions;
using Imageflow.Bindings;
using Imageflow.Fluent;
using Microsoft.Extensions.Options; using Microsoft.Extensions.Options;
using YaeBlog.Abstraction; using YaeBlog.Abstraction;
using YaeBlog.Core.Exceptions; using YaeBlog.Core.Exceptions;
@@ -9,17 +11,30 @@ using YamlDotNet.Serialization;
namespace YaeBlog.Services; namespace YaeBlog.Services;
public partial class EssayScanService( public partial class EssayScanService : IEssayScanService
ISerializer yamlSerializer, {
private readonly BlogOptions _blogOptions;
private readonly ISerializer _yamlSerializer;
private readonly IDeserializer _yamlDeserializer;
private readonly ILogger<EssayScanService> _logger;
public EssayScanService(ISerializer yamlSerializer,
IDeserializer yamlDeserializer, IDeserializer yamlDeserializer,
IOptions<BlogOptions> blogOptions, IOptions<BlogOptions> blogOptions,
ILogger<EssayScanService> logger) : IEssayScanService ILogger<EssayScanService> logger)
{ {
private readonly BlogOptions _blogOptions = blogOptions.Value; _yamlSerializer = yamlSerializer;
_yamlDeserializer = yamlDeserializer;
_logger = logger;
_blogOptions = blogOptions.Value;
RootDirectory = ValidateRootDirectory();
}
private DirectoryInfo RootDirectory { get; }
public async Task<BlogContents> ScanContents() public async Task<BlogContents> ScanContents()
{ {
ValidateDirectory(_blogOptions.Root, out DirectoryInfo drafts, out DirectoryInfo posts); ValidateDirectory(out DirectoryInfo drafts, out DirectoryInfo posts);
return new BlogContents( return new BlogContents(
await ScanContentsInternal(drafts, true), await ScanContentsInternal(drafts, true),
@@ -28,82 +43,92 @@ public partial class EssayScanService(
public async Task SaveBlogContent(BlogContent content, bool isDraft = true) public async Task SaveBlogContent(BlogContent content, bool isDraft = true)
{ {
ValidateDirectory(_blogOptions.Root, out DirectoryInfo drafts, out DirectoryInfo posts); ValidateDirectory(out DirectoryInfo drafts, out DirectoryInfo posts);
FileInfo targetFile = isDraft FileInfo targetFile = isDraft
? new FileInfo(Path.Combine(drafts.FullName, content.FileName + ".md")) ? new FileInfo(Path.Combine(drafts.FullName, content.BlogName + ".md"))
: new FileInfo(Path.Combine(posts.FullName, content.FileName + ".md")); : new FileInfo(Path.Combine(posts.FullName, content.BlogName + ".md"));
if (!isDraft)
{
content.Metadata.Date = DateTime.Now;
}
if (targetFile.Exists) if (targetFile.Exists)
{ {
logger.LogWarning("Blog {} exists, overriding.", targetFile.Name); _logger.LogWarning("Blog {} exists, overriding.", targetFile.Name);
} }
await using StreamWriter writer = targetFile.CreateText(); await using StreamWriter writer = targetFile.CreateText();
await writer.WriteAsync("---\n"); await writer.WriteAsync("---\n");
await writer.WriteAsync(yamlSerializer.Serialize(content.Metadata)); await writer.WriteAsync(_yamlSerializer.Serialize(content.Metadata));
await writer.WriteAsync("---\n"); await writer.WriteAsync("---\n");
if (isDraft) if (string.IsNullOrEmpty(content.Content) && isDraft)
{ {
// 如果博客为操作且内容为空
// 创建简介隔断符号
await writer.WriteLineAsync("<!--more-->"); await writer.WriteLineAsync("<!--more-->");
} }
else else
{ {
await writer.WriteAsync(content.FileContent); await writer.WriteAsync(content.Content);
} }
// 保存图片文件
await Task.WhenAll(from image in content.Images
select File.WriteAllBytesAsync(image.File.FullName, image.Content));
} }
private record struct BlogResult(
FileInfo BlogFile,
string BlogContent,
List<BlogImageInfo> Images,
List<FileInfo> NotFoundImages);
private async Task<ConcurrentBag<BlogContent>> ScanContentsInternal(DirectoryInfo directory, bool isDraft) private async Task<ConcurrentBag<BlogContent>> ScanContentsInternal(DirectoryInfo directory, bool isDraft)
{ {
// 扫描以md结果的但是不是隐藏文件的文件 // 扫描以md结尾且不是隐藏文件的文件
IEnumerable<FileInfo> markdownFiles = from file in directory.EnumerateFiles() IEnumerable<FileInfo> markdownFiles = from file in directory.EnumerateFiles()
where file.Extension == ".md" && !file.Name.StartsWith('.') where file.Extension == ".md" && !file.Name.StartsWith('.')
select file; select file;
ConcurrentBag<(string, string)> fileContents = []; ConcurrentBag<BlogResult> fileContents = [];
await Parallel.ForEachAsync(markdownFiles, async (file, token) => await Parallel.ForEachAsync(markdownFiles, async (file, token) =>
{ {
using StreamReader reader = file.OpenText(); using StreamReader reader = file.OpenText();
fileContents.Add((file.Name, await reader.ReadToEndAsync(token))); string blogName = file.Name.Split('.')[0];
string blogContent = await reader.ReadToEndAsync(token);
ImageResult imageResult =
await ScanImagePreBlog(directory, blogName,
blogContent);
fileContents.Add(new BlogResult(file, blogContent, imageResult.Images, imageResult.NotfoundImages));
}); });
ConcurrentBag<BlogContent> contents = []; ConcurrentBag<BlogContent> contents = [];
await Task.Run(() => await Task.Run(() =>
{ {
foreach ((string filename, string content) in fileContents) foreach (BlogResult blog in fileContents)
{ {
int endPos = content.IndexOf("---", 4, StringComparison.Ordinal); int endPos = blog.BlogContent.IndexOf("---", 4, StringComparison.Ordinal);
if (!content.StartsWith("---") || endPos is -1 or 0) if (!blog.BlogContent.StartsWith("---") || endPos is -1 or 0)
{ {
logger.LogWarning("Failed to parse metadata from {}, skipped.", filename); _logger.LogWarning("Failed to parse metadata from {}, skipped.", blog.BlogFile.Name);
return; return;
} }
string metadataString = content[4..endPos]; string metadataString = blog.BlogContent[4..endPos];
try try
{ {
MarkdownMetadata metadata = yamlDeserializer.Deserialize<MarkdownMetadata>(metadataString); MarkdownMetadata metadata = _yamlDeserializer.Deserialize<MarkdownMetadata>(metadataString);
logger.LogDebug("Scan metadata title: '{}' for {}.", metadata.Title, filename); _logger.LogDebug("Scan metadata title: '{}' for {}.", metadata.Title, blog.BlogFile.Name);
contents.Add(new BlogContent contents.Add(new BlogContent(blog.BlogFile, metadata, blog.BlogContent[(endPos + 3)..], isDraft,
{ blog.Images, blog.NotFoundImages));
FileName = filename[..^3], Metadata = metadata, FileContent = content[(endPos + 3)..],
IsDraft = isDraft
});
} }
catch (YamlException e) catch (YamlException e)
{ {
logger.LogWarning("Failed to parser metadata from {} due to {}, skipping", filename, e); _logger.LogWarning("Failed to parser metadata from {} due to {}, skipping", blog.BlogFile.Name, e);
} }
} }
}); });
@@ -111,99 +136,96 @@ public partial class EssayScanService(
return contents; return contents;
} }
public async Task<ImageScanResult> ScanImages() private record struct ImageResult(List<BlogImageInfo> Images, List<FileInfo> NotfoundImages);
private async Task<ImageResult> ScanImagePreBlog(DirectoryInfo directory, string blogName, string content)
{ {
BlogContents contents = await ScanContents(); MatchCollection matchResult = ImagePattern.Matches(content);
ValidateDirectory(_blogOptions.Root, out DirectoryInfo drafts, out DirectoryInfo posts); DirectoryInfo imageDirectory = new(Path.Combine(directory.FullName, blogName));
List<FileInfo> unusedFiles = []; Dictionary<string, bool> usedImages = imageDirectory.Exists
List<FileInfo> notFoundFiles = []; ? imageDirectory.EnumerateFiles().ToDictionary(file => file.FullName, _ => false)
: [];
List<FileInfo> notFoundImages = [];
ImageScanResult draftResult = await ScanUnusedImagesInternal(contents.Drafts, drafts); foreach (Match match in matchResult)
ImageScanResult postResult = await ScanUnusedImagesInternal(contents.Posts, posts);
unusedFiles.AddRange(draftResult.UnusedImages);
notFoundFiles.AddRange(draftResult.NotFoundImages);
unusedFiles.AddRange(postResult.UnusedImages);
notFoundFiles.AddRange(postResult.NotFoundImages);
return new ImageScanResult(unusedFiles, notFoundFiles);
}
private static Task<ImageScanResult> ScanUnusedImagesInternal(IEnumerable<BlogContent> contents,
DirectoryInfo root)
{
ConcurrentBag<FileInfo> unusedImage = [];
ConcurrentBag<FileInfo> notFoundImage = [];
Parallel.ForEach(contents, content =>
{
MatchCollection result = ImagePattern.Matches(content.FileContent);
DirectoryInfo imageDirectory = new(Path.Combine(root.FullName, content.FileName));
Dictionary<string, bool> usedDictionary;
if (imageDirectory.Exists)
{
usedDictionary = (from file in imageDirectory.EnumerateFiles()
select new KeyValuePair<string, bool>(file.FullName, false)).ToDictionary();
}
else
{
usedDictionary = [];
}
foreach (Match match in result)
{ {
string imageName = match.Groups[1].Value; string imageName = match.Groups[1].Value;
FileInfo usedFile = imageName.Contains(content.FileName) // 判断md文件中的图片名称中是否包含文件夹名称
? new FileInfo(Path.Combine(root.FullName, imageName)) // 例如 blog-1/image.png 或者 image.png
: new FileInfo(Path.Combine(root.FullName, content.FileName, imageName)); // 如果不带文件夹名称
// 默认添加同博客名文件夹
FileInfo usedFile = imageName.Contains(blogName)
? new FileInfo(Path.Combine(directory.FullName, imageName))
: new FileInfo(Path.Combine(directory.FullName, blogName, imageName));
if (usedDictionary.TryGetValue(usedFile.FullName, out _)) if (usedImages.TryGetValue(usedFile.FullName, out _))
{ {
usedDictionary[usedFile.FullName] = true; usedImages[usedFile.FullName] = true;
} }
else else
{ {
notFoundImage.Add(usedFile); notFoundImages.Add(usedFile);
} }
} }
foreach (KeyValuePair<string, bool> pair in usedDictionary.Where(p => !p.Value)) List<BlogImageInfo> images = (await Task.WhenAll((from pair in usedImages
select GetImageInfo(new FileInfo(pair.Key), pair.Value)).ToArray())).ToList();
return new ImageResult(images, notFoundImages);
}
private static async Task<BlogImageInfo> GetImageInfo(FileInfo file, bool isUsed)
{ {
unusedImage.Add(new FileInfo(pair.Key)); byte[] image = await File.ReadAllBytesAsync(file.FullName);
}
});
return Task.FromResult(new ImageScanResult(unusedImage.ToList(), notFoundImage.ToList())); if (file.Extension is ".jpg" or ".jpeg" or ".png")
{
ImageInfo imageInfo =
await ImageJob.GetImageInfoAsync(MemorySource.Borrow(image), SourceLifetime.NowOwnedAndDisposedByTask);
return new BlogImageInfo(file, imageInfo.ImageWidth, imageInfo.ImageWidth, imageInfo.PreferredMimeType,
image, isUsed);
}
return new BlogImageInfo(file, 0, 0, file.Extension switch
{
"svg" => "image/svg",
"avif" => "image/avif",
_ => string.Empty
}, image, isUsed);
} }
[GeneratedRegex(@"\!\[.*?\]\((.*?)\)")] [GeneratedRegex(@"\!\[.*?\]\((.*?)\)")]
private static partial Regex ImagePattern { get; } private static partial Regex ImagePattern { get; }
private void ValidateDirectory(string root, out DirectoryInfo drafts, out DirectoryInfo posts)
private DirectoryInfo ValidateRootDirectory()
{ {
root = Path.Combine(Environment.CurrentDirectory, root); DirectoryInfo rootDirectory = new(Path.Combine(Environment.CurrentDirectory, _blogOptions.Root));
DirectoryInfo rootDirectory = new(root);
if (!rootDirectory.Exists) if (!rootDirectory.Exists)
{ {
throw new BlogFileException($"'{root}' is not a directory."); throw new BlogFileException($"'{_blogOptions.Root}' is not a directory.");
} }
if (rootDirectory.EnumerateDirectories().All(dir => dir.Name != "drafts")) return rootDirectory;
}
private void ValidateDirectory(out DirectoryInfo drafts, out DirectoryInfo posts)
{ {
throw new BlogFileException($"'{root}/drafts' not exists."); if (RootDirectory.EnumerateDirectories().All(dir => dir.Name != "drafts"))
}
if (rootDirectory.EnumerateDirectories().All(dir => dir.Name != "posts"))
{ {
throw new BlogFileException($"'{root}/posts' not exists."); throw new BlogFileException($"'{_blogOptions.Root}/drafts' not exists.");
} }
drafts = new DirectoryInfo(Path.Combine(root, "drafts")); if (RootDirectory.EnumerateDirectories().All(dir => dir.Name != "posts"))
posts = new DirectoryInfo(Path.Combine(root, "posts")); {
throw new BlogFileException($"'{_blogOptions.Root}/posts' not exists.");
}
drafts = new DirectoryInfo(Path.Combine(_blogOptions.Root, "drafts"));
posts = new DirectoryInfo(Path.Combine(_blogOptions.Root, "posts"));
} }
} }

View File

@@ -0,0 +1,108 @@
using Imageflow.Fluent;
using YaeBlog.Abstraction;
using YaeBlog.Core.Exceptions;
using YaeBlog.Models;
namespace YaeBlog.Services;
public sealed class ImageCompressService(IEssayScanService essayScanService, ILogger<ImageCompressService> logger)
{
private record struct CompressResult(BlogImageInfo ImageInfo, byte[] CompressContent);
public async Task<List<BlogImageInfo>> ScanUsedImages()
{
BlogContents contents = await essayScanService.ScanContents();
List<BlogImageInfo> originalImages = (from content in contents.Posts.Concat(contents.Drafts)
from image in content.Images
where image.IsUsed
select image).ToList();
originalImages.Sort();
return originalImages;
}
public async Task Compress(bool dryRun)
{
BlogContents contents = await essayScanService.ScanContents();
// 筛选需要压缩的图片
// 即图片被博客使用且是jpeg/png格式
List<BlogContent> needCompressContents = (from content in contents
where content.Images.Any(i => i is { IsUsed: true } and { File.Extension: ".jpg" or ".jpeg" or ".png" })
select content).ToList();
if (needCompressContents.Count == 0)
{
return;
}
int uncompressedSize = 0;
int compressedSize = 0;
List<BlogContent> compressedContent = new(needCompressContents.Count);
foreach (BlogContent content in needCompressContents)
{
List<BlogImageInfo> uncompressedImages = (from image in content.Images
where image is { IsUsed: true } and { File.Extension: ".jpg" or ".jpeg" or ".png" }
select image).ToList();
uncompressedSize += uncompressedImages.Select(i => i.Size).Sum();
foreach (BlogImageInfo image in uncompressedImages)
{
logger.LogInformation("Uncompressed image: {} belonging to blog {}.", image.File.Name,
content.BlogName);
}
CompressResult[] compressedImages = (await Task.WhenAll(from image in uncompressedImages
select Task.Run(async () => new CompressResult(image, await ConvertToWebp(image.Content))))).ToArray();
compressedSize += compressedImages.Select(i => i.CompressContent.Length).Sum();
// 直接在原有的图片列表上添加图片
List<BlogImageInfo> images = content.Images.Concat(from r in compressedImages
select r.ImageInfo with
{
File = new FileInfo(r.ImageInfo.File.FullName.Split('.')[0] + ".webp"),
Content = r.CompressContent
}).ToList();
// 修改文本
string blogContent = compressedImages.Aggregate(content.Content, (c, r) =>
{
string originalName = r.ImageInfo.File.Name;
string outputName = originalName.Split('.')[0] + ".webp";
return c.Replace(originalName, outputName);
});
compressedContent.Add(content with { Images = images, Content = blogContent });
}
logger.LogInformation("Compression ratio: {}%.", (double)compressedSize / uncompressedSize * 100.0);
if (dryRun is false)
{
await Task.WhenAll(from content in compressedContent
select essayScanService.SaveBlogContent(content, content.IsDraft));
}
}
private static async Task<byte[]> ConvertToWebp(byte[] image)
{
using ImageJob job = new();
BuildJobResult result = await job.Decode(MemorySource.Borrow(image))
.EncodeToBytes(new WebPLossyEncoder(75))
.Finish()
.InProcessAsync();
ArraySegment<byte>? array = result.First?.TryGetBytes();
if (array.HasValue)
{
return array.Value.ToArray();
}
throw new BlogFileException();
}
}

View File

@@ -41,14 +41,14 @@ public partial class RendererService(
uint wordCount = GetWordCount(content); uint wordCount = GetWordCount(content);
BlogEssay essay = new() BlogEssay essay = new()
{ {
Title = content.Metadata.Title ?? content.FileName, Title = content.Metadata.Title ?? content.BlogName,
FileName = content.FileName, FileName = content.BlogName,
IsDraft = content.IsDraft, IsDraft = content.IsDraft,
Description = GetDescription(content), Description = GetDescription(content),
WordCount = wordCount, WordCount = wordCount,
ReadTime = CalculateReadTime(wordCount), ReadTime = CalculateReadTime(wordCount),
PublishTime = content.Metadata.Date ?? DateTime.Now, PublishTime = content.Metadata.Date ?? DateTime.Now,
HtmlContent = content.FileContent HtmlContent = content.Content
}; };
if (content.Metadata.Tags is not null) if (content.Metadata.Tags is not null)
@@ -156,17 +156,17 @@ public partial class RendererService(
private string GetDescription(BlogContent content) private string GetDescription(BlogContent content)
{ {
const string delimiter = "<!--more-->"; const string delimiter = "<!--more-->";
int pos = content.FileContent.IndexOf(delimiter, StringComparison.Ordinal); int pos = content.Content.IndexOf(delimiter, StringComparison.Ordinal);
bool breakSentence = false; bool breakSentence = false;
if (pos == -1) if (pos == -1)
{ {
// 自动截取前50个字符 // 自动截取前50个字符
pos = content.FileContent.Length < 50 ? content.FileContent.Length : 50; pos = content.Content.Length < 50 ? content.Content.Length : 50;
breakSentence = true; breakSentence = true;
} }
string rawContent = content.FileContent[..pos]; string rawContent = content.Content[..pos];
MatchCollection matches = DescriptionPattern.Matches(rawContent); MatchCollection matches = DescriptionPattern.Matches(rawContent);
StringBuilder builder = new(); StringBuilder builder = new();
@@ -182,18 +182,18 @@ public partial class RendererService(
string description = builder.ToString(); string description = builder.ToString();
logger.LogDebug("Description of {} is {}.", content.FileName, logger.LogDebug("Description of {} is {}.", content.BlogName,
description); description);
return description; return description;
} }
private uint GetWordCount(BlogContent content) private uint GetWordCount(BlogContent content)
{ {
int count = (from c in content.FileContent int count = (from c in content.Content
where char.IsLetterOrDigit(c) where char.IsLetterOrDigit(c)
select c).Count(); select c).Count();
logger.LogDebug("Word count of {} is {}", content.FileName, logger.LogDebug("Word count of {} is {}", content.BlogName,
count); count);
return (uint)count; return (uint)count;
} }

View File

@@ -1,6 +1,8 @@
<Project Sdk="Microsoft.NET.Sdk.Web"> <Project Sdk="Microsoft.NET.Sdk.Web">
<ItemGroup> <ItemGroup>
<PackageReference Include="ImageFlow.NativeRuntime.ubuntu-x86_64" Version="2.1.0-rc11"/>
<PackageReference Include="ImageFlow.Net" Version="0.13.2"/>
<PackageReference Include="System.CommandLine" Version="2.0.0-beta4.22272.1"/> <PackageReference Include="System.CommandLine" Version="2.0.0-beta4.22272.1"/>
<PackageReference Include="AngleSharp" Version="1.1.0"/> <PackageReference Include="AngleSharp" Version="1.1.0"/>
<PackageReference Include="Markdig" Version="0.38.0"/> <PackageReference Include="Markdig" Version="0.38.0"/>
@@ -13,7 +15,7 @@
<ImplicitUsings>enable</ImplicitUsings> <ImplicitUsings>enable</ImplicitUsings>
</PropertyGroup> </PropertyGroup>
<Target Name="EnsurePnpmInstalled" BeforeTargets="Build"> <Target Name="EnsurePnpmInstalled" BeforeTargets="BeforeBuild">
<Message Importance="low" Text="Ensure pnpm is installed..."/> <Message Importance="low" Text="Ensure pnpm is installed..."/>
<Exec Command="pnpm --version" ContinueOnError="true"> <Exec Command="pnpm --version" ContinueOnError="true">
<Output TaskParameter="ExitCode" PropertyName="ErrorCode"/> <Output TaskParameter="ExitCode" PropertyName="ErrorCode"/>
@@ -25,9 +27,13 @@
<Exec Command="pnpm install"/> <Exec Command="pnpm install"/>
</Target> </Target>
<Target Name="TailwindGenerate" AfterTargets="EnsurePnpmInstalled"> <Target Name="TailwindGenerate" AfterTargets="EnsurePnpmInstalled" BeforeTargets="BeforeBuild" Condition="'$(_IsPublishing)' == 'yes'">
<Message Importance="normal" Text="Generate css files using tailwind..."/> <Message Importance="normal" Text="Generate css files using tailwind..."/>
<Exec Command="pnpm tailwind -i wwwroot/input.css -o wwwroot/output.css"/> <Exec Command="pnpm tailwindcss -i wwwroot/tailwind.css -o $(IntermediateOutputPath)tailwind.g.css"/>
<ItemGroup>
<Content Include="$(IntermediateOutputPath)tailwind.g.css" Visible="false" TargetPath="wwwroot/tailwind.g.css"/>
</ItemGroup>
</Target> </Target>
</Project> </Project>

View File

@@ -1,12 +1,15 @@
{ {
"name": "YaeBlog", "name": "yae-blog",
"version": "1.0.0", "version": "1.0.0",
"description": "", "description": "",
"scripts": {}, "scripts": {
"dev": "tailwindcss -i wwwroot/tailwind.css -o wwwroot/tailwind.g.css -w"
},
"keywords": [], "keywords": [],
"author": "", "author": "",
"license": "ISC", "license": "ISC",
"devDependencies": { "devDependencies": {
"tailwindcss": "^3.4.16" "tailwindcss": "^4.0.0",
"@tailwindcss/cli": "^4.0.0"
} }
} }

1121
YaeBlog/pnpm-lock.yaml generated

File diff suppressed because it is too large Load Diff

View File

@@ -2,7 +2,8 @@
title: 2021年终总结 title: 2021年终总结
date: 2022-01-12 16:27:19 date: 2022-01-12 16:27:19
tags: tags:
- 随笔 - 杂谈
- 年终总结
--- ---
2021年已经过去2022年已经来临。每每一年开始的时候我都会展开一张纸或者新建一个文档思量着又是一年时光也该同诸大杂志一般写几句意味深长的话语怀念过去的时光也祝福未来的自己。可往往脑海中已是三万字的长篇落在笔头却又是一个字都没有了。 2021年已经过去2022年已经来临。每每一年开始的时候我都会展开一张纸或者新建一个文档思量着又是一年时光也该同诸大杂志一般写几句意味深长的话语怀念过去的时光也祝福未来的自己。可往往脑海中已是三万字的长篇落在笔头却又是一个字都没有了。
@@ -22,7 +23,7 @@ tags:
在前12年的学生生涯中我们都在期待着这一次的暑假以为在这个没有作业的假期里我们就可以充分的享受人间的美好。可是当时我们不知道这人间的烦恼可不止作业这一种无论是突如其来的疫情导致开学延期还是等待录取时的不安。 在前12年的学生生涯中我们都在期待着这一次的暑假以为在这个没有作业的假期里我们就可以充分的享受人间的美好。可是当时我们不知道这人间的烦恼可不止作业这一种无论是突如其来的疫情导致开学延期还是等待录取时的不安。
虽说在暑假时,拥有了自己的笔记本电脑,可是在高中三年屯下的游戏还是没有玩几个,看来我也是“喜加一”的受害者。虽然在高考后入坑了原神,但是假期间我并没有太过投入的玩。 虽说在暑假时,拥有了自己的笔记本电脑,可是在高中三年屯下的游戏还是没有玩几个,看来我也是“喜加一”的受害者。虽然在高考后入坑了原神,但是假期间我并没有太过投入的玩。
暑假下定决心要好好的学一学可是看着我gitee上暑假期间那稀疏的提交我就知道我又摸了一个暑假的鱼。 暑假下定决心要好好的学一学可是看着我gitee上暑假期间那稀疏的提交我就知道我又摸了一个暑假的鱼。
![gitee贡献](./2021-final/1.png) ![gitee贡献](./2021-final/1.webp)
即使我想写的很多项目都没有被扎实的推进下来但是学习的一些的C语言还是让我受益匪浅。 即使我想写的很多项目都没有被扎实的推进下来但是学习的一些的C语言还是让我受益匪浅。
现在看来,这个假期真是,**学也没有学好,耍也没有耍好**的典型。 现在看来,这个假期真是,**学也没有学好,耍也没有耍好**的典型。

BIN
YaeBlog/source/posts/2021-final/1.png (Stored with Git LFS)

Binary file not shown.

Binary file not shown.

After

Width:  |  Height:  |  Size: 9.8 KiB

View File

@@ -1,7 +1,8 @@
--- ---
title: 2022年终总结 title: 2022年终总结
tags: tags:
- 随笔 - 杂谈
- 年终总结
date: 2022-12-30 14:58:12 date: 2022-12-30 14:58:12
--- ---
@@ -56,11 +57,11 @@ date: 2022-12-30 14:58:12
小小的总结一下2022年可以算得上是一事无成的一年还搞砸了不少的事情。在写代码上进展有限成绩上大幅倒退说好的六级英语和大学物理竞赛都没有参加在年末应对疫情进展的时候更是把“不知所措”这个成语诠释的淋漓尽致。 小小的总结一下2022年可以算得上是一事无成的一年还搞砸了不少的事情。在写代码上进展有限成绩上大幅倒退说好的六级英语和大学物理竞赛都没有参加在年末应对疫情进展的时候更是把“不知所措”这个成语诠释的淋漓尽致。
![](./2022-final/2022-12-30-14-26-19-QQ_Image_1672381538441.jpg) ![](./2022-final/2022-12-30-14-26-19-QQ_Image_1672381538441.webp)
关于今年的人际交往和社会关系我愿意用QQ2022年年终总结中的一张截屏来总结这张图片透漏出一种无可救药的悲伤。 关于今年的人际交往和社会关系我愿意用QQ2022年年终总结中的一张截屏来总结这张图片透漏出一种无可救药的悲伤。
![](./2022-final/2022-12-30-14-28-12-QQ_Image_1672381543836.jpg) ![](./2022-final/2022-12-30-14-28-12-QQ_Image_1672381543836.webp)
## 展望 ## 展望

Binary file not shown.

After

Width:  |  Height:  |  Size: 34 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 54 KiB

View File

@@ -1,7 +1,7 @@
--- ---
title: 2022年暑假碎碎念 title: 2022年暑假碎碎念
tags: tags:
- 随笔 - 杂谈
typora-root-url: 2022-summer-vacation typora-root-url: 2022-summer-vacation
date: 2022-08-22 15:39:13 date: 2022-08-22 15:39:13
--- ---
@@ -32,7 +32,7 @@ date: 2022-08-22 15:39:13
- 下定决定要参加下一学期的物理竞赛,但是在听了讲座之后直接决定开学再开始学习,~~我知道我在家没法学习,俗称开摆~~ - 下定决定要参加下一学期的物理竞赛,但是在听了讲座之后直接决定开学再开始学习,~~我知道我在家没法学习,俗称开摆~~
- 又捡起了`Blender`,并在[Github](https://github.com/tanjian1998/bupt_minecraft)上找到了伟大的前辈们在`Minecraft`里复刻的老校区,希望能用`Blender`渲染几张图当作桌面。 - 又捡起了`Blender`,并在[Github](https://github.com/tanjian1998/bupt_minecraft)上找到了伟大的前辈们在`Minecraft`里复刻的老校区,希望能用`Blender`渲染几张图当作桌面。
![唯一的一张成品](result1.png) ![唯一的一张成品](result1.webp)
> 在此感谢所有为此付出过汗水的前辈们,让我这个即将搬入老校区的萌新能提前一睹老校区的风采。 > 在此感谢所有为此付出过汗水的前辈们,让我这个即将搬入老校区的萌新能提前一睹老校区的风采。

Binary file not shown.

Binary file not shown.

After

Width:  |  Height:  |  Size: 134 KiB

View File

@@ -1,7 +1,8 @@
--- ---
title: 2023年年终总结 title: 2023年年终总结
tags: tags:
- 随笔 - 杂谈
- 年终总结
date: 2024-2-29 20:18:19 date: 2024-2-29 20:18:19
--- ---
@@ -43,7 +44,7 @@ date: 2024-2-29 20:18:19
2023年最令我吃惊的事情是我刷B站的时长 2023年最令我吃惊的事情是我刷B站的时长
![image-20240303165826486](2023-final/image-20240303165826486.png) ![image-20240303165826486](2023-final/image-20240303165826486.webp)
容易计算得出我一共看了64天的B站接近六分之一的时间都在看。虽然我确实有着在干活的时候黑听B站和把B站当作音乐播放器的习惯但是这个时间未免有点太长了。下一年一定要在这个方面做出一定的改变将更多的时间放在看书上面去~~虽然写这句话的时候我就在黑听B站~~。 容易计算得出我一共看了64天的B站接近六分之一的时间都在看。虽然我确实有着在干活的时候黑听B站和把B站当作音乐播放器的习惯但是这个时间未免有点太长了。下一年一定要在这个方面做出一定的改变将更多的时间放在看书上面去~~虽然写这句话的时候我就在黑听B站~~。

Binary file not shown.

Binary file not shown.

After

Width:  |  Height:  |  Size: 16 KiB

View File

@@ -70,7 +70,7 @@ tags:
不过我的B站观看时长再度增长30%,这好吗,这不好,~~有这么多时间刷B站鬼知道你匆匆在哪了~~。 不过我的B站观看时长再度增长30%,这好吗,这不好,~~有这么多时间刷B站鬼知道你匆匆在哪了~~。
![image-20250115171809775](./2024-final/image-20250115171809775.png) ![image-20250115171809775](./2024-final/image-20250115171809775.webp)
### 未来 ### 未来

Binary file not shown.

Binary file not shown.

After

Width:  |  Height:  |  Size: 38 KiB

View File

@@ -1,7 +1,7 @@
--- ---
title: 人生代码大作业初体验 title: 人生代码大作业初体验
tags: tags:
- 随笔 - 杂谈
typora-root-url: big-homework typora-root-url: big-homework
date: 2022-07-27 11:34:49 date: 2022-07-27 11:34:49
--- ---
@@ -44,7 +44,7 @@ date: 2022-07-27 11:34:49
而且采用 `Git`还有一个好处,采用 `Github``Insight`功能可以轻松的看出大家的贡献值()。 而且采用 `Git`还有一个好处,采用 `Github``Insight`功能可以轻松的看出大家的贡献值()。
![img](1.png) ![img](1.webp)
## 一些技术上的收获 ## 一些技术上的收获

BIN
YaeBlog/source/posts/big-homework/1.png (Stored with Git LFS)

Binary file not shown.

Binary file not shown.

After

Width:  |  Height:  |  Size: 16 KiB

View File

@@ -131,7 +131,7 @@ Hexo init blog
``` ```
Hexo会以blog为名称创建一个博客文件夹这个文件夹的内容为 Hexo会以blog为名称创建一个博客文件夹这个文件夹的内容为
![文件夹截图](1.png) ![文件夹截图](1.webp)
`node_modules`文件夹是Hexo需要用到的一些npm依赖包的存放地址`public`文件夹下是由Hexo渲染产生的静态博客文件`scaffolds`文件夹是博客用到的模板文件,在默认情况下应该有`draft.md`,`page.md`,`post.md`三个模板文件。`themes`是Hexo中可以使用的主题文件。主题也是Hexo一个非常方便的设计我们可以方便使用其他人编写的Hexo Themes让自己的博客在不同的风格之间变换。`source`文件夹就是存放我们写作的博客的地方。一般这里面会有两个子文件夹,`_draft`, `_posts`。我们在里面在创建一个`img`文件夹,把自己的头像图片和网站的图标文件都放在里面,在之后的设置的时候使用。 `node_modules`文件夹是Hexo需要用到的一些npm依赖包的存放地址`public`文件夹下是由Hexo渲染产生的静态博客文件`scaffolds`文件夹是博客用到的模板文件,在默认情况下应该有`draft.md`,`page.md`,`post.md`三个模板文件。`themes`是Hexo中可以使用的主题文件。主题也是Hexo一个非常方便的设计我们可以方便使用其他人编写的Hexo Themes让自己的博客在不同的风格之间变换。`source`文件夹就是存放我们写作的博客的地方。一般这里面会有两个子文件夹,`_draft`, `_posts`。我们在里面在创建一个`img`文件夹,把自己的头像图片和网站的图标文件都放在里面,在之后的设置的时候使用。
@@ -146,7 +146,7 @@ INFO Hexo is running at http://localhost:4000/ . Press Ctrl+C to stop.
会在本地运行Hexo自带的一台静态博客服务器。我们用浏览器访问http://localhost:4000, 就可以看见Hexo博客的初始界面 会在本地运行Hexo自带的一台静态博客服务器。我们用浏览器访问http://localhost:4000, 就可以看见Hexo博客的初始界面
![初始截图](2.png) ![初始截图](2.webp)
这便说明安装成功了,~~可以开香槟了~~ 这便说明安装成功了,~~可以开香槟了~~

Binary file not shown.

Binary file not shown.

After

Width:  |  Height:  |  Size: 30 KiB

Binary file not shown.

Binary file not shown.

After

Width:  |  Height:  |  Size: 38 KiB

View File

@@ -7,6 +7,7 @@ tags:
--- ---
我们编译是这样的,在本平台上编译只要敲三条命令就好了,而交叉编译要考虑的就很多了。 我们编译是这样的,在本平台上编译只要敲三条命令就好了,而交叉编译要考虑的就很多了。
<!--more--> <!--more-->
@@ -45,7 +46,7 @@ tags:
通常一份GNU工具链只能针对一个平台进行编译但是LLVM工具链是一套先天的交叉编译工具链例如对于`llc`工具,使用`llc --version`命令可以看见该编译器可以生成多种目标平台上的汇编代码: 通常一份GNU工具链只能针对一个平台进行编译但是LLVM工具链是一套先天的交叉编译工具链例如对于`llc`工具,使用`llc --version`命令可以看见该编译器可以生成多种目标平台上的汇编代码:
![image-20240824120646587](./build-dotnet-from-source/image-20240824120646587.png) ![image-20240824120646587](./build-dotnet-from-source/image-20240824120646587.webp)
在使用`clang++`时加上`--target=<triple>`指定目标三元组就可以进行交叉编译。 在使用`clang++`时加上`--target=<triple>`指定目标三元组就可以进行交叉编译。
@@ -62,7 +63,7 @@ int main()
} }
``` ```
![image-20240824121425007](./build-dotnet-from-source/image-20240824121425007.png) ![image-20240824121425007](./build-dotnet-from-source/image-20240824121425007.webp)
看样子交叉编译也不是开箱即用的。最开始我们猜想系统提供的LLVM工具链没有被配置为交叉编译因此尝试在本地自行编译一套LLVM工具链。 看样子交叉编译也不是开箱即用的。最开始我们猜想系统提供的LLVM工具链没有被配置为交叉编译因此尝试在本地自行编译一套LLVM工具链。
@@ -81,7 +82,7 @@ cmake ../llvm-project.src/llvm \
编译之后的成果会安装到`/usr/local/`目录下,而在`$PATH`环境变量中`/usr/local`位置将在`/usr`目录之前因此调用时将会优先调用我们自行编译的LLVM工具链而不是系统中安装的LLVM工具链。 编译之后的成果会安装到`/usr/local/`目录下,而在`$PATH`环境变量中`/usr/local`位置将在`/usr`目录之前因此调用时将会优先调用我们自行编译的LLVM工具链而不是系统中安装的LLVM工具链。
![image-20240824134158262](./build-dotnet-from-source/image-20240824134158262.png) ![image-20240824134158262](./build-dotnet-from-source/image-20240824134158262.webp)
但是使用这套编译工具链仍然会爆出和之前一样的问题。说明这并不是系统安装LLVM工具链的问题。仔细一想也确实这里提示找不到对应的头文件应该是找不到RISC-V架构之下的头文件——这里的也是交叉编译的主要问题所在虽然LLVM工具链宣称自己是原生支持交叉编译的但是没人宣称说标准库和头文件是原生的。这里我们就需要一个根文件系统来提供这些头文件和各种库文件。 但是使用这套编译工具链仍然会爆出和之前一样的问题。说明这并不是系统安装LLVM工具链的问题。仔细一想也确实这里提示找不到对应的头文件应该是找不到RISC-V架构之下的头文件——这里的也是交叉编译的主要问题所在虽然LLVM工具链宣称自己是原生支持交叉编译的但是没人宣称说标准库和头文件是原生的。这里我们就需要一个根文件系统来提供这些头文件和各种库文件。
@@ -198,7 +199,7 @@ clang++ --target=riscv64-linux-gnu --sysroot=$ROOTFS_DIR -fuse-ld=lld hello.cpp
第一个问题的回答是Arch Linux安装的LLVM工具是可以交叉编译的。虽然在Arch Linux官方构建LLVM工具链的[构建脚本](https://gitlab.archlinux.org/archlinux/packaging/packages/clang/-/blob/main/PKGBUILD?ref_type=heads)中没有使用`LLVM_TARGETS_TO_BUILD`参数,但是这个参数的默认值是`all`。这一点我们也可以通过实验来验证。 第一个问题的回答是Arch Linux安装的LLVM工具是可以交叉编译的。虽然在Arch Linux官方构建LLVM工具链的[构建脚本](https://gitlab.archlinux.org/archlinux/packaging/packages/clang/-/blob/main/PKGBUILD?ref_type=heads)中没有使用`LLVM_TARGETS_TO_BUILD`参数,但是这个参数的默认值是`all`。这一点我们也可以通过实验来验证。
![image-20240824153514149](./build-dotnet-from-source/image-20240824153514149.png)于是回到编译`llvm`的目录下执行`cat install_manifest.txt | sudo xargs rm` ![image-20240824153514149](./build-dotnet-from-source/image-20240824153514149.webp)于是回到编译`llvm`的目录下执行`cat install_manifest.txt | sudo xargs rm`
第二个问题的回答可以使用实验来验证,首先安装`riscv64-linux-gnu-gcc`,然后将根文件系统的位置设置为`/usr/riscv64-linux-gnu`,重新编译上面的你好世界样例。编译之后可以正常执行。 第二个问题的回答可以使用实验来验证,首先安装`riscv64-linux-gnu-gcc`,然后将根文件系统的位置设置为`/usr/riscv64-linux-gnu`,重新编译上面的你好世界样例。编译之后可以正常执行。
@@ -229,4 +230,4 @@ export ROOTFS_DIR=<rootfs>
但是现在的.NET在RISC-V平台上还是废物一个甚至连`dotnet new`都跑不过,下一步看看能不能运行一下运行时的测试集看看。 但是现在的.NET在RISC-V平台上还是废物一个甚至连`dotnet new`都跑不过,下一步看看能不能运行一下运行时的测试集看看。
![image-20240824214145759](./build-dotnet-from-source/image-20240824214145759.png) ![image-20240824214145759](./build-dotnet-from-source/image-20240824214145759.webp)

Binary file not shown.

After

Width:  |  Height:  |  Size: 78 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 14 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 72 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 14 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 79 KiB

View File

@@ -17,7 +17,7 @@ date: 2022-05-08 11:35:19
我项目的结构大致如图所示: 我项目的结构大致如图所示:
![](1.png) ![](1.webp)
`include`的头文件目录下有两个头文件,`rail.h``bus.h`,这两个头文件分别定义了两个结构体`rail_node_t``bus_t` `include`的头文件目录下有两个头文件,`rail.h``bus.h`,这两个头文件分别定义了两个结构体`rail_node_t``bus_t`
@@ -68,7 +68,7 @@ typedef struct bus bus_t;
项目的`test`文件夹下是单元测试文件夹,但是在编译的时候会报错 项目的`test`文件夹下是单元测试文件夹,但是在编译的时候会报错
![](2.png) ![](2.webp)
大意就是在一个google test内部的头文件中有几个函数找不到定义这个函数都位于`io.h`这个头文件中。 大意就是在一个google test内部的头文件中有几个函数找不到定义这个函数都位于`io.h`这个头文件中。

Binary file not shown.

Binary file not shown.

After

Width:  |  Height:  |  Size: 5.4 KiB

Binary file not shown.

Binary file not shown.

After

Width:  |  Height:  |  Size: 205 KiB

View File

@@ -5,13 +5,14 @@ tags:
- 杂谈 - 杂谈
--- ---
2024年的中国计算机大会于10月24日到10月26日在浙江省金华市东阳市横店镇举办而鄙人在下不才我有幸受到实验室资助前去参观学习。 2024年的中国计算机大会于10月24日到10月26日在浙江省金华市东阳市横店镇举办而鄙人在下不才我有幸受到实验室资助前去参观学习。
<!--more--> <!--more-->
首先开幕式镇楼。 首先开幕式镇楼。
![image-20241102212738598](./cncc-2024/image-20241102212738598.png) ![image-20241102212738598](./cncc-2024/image-20241102212738598.webp)
## 学术上 ## 学术上
@@ -21,11 +22,11 @@ tags:
第一个报告是华为庞加莱实验室秦彬娟老师的《异构智算时代的操作系统演进》。报告高屋建瓴从比较宏观的角度上介绍了当前异构融合操作系统诞生的背景、发展的方向。在报告中重点介绍了一种异构融合操作系统的设计思路通过三层架构基于互联池化技术构建AI时代的融合算力系统。系统中的三层包括1池化基础底层包括多设备的融合和池化设备虚拟化2异构融合核心子系统例如异构融合调度系统、异构融合内存和异构融合存储系统3异构核心服务。总的来说这个报告在一定程度上勾勒出了未来一个异构融合操作系统应有的各项功能但是显然这一操作系统的实现还存在着明显的困难。 第一个报告是华为庞加莱实验室秦彬娟老师的《异构智算时代的操作系统演进》。报告高屋建瓴从比较宏观的角度上介绍了当前异构融合操作系统诞生的背景、发展的方向。在报告中重点介绍了一种异构融合操作系统的设计思路通过三层架构基于互联池化技术构建AI时代的融合算力系统。系统中的三层包括1池化基础底层包括多设备的融合和池化设备虚拟化2异构融合核心子系统例如异构融合调度系统、异构融合内存和异构融合存储系统3异构核心服务。总的来说这个报告在一定程度上勾勒出了未来一个异构融合操作系统应有的各项功能但是显然这一操作系统的实现还存在着明显的困难。
![image-20241102211959206](./cncc-2024/image-20241102211959206.png) ![image-20241102211959206](./cncc-2024/image-20241102211959206.webp)
下面一个报告是较为有干货的报告北京航空航天大学刘瀚骋老师的《异构融合OS及多样性内存管理框架》。报告中介绍了一个称作`FMMU`的系统是对于异构融合操作系统中内存管理系统的探索。报告中首先介绍了内存池化技术对于异构融合操作系统的重要性指出分布式共享内存Distributed Shared Memory可能是实现内存池化技术的未来。然后介绍了将部分内存管理中的计算卸载到可编程网络硬件中来加速分布式内存访问的新思路。最后在报告中提到了内存管理技术如何解决错误预测和错误回复的问题。虽然在听的时候没太注意但是现在总结的时候才发现这个报告的思路似乎有点混乱尤其是最后一点和内存管理系统并没有什么直接的关系而且这个内存管理系统似乎不是**异构系统**的内存管理,反而是分布式系统的内存管理。不过总的来说,这个报告还是非常实际的,介绍了不少当前异构融合操作系统中的内存管理面临的问题和解决问题的探索。 下面一个报告是较为有干货的报告北京航空航天大学刘瀚骋老师的《异构融合OS及多样性内存管理框架》。报告中介绍了一个称作`FMMU`的系统是对于异构融合操作系统中内存管理系统的探索。报告中首先介绍了内存池化技术对于异构融合操作系统的重要性指出分布式共享内存Distributed Shared Memory可能是实现内存池化技术的未来。然后介绍了将部分内存管理中的计算卸载到可编程网络硬件中来加速分布式内存访问的新思路。最后在报告中提到了内存管理技术如何解决错误预测和错误回复的问题。虽然在听的时候没太注意但是现在总结的时候才发现这个报告的思路似乎有点混乱尤其是最后一点和内存管理系统并没有什么直接的关系而且这个内存管理系统似乎不是**异构系统**的内存管理,反而是分布式系统的内存管理。不过总的来说,这个报告还是非常实际的,介绍了不少当前异构融合操作系统中的内存管理面临的问题和解决问题的探索。
![image-20241102212355390](./cncc-2024/image-20241102212355390.png) ![image-20241102212355390](./cncc-2024/image-20241102212355390.webp)
第三个报告是国防科技大学李东升老师的《异构计算环境下的分布式深度学习训练》。报告首先从李老师的主业——并行计算起手,介绍了深度学习训练过程中主要的各种并行方法,例如数据并行、模型并行和混合并行等,指出目前大模型的并行训练存在着计算/存储/通信难的问题。因此提出了一个智能模型训练并行任务划分方法1基于符号算子的计算图定义方法2面向Transformer模型的流水线并行任务划分方法3异构资源感知的流水线并行任务划分方法。然后针对分布式模型训练中通信调度存在的通信墙、数据依赖关系复杂等的问题提出综合词嵌入表的稀疏通信调度技术、流水线并行的P2P通信调度技术、模型计算的统一操作执行引擎和网络链路感知的通信执行引擎的通信调度技术。最后提到了智能模型训练 的内存优化技术针对现有重计算技术re-computing和存储交换swapping技术存在的问题提出了一种面向大型智能模型训练的细粒度内存优化方法`DELTA` 第三个报告是国防科技大学李东升老师的《异构计算环境下的分布式深度学习训练》。报告首先从李老师的主业——并行计算起手,介绍了深度学习训练过程中主要的各种并行方法,例如数据并行、模型并行和混合并行等,指出目前大模型的并行训练存在着计算/存储/通信难的问题。因此提出了一个智能模型训练并行任务划分方法1基于符号算子的计算图定义方法2面向Transformer模型的流水线并行任务划分方法3异构资源感知的流水线并行任务划分方法。然后针对分布式模型训练中通信调度存在的通信墙、数据依赖关系复杂等的问题提出综合词嵌入表的稀疏通信调度技术、流水线并行的P2P通信调度技术、模型计算的统一操作执行引擎和网络链路感知的通信执行引擎的通信调度技术。最后提到了智能模型训练 的内存优化技术针对现有重计算技术re-computing和存储交换swapping技术存在的问题提出了一种面向大型智能模型训练的细粒度内存优化方法`DELTA`
@@ -49,7 +50,7 @@ Plane讨论没有参加。
第二个报告是南京大学冯新宇老师的《基于仓颉语言的嵌入式DSL开发》同时冯新宇老师也是仓颉语言的首席架构师。冯老师的这个报告主要聚焦于仓颉语言提供的嵌入式DSL能力而嵌入式DSL这一设计范式已经在前端开发中展现了不俗的潜力。报告中介绍了嵌入式DSL出现的背景仓颉中为了提供嵌入式DSL而引入的语法糖、仓颉提供的嵌入式DSL工具箱等。虽然仓颉语言是一个主要面向上层应用开发的语言但是仓颉中丰富的DSL能力还是给异构编程模型的设计提供了不少的启发。而且目前在各种深度学习编译器中DSL的应用也非常广泛例如`triton` 第二个报告是南京大学冯新宇老师的《基于仓颉语言的嵌入式DSL开发》同时冯新宇老师也是仓颉语言的首席架构师。冯老师的这个报告主要聚焦于仓颉语言提供的嵌入式DSL能力而嵌入式DSL这一设计范式已经在前端开发中展现了不俗的潜力。报告中介绍了嵌入式DSL出现的背景仓颉中为了提供嵌入式DSL而引入的语法糖、仓颉提供的嵌入式DSL工具箱等。虽然仓颉语言是一个主要面向上层应用开发的语言但是仓颉中丰富的DSL能力还是给异构编程模型的设计提供了不少的启发。而且目前在各种深度学习编译器中DSL的应用也非常广泛例如`triton`
![image-20241102212536635](./cncc-2024/image-20241102212536635.png) ![image-20241102212536635](./cncc-2024/image-20241102212536635.webp)
第三个报告是在存算一体的芯片上做数据库的加速第四个报告是OpenHarmony上`ArkTS`程序的静态分析,都没怎么听。 第三个报告是在存算一体的芯片上做数据库的加速第四个报告是OpenHarmony上`ArkTS`程序的静态分析,都没怎么听。

Binary file not shown.

Binary file not shown.

After

Width:  |  Height:  |  Size: 136 KiB

Binary file not shown.

Binary file not shown.

After

Width:  |  Height:  |  Size: 238 KiB

Binary file not shown.

Binary file not shown.

After

Width:  |  Height:  |  Size: 47 KiB

Binary file not shown.

Binary file not shown.

After

Width:  |  Height:  |  Size: 639 KiB

View File

@@ -198,7 +198,7 @@ bazel build -c opt --strip=ALWAYS \
如果在编译的过程中提示缺失`dx.jar`这个文件而且你用的SDK版本还是高于31的那可能是SDK中缺失了这个文件可以将SDk降级到30就含有这个文件了。我使用的解决办法比较离奇我是将30版本的SDK文件中的这个文件软链接过来解决了这个问题。 如果在编译的过程中提示缺失`dx.jar`这个文件而且你用的SDK版本还是高于31的那可能是SDK中缺失了这个文件可以将SDk降级到30就含有这个文件了。我使用的解决办法比较离奇我是将30版本的SDK文件中的这个文件软链接过来解决了这个问题。
![](compile-mediapipe/2023-01-15-22-05-41-Screenshot_20230115_220521.png) ![](compile-mediapipe/2023-01-15-22-05-41-Screenshot_20230115_220521.webp)
编译消耗的时间可能比较的长,耐心等待即可。 编译消耗的时间可能比较的长,耐心等待即可。
@@ -227,7 +227,7 @@ bazel build -c opt //mediapipe/graphs/pose_tracking:pose_tracking_gpu_binary_gra
然后还需要从服务器上下载`tflite`文件,`Pose Tracking`这个解决方案需要两个`tflite`文件,第一个是[pose_detection.tflite](https://storage.googleapis.com/mediapipe-assets/pose_detection.tflite),第二个文件则有三个不同的选择,分别对于解决方案中提供的三个质量版本: 然后还需要从服务器上下载`tflite`文件,`Pose Tracking`这个解决方案需要两个`tflite`文件,第一个是[pose_detection.tflite](https://storage.googleapis.com/mediapipe-assets/pose_detection.tflite),第二个文件则有三个不同的选择,分别对于解决方案中提供的三个质量版本:
![](compile-mediapipe/2023-01-19-20-20-40-Screenshot_20230119_202008.png) ![](compile-mediapipe/2023-01-19-20-20-40-Screenshot_20230119_202008.webp)
下载地址是[pose_landmark_full.tflite](https://storage.googleapis.com/mediapipe-assets/pose_landmark_full.tflite)[pose_landmark_heavy.tflite](https://storage.googleapis.com/mediapipe-assets/pose_landmark_heavy.tflite)和[pose_landmark_lite.tflite](https://storage.googleapis.com/mediapipe-assets/pose_landmark_lite.tflite)。 下载地址是[pose_landmark_full.tflite](https://storage.googleapis.com/mediapipe-assets/pose_landmark_full.tflite)[pose_landmark_heavy.tflite](https://storage.googleapis.com/mediapipe-assets/pose_landmark_heavy.tflite)和[pose_landmark_lite.tflite](https://storage.googleapis.com/mediapipe-assets/pose_landmark_lite.tflite)。

Binary file not shown.

After

Width:  |  Height:  |  Size: 36 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 7.9 KiB

View File

@@ -1,20 +1,21 @@
--- ---
title: 计算机系统结构——流水线复习 title: 计算机系统结构——流水线复习
date: 2024-06-12T20:27:25.0000000
tags: tags:
- 计算机系统结构 - 计算机系统结构
- 学习资料 - 学习资料
date: 2024-06-12 20:27:25
--- ---
让指令的各个执行阶段依次进行运行是一个简单而自然的想法,但是这种方式执行速度慢、运行效率低。因此一个很自然的想法就是将指令重叠起来运行,让执行功能部件被充分的利用起来,这就是**流水线**。 让指令的各个执行阶段依次进行运行是一个简单而自然的想法,但是这种方式执行速度慢、运行效率低。因此一个很自然的想法就是将指令重叠起来运行,让执行功能部件被充分的利用起来,这就是**流水线**。
流水线的表示方法有两种。 流水线的表示方法有两种。
![image-20240612184855300](computer-architecture-pipeline/image-20240612184855300.png) ![image-20240612184855300](computer-architecture-pipeline/image-20240612184855300.webp)
第一种被称作**连接图**,清晰的表达出了流水线内部的逻辑关系。 第一种被称作**连接图**,清晰的表达出了流水线内部的逻辑关系。
![image-20240612184949777](computer-architecture-pipeline/image-20240612184949777.png) ![image-20240612184949777](computer-architecture-pipeline/image-20240612184949777.webp)
> 上图中给出了两个流水线中的概念:通过时间和排空时间。其中通过时间又被称作装入时间,是指第一个任务进入流水线到完成的事件;排空时间则相反,是最后一个任务通过流水线的时间。 > 上图中给出了两个流水线中的概念:通过时间和排空时间。其中通过时间又被称作装入时间,是指第一个任务进入流水线到完成的事件;排空时间则相反,是最后一个任务通过流水线的时间。
@@ -40,7 +41,7 @@ date: 2024-06-12 20:27:25
- 静态流水线,同一时间内,多功能流水线的各段只能按照同一种功能的方式连接。 - 静态流水线,同一时间内,多功能流水线的各段只能按照同一种功能的方式连接。
- 动态流水线,同一时间内,多功能流水线的各种可以按照不同的方式连接,执行不同的功能。 - 动态流水线,同一时间内,多功能流水线的各种可以按照不同的方式连接,执行不同的功能。
![image-20240612190426368](computer-architecture-pipeline/image-20240612190426368.png) ![image-20240612190426368](computer-architecture-pipeline/image-20240612190426368.webp)
按照流水线中是否存在反馈回路分类: 按照流水线中是否存在反馈回路分类:
@@ -58,7 +59,7 @@ date: 2024-06-12 20:27:25
- 加速比,同一任务,不使用流水线所使用时间与使用流水线所用时间比。 - 加速比,同一任务,不使用流水线所使用时间与使用流水线所用时间比。
- 效率,流水线设备的利用率。 - 效率,流水线设备的利用率。
![image-20240612192700169](computer-architecture-pipeline/image-20240612192700169.png) ![image-20240612192700169](computer-architecture-pipeline/image-20240612192700169.webp)
在设计流水线的过程中存在若干问题。 在设计流水线的过程中存在若干问题。
@@ -68,7 +69,7 @@ date: 2024-06-12 20:27:25
一个典型的五段流水线MIPS流水线 一个典型的五段流水线MIPS流水线
![image-20240612193301372](computer-architecture-pipeline/image-20240612193301372.png) ![image-20240612193301372](computer-architecture-pipeline/image-20240612193301372.webp)

Binary file not shown.

After

Width:  |  Height:  |  Size: 8.4 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 24 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 23 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 14 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 57 KiB

View File

@@ -2,7 +2,7 @@
title: 日用Linux挑战 第0篇 初见Arch Linux title: 日用Linux挑战 第0篇 初见Arch Linux
tags: tags:
- Linux - Linux
- 随笔 - 杂谈
date: 2023-01-15 22:23:08 date: 2023-01-15 22:23:08
typora-root-url: daily-linux-0 typora-root-url: daily-linux-0
--- ---
@@ -92,7 +92,7 @@ sudo systemctl enable sddm.service
我目前实现的效果大概长这样: 我目前实现的效果大概长这样:
![](2023-01-12-13-28-38-Screenshot_20230112_132829.png) ![](2023-01-12-13-28-38-Screenshot_20230112_132829.webp)
颇有一种`Windows``MacOS`杂交的风格。 颇有一种`Windows``MacOS`杂交的风格。
@@ -106,7 +106,7 @@ sudo systemctl enable sddm.service
先上一张`shell`的系统概览截图: 先上一张`shell`的系统概览截图:
![](2023-01-12-13-36-45-Screenshot_20230112_133628.png) ![](2023-01-12-13-36-45-Screenshot_20230112_133628.webp)
终端模拟器直接使用的`konsole`,目前没有进行改动。 终端模拟器直接使用的`konsole`,目前没有进行改动。

Binary file not shown.

After

Width:  |  Height:  |  Size: 138 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 73 KiB

View File

@@ -2,7 +2,7 @@
title: 日用Linux挑战 第1篇 问题与挑战 title: 日用Linux挑战 第1篇 问题与挑战
tags: tags:
- Linux - Linux
- 随笔 - 杂谈
date: 2023-03-08 22:37:29 date: 2023-03-08 22:37:29
--- ---

View File

@@ -1,7 +1,7 @@
--- ---
title: 日用Linux挑战 第2篇 Wayland title: 日用Linux挑战 第2篇 Wayland
tags: tags:
- 随笔 - 杂谈
- Linux - Linux
date: 2023-07-23 11:44:34 date: 2023-07-23 11:44:34
typora-root-url: daily-linux-2 typora-root-url: daily-linux-2
@@ -18,7 +18,7 @@ typora-root-url: daily-linux-2
最近恰好被平铺式的窗口管理器种草又在B站上看见一个动画绚丽的`wayland`合成器——[Hyprland](https://hyprland.org/),当即脑袋一热,就把`kde`干掉,装上了`hyprland` 最近恰好被平铺式的窗口管理器种草又在B站上看见一个动画绚丽的`wayland`合成器——[Hyprland](https://hyprland.org/),当即脑袋一热,就把`kde`干掉,装上了`hyprland`
![img](df4211f6be2724b3b4725f7ce5a4078818844857.jpg) ![img](df4211f6be2724b3b4725f7ce5a4078818844857.avif)
安装`hyprland`的过程非常舒适,`hyprland`被打包为一个单独的二进制文件,使用`pacman`安装之后直接在`tty`下执行: 安装`hyprland`的过程非常舒适,`hyprland`被打包为一个单独的二进制文件,使用`pacman`安装之后直接在`tty`下执行:
@@ -46,7 +46,7 @@ Hyprland
各种在学习过程中遇到的工具软件基本上都工作运行良好。当然因为没有设置缩放的问题而导致字体都很小。因为如果在配置文件中设置缩放之后会导致字体发虚。下面的截图就是我将我的2K显示屏设置为150%缩放的效果,~~虽然在截图中的效果不明显~~。目前在常用软件中唯一让我十分不满意的软件是`wps`,使用体验完全无法和`offices`相提并论,目前我正在研究使用`wine`运行`offices`,如果成功了就再水一篇博客庆祝一下。 各种在学习过程中遇到的工具软件基本上都工作运行良好。当然因为没有设置缩放的问题而导致字体都很小。因为如果在配置文件中设置缩放之后会导致字体发虚。下面的截图就是我将我的2K显示屏设置为150%缩放的效果,~~虽然在截图中的效果不明显~~。目前在常用软件中唯一让我十分不满意的软件是`wps`,使用体验完全无法和`offices`相提并论,目前我正在研究使用`wine`运行`offices`,如果成功了就再水一篇博客庆祝一下。
![image-20230702205919301](image-20230702205919301.png) ![image-20230702205919301](image-20230702205919301.webp)
> 最新的进展是使用`wine`没法安装学校提供的`office 2021`,同时我又不愿意使用古老的`office`版本,但是我发现一个称作`onlyoffice`的第三方软件蛮好用的,等我试用一段时间再说。 > 最新的进展是使用`wine`没法安装学校提供的`office 2021`,同时我又不愿意使用古老的`office`版本,但是我发现一个称作`onlyoffice`的第三方软件蛮好用的,等我试用一段时间再说。
> >

Binary file not shown.

After

Width:  |  Height:  |  Size: 76 KiB

Binary file not shown.

Binary file not shown.

After

Width:  |  Height:  |  Size: 53 KiB

View File

@@ -1,7 +1,7 @@
--- ---
title: 日用Linux挑战 第3篇 放弃Wayland title: 日用Linux挑战 第3篇 放弃Wayland
tags: tags:
- 随笔 - 杂谈
- Linux - Linux
typora-root-url: daily-linux-3 typora-root-url: daily-linux-3
date: 2023-09-04 14:47:46 date: 2023-09-04 14:47:46
@@ -53,7 +53,7 @@ date: 2023-09-04 14:47:46
- `Meta+F`全屏应用 - `Meta+F`全屏应用
- `Meta+W`关闭应用 - `Meta+W`关闭应用
![](Screenshot_20230904_144149.png) ![](Screenshot_20230904_144149.webp)
### Fuck You NVIDIA ### Fuck You NVIDIA

Binary file not shown.

Some files were not shown because too many files have changed in this diff Show More