乐趣区

使用-jq-与-sed-制作掘金面试文章榜单

月: 听说你这次分享 jq\
我: 恩,是了 \
月:这有啥好分享的,现在大家都用 react/vue 了 \
我: ……\
我: 给你官网链接 https://github.com/stedolan/jq\
月: 原来是命令行下的 json 美化工具,好像也没啥可分享的 \
我: ……\
我: 可以用它分析 nginx 的日志,比如用 json 表示 nginx 的日志 \
月: 线上的 ELK 它不好用吗?还是分布式的,又有索引,还快,就连查询语句都有自动补全 \
我: 好像有点道理 …\
我: 我还可以用它做爬虫!!!

谨以此篇文章讲述 jq 的实际使用案例,如果能提高你使用命令行的乐趣那更再好不过了。关于 jq 的用法可以参考我以前的文章: jq 命令详解及示例

<!–more–>

  • 原文地址: 使用 jq 与 sed 制作掘金面试文章榜单
  • 系列文章: 个人服务器运维指南

预览

使用以下命令,可以直接获取前端面试榜单

$ curl -s 'https://web-api.juejin.im/query' -H 'Content-Type: application/json' -H 'X-Agent: Juejin/Web' --data-binary '{"operationName":"","query":"","variables":{"tags":["55979fe6e4b08a686ce562fe"],"category":"5562b415e4b00c57d9b94ac8","first":100,"after":"","order":"HOTTEST"},"extensions":{"query":{"id":"653b587c5c7c8a00ddf67fc66f989d42"}}}' --compressed | \
 jq -c '.data.articleFeed.items.edges | .[].node | { likeCount, title, originalUrl} | select(.likeCount > 600)' | jq -cs '. | sort_by(-.likeCount) | .[] |"+【???? \(.likeCount)】[\(.title)](\(.originalUrl))"'| sed s/\"//g

+【???? 5059】[一个合格 (优秀) 的前端都应该阅读这些文章](https://juejin.im/post/5d387f696fb9a07eeb13ea60)
+【???? 4695】[2018 前端面试总结,看完弄懂,工资少说加 3K | 掘金技术征文](https://juejin.im/post/5b94d8965188255c5a0cdc02)
+【???? 4425】[中高级前端大厂面试秘籍,为你保驾护航金三银四,直通大厂(上)](https://juejin.im/post/5c64d15d6fb9a049d37f9c20)
+【???? 3013】[2018 春招前端面试: 闯关记(精排精校) | 掘金技术征文](https://juejin.im/post/5a998991f265da237f1dbdf9)
+【???? 2493】[前端面试考点多?看这些文章就够了(2019 年 6 月更新版)](https://juejin.im/post/5aae076d6fb9a028cc6100a9)

获取掘金列表接口

先来看看 http 的 url: https://web-api.juejin.im/query,用的 POST

再看看 body:

{
  "operationName": "","query":"",
  "variables": {
    "first": 20,
    "after": "1.0168277174789",
    "order": "POPULAR"
  },
  "extensions": {
    "query": {"id": "21207e9ddb1de777adeaca7a2fb38030"}
  }
}

最后再看看 http 的 response:

眼前一亮,多么熟悉的数据结构!!!

这不是 graphql 吗,我上个月 (2019/10) 还写了一系列文章介绍它: 使用 graphql 构建 web 应用,一年前 (2018) 还用 graphql 写了一套诗词的前后端: shfshanyue/shici 与 shfshanyue/shici-server

怎么看出来是 graphql 的呢?

  • /query 这是统一的入口
  • extensions.query.id 这是 APQ,为了缓存 gql,减少传输体积,减短网络时延,有利于缓存,当然也减少了安全性问题
  • variables 这是 graphql variables
  • data.items[].edges 这是 graphql 典型的分页 (虽然我不太喜欢,嵌套数据太多 …)

恩,好像跑偏了

总之,拿到了数据 – 关于前端面试的数据

$  curl -s 'https://web-api.juejin.im/query' -H 'Content-Type: application/json' -H 'X-Agent: Juejin/Web' --data-binary '{"operationName":"","query":"","variables":{"tags":["55979fe6e4b08a686ce562fe"],"category":"5562b415e4b00c57d9b94ac8","first":100,"after":"","order":"HOTTEST"},"extensions":{"query":{"id":"653b587c5c7c8a00ddf67fc66f989d42"}}}' --compressed

ETL

还是 etl 这个词高大上啊

先用 jq 取几个数吧: 标题与点赞数。更多用法可以参考我以前的文章: jq 命令详解及示例

为了更容易看到 jq 的用法,把 jq 另起一行,其中

  • -c: 一整行显示
  • .[]: json-array to jsonl
  • {}: 类似于 lodash.pick

我们此时根据命令获取到所有高赞文章,但它此时是无序的

$ curl -s 'https://web-api.juejin.im/query' -H 'Content-Type: application/json' -H 'X-Agent: Juejin/Web' --data-binary '{"operationName":"","query":"","variables":{"tags":["55979fe6e4b08a686ce562fe"],"category":"5562b415e4b00c57d9b94ac8","first":100,"after":"","order":"HOTTEST"},"extensions":{"query":{"id":"653b587c5c7c8a00ddf67fc66f989d42"}}}' --compressed | \
  jq -c '.data.articleFeed.items.edges | .[].node | {title, likeCount}'

{"title":"中高级前端大厂面试秘籍,为你保驾护航金三银四,直通大厂(上)","likeCount":4423}
{"title":"2018 前端面试总结,看完弄懂,工资少说加 3K | 掘金技术征文","likeCount":4690}
{"title":"一个合格 (优秀) 的前端都应该阅读这些文章","likeCount":5052}
{"title":"前端面试考点多?看这些文章就够了(2019 年 6 月更新版)","likeCount":2492}
{"title":"2018 春招前端面试: 闯关记(精排精校) | 掘金技术征文","likeCount":3013}

数据筛选与排序

再来筛选下点赞数大于 600 的

select(.likeCount > 600)

再来倒排个序

jq  -s '. | sort_by(-.likeCount) | .[]'

搞定,此时榜单上全是点赞数大于 600 且排好序的

$ curl -s 'https://web-api.juejin.im/query' -H 'Content-Type: application/json' -H 'X-Agent: Juejin/Web' --data-binary '{"operationName":"","query":"","variables":{"tags":["55979fe6e4b08a686ce562fe"],"category":"5562b415e4b00c57d9b94ac8","first":100,"after":"","order":"HOTTEST"},"extensions":{"query":{"id":"653b587c5c7c8a00ddf67fc66f989d42"}}}' --compressed | \
 jq -c '.data.articleFeed.items.edges | .[].node | {title, likeCount, originalUrl} | select(.likeCount > 600)' | jq -s '. | sort_by(-.likeCount) | .[]'

{"title": "一个合格 (优秀) 的前端都应该阅读这些文章",
  "likeCount": 5052,
  "originalUrl": "https://juejin.im/post/5d387f696fb9a07eeb13ea60"
}
{
  "title": "2018 前端面试总结,看完弄懂,工资少说加 3K | 掘金技术征文",
  "likeCount": 4690,
  "originalUrl": "https://juejin.im/post/5b94d8965188255c5a0cdc02"
}
{"title": "中高级前端大厂面试秘籍,为你保驾护航金三银四,直通大厂(上)",
  "likeCount": 4423,
  "originalUrl": "https://juejin.im/post/5c64d15d6fb9a049d37f9c20"
}

使用 sed 处理生成 markdown

我们此时已经生成了结构化的数据,如果我们使用 react 渲染数据的话,json 自然不错。但我们现在需要生成 markdown,所以再处理处理

先使用 jq 生成链接样式

"+【???? \(.likeCount)】[\(.title)](\(.originalUrl))"

在使用 sed 删除全部双引号,关于 sed,可以参考我的文章: sed 命令详解及示例

sed s/\"//g

此时,成功生成了 markdown 数据

$ curl -s 'https://web-api.juejin.im/query' -H 'Content-Type: application/json' -H 'X-Agent: Juejin/Web' --data-binary '{"operationName":"","query":"","variables":{"tags":["55979fe6e4b08a686ce562fe"],"category":"5562b415e4b00c57d9b94ac8","first":100,"after":"","order":"HOTTEST"},"extensions":{"query":{"id":"653b587c5c7c8a00ddf67fc66f989d42"}}}' --compressed | \
 jq -c '.data.articleFeed.items.edges | .[].node | { likeCount, title, originalUrl} | select(.likeCount > 600)' | jq -cs '. | sort_by(-.likeCount) | .[] |"+【???? \(.likeCount)】[\(.title)](\(.originalUrl))"'| sed s/\"//g

+【???? 5059】[一个合格 (优秀) 的前端都应该阅读这些文章](https://juejin.im/post/5d387f696fb9a07eeb13ea60)
+【???? 4695】[2018 前端面试总结,看完弄懂,工资少说加 3K | 掘金技术征文](https://juejin.im/post/5b94d8965188255c5a0cdc02)
+【???? 4425】[中高级前端大厂面试秘籍,为你保驾护航金三银四,直通大厂(上)](https://juejin.im/post/5c64d15d6fb9a049d37f9c20)
+【???? 3013】[2018 春招前端面试: 闯关记(精排精校) | 掘金技术征文](https://juejin.im/post/5a998991f265da237f1dbdf9)
+【???? 2493】[前端面试考点多?看这些文章就够了(2019 年 6 月更新版)](https://juejin.im/post/5aae076d6fb9a028cc6100a9)
+【???? 2359】[「中高级前端面试」JavaScript 手写代码无敌秘籍](https://juejin.im/post/5c9c3989e51d454e3a3902b6)

我是山月,一个喜欢跑步与爬山的程序员,我会定期分享全栈文章在个人公众号中。如果你对全栈面试,前端工程化,graphql,devops,个人服务器运维以及微服务感兴趣的话,可以关注我

退出移动版