上周我刚注册了一个简书账号,当时想给自己取一个亮眼的昵称。当时就想参照奇舞团月影一样,取一个昵称。月影是多肉的一个品种,那我也从多肉中选一个好了。我找了几个满意的多肉,结果都被注册过了,我也不愿意在昵称后面加是数字,所以干脆想着从百度上拉取多肉的名称列表,来循环请求简书的昵称校验接口,最后从过滤出没被注册过的名称中挑一个。
ok,那么就可以撸起袖子干活了。首先找到百度搜索多肉关键字的接口。通过扒取百度的请求,可以找到她的搜索接口:
https://sp0.baidu.com/8aQDcjqpAAV3otqbppnN2DJv/api.php?format=json&ie=utf-8&oe=utf-8&query=%E5%A4%9A%E8%82%89%E6%A4%8D%E7%89%A9&resource_id=6829&rn=12&from_mid=1&pn=12&&t=1559047427766&cb=jQuery110205850867375586439_1559047036788&_=1559047036821
接下来我们可以分析下这个接口,首先是个get请求,然后注意下请求的参数:query(转义后的关键字),rn(每页的条数),这两个参数最重要的。其他的参数,如时间戳,编码方式等都忽略吧。我们可以尝试使用axios调这个接口,看是否能生效。
function createUri (idx) {
var uri = `https://sp0.baidu.com/8aQDcjqpAAV3otqbppnN2DJv/api.php?format=json&ie=utf-8&oe=utf-8&query=${keyword}&resource_id=6829&rn=12&from_mid=1&pn=${12*idx}&&t=1558925990258&cb=jQuery1102003026095059572187_1558923836101&_=155892383610${(idx)}`
return uri
}
function getData(idx) {
axios.get(createUri(idx)).then(function (response) {
var res = handleData(response, (list) => {
list.forEach(one => checkNickname(one));
});
if (res.length < 12) {
console.log('stop');
} else {
idx++;
getData(idx);
}
}).catch(err => console.log('err:', err));
}
getData(0);
这段代码主要是重复去调用百度的请求接口,获取多肉名称的所有列表。然后将获得的数据过滤,过滤出name字段。
已经有了多肉的namelist了,那么下一步就是循环拿name去请求简书的校验昵称的接口,看昵称是否已被注册了。
function checkNickname (nickname) {
axios.post('https://www.jianshu.com/check_nickname', {
nickname
}).then(function (response) {
if (response.status === 200) {
var name = JSON.parse(response.config.data);
writeFile(name.nickname);
console.log('nickname:', name.nickname);
// return name.nickname;
} else {
}
}).catch(err => {
});
}
其实到这一步,可以把过滤后的昵称输出到终端来看。但是为了方便,我决定把名称写进txt文件里。这边不考虑效率问题,每次都是先读取txt文件的内容,再把新的内容加入的读取的字符串后面,然后再重写写入txt文件。
var pathname = 'nicknames.txt';
function writeFile (data) {
var isexist = fs.existsSync(pathname);
var d = '';
if (isexist) {
d = readFile() + '\n' + data;
}
fs.writeFile(pathname,d,'utf8',function(error){
if(error){
console.log(error);
return false;
}
});
}
function readFile () {
const buffer = fs.readFileSync(pathname);
return buffer.toString();
}
到这一步基本能够满足我们的需求了。但是其实之前的搜索关键字和输出的文件名都是被写死的,那么能不能在交互上能做的更炫酷点呢?ok,那就用目前cli脚手架常用commander
增加以下交互吧。
const program = require('commander');
program
.version('0.1.0')
.option('-k, --keyword [keyword]', 'enter the keyword', '多肉植物')
.option('-d, --dir [dir]', 'enter the direction', 'nicknames.txt')
.parse(process.argv);
怎么用呢?