机核网页端机组插件

Tampermonkey,确实是可以做一些很轻量但是很实用的小工具,总体上四五百行代码,以机核对外暴露的动态查看接口为基础,对机组消息进行遍历查询,然后二次拼装成单独页面进行展示.

核心代码

逻辑其实并不复杂,无非是数据的获取和拼接而已,加一点样式重写,加一点异步请求,加一点二分查找,就成了…

机核的机组是以顺序id进行存储的,但是删除的机博会存在ID占用了,但是数据为空,所以定向查询时我会做一点冗余,比如向后顺序遍历5个ID能返回3条数据就算成功.

另外评论的接口也从机核主站中扒出来了,也能成功做了个简单的评论显示.

唯一美中不足的就是.heic类型的苹果动态图片没法直接在H5中做展示,我也没再做转化,本身就是一个几百行的小脚本,引一堆JS做图片类型转换太蠢了.

锚点更新模块

三方插件怎么能获取到机核那边最新机博的ID以做加载锚点呢?

其实也简单,搞个定时脚本然后每小时做次更新,然后暴露给脚本就可以了.

手头没服务器的我调研一圈后最后敲定了 github-actions 的解决方案.

定时脚本,整点执行后写入到git库指定文件中,然后油猴脚本启动时拿固定路径内的ID值就可以了.

为了减少频繁请求,查询时候做了个伪二分算法,以增量数组的方式写了个小算法.

一切都很合理,唯一不好的地方就是每天固定会有十几条的更新污染git库commit记录.

源码

//锚点更新脚本
const fs = require(‘fs’)
const https = require(‘https’);
const cheerio = require(‘cheerio’);


const Gtalk = ‘https://www.gcores.com/talks/‘
const GitUrl = ‘https://github.com/TOKdawn/gcores-talks-Viewer/blob/main/crawler/TID.html’
const intervalList = [100, 50, 25, 10, 5, 2, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1] //加值率区间
var OLD_ID = ‘’
const add = function (a, b) {
  return a + b
}


function setHTML(TID) {
  if (!TID) {
    console.log(“TID is null”)
    return
  }
  fs.writeFile(‘./crawler/TID.html’, parseInt(TID, 10).toString(), err => {
    if (err) {
      throw err
    } else {
      console.log("TID.html success")
      // console.log(htmlStr)
    }
  })
}

function getTID() {
  return new Promise((resolve) => {
    console.log('开始获取TID')
    https.get(GitUrl, res => {
      let html = ''
      res.on('data', data => {
        html += data
      })
      res.on('end', () => {
        console.log('GIT请求成功')
        let $ = cheerio.load(html);
        let id = $('.blob-code-inner').text();
        if (id) {
          resolve(id)
        } else {
          console.log('HTML解析错误')
          resolve(null)
        }
      })
    })
  })
}

Function tryTID(TID) { //检测TID要同时连续监测三个,因为存在一些删除的机博,占用了TID但是页面拿取为空
  let promiseList = []
  for (let I = 0; I < 3; I++) {
    promiseList[I] = new Promise(resolve => {
      let url = Gtalk + (TID + i)
      https.get(url, res => {
        let html = ‘’
        res.on(‘data’, data => {
          html += data
        })
        res.on(‘end’, () => {
          let $ = cheerio.load(html);
          let DOM = $(‘.aat_container’).children().length;
          console.log(‘DOM’, DOM)
          if (DOM) {
            resolve(1)
          } else {
            resolve(0)
          }
        })
      })
    })
  }
  return promiseList
}

async function run() {
  OLD_ID = await getTID()
  OLD_ID -= 0
  console.log('获取TID成功:', OLD_ID)
  let searchFlag = true;
  let addNum = 0; //初始增值
  let INI = 0; //增值刻度
  let tryFlag = true; // 检测标识
  while (searchFlag) {
    addNum += intervalList[INI];
    await Promise.all(tryTID(OLD_ID + addNum)).then(res => {
      if (res.reduce(add) >= 1) { //三个中有一个有效就成功
        console.log(‘检测成功:’, res, OLD_ID + addNum)
        tryFlag = true
      } else {
        console.log(‘检测失败:’, OLD_ID + addNum)
        tryFlag = false
      }
    })
    if (!tryFlag) { //如果超越边界
      addNum -= intervalList[INI]; //还原成上次增值
      INI++; //维度减小
    }
    if (!tryFlag && INI >= 9) { //结束条件
      searchFlag = false
    }
  }
  console.log(‘最终结果’, OLD_ID + addNum);
  setHTML(OLD_ID + addNum)
  console.log(‘结束 结束’)
}

run();
// ==UserScript==
// @name         机核网页端机组插件
// @namespace    https://github.com/TOKdawn
// @version      1.1.0
// @description  机核网页端查看机组辅助工具
// @icon         
// @author       TOKdawn
// @include      *gcores.com/*
// @require      http://cdn.staticfile.org/jquery/2.1.4/jquery.min.js
// @license      MIT
// @supportURL   https://github.com/TOKdawn/gcores-talks-Viewer/issues
// @compatible   chrome
// @compatible   firefox
// @grant        GM_addStyle
// @grant        GM_xmlhttpRequest
// @run-at       document-start
// ==/UserScript==
/* globals $, Highcharts, tippy */
(function() {
// ==UserScript==
// @description    Usercript with GM_addStyle method.
// ==/UserScript==
function GM_addStyle(css) {
  var style = document.getElementById("GM_addStyleByTOKDawn") || (function() {
    var style = document.createElement('style');
    style.type = 'text/css';
    style.id = "GM_addStyleByTOKDawn";
    document.head.appendChild(style);
    return style;
  })();
  var sheet = style.sheet;
  sheet.insertRule(css, (sheet.rules || sheet.cssRules || []).length);
}
 
function random(){
  return  Math.ceil(Math.random()*500); 
}
var add=function(a,b){
  return a+b
}
 
var GURL =  'https://www.gcores.com/talks/'
var CURL =  'https://www.gcores.com/gapi/v1/talks/'
var domparser = new DOMParser()
var GTbtn = '<span class="globalActions_item GTK_btn"  target="_blank"><div><svg t="1648725115646" class="icon" viewBox="0 0 1024 1024" version="1.1" xmlns="http://www.w3.org/2000/svg" p-id="21619" width="24" height="24"><path d="M677.6 389.78v-82H343.21v82z m186.5 369.07H274.62c-21.71 0-39.3 18.36-39.3 41s17.59 41 39.3 41H864.1v82H274.62c-65.12 0-117.9-55.08-117.9-123v-615.1c0-45.3 35.19-82 78.6-82H864.1z" p-id="21620" fill="#f44336"></path></svg></div><p>机组</p></span>';
var GTCon = '<span class="GTK_con"> <div class="GTK_content"><div class="GTK_talk_go_top">↑回到顶部↑</div><div class="GTK_talk_load_new">加载更新机博</div><div class="GTK_talk_list"></div><div class="GTK_talk_load_old">加载更多</div></div></span>';
var GitUrl = 'https://github.com/TOKdawn/gcores-talks-Viewer/blob/main/crawler/TID.html'
var GTcommit = '<div class="GTK_commit"><ul style="margin-bottom: 0px;" class="GTK_commit_list">';
var loadFlag = false
var startTID = 1232; //已经加载的最新鸡脖 大值
var endTID = 1232; //已经加载的最旧鸡脖 小值
var commitFlag = {}
var STEP = 10; //每次加载的机组数
var MAX = 500; //最大加载的机组数
var MIN = 4; //最小成功数
var topTID = { // 目前最新的鸡脖ID 更新时间 最大值
  id: 999999999,
  time: '2019-01-01 00:00:00'
}; // 目前最新的鸡脖ID
var anchorTID = {// 远程获取锚点鸡脖ID 更新时间
 id: 223644,
 time: '2019-01-01 00:00:00'
}
var gitTID = {
  id: 223644,
  time: '2019-01-01 00:00:00'
}
var carouselsList = { //存储带轮播图机组信息
}
 

//初始化组件
function onDocumentStart(){
  $('body').append($(GTbtn))
  $('body').append($(GTCon))
  $('.GTK_btn').off().on('click', showGTK)
  
  // var time = Date.parse( new Date())
  var localTID = localStorage.getItem('GTK_TID_DATA')
  if(localTID){
    localTID = JSON.parse(localTID)
    anchorTID.id = localTID.id-0 //拿取缓存的TID
    updataTID(false)
  }else{
    updataTID(true)//没有本地缓存ID
  }
  addStyle();
}
 
function updataTID(fisrtFlag){
  GM_xmlhttpRequest({
    method: 'GET',
    url: GitUrl, //获取对应编号鸡脖
    timeout: 2000,
    onload: function (xhr) {
        var html = domparser.parseFromString(xhr.responseText,'text/html') 
        if(html){
          var content = html.querySelector('.blob-code-inner') //拿取git库中锚点ID
          if(content.innerText){
            // anchorTID.id = content.innerText-0
            // anchorTID.time = Date.parse( new Date())
            // localStorage.setItem('GTK_TID_DATA', JSON.stringify(anchorTID))
            // gitTID.time = Date.parse( new Date())
            gitTID.id = content.innerText-0
            if(fisrtFlag){
              anchorTID.id = gitTID.id //没有本地缓存ID直接用远程ID
            }else{
              if(gitTID.id - anchorTID.id > STEP){//远程ID大于本地ID 则显示回到顶部
                $('.GTK_talk_go_top').show()
              }else{
                $('.GTK_talk_go_top').hide()//本地的大于远程ID 则隐藏回到顶部
              }
            }
          }else{
            console.log('获取gitTID失败')
          }
        }
    },
    onerror: function (xhr) {
      console.log('拿取gitID失败', xhr,'id:',anchorTID.id - i);
    },
    ontimeout: function (xhr) {
      console.log('拿取gitID超时:', xhr, 'id:',anchorTID.id - i);
    },
    onabort:function (xhr) {
      console.log('拿取gitID中止:', xhr, 'id:',anchorTID.id - i);
    }
  })
}
function addStyle(){
  //入口Btn样式
  GM_addStyle(`
  .GTK_btn {
    font-size: .875rem;
    position: fixed;
    display: block;
    right: calc(1.125rem + 0.83333vw);
    bottom: calc(10rem + 7.33333vw );
    background: var(--gray-global-action-bg);
    color: #f44336;
    cursor: pointer;
    z-index: 5;
  }`);
  GM_addStyle(`
  .GTK_con {
    position: fixed;
    display: none;
    top: 60px;
    left: 250px;
    height: calc(100vh - 60px);
    outline: red 1px solid;
    right:0px;
    background: var(--gray-global-action-bg);
    color:  var(--gray-text);
    cursor: pointer;
    z-index: 9;
    overflow: auto;
    text-align: center;
  }`);
  GM_addStyle(`
    @media (max-width: 599px){
      .GTK_con {
        left: 0px;
        top:0px;
      }
    }
  `)
  GM_addStyle(`
    @media (max-width: 1279px) and (min-width: 600px){
      .GTK_con {
        left: 75px;
      }
    }
  `)
  GM_addStyle(`
    .GTK_talk_list{
      with:100%;
      text-align: left;
     
    }
  `)
  GM_addStyle(`
    .GTK_content {  
      min-width: 350px;
      max-width: 660px;
      display: inline-block;
      text-align: center;
      
    }
  `)
  GM_addStyle(`
  .GTK_talk_go_top {  
    with:100%;
    height:30px;
    line-height:30px;
    font-size:14px;
    text-align: center;
    cursor: pointer;
    background-color: #f44336;
    color:#fff;
    display:none;
  }`)
  GM_addStyle(`
  .GTK_talk_go_top:hover {  
    color: #f7f7f7;
  }`)
  GM_addStyle(`
  .GTK_talk_load_new {
    with:100%;
    height:30px;
    line-height:30px;
    font-size:14px;
    text-align: center;
    cursor: pointer;
  }
  `)
  GM_addStyle(`
    .GTK_talk_load_old {  
      with:100%;
      height:30px;
      line-height:30px;
      font-size:14px;
      border-radius: 3px;
      text-align: center;
      cursor: pointer;
      
    }
  `)
  GM_addStyle(`
    .GTK_talk_load_old:hover  {  
      color: #f44336;
      background-color: #f7f7f7;
    }
  `)
  GM_addStyle(`
  .GTK_talk_load_new:hover  {  
    color: #f44336;
    background-color: #f7f7f7;
  }
`)
  GM_addStyle(`
  .GTK_talk_item {  
    border-bottom: 1px solid #f4f4f4;
  }
  `)
  GM_addStyle(`
    .talk_actions{
      flex-wrap: wrap;
  }`)
  GM_addStyle(`
    .GTK_commit{
      border-top: 1px solid #f4f4f4;
      flex: 1 1 100%;
      height: auto;
  }`)
  GM_addStyle(`
  .GTK_commit_img{
    width: 20px;
    height: 20px;
    line-height: 19px;
    text-align: center;
    border-radius: 50%;
    font-weight: 800;
    display: inline-block;
    background-color: #f44336;
    color: #fff;
    margin-right: 10px;
  }`)
  GM_addStyle(`
  .GTK_commit_item{
    margin-top: 5px;
    line-height: 28px;
    text-align: left;
    margin-right: 12px;
   
  }`)
  GM_addStyle(`
  .GTK_sub_commit{
    border-bottom: 2px dashed #eee;
    padding-left: 30px;
  }`)
  GM_addStyle(`
  .GTK_main_commit{
    border-bottom: 2px dashed #eee;
  }`)
}
 
function showGTK() {
  $('.navLayout_nav').removeClass('gnav-hide')
  $('.pageContainer').hide() //阻止主页内容滚动影响导航栏隐藏
  $('.GTK_con').show()
  $('.navLayout').on('click',hideGTK)//进入其他标签页自动取消机组显示
  $('.GTK_talk_load_new').off().on('click', _=>{loadData('TOP')}) //向新加载
  $('.GTK_talk_load_old').off().on('click', _=>{loadData('Down')})//向旧加载
  $('.GTK_talk_go_top').off().on('click', _=>{goTop()})//向旧加载
  $('.GTK_con').on('click',function(e){
    e.stopPropagation() //阻止冒泡
  })
  loadData('NEW');
}
 
function hideGTK() {
  $('.pageContainer').show()
  $('.GTK_con').hide()
  $('.navLayout').unbind('click',hideGTK)
}
function goTop(){
  if(gitTID.id){
    anchorTID.id = gitTID.id
    $('.GTK_talk_list').empty()
    loadData('NEW')
    $('.GTK_talk_go_top').hide();
  }
}
function loadData(type) {
  var DOM = $('.GTK_talk_list')
  var tklist = []
  var PromiseList = []
  var url = ''
  if(type == 'Down'){
    if( anchorTID.id - endTID > MAX ){//已经加载200条了
      $('.GTK_talk_load_old').text('无法加载更多...')
      return;
    }
  }else if(type == 'TOP'){
    if(startTID > topTID.id - STEP){//最新机博不足
      $('.GTK_talk_load_new').text('已加载到最新机博...')
      return;
    }
  }else {
    startTID =  anchorTID.id
  }
  for(var index = 0; index < STEP; index++){ //每次加载5条
    PromiseList[index] = new Promise(function(resolve, reject){//构建Promise
      setTimeout((i,type)=>{
        var TID = ''
        if(type == 'Down'){
          TID = (endTID - i) //获取对应编号鸡脖
        }else if(type == 'TOP'){
          TID =  (startTID + i + 1 ) //获取对应编号鸡脖 加零减零都是零兄弟
        }else{
          TID = (anchorTID.id - i)
        }
        GM_xmlhttpRequest({
          method: 'GET',
          url: GURL + TID, //获取对应编号鸡脖
          timeout: 2000,
          onload: function (xhr) {
              var html = domparser.parseFromString(xhr.responseText.replace('visibility:hidden',' '),'text/html') //处理带图像鸡脖显示并且转换为Document
              if(html){
                var tk = html.querySelector('.aat_container')
                if(tk && tk.hasChildNodes()){
                  tklist[i] = {
                    'id': TID,
                    'dom': tk
                  }// 存储到对应鸡脖列表
                  resolve(1)
                }else{
                  resolve(0)
                }
              }else{
                resolve(0)
              }
          },
          onerror: function (xhr) {
            console.log('失败', xhr,'id:',anchorTID.id - i);
            resolve(0) //失败
          },
          ontimeout: function (xhr) {
            console.log('超时:', xhr, 'id:',anchorTID.id - i);
            resolve(0) //失败
          },
          onabort:function (xhr) {
            console.log('中止:', xhr, 'id:',anchorTID.id - i);
            resolve(0) //失败
          }
        })
      },random(),index,type) //随机延迟防止被封
    })
  }
  if(type == 'Down'){
    Promise.all(PromiseList).then(Pres => {
      if(Pres.reduce(add) >= MIN){ //成功两条就算 因为有些删除的机博ID占用了但是没内容
        endTID -= STEP //更新已加载的最后一条ID
        tklist.forEach(tk => {
          var dom =$('<div class="GTK_talk_item" data-tid='+ tk.id +'></div>')
          DOM.append( dom.append(tk.dom)) //插入到最后
        })
        DOM.find('.gallery_item').css({'height':'375px','width':'563px'})
        DOM.find('.gallery').css({'height':'375px','width':'563px'})
        carouselsInit()
      }else{
        $('.GTK_talk_load_old').text('无法加载更多...')
      }
    });
  }else if(type == 'TOP'){
    Promise.all(PromiseList).then(Pres => {
      if(Pres.reduce(add) >= MIN){ //成功4条就算 因为有些删除的机博ID占用了但是没内容
        startTID += STEP//更新已加载的最新一条ID
        tklist.forEach(tk => {
          var dom =$('<div class="GTK_talk_item" data-tid='+ tk.id +'></div>')
          DOM.prepend( dom.append(tk.dom)) //插入到最前面
        })
        DOM.find('.gallery_item').css({'height':'375px','width':'563px'})
        DOM.find('.gallery').css({'height':'375px','width':'563px'})
        localStorage.setItem('GTK_TID_DATA', JSON.stringify({id:startTID})) //更新最新机博ID
        carouselsInit()
      }else{
        topTID.id = startTID //已经查到最顶部
        $('.GTK_talk_load_new').text('已加载到最新机博...')
      }
    });
  }else {
    Promise.all(PromiseList).then(Pres => {
      if(Pres.reduce(add) >= MIN){ //成功4条就算 因为有些删除的机博ID占用了但是没内容
        endTID = anchorTID.id - STEP //更新已加载的最后一条ID
        tklist.forEach(tk => {
          var dom =$('<div class="GTK_talk_item" data-tid='+ tk.id +'></div>')
          DOM.append(dom.append(tk.dom))
        })
        DOM.find('.gallery_item').css({'height':'375px','width':'563px'})
        DOM.find('.gallery').css({'height':'375px','width':'563px'})
        localStorage.setItem('GTK_TID_DATA', JSON.stringify(anchorTID))
        carouselsInit()
        loadData('TOP') //自动向前搜寻一次
      }else{
        $('.GTK_talk_list').text('呀!,出错了QAQ')
      }
    });
  }
}
function carouselsInit(){
  carouselsList = {} //初始化数据
  $('.slick-arrow').off().on('click',imgChange)
  $('.talk_actions').off().on('click',showComment)
}
function showComment(){
  var index, talkDOM;
  var Tid,html = '';
  TDOM =  $(this).parents('.GTK_talk_item')
  TDOM && (Tid = TDOM.attr('data-tid'))
  TDOM && (talkDOM = TDOM.find('.talk_actions'))
  talkDOM && (index = talkDOM.find('.u_plainButton')[1].innerText) //获取评论数量
  if(isNaN(index-0)){//评论是文本不是数字
    return
  }
  if(commitFlag[Tid] || loadFlag){
    return
  }
  loadFlag = true
  html+=GTcommit
  GM_xmlhttpRequest({
    method: 'GET',
    url: CURL+ Tid +'?include=comments', //获取对应编号鸡脖
    timeout: 2000,
    headers:{
      'Content-Type': 'application/vnd.api+json'
    },
    onload: function(xhr) {
      loadFlag = false
      var data = JSON.parse(xhr.responseText)
      for(var i = 0;i < data.included.length;i++){
        var item = data.included[i]
        html+= '<li class="GTK_commit_item '+ (item.attributes.depth > 0 ?"GTK_sub_commit":"GTK_main_commit") + '"  cid = '+item.id +'><span class="GTK_commit_img">G</span>' + item.attributes.body + '</li>'
      }
      html+='</ul></div>'
      talkDOM && talkDOM.append($(html))
      commitFlag[Tid] = true
    },
    onerror: function (xhr) {
      loadFlag = false
      console.log('拿取gitID失败', xhr,'id:',anchorTID.id - i);
    },
    ontimeout: function (xhr) {
      loadFlag = false
      console.log('拿取gitID超时:', xhr, 'id:',anchorTID.id - i);
    },
    onabort:function (xhr) {
      loadFlag = false
      console.log('拿取gitID中止:', xhr, 'id:',anchorTID.id - i);
    }
  })
}
function imgChange(){
  var button,target,TDOM,indexDOM,trackDOM;
  var nowNUM,allNUM,Tid;
  var resNUM = 1
  button = $(this);
  target = button.hasClass('slick-prev') ? 'prev' : 'next';
  TDOM =  $(this).parents('.GTK_talk_item')
  TDOM && (indexDOM = TDOM.find('.gallery_indexOf'))
  TDOM && (Tid = TDOM.attr('data-tid'))
  TDOM && (trackDOM = TDOM.find('.slick-track'))
  if(Tid && carouselsList[Tid] == undefined){ //第一次操作
    indexDOM && (nowNUM = indexDOM.find('span')[0].innerText) && (allNUM = indexDOM.find('span')[1].innerText)//拿取轮播图总数和当前编号
    carouselsList[Tid] = { //初始化
      'all':allNUM,
      'now':nowNUM,
    }
  }else{
    nowNUM = carouselsList[Tid].now;
    allNUM = carouselsList[Tid].all;
  }
  allNUM -= 0 //强制类型转换
  nowNUM -= 0 //强制类型转换
  target == 'prev' ?  nowNUM -= 1 : nowNUM += 1;
  nowNUM == 0 ? nowNUM = allNUM : nowNUM == allNUM + 1 ? nowNUM = 1 : nowNUM; //边界处理
  if( typeof nowNUM == 'number'){
    carouselsList[Tid].now = resNUM = nowNUM //更新数据
    indexDOM.find('span')[0].innerText = resNUM
    trackDOM.css('left','-'+resNUM+'00%')
  }else{
    console.log('轮播图错误TID:',Tid,'index:', nowNUM);
  }
  return 
}
onDocumentStart();
})();

最后更新于