Scriptable 对接 ☁ Cloudflare Workers/Pages Functions Metrics

Scriptable 对接 ☁ Cloudflare Workers/Pages Functions Metrics

效果如下

🟢支持自定义文案
🟢支持设置文字是否居中
🟢支持设置格子数
🟢支持设置格子是否为正方形

源码如下 CloudflareWorkersPagesFunctionsMetrics.js

// Variables used by Scriptable.
// These must be at the very top of the file. Do not edit.
// icon-color: orange; icon-glyph: cloud;

// 自定义一些文案
// https://www.namecheap.com/visual/font-generator/fancy
let title = '𝗖𝗹𝗼𝘂𝗱𝗳𝗹𝗮𝗿𝗲'
let workersLabel = '𝗪𝗼𝗿𝗸𝗲𝗿𝘀'
let pagesLabel = '𝗣𝗮𝗴𝗲𝘀'
let remainingLabel = '𝗥𝗲𝗺𝗮𝗶𝗻𝗶𝗻𝗴'
// 总量(免费账号为 100,000)
let total = 100000
// 如果知道就填上 不知道就会自动取 懒得做缓存了
let accountId = ''
// 默认取第几项账号
const accountIndex = 0
// 账号
const email = ''
// API Key
// https://dash.cloudflare.com/profile/api-tokens 里生成 Global API Key
const key = ''
// 数字是否居中
const isNumberCenter = false
// 下方格子数 自己根据机型尺寸调整
const boxsCount = 10
// 下方格子是否为正方形
const isSquare = true

const now = new Date()
now.setUTCHours(0, 0, 0, 0)
const startDate = now.toISOString()
const endDate = new Date().toISOString()

const logoIcon = ''
const pageIcon = ''
const workerIcon = ''

if (!accountId) {
  accountId = await getAccountId()
}

const { pagesSum = 0, workersSum = 0 } = await getSum()



let widget = new ListWidget()
const top_stack = widget.addStack();
top_stack.topAlignContent();
const container = top_stack.addStack();
container.layoutVertically();
const title_info = container.addStack();
title_info.layoutHorizontally();
title_info.centerAlignContent();
let req = new Request(logoIcon)
let logo = await req.loadImage()
const fuelpumpIcon = title_info.addImage(logo);
fuelpumpIcon.imageSize = scaleImage(fuelpumpIcon.image.size, 13);
title_info.addSpacer(5)
const t = title_info.addText(title)
t.font = Font.boldSystemFont(18);
t.textColor = Color.dynamic(Color.black(), Color.white())



widget.addSpacer(10)


const metricStack = widget.addStack()
metricStack.layoutHorizontally()

const pageContainer = metricStack.addStack()
pageContainer.layoutVertically()
const titleContainer = pageContainer.addStack()
titleContainer.layoutHorizontally()
titleContainer.centerAlignContent()
let pageIconReq = new Request(pageIcon)
let pageIconLoader = await pageIconReq.loadImage()
const pageLogo = titleContainer.addImage(pageIconLoader);
pageLogo.imageSize = scaleImage(pageLogo.image.size, 10);
titleContainer.addSpacer(2)
const titleLabel = titleContainer.addText(pagesLabel)
titleLabel.font = Font.systemFont(12)
titleLabel.textColor = Color.gray()
pageContainer.addSpacer(3)
const valueLabel = pageContainer.addText(centerNumber(formatNumber(pagesSum),8));
valueLabel.font = Font.boldRoundedSystemFont(15)

metricStack.addSpacer(20)

const workerContainer = metricStack.addStack()
workerContainer.layoutVertically()
const workerTitleContainer = workerContainer.addStack()
workerTitleContainer.layoutHorizontally()
workerTitleContainer.centerAlignContent()
let workerIconReq = new Request(workerIcon)
let workerIconLoader = await workerIconReq.loadImage()
const workerLogo = workerTitleContainer.addImage(workerIconLoader);
workerLogo.imageSize = scaleImage(workerLogo.image.size, 10);
workerTitleContainer.addSpacer(2)
const workerTitleLabel = workerTitleContainer.addText(workersLabel)
workerTitleLabel.font = Font.systemFont(12)
workerTitleLabel.textColor = Color.gray()
workerContainer.addSpacer(3)
const workerValueLabel = workerContainer.addText(centerNumber(formatNumber(workersSum),15));workerValueLabel.font = Font.boldRoundedSystemFont(15)


widget.addSpacer(5)
const remainingContainer = widget.addStack()
remainingContainer.layoutVertically()
const remainingTitleContainer = remainingContainer.addStack()
remainingTitleContainer.layoutHorizontally()
remainingTitleContainer.centerAlignContent()
let remainingIcon = remainingTitleContainer.addImage(SFSymbol.named("network").image);
remainingIcon.imageSize = scaleImage(remainingIcon.image.size, 10)
remainingIcon.tintColor = Color.orange()
remainingTitleContainer.addSpacer(2)
const remainingTitleLabel = remainingTitleContainer.addText(remainingLabel)
remainingTitleLabel.font = Font.systemFont(12)
remainingTitleLabel.textColor = Color.gray()
remainingTitleContainer.addSpacer(10)
const clockIcon = remainingTitleContainer.addImage(SFSymbol.named("clock").image)
clockIcon.imageSize = scaleImage(clockIcon.image.size, 9)
clockIcon.tintColor = Color.gray()
remainingTitleContainer.addSpacer(2);
const date = new Date()
const month = String(date.getMonth() + 1).padStart(2, '0');
const day = String(date.getDate()).padStart(2, '0');
const dateLabel = remainingTitleContainer.addText(`${month}-${day}`)
dateLabel.font = Font.boldRoundedSystemFont(10)
dateLabel.textColor = Color.gray()
dateLabel.rightAlignText()
remainingContainer.addSpacer(3)
const remainingValueLabel = remainingContainer.addText(centerNumber(formatNumber(total - pagesSum - workersSum) + ' / ' + (formatNumber(total)), 24))
remainingValueLabel.font = Font.boldRoundedSystemFont(15)
remainingContainer.addSpacer(5)

const progressContainer = widget.addStack()
progressContainer.layoutHorizontally()
progressContainer.centerAlignContent()
const remaining = total - pagesSum - workersSum;
const percent = (remaining / total) * 100;
const remainingNum = Math.ceil((percent / 100) * boxsCount);
const isHalfFilled = (remaining % total) != 0

const boxs = []
const boxIcons = []
for(let i = 0 ; i < boxsCount; i++) {
  if (isSquare) {
    boxIcons[i] = SFSymbol.named("square").image;
    for(let j = 0; j < remainingNum; j++) {
      boxIcons[j] = SFSymbol.named("square.fill").image;
    }
    if (isHalfFilled) {
      boxIcons[remainingNum - 1] = SFSymbol.named("square.lefthalf.fill").image;
    }
  } else {
    boxIcons[i] = SFSymbol.named("rectangle.portrait").image;
    for(let j = 0; j < remainingNum; j++) {
      boxIcons[j] = SFSymbol.named("rectangle.portrait.fill").image;
    }
    if (isHalfFilled) {
      boxIcons[remainingNum - 1] = SFSymbol.named("rectangle.lefthalf.fill").image;
    }
  }
}
for(let i = 0; i < boxsCount; i++) {
  boxs[i] = progressContainer.addImage(boxIcons[i]);
  boxs[i].imageSize = scaleImage(boxs[i].image.size, 13);
  boxs[i].tintColor = Color.orange();
}

Script.setWidget(widget)
if (!config.runsInWidget) {
  await widget.presentSmall()
}
Script.complete()

// 定义一个方法,将数字居中显示
function centerNumber(num, totalWidth) {
  if(!isNumberCenter) return num
  // 将数字转换为字符串
  let numStr = num.toString();

  // 计算数字需要的空格数
  let numSpaces = Math.max(0, Math.floor((totalWidth - numStr.length) / 2));

  // 构建居中显示的字符串
  let centeredNum = ' '.repeat(numSpaces) + numStr + ' '.repeat(numSpaces);

  // 如果总宽度为奇数,需要再加一个空格来保持整体居中
  if (centeredNum.length < totalWidth) {
      centeredNum += ' ';
  }

  return centeredNum;
}

async function getAccountId() {
  const req = new Request(`https://api.cloudflare.com/client/v4/accounts`)
  req.method = 'GET'
  req.headers = {
    'content-type': 'application/json',
    'X-AUTH-EMAIL': email,
    'X-AUTH-KEY': key,
  }
  const res = await req.loadJSON()
  // console.log(res)
  const name = res?.result?.[accountIndex]?.name
  const id = res?.result?.[accountIndex]?.id
  console.log(`默认取第 ${accountIndex} 项\n名称: ${name}, 账号 ID: ${id}`)
  if (!id) throw new Error('找不到账号 ID')
  return id
}

async function getSum() {
  const req = new Request(`https://api.cloudflare.com/client/v4/graphql`)
  req.method = 'POST'
  req.headers = {
    'content-type': 'application/json',
    'X-AUTH-EMAIL': email,
    'X-AUTH-KEY': key,
  }
  req.body = JSON.stringify({
    query: `query getBillingMetrics($accountId: string!, $filter: AccountWorkersInvocationsAdaptiveFilter_InputObject) {
      viewer {
        accounts(filter: {accountTag: $accountId}) {
          pagesFunctionsInvocationsAdaptiveGroups(limit: 1000, filter: $filter) {
            sum {
              requests
            }
          }
          workersInvocationsAdaptive(limit: 10000, filter: $filter) {
            sum {
              requests
            }
          }
        }
      }
    }`,
    variables: {
      accountId,
      filter:{ datetime_geq: startDate, datetime_leq: endDate}
    },
  })
  const res = await req.loadJSON()
  // console.log(res)
  const pagesFunctionsInvocationsAdaptiveGroups = res?.data?.viewer?.accounts?.[accountIndex]?.pagesFunctionsInvocationsAdaptiveGroups
  const workersInvocationsAdaptive = res?.data?.viewer?.accounts?.[accountIndex]?.workersInvocationsAdaptive
  if (!pagesFunctionsInvocationsAdaptiveGroups && !workersInvocationsAdaptive) throw new Error('找不到数据')
  const pagesSum = pagesFunctionsInvocationsAdaptiveGroups.reduce((a, b) => a + b?.sum.requests, 0) 
  const workersSum = workersInvocationsAdaptive.reduce((a, b) => a + b?.sum.requests, 0) 
  console.log(`范围: ${startDate} ~ ${endDate}\n默认取第 ${accountIndex} 项`)

  return { pagesSum, workersSum }
}


// 缩放图片
function scaleImage(imageSize, height) {
    scale = height / imageSize.height
    return new Size(scale * imageSize.width, height)
}
function formatNumber(num) {
  // 如果数字小于1万,直接返回原始数字
  if (num < 1000) {
      return num.toString();
  }
  // 如果数字大于等于1万,进行格式化
  const suffixes = ['', 'k', 'm', 'b', 't']; // 数字后缀,可根据需要扩展
  let suffixIndex = 0;
  let formattedNum = num;

  while (formattedNum >= 1000 && suffixIndex < suffixes.length - 1) {
      formattedNum /= 1000;
      suffixIndex++;
  }

  // 使用 toFixed(1) 将数字保留一位小数,然后连接上对应的后缀
  return formattedNum.toFixed(1) + suffixes[suffixIndex];
}