修复 CI 构建博客造成的更新时间错误

修复 CI 构建博客造成的更新时间错误

当使用 Travis CI or Github Actions 自动化部署时,发现部署成功后,所有文章的更新时间都变成了此次提交修改的时间,但有些文章在上一次提交后是没有发生过任何修改的。

这是因为 git 在推送更新时,并不记录保存文件的访问时间、修改时间等元信息,(原因在这里)所以每次使用 git 把项目 clone 下来时,文件的时间都是克隆时的时间。又因为如果没有在 front-matter 中指定 updatedHexo 会默认使用文件的最后修改时间作为文章的更新时间,所以会出现所有文章的更新时间都发生变化的情况。

总的来说,使用 git clone 下来的文件的时间都不是原来文件的时间,而自动化部署每次都需要 clone 源码才能进行后面的生成和部署操作,所以目前如果想正确显示更新时间。需要:

方法一:使用 updated 属性字段

在文章中的front-matter中添加updated: 更新时间

front-matter
1
2
3
4
5
title: 修复 CI 构建博客造成的更新时间错误
author: Sea
toc: true
date: 2021-01-07 15:53:16
updated: 2021-01-07 15:53:16

image

以后修改文件的话也只需修改updated项对应日期即可,但是这样有个弊端,完全需要靠自己自觉去修改更新时间,自己一旦忘掉,那么更新时间的意义也就荡然无存了,因此这里推荐第二种方法。

方法二:使用 git 推送时间

如果你用的是Travis Ci的话,只需要在.travis.yml添加如下配置:

.travis.yml
1
2
3
before_install:
# Restore last modified time
- 'git ls-files -z | while read -d '''' path; do touch -d "$(git log -1 --format="@%ct" "$path")" "$path"; done'

如果你用的是Github Actions的话,只需要在对应的yml添加如下配置:

actions
1
2
3
4
5
6
jobs:
<jobs_id>:
steps:
- name: Restore file modification time
run: |
git ls-files -z | while read -d '' path; do touch -d "$(git log -1 --format="@%ct" "$path")" "$path"; done

当然git ls-files 如果不好用可以改成 find

bash
1
find source/_posts -name '*.md' | while read file; do touch -d "$(git log -1 --format="@%ct" "$file")" "$file"; done

实际上,clone 下来的文件的时间还是克隆时的时间,然后通过上面的命令,它将 clone 下来的文件的时间改成了该文件最近一次变动的推送时间(也即文件最后一次修改的 push 时间)。

:如果github actions中使用actions/checkout@v2,请设定它的参数fetch-depth: 0,因为0表示获取所有分支和标签的所有历史记录。默认值为1

小知识:

bash
1
2
3
4
5
6
7
8
# 获取 git 仓库中所有文件的最新修改时间
git ls-tree -r --name-only HEAD | while read filename; do
echo "$(git log -1 --format="%ad" -- $filename) $filename"
done
# 获取 git 仓库中所有文件的最初创建时间
git ls-tree -r --name-only HEAD | while read filename; do
echo "$(git log --format="%ad" -- $filename | tail -1) $filename"
done

参考链接

修复 CI 构建博客造成的更新时间错误

https://hailangya.com/articles/2021/01/07/ci-hexo-update-time/

作者

Sea

发布于

2021-01-07

更新于

2023-01-10

许可协议

评论