mirror of
https://github.com/misskey-dev/summaly.git
synced 2025-08-07 08:43:56 +09:00
Compare commits
31 Commits
Author | SHA1 | Date | |
---|---|---|---|
7e94c26e9f | |||
2f5ab74f52 | |||
7902ded327 | |||
67909d360d | |||
0606cd3fae | |||
52f5692199 | |||
3ecb30aefc | |||
c45b3a1c63 | |||
81de408b54 | |||
38000dd462 | |||
d7f6a88aee | |||
d2a3e07205 | |||
d2d8db4994 | |||
089a0ad8e8 | |||
77dd5654bb | |||
2d63e2a006 | |||
c7d71a9ec2 | |||
994f420b46 | |||
5a3321a04f | |||
1bab7afee6 | |||
441e8c22f9 | |||
376bba9c61 | |||
028b2fed2f | |||
90d5d0f33b | |||
9e955d8d04 | |||
a36652c859 | |||
eab3766db9 | |||
51f3870e1f | |||
5684f116c9 | |||
709ca51b6c | |||
f4a180907c |
@ -5,6 +5,11 @@ indent_style = tab
|
|||||||
indent_size = 2
|
indent_size = 2
|
||||||
charset = utf-8
|
charset = utf-8
|
||||||
insert_final_newline = true
|
insert_final_newline = true
|
||||||
|
end_of_line = lf
|
||||||
|
trim_trailing_whitespace = true
|
||||||
|
|
||||||
[*.json]
|
[*.md]
|
||||||
|
trim_trailing_whitespace = false
|
||||||
|
|
||||||
|
[*.{yml,yaml}]
|
||||||
indent_style = space
|
indent_style = space
|
||||||
|
122
.eslintrc.cjs
Normal file
122
.eslintrc.cjs
Normal file
@ -0,0 +1,122 @@
|
|||||||
|
module.exports = {
|
||||||
|
root: true,
|
||||||
|
parser: '@typescript-eslint/parser',
|
||||||
|
parserOptions: {
|
||||||
|
tsconfigRootDir: __dirname,
|
||||||
|
project: ['./tsconfig.json'],
|
||||||
|
},
|
||||||
|
plugins: [
|
||||||
|
'@typescript-eslint',
|
||||||
|
'import'
|
||||||
|
],
|
||||||
|
extends: [
|
||||||
|
'eslint:recommended',
|
||||||
|
'plugin:@typescript-eslint/recommended',
|
||||||
|
'plugin:import/recommended',
|
||||||
|
'plugin:import/typescript'
|
||||||
|
],
|
||||||
|
rules: {
|
||||||
|
'indent': ['warn', 'tab', {
|
||||||
|
'SwitchCase': 1,
|
||||||
|
'MemberExpression': 1,
|
||||||
|
'flatTernaryExpressions': true,
|
||||||
|
'ArrayExpression': 'first',
|
||||||
|
'ObjectExpression': 'first',
|
||||||
|
}],
|
||||||
|
'eol-last': ['error', 'always'],
|
||||||
|
'semi': ['error', 'always'],
|
||||||
|
'semi-spacing': ['error', { 'before': false, 'after': true }],
|
||||||
|
'quotes': ['warn', 'single'],
|
||||||
|
'comma-dangle': ['warn', 'always-multiline'],
|
||||||
|
'comma-spacing': ['error', { 'before': false, 'after': true }],
|
||||||
|
'array-bracket-spacing': ['error', 'never'],
|
||||||
|
'keyword-spacing': ['error', {
|
||||||
|
'before': true,
|
||||||
|
'after': true,
|
||||||
|
}],
|
||||||
|
'key-spacing': ['error', {
|
||||||
|
'beforeColon': false,
|
||||||
|
'afterColon': true,
|
||||||
|
}],
|
||||||
|
'arrow-spacing': ['error', {
|
||||||
|
'before': true,
|
||||||
|
'after': true,
|
||||||
|
}],
|
||||||
|
'brace-style': ['error', '1tbs', {
|
||||||
|
'allowSingleLine': true,
|
||||||
|
}],
|
||||||
|
'padded-blocks': ['error', 'never'],
|
||||||
|
/* TODO: path aliasを使わないとwarnする
|
||||||
|
'no-restricted-imports': ['warn', {
|
||||||
|
'patterns': [
|
||||||
|
]
|
||||||
|
}],
|
||||||
|
*/
|
||||||
|
'eqeqeq': ['error', 'always', { 'null': 'ignore' }],
|
||||||
|
'no-multi-spaces': ['error'],
|
||||||
|
'no-var': ['error'],
|
||||||
|
'prefer-arrow-callback': ['error'],
|
||||||
|
'no-throw-literal': ['error'],
|
||||||
|
'no-param-reassign': ['warn'],
|
||||||
|
'no-constant-condition': ['warn'],
|
||||||
|
'no-empty-pattern': ['warn'],
|
||||||
|
'no-async-promise-executor': ['off'],
|
||||||
|
'no-useless-escape': ['off'],
|
||||||
|
'no-multiple-empty-lines': ['error', { 'max': 1 }],
|
||||||
|
'no-control-regex': ['warn'],
|
||||||
|
'no-empty': ['warn'],
|
||||||
|
'no-inner-declarations': ['off'],
|
||||||
|
'no-sparse-arrays': ['off'],
|
||||||
|
'nonblock-statement-body-position': ['error', 'beside'],
|
||||||
|
'object-curly-spacing': ['error', 'always'],
|
||||||
|
'space-infix-ops': ['error'],
|
||||||
|
'space-before-blocks': ['error', 'always'],
|
||||||
|
'padding-line-between-statements': [
|
||||||
|
'error',
|
||||||
|
{ 'blankLine': 'always', 'prev': 'function', 'next': '*' },
|
||||||
|
{ 'blankLine': 'always', 'prev': '*', 'next': 'function' },
|
||||||
|
],
|
||||||
|
"lines-between-class-members": "off",
|
||||||
|
/* typescript-eslint では enforce に対応してないっぽい
|
||||||
|
'@typescript-eslint/lines-between-class-members': ['error', {
|
||||||
|
enforce: [{
|
||||||
|
blankLine: 'always',
|
||||||
|
prev: 'method',
|
||||||
|
next: '*',
|
||||||
|
}]
|
||||||
|
}],
|
||||||
|
*/
|
||||||
|
'@typescript-eslint/func-call-spacing': ['error', 'never'],
|
||||||
|
'@typescript-eslint/no-explicit-any': ['warn'],
|
||||||
|
'@typescript-eslint/no-unused-vars': ['warn'],
|
||||||
|
'@typescript-eslint/no-unnecessary-condition': ['warn'],
|
||||||
|
'@typescript-eslint/no-var-requires': ['warn'],
|
||||||
|
'@typescript-eslint/no-inferrable-types': ['warn'],
|
||||||
|
'@typescript-eslint/no-empty-function': ['off'],
|
||||||
|
'@typescript-eslint/no-non-null-assertion': ['warn'],
|
||||||
|
'@typescript-eslint/explicit-function-return-type': ['off'],
|
||||||
|
'@typescript-eslint/no-misused-promises': ['error', {
|
||||||
|
'checksVoidReturn': false,
|
||||||
|
}],
|
||||||
|
'@typescript-eslint/consistent-type-imports': 'off',
|
||||||
|
'@typescript-eslint/prefer-nullish-coalescing': [
|
||||||
|
'warn',
|
||||||
|
],
|
||||||
|
'@typescript-eslint/naming-convention': [
|
||||||
|
'error',
|
||||||
|
{
|
||||||
|
"selector": "typeLike",
|
||||||
|
"format": ["PascalCase"]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"selector": "typeParameter",
|
||||||
|
"format": []
|
||||||
|
}
|
||||||
|
],
|
||||||
|
'import/no-unresolved': ['off'],
|
||||||
|
'import/no-default-export': ['warn'],
|
||||||
|
'import/order': ['warn', {
|
||||||
|
'groups': ['builtin', 'external', 'internal', 'parent', 'sibling', 'index', 'object', 'type'],
|
||||||
|
}]
|
||||||
|
},
|
||||||
|
};
|
31
.github/workflows/lint.yml
vendored
Normal file
31
.github/workflows/lint.yml
vendored
Normal file
@ -0,0 +1,31 @@
|
|||||||
|
name: Lint
|
||||||
|
|
||||||
|
on: [push, pull_request]
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
lint:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
|
||||||
|
strategy:
|
||||||
|
matrix:
|
||||||
|
node-version: [20.10.0]
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- uses: actions/checkout@v4
|
||||||
|
- name: Install pnpm
|
||||||
|
uses: pnpm/action-setup@v2
|
||||||
|
with:
|
||||||
|
version: 8
|
||||||
|
run_install: false
|
||||||
|
- name: Use Node.js ${{ matrix.node-version }}
|
||||||
|
uses: actions/setup-node@v4
|
||||||
|
with:
|
||||||
|
node-version: ${{ matrix.node-version }}
|
||||||
|
cache: 'pnpm'
|
||||||
|
- name: Install
|
||||||
|
run: |
|
||||||
|
corepack enable
|
||||||
|
pnpm i --frozen-lockfile
|
||||||
|
- name: eslint
|
||||||
|
run: |
|
||||||
|
pnpm eslint
|
36
.github/workflows/npm-publish.yml
vendored
Normal file
36
.github/workflows/npm-publish.yml
vendored
Normal file
@ -0,0 +1,36 @@
|
|||||||
|
name: Publish Node.js Package
|
||||||
|
|
||||||
|
on:
|
||||||
|
release:
|
||||||
|
types: [created]
|
||||||
|
|
||||||
|
workflow_dispatch:
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
publish:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
|
||||||
|
strategy:
|
||||||
|
matrix:
|
||||||
|
node-version: [20.10.0]
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- uses: actions/checkout@v4
|
||||||
|
- name: Install pnpm
|
||||||
|
uses: pnpm/action-setup@v2
|
||||||
|
with:
|
||||||
|
version: 8
|
||||||
|
run_install: false
|
||||||
|
- name: Use Node.js ${{ matrix.node-version }}
|
||||||
|
uses: actions/setup-node@v4
|
||||||
|
with:
|
||||||
|
node-version: ${{ matrix.node-version }}
|
||||||
|
cache: 'pnpm'
|
||||||
|
- name: Publish package
|
||||||
|
run: |
|
||||||
|
corepack enable
|
||||||
|
pnpm i --frozen-lockfile
|
||||||
|
pnpm build
|
||||||
|
pnpm publish --access public --no-git-checks --provenance
|
||||||
|
env:
|
||||||
|
NODE_AUTH_TOKEN: ${{ secrets.NODE_AUTH_TOKEN }}
|
34
.github/workflows/test.yml
vendored
Normal file
34
.github/workflows/test.yml
vendored
Normal file
@ -0,0 +1,34 @@
|
|||||||
|
name: Test
|
||||||
|
|
||||||
|
on: [push, pull_request]
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
test:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
|
||||||
|
strategy:
|
||||||
|
matrix:
|
||||||
|
node-version: [20.10.0]
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- uses: actions/checkout@v4
|
||||||
|
- name: Install pnpm
|
||||||
|
uses: pnpm/action-setup@v2
|
||||||
|
with:
|
||||||
|
version: 8
|
||||||
|
run_install: false
|
||||||
|
- name: Use Node.js ${{ matrix.node-version }}
|
||||||
|
uses: actions/setup-node@v4
|
||||||
|
with:
|
||||||
|
node-version: ${{ matrix.node-version }}
|
||||||
|
cache: 'pnpm'
|
||||||
|
- name: Install
|
||||||
|
run: |
|
||||||
|
corepack enable
|
||||||
|
pnpm i --frozen-lockfile
|
||||||
|
- name: Build
|
||||||
|
run: |
|
||||||
|
pnpm build
|
||||||
|
- name: Test
|
||||||
|
run: |
|
||||||
|
pnpm test
|
1
.gitignore
vendored
1
.gitignore
vendored
@ -1,2 +1,3 @@
|
|||||||
/node_modules
|
/node_modules
|
||||||
|
/built
|
||||||
npm-debug.log
|
npm-debug.log
|
||||||
|
@ -5,5 +5,5 @@
|
|||||||
npm-debug.log
|
npm-debug.log
|
||||||
gulpfile.js
|
gulpfile.js
|
||||||
tsconfig.json
|
tsconfig.json
|
||||||
tslint.json
|
.eslintrc.cjs
|
||||||
.editorconfig
|
.editorconfig
|
||||||
|
14
.travis.yml
14
.travis.yml
@ -1,14 +0,0 @@
|
|||||||
# travis file
|
|
||||||
# https://docs.travis-ci.com/user/customizing-the-build
|
|
||||||
|
|
||||||
language: node_js
|
|
||||||
|
|
||||||
node_js:
|
|
||||||
- 7.5.0
|
|
||||||
|
|
||||||
before_script:
|
|
||||||
- npm run build
|
|
||||||
|
|
||||||
cache:
|
|
||||||
directories:
|
|
||||||
- node_modules
|
|
36
CHANGELOG.md
36
CHANGELOG.md
@ -1,3 +1,39 @@
|
|||||||
|
5.0.0 / 2023-12-30
|
||||||
|
------------------
|
||||||
|
* support `<link rel="alternate" type="application/activitypub+json" href="{href}">` https://github.com/misskey-dev/summaly/pull/10, https://github.com/misskey-dev/summaly/pull/11
|
||||||
|
* 結果の`activityPub`プロパティでherfの内容を取得できます
|
||||||
|
* branch.ioを用いたディープリンク(spotify.link)などでパースに失敗する問題を修正 https://github.com/misskey-dev/summaly/pull/13
|
||||||
|
* Twitter Cardが読めていない問題を修正 https://github.com/misskey-dev/summaly/pull/15
|
||||||
|
* 'mixi:content-rating'をsensitive判定で見ることで、dlsiteなどでセンシティブ情報を得れるように https://github.com/misskey-dev/summaly/pull/16
|
||||||
|
* sitenameをURLから生成する場合、ポートを含むように (URL.hostname → URL.host)
|
||||||
|
* `Summary`型に`url`プロパティを追加した`SummalyResult`型をexportするように
|
||||||
|
* `IPlugin`インターフェースを`SummalyPlugin`に改称
|
||||||
|
|
||||||
|
4.0.2 / 2023-04-20
|
||||||
|
------------------
|
||||||
|
* YouTubeをフルスクリーンにできない問題を修正
|
||||||
|
|
||||||
|
4.0.1 / 2023-03-16
|
||||||
|
------------------
|
||||||
|
* oEmbedの読み込みでエラーが発生した際は、エラーにせずplayerの中身をnullにするように
|
||||||
|
|
||||||
|
4.0.0 / 2023-03-14
|
||||||
|
------------------
|
||||||
|
* oEmbed type=richの制限的なサポート
|
||||||
|
* プラグインの引数がWHATWG URLになりました
|
||||||
|
|
||||||
|
3.0.4 / 2023-02-12
|
||||||
|
------------------
|
||||||
|
* 不要な依存関係を除去
|
||||||
|
|
||||||
|
3.0.3 / 2023-02-12
|
||||||
|
------------------
|
||||||
|
* agentが指定されている(もしくはagentが空のオブジェクトの)場合はプライベートIPのリクエストを許可
|
||||||
|
|
||||||
|
3.0.2 / 2023-02-12
|
||||||
|
------------------
|
||||||
|
* Fastifyのルーティングを'/url'から'/'に
|
||||||
|
|
||||||
3.0.1 / 2023-02-12
|
3.0.1 / 2023-02-12
|
||||||
------------------
|
------------------
|
||||||
* ES Moduleになりました
|
* ES Moduleになりました
|
||||||
|
2
LICENSE
2
LICENSE
@ -1,6 +1,6 @@
|
|||||||
The MIT License (MIT)
|
The MIT License (MIT)
|
||||||
|
|
||||||
Copyright (c) 2016-2019 syuilo
|
Copyright (c) 2016-2024 syuilo
|
||||||
|
|
||||||
Permission is hereby granted, free of charge, to any person obtaining a copy
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
of this software and associated documentation files (the "Software"), to deal
|
of this software and associated documentation files (the "Software"), to deal
|
||||||
|
133
README.md
133
README.md
@ -1,6 +1,7 @@
|
|||||||
summaly
|
summaly
|
||||||
================================================================
|
================================================================
|
||||||
|
|
||||||
|
[![][npm-badge]][npm-link]
|
||||||
[![][mit-badge]][mit]
|
[![][mit-badge]][mit]
|
||||||
[![][himawari-badge]][himasaku]
|
[![][himawari-badge]][himasaku]
|
||||||
[![][sakurako-badge]][himasaku]
|
[![][sakurako-badge]][himasaku]
|
||||||
@ -8,7 +9,7 @@ summaly
|
|||||||
Installation
|
Installation
|
||||||
----------------------------------------------------------------
|
----------------------------------------------------------------
|
||||||
```
|
```
|
||||||
npm install git+https://github.com/misskey-dev/summaly.git
|
npm install @misskey-dev/summaly
|
||||||
```
|
```
|
||||||
|
|
||||||
Usage
|
Usage
|
||||||
@ -21,8 +22,8 @@ import { summaly } from 'summaly';
|
|||||||
summaly(url[, opts])
|
summaly(url[, opts])
|
||||||
```
|
```
|
||||||
|
|
||||||
As Fastify plugin:
|
As Fastify plugin:
|
||||||
(will listen `GET` of `/url`)
|
(will listen `GET` of `/`)
|
||||||
|
|
||||||
```javascript
|
```javascript
|
||||||
import Summaly from 'summaly';
|
import Summaly from 'summaly';
|
||||||
@ -40,71 +41,115 @@ npm run build
|
|||||||
npm run serve
|
npm run serve
|
||||||
```
|
```
|
||||||
|
|
||||||
### Options
|
#### opts (SummalyOptions)
|
||||||
|
|
||||||
| Property | Type | Description | Default |
|
| Property | Type | Description | Default |
|
||||||
| :------------------ | :--------------------- | :----------------------- | :------ |
|
| :------------------ | :--------------------- | :------------------------------ | :------ |
|
||||||
| **followRedirects** | *boolean* | Whether follow redirects | `true` |
|
| **lang** | *string* | Accept-Language for the request | `null` |
|
||||||
| **plugins** | *plugin[]* (see below) | Custom plugins | `null` |
|
| **followRedirects** | *boolean* | Whether follow redirects | `true` |
|
||||||
|
| **plugins** | *plugin[]* (see below) | Custom plugins | `null` |
|
||||||
|
| **agent** | *Got.Agents* | Custom HTTP agent (see below) | `null` |
|
||||||
|
|
||||||
#### Plugin
|
#### Plugin
|
||||||
|
|
||||||
``` typescript
|
``` typescript
|
||||||
interface IPlugin {
|
interface SummalyPlugin {
|
||||||
test: (url: URL.Url) => boolean;
|
test: (url: URL) => boolean;
|
||||||
summarize: (url: URL.Url) => Promise<Summary>;
|
summarize: (url: URL) => Promise<Summary>;
|
||||||
}
|
}
|
||||||
```
|
```
|
||||||
|
|
||||||
|
urls are WHATWG URL since v4.
|
||||||
|
|
||||||
|
#### Custom HTTP agent for proxy
|
||||||
|
You can specify agents to be passed to Got for proxy use, etc.
|
||||||
|
https://github.com/sindresorhus/got/blob/v12.6.0/documentation/tips.md#proxying
|
||||||
|
|
||||||
|
**⚠️If you set some agent, local IP rejecting will not work.⚠️**
|
||||||
|
(Summaly usually rejects local IPs.)
|
||||||
|
|
||||||
|
(Summaly currently does not support http2.)
|
||||||
|
|
||||||
### Returns
|
### Returns
|
||||||
|
|
||||||
A Promise of an Object that contains properties below:
|
A Promise of an Object that contains properties below:
|
||||||
|
|
||||||
※ Almost all values are nullable. player shoud not be null.
|
※ Almost all values are nullable. player should not be null.
|
||||||
|
|
||||||
#### Root
|
#### SummalyResult
|
||||||
|
|
||||||
| Property | Type | Description |
|
| Property | Type | Description |
|
||||||
| :-------------- | :------- | :--------------------------------------- |
|
| :-------------- | :------- | :------------------------------------------ |
|
||||||
| **description** | *string* | The description of the web page |
|
| **title** | *string* \| *null* | The title of the web page |
|
||||||
| **icon** | *string* | The url of the icon of the web page |
|
| **icon** | *string* \| *null* | The url of the icon of the web page |
|
||||||
| **sitename** | *string* | The name of the web site |
|
| **description** | *string* \| *null* | The description of the web page |
|
||||||
| **thumbnail** | *string* | The url of the thumbnail of the web page |
|
| **thumbnail** | *string* \| *null* | The url of the thumbnail of the web page |
|
||||||
| **player** | *Player* | The player of the web page |
|
| **sitename** | *string* \| *null* | The name of the web site |
|
||||||
| **title** | *string* | The title of the web page |
|
| **player** | *Player* | The player of the web page |
|
||||||
| **url** | *string* | The url of the web page |
|
| **sensitive** | *boolean* | Whether the url is sensitive |
|
||||||
|
| **activityPub** | *string* \| *null* | The url of the ActivityPub representation of that web page |
|
||||||
|
| **url** | *string* | The url of the web page |
|
||||||
|
|
||||||
|
#### Summary
|
||||||
|
|
||||||
|
`Omit<SummalyResult, "url">`
|
||||||
|
|
||||||
#### Player
|
#### Player
|
||||||
|
|
||||||
| Property | Type | Description |
|
| Property | Type | Description |
|
||||||
| :-------------- | :------- | :--------------------------------------- |
|
| :-------------- | :--------- | :---------------------------------------------- |
|
||||||
| **url** | *string* | The url of the player |
|
| **url** | *string* \| *null* | The url of the player |
|
||||||
| **width** | *number* | The width of the player |
|
| **width** | *number* \| *null* | The width of the player |
|
||||||
| **height** | *number* | The height of the player |
|
| **height** | *number* \| *null* | The height of the player |
|
||||||
|
| **allow** | *string[]* | The names of the allowed permissions for iframe |
|
||||||
|
|
||||||
|
Currently the possible items in `allow` are:
|
||||||
|
|
||||||
|
* `autoplay`
|
||||||
|
* `clipboard-write`
|
||||||
|
* `fullscreen`
|
||||||
|
* `encrypted-media`
|
||||||
|
* `picture-in-picture`
|
||||||
|
* `web-share`
|
||||||
|
|
||||||
|
See [Permissions Policy](https://developer.mozilla.org/en-US/docs/Web/HTTP/Permissions_Policy) in MDN for details of them.
|
||||||
|
|
||||||
### Example
|
### Example
|
||||||
|
|
||||||
``` javascript
|
```javascript
|
||||||
import { summaly } from 'summaly';
|
import { summaly } from 'summaly';
|
||||||
|
|
||||||
const summary = await summaly('https://www.youtube.com/watch?v=NMIEAhH_fTU');
|
const summary = await summaly('https://www.youtube.com/watch?v=NMIEAhH_fTU');
|
||||||
|
|
||||||
console.log(summary); // will be ... ↓
|
console.log(summary);
|
||||||
/*
|
```
|
||||||
|
|
||||||
|
will be ... ↓
|
||||||
|
|
||||||
|
```json
|
||||||
{
|
{
|
||||||
title: '【楽曲試聴】「Stage Bye Stage」(歌:島村卯月、渋谷凛、本田未央)',
|
"title": "【アイドルマスター】「Stage Bye Stage」(歌:島村卯月、渋谷凛、本田未央)",
|
||||||
icon: 'https://s.ytimg.com/yts/img/favicon-vfl8qSV2F.ico',
|
"icon": "https://www.youtube.com/s/desktop/28b0985e/img/favicon.ico",
|
||||||
description: 'http://columbia.jp/idolmaster/ 2018年7月18日発売予定 THE IDOLM@STER CINDERELLA GIRLS CG STAR LIVE Stage Bye Stage 歌:島村卯月、渋谷凛、本田未央 COCC-17495[CD1枚組] ¥1,200+税 収録内容 Tr...',
|
"description": "Website▶https://columbia.jp/idolmaster/Playlist▶https://www.youtube.com/playlist?list=PL83A2998CF3BBC86D2018年7月18日発売予定THE IDOLM@STER CINDERELLA GIRLS CG STAR...",
|
||||||
thumbnail: 'https://i.ytimg.com/vi/NMIEAhH_fTU/maxresdefault.jpg',
|
"thumbnail": "https://i.ytimg.com/vi/NMIEAhH_fTU/maxresdefault.jpg",
|
||||||
player: {
|
"player": {
|
||||||
url: 'https://www.youtube.com/embed/NMIEAhH_fTU',
|
"url": "https://www.youtube.com/embed/NMIEAhH_fTU?feature=oembed",
|
||||||
width: 1280,
|
"width": 200,
|
||||||
height: 720
|
"height": 113,
|
||||||
|
"allow": [
|
||||||
|
"autoplay",
|
||||||
|
"clipboard-write",
|
||||||
|
"encrypted-media",
|
||||||
|
"picture-in-picture",
|
||||||
|
"web-share",
|
||||||
|
"fullscreen",
|
||||||
|
]
|
||||||
},
|
},
|
||||||
sitename: 'YouTube',
|
"sitename": "YouTube",
|
||||||
url: 'https://www.youtube.com/watch?v=NMIEAhH_fTU'
|
"sensitive": false,
|
||||||
|
"activityPub": null,
|
||||||
|
"url": "https://www.youtube.com/watch?v=NMIEAhH_fTU"
|
||||||
}
|
}
|
||||||
*/
|
|
||||||
```
|
```
|
||||||
|
|
||||||
Testing
|
Testing
|
||||||
@ -115,12 +160,10 @@ License
|
|||||||
----------------------------------------------------------------
|
----------------------------------------------------------------
|
||||||
[MIT](LICENSE)
|
[MIT](LICENSE)
|
||||||
|
|
||||||
[npm-link]: https://www.npmjs.com/package/summaly
|
|
||||||
[npm-badge]: https://img.shields.io/npm/v/summaly.svg?style=flat-square
|
|
||||||
[mit]: http://opensource.org/licenses/MIT
|
[mit]: http://opensource.org/licenses/MIT
|
||||||
[mit-badge]: https://img.shields.io/badge/license-MIT-444444.svg?style=flat-square
|
[mit-badge]: https://img.shields.io/badge/license-MIT-444444.svg?style=flat-square
|
||||||
[travis-link]: https://travis-ci.org/syuilo/summaly
|
|
||||||
[travis-badge]: http://img.shields.io/travis/syuilo/summaly.svg?style=flat-square
|
|
||||||
[himasaku]: https://himasaku.net
|
[himasaku]: https://himasaku.net
|
||||||
[himawari-badge]: https://img.shields.io/badge/%E5%8F%A4%E8%B0%B7-%E5%90%91%E6%97%A5%E8%91%B5-1684c5.svg?style=flat-square
|
[himawari-badge]: https://img.shields.io/badge/%E5%8F%A4%E8%B0%B7-%E5%90%91%E6%97%A5%E8%91%B5-1684c5.svg?style=flat-square
|
||||||
[sakurako-badge]: https://img.shields.io/badge/%E5%A4%A7%E5%AE%A4-%E6%AB%BB%E5%AD%90-efb02a.svg?style=flat-square
|
[sakurako-badge]: https://img.shields.io/badge/%E5%A4%A7%E5%AE%A4-%E6%AB%BB%E5%AD%90-efb02a.svg?style=flat-square
|
||||||
|
[npm-link]: https://www.npmjs.com/package/@misskey-dev/summaly
|
||||||
|
[npm-badge]: https://img.shields.io/npm/v/@misskey-dev/summaly.svg?style=flat-square
|
||||||
|
4
built/general.d.ts
vendored
4
built/general.d.ts
vendored
@ -1,4 +0,0 @@
|
|||||||
import * as URL from 'node:url';
|
|
||||||
import Summary from './summary.js';
|
|
||||||
declare const _default: (url: URL.Url, lang?: string | null) => Promise<Summary | null>;
|
|
||||||
export default _default;
|
|
@ -1,98 +0,0 @@
|
|||||||
import * as URL from 'node:url';
|
|
||||||
import clip from './utils/clip.js';
|
|
||||||
import cleanupTitle from './utils/cleanup-title.js';
|
|
||||||
import { decode as decodeHtml } from 'html-entities';
|
|
||||||
import { head, scpaping } from './utils/got.js';
|
|
||||||
export default async (url, lang = null) => {
|
|
||||||
if (lang && !lang.match(/^[\w-]+(\s*,\s*[\w-]+)*$/))
|
|
||||||
lang = null;
|
|
||||||
const res = await scpaping(url.href, { lang: lang || undefined });
|
|
||||||
const $ = res.$;
|
|
||||||
const twitterCard = $('meta[property="twitter:card"]').attr('content');
|
|
||||||
let title = $('meta[property="og:title"]').attr('content') ||
|
|
||||||
$('meta[property="twitter:title"]').attr('content') ||
|
|
||||||
$('title').text();
|
|
||||||
if (title === undefined || title === null) {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
title = clip(decodeHtml(title), 100);
|
|
||||||
let image = $('meta[property="og:image"]').attr('content') ||
|
|
||||||
$('meta[property="twitter:image"]').attr('content') ||
|
|
||||||
$('link[rel="image_src"]').attr('href') ||
|
|
||||||
$('link[rel="apple-touch-icon"]').attr('href') ||
|
|
||||||
$('link[rel="apple-touch-icon image_src"]').attr('href');
|
|
||||||
image = image ? URL.resolve(url.href, image) : null;
|
|
||||||
const playerUrl = (twitterCard !== 'summary_large_image' && $('meta[property="twitter:player"]').attr('content')) ||
|
|
||||||
(twitterCard !== 'summary_large_image' && $('meta[name="twitter:player"]').attr('content')) ||
|
|
||||||
$('meta[property="og:video"]').attr('content') ||
|
|
||||||
$('meta[property="og:video:secure_url"]').attr('content') ||
|
|
||||||
$('meta[property="og:video:url"]').attr('content');
|
|
||||||
const playerWidth = parseInt($('meta[property="twitter:player:width"]').attr('content') ||
|
|
||||||
$('meta[name="twitter:player:width"]').attr('content') ||
|
|
||||||
$('meta[property="og:video:width"]').attr('content') ||
|
|
||||||
'');
|
|
||||||
const playerHeight = parseInt($('meta[property="twitter:player:height"]').attr('content') ||
|
|
||||||
$('meta[name="twitter:player:height"]').attr('content') ||
|
|
||||||
$('meta[property="og:video:height"]').attr('content') ||
|
|
||||||
'');
|
|
||||||
let description = $('meta[property="og:description"]').attr('content') ||
|
|
||||||
$('meta[property="twitter:description"]').attr('content') ||
|
|
||||||
$('meta[name="description"]').attr('content');
|
|
||||||
description = description
|
|
||||||
? clip(decodeHtml(description), 300)
|
|
||||||
: null;
|
|
||||||
if (title === description) {
|
|
||||||
description = null;
|
|
||||||
}
|
|
||||||
let siteName = $('meta[property="og:site_name"]').attr('content') ||
|
|
||||||
$('meta[name="application-name"]').attr('content') ||
|
|
||||||
url.hostname;
|
|
||||||
siteName = siteName ? decodeHtml(siteName) : null;
|
|
||||||
const favicon = $('link[rel="shortcut icon"]').attr('href') ||
|
|
||||||
$('link[rel="icon"]').attr('href') ||
|
|
||||||
'/favicon.ico';
|
|
||||||
const sensitive = $('.tweet').attr('data-possibly-sensitive') === 'true';
|
|
||||||
const find = async (path) => {
|
|
||||||
const target = URL.resolve(url.href, path);
|
|
||||||
try {
|
|
||||||
await head(target);
|
|
||||||
return target;
|
|
||||||
}
|
|
||||||
catch (e) {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
// 相対的なURL (ex. test) を絶対的 (ex. /test) に変換
|
|
||||||
const toAbsolute = (relativeURLString) => {
|
|
||||||
const relativeURL = URL.parse(relativeURLString);
|
|
||||||
const isAbsolute = relativeURL.slashes || relativeURL.path !== null && relativeURL.path[0] === '/';
|
|
||||||
// 既に絶対的なら、即座に値を返却
|
|
||||||
if (isAbsolute) {
|
|
||||||
return relativeURLString;
|
|
||||||
}
|
|
||||||
// スラッシュを付けて返却
|
|
||||||
return '/' + relativeURLString;
|
|
||||||
};
|
|
||||||
const icon = await find(favicon) ||
|
|
||||||
// 相対指定を絶対指定に変換し再試行
|
|
||||||
await find(toAbsolute(favicon)) ||
|
|
||||||
null;
|
|
||||||
// Clean up the title
|
|
||||||
title = cleanupTitle(title, siteName);
|
|
||||||
if (title === '') {
|
|
||||||
title = siteName;
|
|
||||||
}
|
|
||||||
return {
|
|
||||||
title: title || null,
|
|
||||||
icon: icon || null,
|
|
||||||
description: description || null,
|
|
||||||
thumbnail: image || null,
|
|
||||||
player: {
|
|
||||||
url: playerUrl || null,
|
|
||||||
width: Number.isNaN(playerWidth) ? null : playerWidth,
|
|
||||||
height: Number.isNaN(playerHeight) ? null : playerHeight
|
|
||||||
},
|
|
||||||
sitename: siteName || null,
|
|
||||||
sensitive,
|
|
||||||
};
|
|
||||||
};
|
|
39
built/index.d.ts
vendored
39
built/index.d.ts
vendored
@ -1,39 +0,0 @@
|
|||||||
/**
|
|
||||||
* summaly
|
|
||||||
* https://github.com/syuilo/summaly
|
|
||||||
*/
|
|
||||||
import Summary from './summary.js';
|
|
||||||
import type { IPlugin as _IPlugin } from './iplugin.js';
|
|
||||||
export declare type IPlugin = _IPlugin;
|
|
||||||
import * as Got from 'got';
|
|
||||||
import type { FastifyInstance } from 'fastify';
|
|
||||||
declare type Options = {
|
|
||||||
/**
|
|
||||||
* Accept-Language for the request
|
|
||||||
*/
|
|
||||||
lang?: string | null;
|
|
||||||
/**
|
|
||||||
* Whether follow redirects
|
|
||||||
*/
|
|
||||||
followRedirects?: boolean;
|
|
||||||
/**
|
|
||||||
* Custom Plugins
|
|
||||||
*/
|
|
||||||
plugins?: IPlugin[];
|
|
||||||
/**
|
|
||||||
* Custom HTTP agent
|
|
||||||
*/
|
|
||||||
agent?: Got.Agents;
|
|
||||||
};
|
|
||||||
declare type Result = Summary & {
|
|
||||||
/**
|
|
||||||
* The actual url of that web page
|
|
||||||
*/
|
|
||||||
url: string;
|
|
||||||
};
|
|
||||||
/**
|
|
||||||
* Summarize an web page
|
|
||||||
*/
|
|
||||||
export declare const summaly: (url: string, options?: Options | undefined) => Promise<Result>;
|
|
||||||
export default function (fastify: FastifyInstance, options: Options, done: (err?: Error) => void): void;
|
|
||||||
export {};
|
|
@ -1,68 +0,0 @@
|
|||||||
/**
|
|
||||||
* summaly
|
|
||||||
* https://github.com/syuilo/summaly
|
|
||||||
*/
|
|
||||||
import * as URL from 'node:url';
|
|
||||||
import tracer from 'trace-redirect';
|
|
||||||
import general from './general.js';
|
|
||||||
import { setAgent } from './utils/got.js';
|
|
||||||
import { plugins as builtinPlugins } from './plugins/index.js';
|
|
||||||
const defaultOptions = {
|
|
||||||
lang: null,
|
|
||||||
followRedirects: true,
|
|
||||||
plugins: [],
|
|
||||||
};
|
|
||||||
/**
|
|
||||||
* Summarize an web page
|
|
||||||
*/
|
|
||||||
export const summaly = async (url, options) => {
|
|
||||||
if (options?.agent)
|
|
||||||
setAgent(options.agent);
|
|
||||||
const opts = Object.assign(defaultOptions, options);
|
|
||||||
const plugins = builtinPlugins.concat(opts.plugins || []);
|
|
||||||
let actualUrl = url;
|
|
||||||
if (opts.followRedirects) {
|
|
||||||
// .catch(() => url)にすればいいけど、jestにtrace-redirectを食わせるのが面倒なのでtry-catch
|
|
||||||
try {
|
|
||||||
actualUrl = await tracer(url);
|
|
||||||
}
|
|
||||||
catch (e) {
|
|
||||||
actualUrl = url;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
const _url = URL.parse(actualUrl, true);
|
|
||||||
// Find matching plugin
|
|
||||||
const match = plugins.filter(plugin => plugin.test(_url))[0];
|
|
||||||
// Get summary
|
|
||||||
const summary = await (match ? match.summarize : general)(_url, opts.lang || undefined);
|
|
||||||
if (summary == null) {
|
|
||||||
throw 'failed summarize';
|
|
||||||
}
|
|
||||||
return Object.assign(summary, {
|
|
||||||
url: actualUrl
|
|
||||||
});
|
|
||||||
};
|
|
||||||
export default function (fastify, options, done) {
|
|
||||||
fastify.get('/url', async (req, reply) => {
|
|
||||||
const url = req.query.url;
|
|
||||||
if (url == null) {
|
|
||||||
return reply.status(400).send({
|
|
||||||
error: 'url is required'
|
|
||||||
});
|
|
||||||
}
|
|
||||||
try {
|
|
||||||
const summary = await summaly(url, {
|
|
||||||
lang: req.query.lang,
|
|
||||||
followRedirects: false,
|
|
||||||
...options,
|
|
||||||
});
|
|
||||||
return summary;
|
|
||||||
}
|
|
||||||
catch (e) {
|
|
||||||
return reply.status(500).send({
|
|
||||||
error: e
|
|
||||||
});
|
|
||||||
}
|
|
||||||
});
|
|
||||||
done();
|
|
||||||
}
|
|
7
built/iplugin.d.ts
vendored
7
built/iplugin.d.ts
vendored
@ -1,7 +0,0 @@
|
|||||||
/// <reference types="node" />
|
|
||||||
import * as URL from 'node:url';
|
|
||||||
import Summary from './summary.js';
|
|
||||||
export interface IPlugin {
|
|
||||||
test: (url: URL.Url) => boolean;
|
|
||||||
summarize: (url: URL.Url, lang?: string) => Promise<Summary>;
|
|
||||||
}
|
|
@ -1 +0,0 @@
|
|||||||
export {};
|
|
5
built/plugins/amazon.d.ts
vendored
5
built/plugins/amazon.d.ts
vendored
@ -1,5 +0,0 @@
|
|||||||
/// <reference types="node" />
|
|
||||||
import * as URL from 'node:url';
|
|
||||||
import summary from '../summary.js';
|
|
||||||
export declare function test(url: URL.Url): boolean;
|
|
||||||
export declare function summarize(url: URL.Url): Promise<summary>;
|
|
@ -1,43 +0,0 @@
|
|||||||
import { scpaping } from '../utils/got.js';
|
|
||||||
export function test(url) {
|
|
||||||
return url.hostname === 'www.amazon.com' ||
|
|
||||||
url.hostname === 'www.amazon.co.jp' ||
|
|
||||||
url.hostname === 'www.amazon.ca' ||
|
|
||||||
url.hostname === 'www.amazon.com.br' ||
|
|
||||||
url.hostname === 'www.amazon.com.mx' ||
|
|
||||||
url.hostname === 'www.amazon.co.uk' ||
|
|
||||||
url.hostname === 'www.amazon.de' ||
|
|
||||||
url.hostname === 'www.amazon.fr' ||
|
|
||||||
url.hostname === 'www.amazon.it' ||
|
|
||||||
url.hostname === 'www.amazon.es' ||
|
|
||||||
url.hostname === 'www.amazon.nl' ||
|
|
||||||
url.hostname === 'www.amazon.cn' ||
|
|
||||||
url.hostname === 'www.amazon.in' ||
|
|
||||||
url.hostname === 'www.amazon.au';
|
|
||||||
}
|
|
||||||
export async function summarize(url) {
|
|
||||||
const res = await scpaping(url.href);
|
|
||||||
const $ = res.$;
|
|
||||||
const title = $('#title').text();
|
|
||||||
const description = $('#productDescription').text() ||
|
|
||||||
$('meta[name="description"]').attr('content');
|
|
||||||
const thumbnail = $('#landingImage').attr('src');
|
|
||||||
const playerUrl = $('meta[property="twitter:player"]').attr('content') ||
|
|
||||||
$('meta[name="twitter:player"]').attr('content');
|
|
||||||
const playerWidth = $('meta[property="twitter:player:width"]').attr('content') ||
|
|
||||||
$('meta[name="twitter:player:width"]').attr('content');
|
|
||||||
const playerHeight = $('meta[property="twitter:player:height"]').attr('content') ||
|
|
||||||
$('meta[name="twitter:player:height"]').attr('content');
|
|
||||||
return {
|
|
||||||
title: title ? title.trim() : null,
|
|
||||||
icon: 'https://www.amazon.com/favicon.ico',
|
|
||||||
description: description ? description.trim() : null,
|
|
||||||
thumbnail: thumbnail ? thumbnail.trim() : null,
|
|
||||||
player: {
|
|
||||||
url: playerUrl || null,
|
|
||||||
width: playerWidth ? parseInt(playerWidth) : null,
|
|
||||||
height: playerHeight ? parseInt(playerHeight) : null
|
|
||||||
},
|
|
||||||
sitename: 'Amazon'
|
|
||||||
};
|
|
||||||
}
|
|
2
built/plugins/index.d.ts
vendored
2
built/plugins/index.d.ts
vendored
@ -1,2 +0,0 @@
|
|||||||
import { IPlugin } from '@/iplugin.js';
|
|
||||||
export declare const plugins: IPlugin[];
|
|
@ -1,6 +0,0 @@
|
|||||||
import * as amazon from './amazon.js';
|
|
||||||
import * as wikipedia from './wikipedia.js';
|
|
||||||
export const plugins = [
|
|
||||||
amazon,
|
|
||||||
wikipedia,
|
|
||||||
];
|
|
5
built/plugins/wikipedia.d.ts
vendored
5
built/plugins/wikipedia.d.ts
vendored
@ -1,5 +0,0 @@
|
|||||||
/// <reference types="node" />
|
|
||||||
import * as URL from 'node:url';
|
|
||||||
import summary from '../summary.js';
|
|
||||||
export declare function test(url: URL.Url): boolean;
|
|
||||||
export declare function summarize(url: URL.Url): Promise<summary>;
|
|
@ -1,36 +0,0 @@
|
|||||||
import { get } from '../utils/got.js';
|
|
||||||
import debug from 'debug';
|
|
||||||
import clip from './../utils/clip.js';
|
|
||||||
const log = debug('summaly:plugins:wikipedia');
|
|
||||||
export function test(url) {
|
|
||||||
if (!url.hostname)
|
|
||||||
return false;
|
|
||||||
return /\.wikipedia\.org$/.test(url.hostname);
|
|
||||||
}
|
|
||||||
export async function summarize(url) {
|
|
||||||
const lang = url.host ? url.host.split('.')[0] : null;
|
|
||||||
const title = url.pathname ? url.pathname.split('/')[2] : null;
|
|
||||||
const endpoint = `https://${lang}.wikipedia.org/w/api.php?format=json&action=query&prop=extracts&exintro=&explaintext=&titles=${title}`;
|
|
||||||
log(`lang is ${lang}`);
|
|
||||||
log(`title is ${title}`);
|
|
||||||
log(`endpoint is ${endpoint}`);
|
|
||||||
let body = await get(endpoint);
|
|
||||||
body = JSON.parse(body);
|
|
||||||
log(body);
|
|
||||||
if (!('query' in body) || !('pages' in body.query)) {
|
|
||||||
throw 'fetch failed';
|
|
||||||
}
|
|
||||||
const info = body.query.pages[Object.keys(body.query.pages)[0]];
|
|
||||||
return {
|
|
||||||
title: info.title,
|
|
||||||
icon: 'https://wikipedia.org/static/favicon/wikipedia.ico',
|
|
||||||
description: clip(info.extract, 300),
|
|
||||||
thumbnail: `https://wikipedia.org/static/images/project-logos/${lang}wiki.png`,
|
|
||||||
player: {
|
|
||||||
url: null,
|
|
||||||
width: null,
|
|
||||||
height: null
|
|
||||||
},
|
|
||||||
sitename: 'Wikipedia'
|
|
||||||
};
|
|
||||||
}
|
|
1
built/server/index.d.ts
vendored
1
built/server/index.d.ts
vendored
@ -1 +0,0 @@
|
|||||||
export {};
|
|
@ -1,22 +0,0 @@
|
|||||||
import * as http from 'http';
|
|
||||||
import * as Koa from 'koa';
|
|
||||||
import summaly from '../';
|
|
||||||
const app = new Koa();
|
|
||||||
app.use(async (ctx) => {
|
|
||||||
if (!ctx.query.url) {
|
|
||||||
ctx.status = 400;
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
try {
|
|
||||||
const summary = await summaly(ctx.query.url, {
|
|
||||||
lang: ctx.query.lang,
|
|
||||||
followRedirects: false
|
|
||||||
});
|
|
||||||
ctx.body = summary;
|
|
||||||
}
|
|
||||||
catch (e) {
|
|
||||||
ctx.status = 500;
|
|
||||||
}
|
|
||||||
});
|
|
||||||
const server = http.createServer(app.callback());
|
|
||||||
server.listen(process.env.PORT || 80);
|
|
45
built/summary.d.ts
vendored
45
built/summary.d.ts
vendored
@ -1,45 +0,0 @@
|
|||||||
declare type Summary = {
|
|
||||||
/**
|
|
||||||
* The description of that web page
|
|
||||||
*/
|
|
||||||
description: string | null;
|
|
||||||
/**
|
|
||||||
* The url of the icon of that web page
|
|
||||||
*/
|
|
||||||
icon: string | null;
|
|
||||||
/**
|
|
||||||
* The name of site of that web page
|
|
||||||
*/
|
|
||||||
sitename: string | null;
|
|
||||||
/**
|
|
||||||
* The url of the thumbnail of that web page
|
|
||||||
*/
|
|
||||||
thumbnail: string | null;
|
|
||||||
/**
|
|
||||||
* The player of that web page
|
|
||||||
*/
|
|
||||||
player: Player;
|
|
||||||
/**
|
|
||||||
* The title of that web page
|
|
||||||
*/
|
|
||||||
title: string | null;
|
|
||||||
/**
|
|
||||||
* Possibly sensitive
|
|
||||||
*/
|
|
||||||
sensitive?: boolean;
|
|
||||||
};
|
|
||||||
export default Summary;
|
|
||||||
export declare type Player = {
|
|
||||||
/**
|
|
||||||
* The url of the player
|
|
||||||
*/
|
|
||||||
url: string | null;
|
|
||||||
/**
|
|
||||||
* The width of the player
|
|
||||||
*/
|
|
||||||
width: number | null;
|
|
||||||
/**
|
|
||||||
* The height of the player
|
|
||||||
*/
|
|
||||||
height: number | null;
|
|
||||||
};
|
|
@ -1 +0,0 @@
|
|||||||
export {};
|
|
1
built/utils/cleanup-title.d.ts
vendored
1
built/utils/cleanup-title.d.ts
vendored
@ -1 +0,0 @@
|
|||||||
export default function (title: string, siteName?: string | null): string;
|
|
@ -1,19 +0,0 @@
|
|||||||
import escapeRegExp from 'escape-regexp';
|
|
||||||
export default function (title, siteName) {
|
|
||||||
title = title.trim();
|
|
||||||
if (siteName) {
|
|
||||||
siteName = siteName.trim();
|
|
||||||
const x = escapeRegExp(siteName);
|
|
||||||
const patterns = [
|
|
||||||
`^(.+?)\\s?[\\-\\|:・]\\s?${x}$`
|
|
||||||
];
|
|
||||||
for (let i = 0; i < patterns.length; i++) {
|
|
||||||
const pattern = new RegExp(patterns[i]);
|
|
||||||
const [, match] = pattern.exec(title) || [null, null];
|
|
||||||
if (match) {
|
|
||||||
return match;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return title;
|
|
||||||
}
|
|
1
built/utils/clip.d.ts
vendored
1
built/utils/clip.d.ts
vendored
@ -1 +0,0 @@
|
|||||||
export default function (s: string, max: number): string;
|
|
@ -1,13 +0,0 @@
|
|||||||
import nullOrEmpty from './null-or-empty.js';
|
|
||||||
export default function (s, max) {
|
|
||||||
if (nullOrEmpty(s)) {
|
|
||||||
return s;
|
|
||||||
}
|
|
||||||
s = s.trim();
|
|
||||||
if (s.length > max) {
|
|
||||||
return s.substr(0, max) + '...';
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
return s;
|
|
||||||
}
|
|
||||||
}
|
|
8
built/utils/encoding.d.ts
vendored
8
built/utils/encoding.d.ts
vendored
@ -1,8 +0,0 @@
|
|||||||
/// <reference types="node" />
|
|
||||||
/**
|
|
||||||
* Detect HTML encoding
|
|
||||||
* @param body Body in Buffer
|
|
||||||
* @returns encoding
|
|
||||||
*/
|
|
||||||
export declare function detectEncoding(body: Buffer): string;
|
|
||||||
export declare function toUtf8(body: Buffer, encoding: string): string;
|
|
@ -1,40 +0,0 @@
|
|||||||
import iconv from 'iconv-lite';
|
|
||||||
import jschardet from 'jschardet';
|
|
||||||
const regCharset = new RegExp(/charset\s*=\s*["']?([\w-]+)/, 'i');
|
|
||||||
/**
|
|
||||||
* Detect HTML encoding
|
|
||||||
* @param body Body in Buffer
|
|
||||||
* @returns encoding
|
|
||||||
*/
|
|
||||||
export function detectEncoding(body) {
|
|
||||||
// By detection
|
|
||||||
const detected = jschardet.detect(body, { minimumThreshold: 0.99 });
|
|
||||||
if (detected) {
|
|
||||||
const candicate = detected.encoding;
|
|
||||||
const encoding = toEncoding(candicate);
|
|
||||||
if (encoding != null)
|
|
||||||
return encoding;
|
|
||||||
}
|
|
||||||
// From meta
|
|
||||||
const matchMeta = body.toString('ascii').match(regCharset);
|
|
||||||
if (matchMeta) {
|
|
||||||
const candicate = matchMeta[1];
|
|
||||||
const encoding = toEncoding(candicate);
|
|
||||||
if (encoding != null)
|
|
||||||
return encoding;
|
|
||||||
}
|
|
||||||
return 'utf-8';
|
|
||||||
}
|
|
||||||
export function toUtf8(body, encoding) {
|
|
||||||
return iconv.decode(body, encoding);
|
|
||||||
}
|
|
||||||
function toEncoding(candicate) {
|
|
||||||
if (iconv.encodingExists(candicate)) {
|
|
||||||
if (['shift_jis', 'shift-jis', 'windows-31j', 'x-sjis'].includes(candicate.toLowerCase()))
|
|
||||||
return 'cp932';
|
|
||||||
return candicate;
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
}
|
|
20
built/utils/got.d.ts
vendored
20
built/utils/got.d.ts
vendored
@ -1,20 +0,0 @@
|
|||||||
import * as Got from 'got';
|
|
||||||
import * as cheerio from 'cheerio';
|
|
||||||
export declare let agent: Got.Agents;
|
|
||||||
export declare function setAgent(_agent: Got.Agents): void;
|
|
||||||
export declare type GotOptions = {
|
|
||||||
url: string;
|
|
||||||
method: 'GET' | 'POST' | 'HEAD';
|
|
||||||
body?: string;
|
|
||||||
headers: Record<string, string | undefined>;
|
|
||||||
typeFilter?: RegExp;
|
|
||||||
};
|
|
||||||
export declare function scpaping(url: string, opts?: {
|
|
||||||
lang?: string;
|
|
||||||
}): Promise<{
|
|
||||||
body: string;
|
|
||||||
$: cheerio.CheerioAPI;
|
|
||||||
response: Got.Response<string>;
|
|
||||||
}>;
|
|
||||||
export declare function get(url: string): Promise<string>;
|
|
||||||
export declare function head(url: string): Promise<Got.Response<string>>;
|
|
@ -1,123 +0,0 @@
|
|||||||
import got, * as Got from 'got';
|
|
||||||
import { StatusError } from './status-error.js';
|
|
||||||
import { detectEncoding, toUtf8 } from './encoding.js';
|
|
||||||
import * as cheerio from 'cheerio';
|
|
||||||
import PrivateIp from 'private-ip';
|
|
||||||
import { dirname } from 'node:path';
|
|
||||||
import { fileURLToPath } from 'node:url';
|
|
||||||
import { readFileSync } from 'node:fs';
|
|
||||||
const _filename = fileURLToPath(import.meta.url);
|
|
||||||
const _dirname = dirname(_filename);
|
|
||||||
export let agent = {};
|
|
||||||
export function setAgent(_agent) {
|
|
||||||
agent = _agent || {};
|
|
||||||
}
|
|
||||||
const repo = JSON.parse(readFileSync(`${_dirname}/../../package.json`, 'utf8'));
|
|
||||||
const RESPONSE_TIMEOUT = 20 * 1000;
|
|
||||||
const OPERATION_TIMEOUT = 60 * 1000;
|
|
||||||
const MAX_RESPONSE_SIZE = 10 * 1024 * 1024;
|
|
||||||
const BOT_UA = `SummalyBot/${repo.version}`;
|
|
||||||
export async function scpaping(url, opts) {
|
|
||||||
const response = await getResponse({
|
|
||||||
url,
|
|
||||||
method: 'GET',
|
|
||||||
headers: {
|
|
||||||
'accept': 'text/html,application/xhtml+xml',
|
|
||||||
'user-agent': BOT_UA,
|
|
||||||
'accept-language': opts?.lang
|
|
||||||
},
|
|
||||||
typeFilter: /^(text\/html|application\/xhtml\+xml)/,
|
|
||||||
});
|
|
||||||
// テスト用
|
|
||||||
const allowPrivateIp = process.env.SUMMALY_ALLOW_PRIVATE_IP === 'true';
|
|
||||||
if (!allowPrivateIp && response.ip && PrivateIp(response.ip)) {
|
|
||||||
throw new StatusError(`Private IP rejected ${response.ip}`, 400, 'Private IP Rejected');
|
|
||||||
}
|
|
||||||
const encoding = detectEncoding(response.rawBody);
|
|
||||||
const body = toUtf8(response.rawBody, encoding);
|
|
||||||
const $ = cheerio.load(body);
|
|
||||||
return {
|
|
||||||
body,
|
|
||||||
$,
|
|
||||||
response,
|
|
||||||
};
|
|
||||||
}
|
|
||||||
export async function get(url) {
|
|
||||||
const res = await getResponse({
|
|
||||||
url,
|
|
||||||
method: 'GET',
|
|
||||||
headers: {
|
|
||||||
'accept': '*/*',
|
|
||||||
},
|
|
||||||
});
|
|
||||||
return await res.body;
|
|
||||||
}
|
|
||||||
export async function head(url) {
|
|
||||||
const res = await getResponse({
|
|
||||||
url,
|
|
||||||
method: 'HEAD',
|
|
||||||
headers: {
|
|
||||||
'accept': '*/*',
|
|
||||||
},
|
|
||||||
});
|
|
||||||
return await res;
|
|
||||||
}
|
|
||||||
async function getResponse(args) {
|
|
||||||
const timeout = RESPONSE_TIMEOUT;
|
|
||||||
const operationTimeout = OPERATION_TIMEOUT;
|
|
||||||
const req = got(args.url, {
|
|
||||||
method: args.method,
|
|
||||||
headers: args.headers,
|
|
||||||
body: args.body,
|
|
||||||
timeout: {
|
|
||||||
lookup: timeout,
|
|
||||||
connect: timeout,
|
|
||||||
secureConnect: timeout,
|
|
||||||
socket: timeout,
|
|
||||||
response: timeout,
|
|
||||||
send: timeout,
|
|
||||||
request: operationTimeout, // whole operation timeout
|
|
||||||
},
|
|
||||||
agent,
|
|
||||||
http2: false,
|
|
||||||
retry: {
|
|
||||||
limit: 0,
|
|
||||||
},
|
|
||||||
});
|
|
||||||
return await receiveResponce({ req, typeFilter: args.typeFilter });
|
|
||||||
}
|
|
||||||
async function receiveResponce(args) {
|
|
||||||
const req = args.req;
|
|
||||||
const maxSize = MAX_RESPONSE_SIZE;
|
|
||||||
req.on('response', (res) => {
|
|
||||||
// Check html
|
|
||||||
if (args.typeFilter && !res.headers['content-type']?.match(args.typeFilter)) {
|
|
||||||
req.cancel(`Rejected by type filter ${res.headers['content-type']}`);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
// 応答ヘッダでサイズチェック
|
|
||||||
const contentLength = res.headers['content-length'];
|
|
||||||
if (contentLength != null) {
|
|
||||||
const size = Number(contentLength);
|
|
||||||
if (size > maxSize) {
|
|
||||||
req.cancel(`maxSize exceeded (${size} > ${maxSize}) on response`);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
});
|
|
||||||
// 受信中のデータでサイズチェック
|
|
||||||
req.on('downloadProgress', (progress) => {
|
|
||||||
if (progress.transferred > maxSize && progress.percent !== 1) {
|
|
||||||
req.cancel(`maxSize exceeded (${progress.transferred} > ${maxSize}) on response`);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
// 応答取得 with ステータスコードエラーの整形
|
|
||||||
const res = await req.catch(e => {
|
|
||||||
if (e instanceof Got.HTTPError) {
|
|
||||||
throw new StatusError(`${e.response.statusCode} ${e.response.statusMessage}`, e.response.statusCode, e.response.statusMessage);
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
throw e;
|
|
||||||
}
|
|
||||||
});
|
|
||||||
return res;
|
|
||||||
}
|
|
1
built/utils/null-or-empty.d.ts
vendored
1
built/utils/null-or-empty.d.ts
vendored
@ -1 +0,0 @@
|
|||||||
export default function (val: string): boolean;
|
|
@ -1,14 +0,0 @@
|
|||||||
export default function (val) {
|
|
||||||
if (val === undefined) {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
else if (val === null) {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
else if (val.trim() === '') {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
6
built/utils/status-error.d.ts
vendored
6
built/utils/status-error.d.ts
vendored
@ -1,6 +0,0 @@
|
|||||||
export declare class StatusError extends Error {
|
|
||||||
statusCode: number;
|
|
||||||
statusMessage?: string;
|
|
||||||
isPermanentError: boolean;
|
|
||||||
constructor(message: string, statusCode: number, statusMessage?: string);
|
|
||||||
}
|
|
@ -1,9 +0,0 @@
|
|||||||
export class StatusError extends Error {
|
|
||||||
constructor(message, statusCode, statusMessage) {
|
|
||||||
super(message);
|
|
||||||
this.name = 'StatusError';
|
|
||||||
this.statusCode = statusCode;
|
|
||||||
this.statusMessage = statusMessage;
|
|
||||||
this.isPermanentError = typeof this.statusCode === 'number' && this.statusCode >= 400 && this.statusCode < 500;
|
|
||||||
}
|
|
||||||
}
|
|
10668
package-lock.json
generated
10668
package-lock.json
generated
File diff suppressed because it is too large
Load Diff
36
package.json
36
package.json
@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "summaly",
|
"name": "@misskey-dev/summaly",
|
||||||
"version": "3.0.1",
|
"version": "5.0.0",
|
||||||
"description": "Get web page's summary",
|
"description": "Get web page's summary",
|
||||||
"author": "syuilo <syuilotan@yahoo.co.jp>",
|
"author": "syuilo <syuilotan@yahoo.co.jp>",
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
@ -9,42 +9,42 @@
|
|||||||
"main": "./built/index.js",
|
"main": "./built/index.js",
|
||||||
"type": "module",
|
"type": "module",
|
||||||
"types": "./built/index.d.ts",
|
"types": "./built/index.d.ts",
|
||||||
|
"packageManager": "pnpm@8.13.1",
|
||||||
"files": [
|
"files": [
|
||||||
"built",
|
"built",
|
||||||
"LICENSE"
|
"LICENSE"
|
||||||
],
|
],
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"build": "tsc",
|
"build": "tsc",
|
||||||
|
"eslint": "eslint --quiet \"src/**/*.ts\"",
|
||||||
"test": "node --experimental-vm-modules node_modules/jest/bin/jest.js --silent=false --verbose false",
|
"test": "node --experimental-vm-modules node_modules/jest/bin/jest.js --silent=false --verbose false",
|
||||||
"serve": "fastify start ./built/index.js"
|
"serve": "fastify start ./built/index.js"
|
||||||
},
|
},
|
||||||
"optionalDependencies": {
|
|
||||||
"fastify": "3.24.1"
|
|
||||||
},
|
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@jest/globals": "^29.4.2",
|
"@jest/globals": "^29.7.0",
|
||||||
"@swc/core": "^1.3.35",
|
"@swc/core": "^1.3.101",
|
||||||
"@swc/jest": "^0.2.24",
|
"@swc/jest": "^0.2.29",
|
||||||
"@types/cheerio": "0.22.18",
|
"@types/cheerio": "0.22.18",
|
||||||
"@types/debug": "4.1.7",
|
"@types/debug": "4.1.7",
|
||||||
"@types/escape-regexp": "^0.0.1",
|
"@types/escape-regexp": "^0.0.1",
|
||||||
"@types/html-entities": "1.3.4",
|
"@types/node": "20.10.6",
|
||||||
"@types/node": "16.11.12",
|
"@typescript-eslint/eslint-plugin": "^6.16.0",
|
||||||
|
"@typescript-eslint/parser": "^6.16.0",
|
||||||
"debug": "^4.3.4",
|
"debug": "^4.3.4",
|
||||||
"express": "^4.18.2",
|
"eslint": "^8.56.0",
|
||||||
"fastify": "^4.13.0",
|
"eslint-plugin-import": "^2.29.1",
|
||||||
"fastify-cli": "^5.7.1",
|
"fastify": "^4.25.2",
|
||||||
"jest": "^29.4.2",
|
"fastify-cli": "^5.9.0",
|
||||||
"typescript": "4.5.3"
|
"jest": "^29.7.0",
|
||||||
|
"typescript": "5.3.3"
|
||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"cheerio": "^1.0.0-rc.12",
|
"cheerio": "1.0.0-rc.12",
|
||||||
"escape-regexp": "0.0.1",
|
"escape-regexp": "0.0.1",
|
||||||
"got": "^12.5.3",
|
"got": "^12.6.1",
|
||||||
"html-entities": "2.3.2",
|
"html-entities": "2.3.2",
|
||||||
"iconv-lite": "0.6.3",
|
"iconv-lite": "0.6.3",
|
||||||
"jschardet": "3.0.0",
|
"jschardet": "3.0.0",
|
||||||
"koa": "2.13.4",
|
|
||||||
"private-ip": "2.3.3",
|
"private-ip": "2.3.3",
|
||||||
"trace-redirect": "1.0.6"
|
"trace-redirect": "1.0.6"
|
||||||
}
|
}
|
||||||
|
4667
pnpm-lock.yaml
generated
Normal file
4667
pnpm-lock.yaml
generated
Normal file
File diff suppressed because it is too large
Load Diff
204
src/general.ts
204
src/general.ts
@ -1,21 +1,155 @@
|
|||||||
import * as URL from 'node:url';
|
import { URL } from 'node:url';
|
||||||
|
import { decode as decodeHtml } from 'html-entities';
|
||||||
|
import * as cheerio from 'cheerio';
|
||||||
import clip from './utils/clip.js';
|
import clip from './utils/clip.js';
|
||||||
import cleanupTitle from './utils/cleanup-title.js';
|
import cleanupTitle from './utils/cleanup-title.js';
|
||||||
|
|
||||||
import { decode as decodeHtml } from 'html-entities';
|
import { get, head, scpaping } from './utils/got.js';
|
||||||
|
import type { default as Summary, Player } from './summary.js';
|
||||||
|
|
||||||
import { head, scpaping } from './utils/got.js';
|
/**
|
||||||
import Summary from './summary.js';
|
* Contains only the html snippet for a sanitized iframe as the thumbnail is
|
||||||
|
* mostly covered in OpenGraph instead.
|
||||||
|
*
|
||||||
|
* Width should always be 100%.
|
||||||
|
*/
|
||||||
|
async function getOEmbedPlayer($: cheerio.CheerioAPI, pageUrl: string): Promise<Player | null> {
|
||||||
|
const href = $('link[type="application/json+oembed"]').attr('href');
|
||||||
|
if (!href) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
export default async (url: URL.Url, lang: string | null = null): Promise<Summary | null> => {
|
const oEmbedUrl = (() => {
|
||||||
|
try {
|
||||||
|
return new URL(href, pageUrl);
|
||||||
|
} catch { return null; }
|
||||||
|
})();
|
||||||
|
if (!oEmbedUrl) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
const oEmbed = await get(oEmbedUrl.href).catch(() => null);
|
||||||
|
if (!oEmbed) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
const body = (() => {
|
||||||
|
try {
|
||||||
|
return JSON.parse(oEmbed);
|
||||||
|
} catch {}
|
||||||
|
})();
|
||||||
|
|
||||||
|
if (!body || body.version !== '1.0' || !['rich', 'video'].includes(body.type)) {
|
||||||
|
// Not a well formed rich oEmbed
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!body.html.startsWith('<iframe ') || !body.html.endsWith('</iframe>')) {
|
||||||
|
// It includes something else than an iframe
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
const oEmbedHtml = cheerio.load(body.html);
|
||||||
|
const iframe = oEmbedHtml('iframe');
|
||||||
|
|
||||||
|
if (iframe.length !== 1) {
|
||||||
|
// Somehow we either have multiple iframes or none
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (iframe.parents().length !== 2) {
|
||||||
|
// Should only have the body and html elements as the parents
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
const url = iframe.attr('src');
|
||||||
|
if (!url) {
|
||||||
|
// No src?
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
if ((new URL(url)).protocol !== 'https:') {
|
||||||
|
// Allow only HTTPS for best security
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
} catch (e) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Height is the most important, width is okay to be null. The implementer
|
||||||
|
// should choose fixed height instead of fixed aspect ratio if width is null.
|
||||||
|
//
|
||||||
|
// For example, Spotify's embed page does not strictly follow aspect ratio
|
||||||
|
// and thus keeping the height is better than keeping the aspect ratio.
|
||||||
|
//
|
||||||
|
// Spotify gives `width: 100%, height: 152px` for iframe while `width: 456,
|
||||||
|
// height: 152` for oEmbed data, and we treat any percentages as null here.
|
||||||
|
let width: number | null = Number(iframe.attr('width') ?? body.width);
|
||||||
|
if (Number.isNaN(width)) {
|
||||||
|
width = null;
|
||||||
|
}
|
||||||
|
const height = Math.min(Number(iframe.attr('height') ?? body.height), 1024);
|
||||||
|
if (Number.isNaN(height)) {
|
||||||
|
// No proper height info
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
// TODO: This implementation only allows basic syntax of `allow`.
|
||||||
|
// Might need to implement better later.
|
||||||
|
const safeList = [
|
||||||
|
'autoplay',
|
||||||
|
'clipboard-write',
|
||||||
|
'fullscreen',
|
||||||
|
'encrypted-media',
|
||||||
|
'picture-in-picture',
|
||||||
|
'web-share',
|
||||||
|
];
|
||||||
|
// YouTube has these but they are almost never used.
|
||||||
|
const ignoredList = [
|
||||||
|
'gyroscope',
|
||||||
|
'accelerometer',
|
||||||
|
];
|
||||||
|
const allowedPermissions =
|
||||||
|
(iframe.attr('allow') ?? '').split(/\s*;\s*/g)
|
||||||
|
.filter(s => s)
|
||||||
|
.filter(s => !ignoredList.includes(s));
|
||||||
|
if (iframe.attr('allowfullscreen') === '') {
|
||||||
|
allowedPermissions.push('fullscreen');
|
||||||
|
}
|
||||||
|
if (allowedPermissions.some(allow => !safeList.includes(allow))) {
|
||||||
|
// This iframe is probably too powerful to be embedded
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
url,
|
||||||
|
width,
|
||||||
|
height,
|
||||||
|
allow: allowedPermissions,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
export default async (_url: URL | string, lang: string | null = null): Promise<Summary | null> => {
|
||||||
if (lang && !lang.match(/^[\w-]+(\s*,\s*[\w-]+)*$/)) lang = null;
|
if (lang && !lang.match(/^[\w-]+(\s*,\s*[\w-]+)*$/)) lang = null;
|
||||||
|
|
||||||
|
const url = typeof _url === 'string' ? new URL(_url) : _url;
|
||||||
|
|
||||||
const res = await scpaping(url.href, { lang: lang || undefined });
|
const res = await scpaping(url.href, { lang: lang || undefined });
|
||||||
const $ = res.$;
|
const $ = res.$;
|
||||||
const twitterCard = $('meta[property="twitter:card"]').attr('content');
|
const twitterCard =
|
||||||
|
$('meta[name="twitter:card"]').attr('content') ||
|
||||||
|
$('meta[property="twitter:card"]').attr('content');
|
||||||
|
|
||||||
|
// According to docs, name attribute of meta tag is used for twitter card but for compatibility,
|
||||||
|
// this library will also look for property attribute.
|
||||||
|
// See https://developer.twitter.com/en/docs/twitter-for-websites/cards/overview/summary
|
||||||
|
// Property attribute is used for open graph.
|
||||||
|
// See https://ogp.me/
|
||||||
|
|
||||||
let title: string | null | undefined =
|
let title: string | null | undefined =
|
||||||
$('meta[property="og:title"]').attr('content') ||
|
$('meta[property="og:title"]').attr('content') ||
|
||||||
|
$('meta[name="twitter:title"]').attr('content') ||
|
||||||
$('meta[property="twitter:title"]').attr('content') ||
|
$('meta[property="twitter:title"]').attr('content') ||
|
||||||
$('title').text();
|
$('title').text();
|
||||||
|
|
||||||
@ -27,34 +161,36 @@ export default async (url: URL.Url, lang: string | null = null): Promise<Summary
|
|||||||
|
|
||||||
let image: string | null | undefined =
|
let image: string | null | undefined =
|
||||||
$('meta[property="og:image"]').attr('content') ||
|
$('meta[property="og:image"]').attr('content') ||
|
||||||
|
$('meta[name="twitter:image"]').attr('content') ||
|
||||||
$('meta[property="twitter:image"]').attr('content') ||
|
$('meta[property="twitter:image"]').attr('content') ||
|
||||||
$('link[rel="image_src"]').attr('href') ||
|
$('link[rel="image_src"]').attr('href') ||
|
||||||
$('link[rel="apple-touch-icon"]').attr('href') ||
|
$('link[rel="apple-touch-icon"]').attr('href') ||
|
||||||
$('link[rel="apple-touch-icon image_src"]').attr('href');
|
$('link[rel="apple-touch-icon image_src"]').attr('href');
|
||||||
|
|
||||||
image = image ? URL.resolve(url.href, image) : null;
|
image = image ? (new URL(image, url.href)).href : null;
|
||||||
|
|
||||||
const playerUrl =
|
const playerUrl =
|
||||||
(twitterCard !== 'summary_large_image' && $('meta[property="twitter:player"]').attr('content')) ||
|
|
||||||
(twitterCard !== 'summary_large_image' && $('meta[name="twitter:player"]').attr('content')) ||
|
(twitterCard !== 'summary_large_image' && $('meta[name="twitter:player"]').attr('content')) ||
|
||||||
|
(twitterCard !== 'summary_large_image' && $('meta[property="twitter:player"]').attr('content')) ||
|
||||||
$('meta[property="og:video"]').attr('content') ||
|
$('meta[property="og:video"]').attr('content') ||
|
||||||
$('meta[property="og:video:secure_url"]').attr('content') ||
|
$('meta[property="og:video:secure_url"]').attr('content') ||
|
||||||
$('meta[property="og:video:url"]').attr('content');
|
$('meta[property="og:video:url"]').attr('content');
|
||||||
|
|
||||||
const playerWidth = parseInt(
|
const playerWidth = parseInt(
|
||||||
$('meta[property="twitter:player:width"]').attr('content') ||
|
|
||||||
$('meta[name="twitter:player:width"]').attr('content') ||
|
$('meta[name="twitter:player:width"]').attr('content') ||
|
||||||
|
$('meta[property="twitter:player:width"]').attr('content') ||
|
||||||
$('meta[property="og:video:width"]').attr('content') ||
|
$('meta[property="og:video:width"]').attr('content') ||
|
||||||
'');
|
'');
|
||||||
|
|
||||||
const playerHeight = parseInt(
|
const playerHeight = parseInt(
|
||||||
$('meta[property="twitter:player:height"]').attr('content') ||
|
|
||||||
$('meta[name="twitter:player:height"]').attr('content') ||
|
$('meta[name="twitter:player:height"]').attr('content') ||
|
||||||
|
$('meta[property="twitter:player:height"]').attr('content') ||
|
||||||
$('meta[property="og:video:height"]').attr('content') ||
|
$('meta[property="og:video:height"]').attr('content') ||
|
||||||
'');
|
'');
|
||||||
|
|
||||||
let description: string | null | undefined =
|
let description: string | null | undefined =
|
||||||
$('meta[property="og:description"]').attr('content') ||
|
$('meta[property="og:description"]').attr('content') ||
|
||||||
|
$('meta[name="twitter:description"]').attr('content') ||
|
||||||
$('meta[property="twitter:description"]').attr('content') ||
|
$('meta[property="twitter:description"]').attr('content') ||
|
||||||
$('meta[name="description"]').attr('content');
|
$('meta[name="description"]').attr('content');
|
||||||
|
|
||||||
@ -66,48 +202,42 @@ export default async (url: URL.Url, lang: string | null = null): Promise<Summary
|
|||||||
description = null;
|
description = null;
|
||||||
}
|
}
|
||||||
|
|
||||||
let siteName =
|
const siteName = decodeHtml(
|
||||||
$('meta[property="og:site_name"]').attr('content') ||
|
$('meta[property="og:site_name"]').attr('content') ||
|
||||||
$('meta[name="application-name"]').attr('content') ||
|
$('meta[name="application-name"]').attr('content') ||
|
||||||
url.hostname;
|
url.host,
|
||||||
|
);
|
||||||
siteName = siteName ? decodeHtml(siteName) : null;
|
|
||||||
|
|
||||||
const favicon =
|
const favicon =
|
||||||
$('link[rel="shortcut icon"]').attr('href') ||
|
$('link[rel="shortcut icon"]').attr('href') ||
|
||||||
$('link[rel="icon"]').attr('href') ||
|
$('link[rel="icon"]').attr('href') ||
|
||||||
'/favicon.ico';
|
'/favicon.ico';
|
||||||
|
|
||||||
const sensitive = $('.tweet').attr('data-possibly-sensitive') === 'true'
|
const activityPub =
|
||||||
|
$('link[rel="alternate"][type="application/activity+json"]').attr('href') || null;
|
||||||
|
|
||||||
|
// https://developer.mixi.co.jp/connect/mixi_plugin/mixi_check/spec_mixi_check/#toc-18-
|
||||||
|
const sensitive =
|
||||||
|
$('meta[property=\'mixi:content-rating\']').attr('content') === '1';
|
||||||
|
|
||||||
const find = async (path: string) => {
|
const find = async (path: string) => {
|
||||||
const target = URL.resolve(url.href, path);
|
const target = new URL(path, url.href);
|
||||||
try {
|
try {
|
||||||
await head(target);
|
await head(target.href);
|
||||||
return target;
|
return target;
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
// 相対的なURL (ex. test) を絶対的 (ex. /test) に変換
|
const getIcon = async () => {
|
||||||
const toAbsolute = (relativeURLString: string): string => {
|
return (await find(favicon)) || null;
|
||||||
const relativeURL = URL.parse(relativeURLString);
|
|
||||||
const isAbsolute = relativeURL.slashes || relativeURL.path !== null && relativeURL.path[0] === '/';
|
|
||||||
|
|
||||||
// 既に絶対的なら、即座に値を返却
|
|
||||||
if (isAbsolute) {
|
|
||||||
return relativeURLString;
|
|
||||||
}
|
|
||||||
|
|
||||||
// スラッシュを付けて返却
|
|
||||||
return '/' + relativeURLString;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
const icon = await find(favicon) ||
|
const [icon, oEmbed] = await Promise.all([
|
||||||
// 相対指定を絶対指定に変換し再試行
|
getIcon(),
|
||||||
await find(toAbsolute(favicon)) ||
|
getOEmbedPlayer($, url.href),
|
||||||
null;
|
]);
|
||||||
|
|
||||||
// Clean up the title
|
// Clean up the title
|
||||||
title = cleanupTitle(title, siteName);
|
title = cleanupTitle(title, siteName);
|
||||||
@ -118,15 +248,17 @@ export default async (url: URL.Url, lang: string | null = null): Promise<Summary
|
|||||||
|
|
||||||
return {
|
return {
|
||||||
title: title || null,
|
title: title || null,
|
||||||
icon: icon || null,
|
icon: icon?.href || null,
|
||||||
description: description || null,
|
description: description || null,
|
||||||
thumbnail: image || null,
|
thumbnail: image || null,
|
||||||
player: {
|
player: oEmbed ?? {
|
||||||
url: playerUrl || null,
|
url: playerUrl || null,
|
||||||
width: Number.isNaN(playerWidth) ? null : playerWidth,
|
width: Number.isNaN(playerWidth) ? null : playerWidth,
|
||||||
height: Number.isNaN(playerHeight) ? null : playerHeight
|
height: Number.isNaN(playerHeight) ? null : playerHeight,
|
||||||
|
allow: ['autoplay', 'encrypted-media', 'fullscreen'],
|
||||||
},
|
},
|
||||||
sitename: siteName || null,
|
sitename: siteName || null,
|
||||||
sensitive,
|
sensitive,
|
||||||
|
activityPub,
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
41
src/index.ts
41
src/index.ts
@ -1,20 +1,20 @@
|
|||||||
/**
|
/**
|
||||||
* summaly
|
* summaly
|
||||||
* https://github.com/syuilo/summaly
|
* https://github.com/misskey-dev/summaly
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import * as URL from 'node:url';
|
import { URL } from 'node:url';
|
||||||
import tracer from 'trace-redirect';
|
import tracer from 'trace-redirect';
|
||||||
import Summary from './summary.js';
|
import { SummalyResult } from './summary.js';
|
||||||
import type { IPlugin as _IPlugin } from './iplugin.js';
|
import { SummalyPlugin } from './iplugin.js';
|
||||||
export type IPlugin = _IPlugin;
|
export * from './iplugin.js';
|
||||||
import general from './general.js';
|
import general from './general.js';
|
||||||
import * as Got from 'got';
|
import * as Got from 'got';
|
||||||
import { setAgent } from './utils/got.js';
|
import { setAgent } from './utils/got.js';
|
||||||
import type { FastifyInstance } from 'fastify';
|
import type { FastifyInstance } from 'fastify';
|
||||||
import { plugins as builtinPlugins } from './plugins/index.js';
|
import { plugins as builtinPlugins } from './plugins/index.js';
|
||||||
|
|
||||||
type Options = {
|
export type SummalyOptions = {
|
||||||
/**
|
/**
|
||||||
* Accept-Language for the request
|
* Accept-Language for the request
|
||||||
*/
|
*/
|
||||||
@ -28,7 +28,7 @@ type Options = {
|
|||||||
/**
|
/**
|
||||||
* Custom Plugins
|
* Custom Plugins
|
||||||
*/
|
*/
|
||||||
plugins?: IPlugin[];
|
plugins?: SummalyPlugin[];
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Custom HTTP agent
|
* Custom HTTP agent
|
||||||
@ -36,26 +36,19 @@ type Options = {
|
|||||||
agent?: Got.Agents;
|
agent?: Got.Agents;
|
||||||
};
|
};
|
||||||
|
|
||||||
type Result = Summary & {
|
export const summalyDefaultOptions = {
|
||||||
/**
|
|
||||||
* The actual url of that web page
|
|
||||||
*/
|
|
||||||
url: string;
|
|
||||||
};
|
|
||||||
|
|
||||||
const defaultOptions = {
|
|
||||||
lang: null,
|
lang: null,
|
||||||
followRedirects: true,
|
followRedirects: true,
|
||||||
plugins: [],
|
plugins: [],
|
||||||
} as Options;
|
} as SummalyOptions;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Summarize an web page
|
* Summarize an web page
|
||||||
*/
|
*/
|
||||||
export const summaly = async (url: string, options?: Options): Promise<Result> => {
|
export const summaly = async (url: string, options?: SummalyOptions): Promise<SummalyResult> => {
|
||||||
if (options?.agent) setAgent(options.agent);
|
if (options?.agent) setAgent(options.agent);
|
||||||
|
|
||||||
const opts = Object.assign(defaultOptions, options);
|
const opts = Object.assign(summalyDefaultOptions, options);
|
||||||
|
|
||||||
const plugins = builtinPlugins.concat(opts.plugins || []);
|
const plugins = builtinPlugins.concat(opts.plugins || []);
|
||||||
|
|
||||||
@ -68,8 +61,8 @@ export const summaly = async (url: string, options?: Options): Promise<Result> =
|
|||||||
actualUrl = url;
|
actualUrl = url;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
const _url = URL.parse(actualUrl, true);
|
const _url = new URL(actualUrl);
|
||||||
|
|
||||||
// Find matching plugin
|
// Find matching plugin
|
||||||
const match = plugins.filter(plugin => plugin.test(_url))[0];
|
const match = plugins.filter(plugin => plugin.test(_url))[0];
|
||||||
@ -78,7 +71,7 @@ export const summaly = async (url: string, options?: Options): Promise<Result> =
|
|||||||
const summary = await (match ? match.summarize : general)(_url, opts.lang || undefined);
|
const summary = await (match ? match.summarize : general)(_url, opts.lang || undefined);
|
||||||
|
|
||||||
if (summary == null) {
|
if (summary == null) {
|
||||||
throw 'failed summarize';
|
throw new Error('failed summarize');
|
||||||
}
|
}
|
||||||
|
|
||||||
return Object.assign(summary, {
|
return Object.assign(summary, {
|
||||||
@ -86,13 +79,13 @@ export const summaly = async (url: string, options?: Options): Promise<Result> =
|
|||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
export default function (fastify: FastifyInstance, options: Options, done: (err?: Error) => void) {
|
export default function (fastify: FastifyInstance, options: SummalyOptions, done: (err?: Error) => void) {
|
||||||
fastify.get<{
|
fastify.get<{
|
||||||
Querystring: {
|
Querystring: {
|
||||||
url?: string;
|
url?: string;
|
||||||
lang?: string;
|
lang?: string;
|
||||||
};
|
};
|
||||||
}>('/url', async (req, reply) => {
|
}>('/', async (req, reply) => {
|
||||||
const url = req.query.url as string;
|
const url = req.query.url as string;
|
||||||
if (url == null) {
|
if (url == null) {
|
||||||
return reply.status(400).send({
|
return reply.status(400).send({
|
||||||
@ -116,4 +109,4 @@ export default function (fastify: FastifyInstance, options: Options, done: (err?
|
|||||||
});
|
});
|
||||||
|
|
||||||
done();
|
done();
|
||||||
}
|
}
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
import * as URL from 'node:url';
|
import type { URL } from 'node:url';
|
||||||
import Summary from './summary.js';
|
import Summary from './summary.js';
|
||||||
|
|
||||||
export interface IPlugin {
|
export interface SummalyPlugin {
|
||||||
test: (url: URL.Url) => boolean;
|
test: (url: URL) => boolean;
|
||||||
summarize: (url: URL.Url, lang?: string) => Promise<Summary>;
|
summarize: (url: URL, lang?: string) => Promise<Summary | null>;
|
||||||
}
|
}
|
||||||
|
@ -1,8 +1,8 @@
|
|||||||
import * as URL from 'node:url';
|
import { URL } from 'node:url';
|
||||||
import { scpaping } from '../utils/got.js';
|
import { scpaping } from '../utils/got.js';
|
||||||
import summary from '../summary.js';
|
import summary from '../summary.js';
|
||||||
|
|
||||||
export function test(url: URL.Url): boolean {
|
export function test(url: URL): boolean {
|
||||||
return url.hostname === 'www.amazon.com' ||
|
return url.hostname === 'www.amazon.com' ||
|
||||||
url.hostname === 'www.amazon.co.jp' ||
|
url.hostname === 'www.amazon.co.jp' ||
|
||||||
url.hostname === 'www.amazon.ca' ||
|
url.hostname === 'www.amazon.ca' ||
|
||||||
@ -19,7 +19,7 @@ export function test(url: URL.Url): boolean {
|
|||||||
url.hostname === 'www.amazon.au';
|
url.hostname === 'www.amazon.au';
|
||||||
}
|
}
|
||||||
|
|
||||||
export async function summarize(url: URL.Url): Promise<summary> {
|
export async function summarize(url: URL): Promise<summary> {
|
||||||
const res = await scpaping(url.href);
|
const res = await scpaping(url.href);
|
||||||
const $ = res.$;
|
const $ = res.$;
|
||||||
|
|
||||||
@ -51,8 +51,10 @@ export async function summarize(url: URL.Url): Promise<summary> {
|
|||||||
player: {
|
player: {
|
||||||
url: playerUrl || null,
|
url: playerUrl || null,
|
||||||
width: playerWidth ? parseInt(playerWidth) : null,
|
width: playerWidth ? parseInt(playerWidth) : null,
|
||||||
height: playerHeight ? parseInt(playerHeight) : null
|
height: playerHeight ? parseInt(playerHeight) : null,
|
||||||
|
allow: playerUrl ? ['fullscreen', 'encrypted-media'] : [],
|
||||||
},
|
},
|
||||||
sitename: 'Amazon'
|
sitename: 'Amazon',
|
||||||
|
activityPub: null,
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
18
src/plugins/branchio-deeplinks.ts
Normal file
18
src/plugins/branchio-deeplinks.ts
Normal file
@ -0,0 +1,18 @@
|
|||||||
|
import { URL } from 'node:url';
|
||||||
|
import { scpaping } from '../utils/got.js';
|
||||||
|
import general from '../general.js';
|
||||||
|
import Summary from '../summary.js';
|
||||||
|
|
||||||
|
export function test(url: URL): boolean {
|
||||||
|
// Branch.io を使用したディープリンクにマッチ
|
||||||
|
return /^[a-zA-Z0-9]+\.app\.link$/.test(url.hostname) ||
|
||||||
|
url.hostname === 'spotify.link';
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function summarize(url: URL, lang: string | null = null): Promise<Summary | null> {
|
||||||
|
// https://help.branch.io/using-branch/docs/creating-a-deep-link#redirections
|
||||||
|
// Web版に強制リダイレクトすることでbranch.ioの独自ページが開くのを防ぐ
|
||||||
|
url.searchParams.append('$web_only', 'true');
|
||||||
|
|
||||||
|
return await general(url, lang);
|
||||||
|
}
|
@ -1,8 +1,10 @@
|
|||||||
import { IPlugin } from '@/iplugin.js';
|
import { SummalyPlugin } from '@/iplugin.js';
|
||||||
import * as amazon from './amazon.js';
|
import * as amazon from './amazon.js';
|
||||||
import * as wikipedia from './wikipedia.js';
|
import * as wikipedia from './wikipedia.js';
|
||||||
|
import * as branchIoDeeplinks from './branchio-deeplinks.js';
|
||||||
|
|
||||||
export const plugins: IPlugin[] = [
|
export const plugins: SummalyPlugin[] = [
|
||||||
amazon,
|
amazon,
|
||||||
wikipedia,
|
wikipedia,
|
||||||
|
branchIoDeeplinks,
|
||||||
];
|
];
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
import * as URL from 'node:url';
|
import { URL } from 'node:url';
|
||||||
import { get } from '../utils/got.js';
|
import { get } from '../utils/got.js';
|
||||||
import debug from 'debug';
|
import debug from 'debug';
|
||||||
import summary from '../summary.js';
|
import summary from '../summary.js';
|
||||||
@ -6,12 +6,12 @@ import clip from './../utils/clip.js';
|
|||||||
|
|
||||||
const log = debug('summaly:plugins:wikipedia');
|
const log = debug('summaly:plugins:wikipedia');
|
||||||
|
|
||||||
export function test(url: URL.Url): boolean {
|
export function test(url: URL): boolean {
|
||||||
if (!url.hostname) return false;
|
if (!url.hostname) return false;
|
||||||
return /\.wikipedia\.org$/.test(url.hostname);
|
return /\.wikipedia\.org$/.test(url.hostname);
|
||||||
}
|
}
|
||||||
|
|
||||||
export async function summarize(url: URL.Url): Promise<summary> {
|
export async function summarize(url: URL): Promise<summary> {
|
||||||
const lang = url.host ? url.host.split('.')[0] : null;
|
const lang = url.host ? url.host.split('.')[0] : null;
|
||||||
const title = url.pathname ? url.pathname.split('/')[2] : null;
|
const title = url.pathname ? url.pathname.split('/')[2] : null;
|
||||||
const endpoint = `https://${lang}.wikipedia.org/w/api.php?format=json&action=query&prop=extracts&exintro=&explaintext=&titles=${title}`;
|
const endpoint = `https://${lang}.wikipedia.org/w/api.php?format=json&action=query&prop=extracts&exintro=&explaintext=&titles=${title}`;
|
||||||
@ -25,7 +25,7 @@ export async function summarize(url: URL.Url): Promise<summary> {
|
|||||||
log(body);
|
log(body);
|
||||||
|
|
||||||
if (!('query' in body) || !('pages' in body.query)) {
|
if (!('query' in body) || !('pages' in body.query)) {
|
||||||
throw 'fetch failed';
|
throw new Error('fetch failed');
|
||||||
}
|
}
|
||||||
|
|
||||||
const info = body.query.pages[Object.keys(body.query.pages)[0]];
|
const info = body.query.pages[Object.keys(body.query.pages)[0]];
|
||||||
@ -38,8 +38,10 @@ export async function summarize(url: URL.Url): Promise<summary> {
|
|||||||
player: {
|
player: {
|
||||||
url: null,
|
url: null,
|
||||||
width: null,
|
width: null,
|
||||||
height: null
|
height: null,
|
||||||
|
allow: [],
|
||||||
},
|
},
|
||||||
sitename: 'Wikipedia'
|
sitename: 'Wikipedia',
|
||||||
|
activityPub: null,
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
@ -1,8 +1,8 @@
|
|||||||
type Summary = {
|
type Summary = {
|
||||||
/**
|
/**
|
||||||
* The description of that web page
|
* The title of that web page
|
||||||
*/
|
*/
|
||||||
description: string | null;
|
title: string | null;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The url of the icon of that web page
|
* The url of the icon of that web page
|
||||||
@ -10,29 +10,41 @@ type Summary = {
|
|||||||
icon: string | null;
|
icon: string | null;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The name of site of that web page
|
* The description of that web page
|
||||||
*/
|
*/
|
||||||
sitename: string | null;
|
description: string | null;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The url of the thumbnail of that web page
|
* The url of the thumbnail of that web page
|
||||||
*/
|
*/
|
||||||
thumbnail: string | null;
|
thumbnail: string | null;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The name of site of that web page
|
||||||
|
*/
|
||||||
|
sitename: string | null;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The player of that web page
|
* The player of that web page
|
||||||
*/
|
*/
|
||||||
player: Player;
|
player: Player;
|
||||||
|
|
||||||
/**
|
|
||||||
* The title of that web page
|
|
||||||
*/
|
|
||||||
title: string | null;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Possibly sensitive
|
* Possibly sensitive
|
||||||
*/
|
*/
|
||||||
sensitive?: boolean;
|
sensitive?: boolean;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The url of the ActivityPub representation of that web page
|
||||||
|
*/
|
||||||
|
activityPub: string | null;
|
||||||
|
};
|
||||||
|
|
||||||
|
export type SummalyResult = Summary & {
|
||||||
|
/**
|
||||||
|
* The actual url of that web page
|
||||||
|
*/
|
||||||
|
url: string;
|
||||||
};
|
};
|
||||||
|
|
||||||
export default Summary;
|
export default Summary;
|
||||||
@ -52,4 +64,9 @@ export type Player = {
|
|||||||
* The height of the player
|
* The height of the player
|
||||||
*/
|
*/
|
||||||
height: number | null;
|
height: number | null;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The allowed permissions of the iframe
|
||||||
|
*/
|
||||||
|
allow: string[];
|
||||||
};
|
};
|
||||||
|
@ -42,8 +42,8 @@ export async function scpaping(url: string, opts?: { lang?: string; }) {
|
|||||||
typeFilter: /^(text\/html|application\/xhtml\+xml)/,
|
typeFilter: /^(text\/html|application\/xhtml\+xml)/,
|
||||||
});
|
});
|
||||||
|
|
||||||
// テスト用
|
// SUMMALY_ALLOW_PRIVATE_IPはテスト用
|
||||||
const allowPrivateIp = process.env.SUMMALY_ALLOW_PRIVATE_IP === 'true';
|
const allowPrivateIp = process.env.SUMMALY_ALLOW_PRIVATE_IP === 'true' || Object.keys(agent).length > 0;
|
||||||
|
|
||||||
if (!allowPrivateIp && response.ip && PrivateIp(response.ip)) {
|
if (!allowPrivateIp && response.ip && PrivateIp(response.ip)) {
|
||||||
throw new StatusError(`Private IP rejected ${response.ip}`, 400, 'Private IP Rejected');
|
throw new StatusError(`Private IP rejected ${response.ip}`, 400, 'Private IP Rejected');
|
||||||
@ -108,16 +108,17 @@ async function getResponse(args: GotOptions) {
|
|||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
return await receiveResponce({ req, typeFilter: args.typeFilter });
|
return await receiveResponse({ req, typeFilter: args.typeFilter });
|
||||||
}
|
}
|
||||||
|
|
||||||
async function receiveResponce<T>(args: { req: Got.CancelableRequest<Got.Response<T>>, typeFilter?: RegExp }) {
|
async function receiveResponse<T>(args: { req: Got.CancelableRequest<Got.Response<T>>, typeFilter?: RegExp }) {
|
||||||
const req = args.req;
|
const req = args.req;
|
||||||
const maxSize = MAX_RESPONSE_SIZE;
|
const maxSize = MAX_RESPONSE_SIZE;
|
||||||
|
|
||||||
req.on('response', (res: Got.Response) => {
|
req.on('response', (res: Got.Response) => {
|
||||||
// Check html
|
// Check html
|
||||||
if (args.typeFilter && !res.headers['content-type']?.match(args.typeFilter)) {
|
if (args.typeFilter && !res.headers['content-type']?.match(args.typeFilter)) {
|
||||||
|
// console.warn(res.headers['content-type']);
|
||||||
req.cancel(`Rejected by type filter ${res.headers['content-type']}`);
|
req.cancel(`Rejected by type filter ${res.headers['content-type']}`);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
@ -1,4 +1,5 @@
|
|||||||
export class StatusError extends Error {
|
export class StatusError extends Error {
|
||||||
|
public name: string;
|
||||||
public statusCode: number;
|
public statusCode: number;
|
||||||
public statusMessage?: string;
|
public statusMessage?: string;
|
||||||
public isPermanentError: boolean;
|
public isPermanentError: boolean;
|
||||||
|
3
test/htmls/activitypub.html
Normal file
3
test/htmls/activitypub.html
Normal file
@ -0,0 +1,3 @@
|
|||||||
|
<!DOCTYPE html>
|
||||||
|
<meta charset="utf-8">
|
||||||
|
<link rel="alternate" type="application/activity+json" href="https://misskey.test/notes/abcdefg">
|
@ -7,6 +7,6 @@
|
|||||||
</head>
|
</head>
|
||||||
<body>
|
<body>
|
||||||
<h1>KISS principle</h1>
|
<h1>KISS principle</h1>
|
||||||
<p>KISS is an acronym for "Keep it simple, stupid" as a design principle noted by the U.S. Navy in 1960.</p>
|
<p>KISS is an acronym for ”Keep it simple, stupid” as a design principle noted by the U.S. Navy in 1960.</p>
|
||||||
</body>
|
</body>
|
||||||
</html>
|
</html>
|
||||||
|
@ -1,9 +1,13 @@
|
|||||||
<!doctype html>
|
<!doctype html>
|
||||||
|
|
||||||
<html>
|
<html lang="en">
|
||||||
<head>
|
<head>
|
||||||
<meta charset="utf-8">
|
<meta charset="utf-8">
|
||||||
<meta property="og:site_name" content="Alice's Site">
|
<meta property="og:site_name" content="Alice's Site">
|
||||||
<title>Strawberry Pasta | Alice's Site</title>
|
<title>Strawberry Pasta | Alice's Site</title>
|
||||||
</head>
|
</head>
|
||||||
|
<body>
|
||||||
|
<h1>Strawberry Pasta</h1>
|
||||||
|
<p>Strawberry pasta is a kind of pasta with strawberry sauce.</p>
|
||||||
|
</body>
|
||||||
</html>
|
</html>
|
||||||
|
13
test/htmls/mixi-sensitive.html
Normal file
13
test/htmls/mixi-sensitive.html
Normal file
@ -0,0 +1,13 @@
|
|||||||
|
<!doctype html>
|
||||||
|
|
||||||
|
<html lang="en">
|
||||||
|
<head>
|
||||||
|
<meta charset="utf-8">
|
||||||
|
<meta property="mixi:content-rating" content="1">
|
||||||
|
<title>SENSITIVE CONTENT!!</title>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<h1>Yo</h1>
|
||||||
|
<p>Hey hey hey syuilo.</p>
|
||||||
|
</body>
|
||||||
|
</html>
|
@ -3,9 +3,10 @@
|
|||||||
<html lang="en">
|
<html lang="en">
|
||||||
<head>
|
<head>
|
||||||
<meta charset="utf-8">
|
<meta charset="utf-8">
|
||||||
|
<title>KISS principle</title>
|
||||||
</head>
|
</head>
|
||||||
<body>
|
<body>
|
||||||
<h1>KISS principle</h1>
|
<h1>KISS principle</h1>
|
||||||
<p>KISS is an acronym for "Keep it simple, stupid" as a design principle noted by the U.S. Navy in 1960.</p>
|
<p>KISS is an acronym for ”Keep it simple, stupid” as a design principle noted by the U.S. Navy in 1960.</p>
|
||||||
</body>
|
</body>
|
||||||
</html>
|
</html>
|
||||||
|
3
test/htmls/oembed-and-og-video.html
Normal file
3
test/htmls/oembed-and-og-video.html
Normal file
@ -0,0 +1,3 @@
|
|||||||
|
<!DOCTYPE html>
|
||||||
|
<meta property="og:video:url" content="https://example.com/embedurl" />
|
||||||
|
<link type="application/json+oembed" href="http://localhost:3060/oembed.json" />
|
3
test/htmls/oembed-and-og.html
Normal file
3
test/htmls/oembed-and-og.html
Normal file
@ -0,0 +1,3 @@
|
|||||||
|
<!DOCTYPE html>
|
||||||
|
<meta property="og:description" content="blobcats rule the world">
|
||||||
|
<link type="application/json+oembed" href="http://localhost:3060/oembed.json" />
|
3
test/htmls/oembed-nonexistent-path.html
Normal file
3
test/htmls/oembed-nonexistent-path.html
Normal file
@ -0,0 +1,3 @@
|
|||||||
|
<!DOCTYPE html>
|
||||||
|
<link type="application/json+oembed" href="http://localhost:3060/oembe.json" />
|
||||||
|
<meta property="og:description" content="nonexistent">
|
2
test/htmls/oembed-relative.html
Normal file
2
test/htmls/oembed-relative.html
Normal file
@ -0,0 +1,2 @@
|
|||||||
|
<!DOCTYPE html>
|
||||||
|
<link type="application/json+oembed" href="oembed.json" />
|
3
test/htmls/oembed-wrong-path.html
Normal file
3
test/htmls/oembed-wrong-path.html
Normal file
@ -0,0 +1,3 @@
|
|||||||
|
<!DOCTYPE html>
|
||||||
|
<link type="application/json+oembed" href="http://localhost:+3060/oembed.json" />
|
||||||
|
<meta property="og:description" content="wrong url">
|
2
test/htmls/oembed.html
Normal file
2
test/htmls/oembed.html
Normal file
@ -0,0 +1,2 @@
|
|||||||
|
<!DOCTYPE html>
|
||||||
|
<link type="application/json+oembed" href="http://localhost:3060/oembed.json" />
|
290
test/index.ts
290
test/index.ts
@ -6,13 +6,16 @@
|
|||||||
|
|
||||||
/* dependencies below */
|
/* dependencies below */
|
||||||
|
|
||||||
import fs from 'node:fs';
|
import fs, { readdirSync } from 'node:fs';
|
||||||
import process from 'node:process';
|
import process from 'node:process';
|
||||||
import fastify from 'fastify';
|
import fastify from 'fastify';
|
||||||
import { summaly } from '../src/index.js';
|
import { summaly } from '../src/index.js';
|
||||||
import { dirname } from 'node:path';
|
import { dirname } from 'node:path';
|
||||||
import { fileURLToPath } from 'node:url';
|
import { fileURLToPath } from 'node:url';
|
||||||
import {expect, jest, test, describe, beforeEach, afterEach} from '@jest/globals';
|
import { expect, jest, test, describe, beforeEach, afterEach } from '@jest/globals';
|
||||||
|
import { Agent as httpAgent } from 'node:http';
|
||||||
|
import { Agent as httpsAgent } from 'node:https';
|
||||||
|
import { StatusError } from '../src/utils/status-error.js';
|
||||||
|
|
||||||
const _filename = fileURLToPath(import.meta.url);
|
const _filename = fileURLToPath(import.meta.url);
|
||||||
const _dirname = dirname(_filename);
|
const _dirname = dirname(_filename);
|
||||||
@ -31,14 +34,77 @@ const host = `http://localhost:${port}`;
|
|||||||
// Display detail of unhandled promise rejection
|
// Display detail of unhandled promise rejection
|
||||||
process.on('unhandledRejection', console.dir);
|
process.on('unhandledRejection', console.dir);
|
||||||
|
|
||||||
let app: ReturnType<typeof fastify>;
|
let app: ReturnType<typeof fastify> | null = null;
|
||||||
|
let n = 0;
|
||||||
|
|
||||||
afterEach(() => {
|
afterEach(async () => {
|
||||||
if (app) return app.close();
|
if (app) {
|
||||||
|
await app.close();
|
||||||
|
app = null;
|
||||||
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
/* tests below */
|
/* tests below */
|
||||||
|
|
||||||
|
test('basic', async () => {
|
||||||
|
app = fastify();
|
||||||
|
app.get('/', (request, reply) => {
|
||||||
|
return reply.send(fs.createReadStream(_dirname + '/htmls/basic.html'));
|
||||||
|
});
|
||||||
|
await app.listen({ port });
|
||||||
|
expect(await summaly(host)).toEqual({
|
||||||
|
title: 'KISS principle',
|
||||||
|
icon: null,
|
||||||
|
description: null,
|
||||||
|
thumbnail: null,
|
||||||
|
player: {
|
||||||
|
url: null,
|
||||||
|
width: null,
|
||||||
|
height: null,
|
||||||
|
"allow": [
|
||||||
|
"autoplay",
|
||||||
|
"encrypted-media",
|
||||||
|
"fullscreen",
|
||||||
|
],
|
||||||
|
},
|
||||||
|
sitename: 'localhost:3060',
|
||||||
|
sensitive: false,
|
||||||
|
url: host,
|
||||||
|
activityPub: null,
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
test('Stage Bye Stage', async () => {
|
||||||
|
// If this test fails, you must rewrite the result data and the example in README.md.
|
||||||
|
|
||||||
|
const summary = await summaly('https://www.youtube.com/watch?v=NMIEAhH_fTU');
|
||||||
|
expect(summary).toEqual(
|
||||||
|
{
|
||||||
|
"title": "【アイドルマスター】「Stage Bye Stage」(歌:島村卯月、渋谷凛、本田未央)",
|
||||||
|
"icon": "https://www.youtube.com/s/desktop/28b0985e/img/favicon.ico",
|
||||||
|
"description": "Website▶https://columbia.jp/idolmaster/Playlist▶https://www.youtube.com/playlist?list=PL83A2998CF3BBC86D2018年7月18日発売予定THE IDOLM@STER CINDERELLA GIRLS CG STAR...",
|
||||||
|
"thumbnail": "https://i.ytimg.com/vi/NMIEAhH_fTU/maxresdefault.jpg",
|
||||||
|
"player": {
|
||||||
|
"url": "https://www.youtube.com/embed/NMIEAhH_fTU?feature=oembed",
|
||||||
|
"width": 200,
|
||||||
|
"height": 113,
|
||||||
|
"allow": [
|
||||||
|
"autoplay",
|
||||||
|
"clipboard-write",
|
||||||
|
"encrypted-media",
|
||||||
|
"picture-in-picture",
|
||||||
|
"web-share",
|
||||||
|
"fullscreen",
|
||||||
|
]
|
||||||
|
},
|
||||||
|
"sitename": "YouTube",
|
||||||
|
"sensitive": false,
|
||||||
|
"activityPub": null,
|
||||||
|
"url": "https://www.youtube.com/watch?v=NMIEAhH_fTU"
|
||||||
|
}
|
||||||
|
);
|
||||||
|
});
|
||||||
|
|
||||||
test('faviconがHTML上で指定されていないが、ルートに存在する場合、正しく設定される', async () => {
|
test('faviconがHTML上で指定されていないが、ルートに存在する場合、正しく設定される', async () => {
|
||||||
app = fastify();
|
app = fastify();
|
||||||
app.get('/', (request, reply) => {
|
app.get('/', (request, reply) => {
|
||||||
@ -66,7 +132,7 @@ test('faviconがHTML上で指定されていなくて、ルートにも存在し
|
|||||||
test('titleがcleanupされる', async () => {
|
test('titleがcleanupされる', async () => {
|
||||||
app = fastify();
|
app = fastify();
|
||||||
app.get('/', (request, reply) => {
|
app.get('/', (request, reply) => {
|
||||||
return reply.send(fs.createReadStream(_dirname + '/htmls/ditry-title.html'));
|
return reply.send(fs.createReadStream(_dirname + '/htmls/dirty-title.html'));
|
||||||
});
|
});
|
||||||
await app.listen({ port });
|
await app.listen({ port });
|
||||||
|
|
||||||
@ -77,15 +143,39 @@ test('titleがcleanupされる', async () => {
|
|||||||
describe('Private IP blocking', () => {
|
describe('Private IP blocking', () => {
|
||||||
beforeEach(() => {
|
beforeEach(() => {
|
||||||
process.env.SUMMALY_ALLOW_PRIVATE_IP = 'false';
|
process.env.SUMMALY_ALLOW_PRIVATE_IP = 'false';
|
||||||
|
app = fastify();
|
||||||
|
app.get('*', (request, reply) => {
|
||||||
|
return reply.send(fs.createReadStream(_dirname + '/htmls/og-title.html'));
|
||||||
|
});
|
||||||
|
return app.listen({ port });
|
||||||
});
|
});
|
||||||
|
|
||||||
test('private ipなサーバーの情報を取得できない', async () => {
|
test('private ipなサーバーの情報を取得できない', async () => {
|
||||||
app = fastify();
|
const summary = await summaly(host).catch((e: StatusError) => e);
|
||||||
app.get('/', (request, reply) => {
|
if (summary instanceof StatusError) {
|
||||||
return reply.send(fs.createReadStream(_dirname + '/htmls/og-title.html'));
|
expect(summary.name).toBe('StatusError');
|
||||||
|
} else {
|
||||||
|
expect(summary).toBeInstanceOf(StatusError);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
test('agentが指定されている場合はprivate ipを許可', async () => {
|
||||||
|
const summary = await summaly(host, {
|
||||||
|
agent: {
|
||||||
|
http: new httpAgent({ keepAlive: true }),
|
||||||
|
https: new httpsAgent({ keepAlive: true }),
|
||||||
|
}
|
||||||
});
|
});
|
||||||
await app.listen({ port });
|
expect(summary.title).toBe('Strawberry Pasta');
|
||||||
expect(() => summaly(host)).rejects.toMatch('Private IP rejected 127.0.0.1');
|
});
|
||||||
|
|
||||||
|
test('agentが空のオブジェクトの場合はprivate ipを許可しない', async () => {
|
||||||
|
const summary = await summaly(host, { agent: {} }).catch((e: StatusError) => e);
|
||||||
|
if (summary instanceof StatusError) {
|
||||||
|
expect(summary.name).toBe('StatusError');
|
||||||
|
} else {
|
||||||
|
expect(summary).toBeInstanceOf(StatusError);
|
||||||
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
afterEach(() => {
|
afterEach(() => {
|
||||||
@ -96,7 +186,7 @@ describe('Private IP blocking', () => {
|
|||||||
describe('OGP', () => {
|
describe('OGP', () => {
|
||||||
test('title', async () => {
|
test('title', async () => {
|
||||||
app = fastify();
|
app = fastify();
|
||||||
app.get('/', (request, reply) => {
|
app.get('*', (request, reply) => {
|
||||||
return reply.send(fs.createReadStream(_dirname + '/htmls/og-title.html'));
|
return reply.send(fs.createReadStream(_dirname + '/htmls/og-title.html'));
|
||||||
});
|
});
|
||||||
await app.listen({ port });
|
await app.listen({ port });
|
||||||
@ -182,6 +272,7 @@ describe('TwitterCard', () => {
|
|||||||
|
|
||||||
const summary = await summaly(host);
|
const summary = await summaly(host);
|
||||||
expect(summary.player.url).toBe('https://example.com/embedurl');
|
expect(summary.player.url).toBe('https://example.com/embedurl');
|
||||||
|
expect(summary.player.allow).toStrictEqual(['autoplay', 'encrypted-media', 'fullscreen']);
|
||||||
});
|
});
|
||||||
|
|
||||||
test('Player detection - Pleroma:video => video', async () => {
|
test('Player detection - Pleroma:video => video', async () => {
|
||||||
@ -193,6 +284,7 @@ describe('TwitterCard', () => {
|
|||||||
|
|
||||||
const summary = await summaly(host);
|
const summary = await summaly(host);
|
||||||
expect(summary.player.url).toBe('https://example.com/embedurl');
|
expect(summary.player.url).toBe('https://example.com/embedurl');
|
||||||
|
expect(summary.player.allow).toStrictEqual(['autoplay', 'encrypted-media', 'fullscreen']);
|
||||||
});
|
});
|
||||||
|
|
||||||
test('Player detection - Pleroma:image => image', async () => {
|
test('Player detection - Pleroma:image => image', async () => {
|
||||||
@ -206,3 +298,177 @@ describe('TwitterCard', () => {
|
|||||||
expect(summary.thumbnail).toBe('https://example.com/imageurl');
|
expect(summary.thumbnail).toBe('https://example.com/imageurl');
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
describe("oEmbed", () => {
|
||||||
|
const setUpFastify = async (oEmbedPath: string, htmlPath = 'htmls/oembed.html') => {
|
||||||
|
app = fastify();
|
||||||
|
app.get('/', (request, reply) => {
|
||||||
|
return reply.send(fs.createReadStream(new URL(htmlPath, import.meta.url)));
|
||||||
|
});
|
||||||
|
app.get('/oembed.json', (request, reply) => {
|
||||||
|
return reply.send(fs.createReadStream(
|
||||||
|
new URL(oEmbedPath, new URL('oembed/', import.meta.url))
|
||||||
|
));
|
||||||
|
});
|
||||||
|
await app.listen({ port });
|
||||||
|
}
|
||||||
|
|
||||||
|
for (const filename of readdirSync(new URL('oembed/invalid', import.meta.url))) {
|
||||||
|
test(`Invalidity test: ${filename}`, async () => {
|
||||||
|
await setUpFastify(`invalid/${filename}`);
|
||||||
|
const summary = await summaly(host);
|
||||||
|
expect(summary.player.url).toBe(null);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
test('basic properties', async () => {
|
||||||
|
await setUpFastify('oembed.json');
|
||||||
|
const summary = await summaly(host);
|
||||||
|
expect(summary.player.url).toBe('https://example.com/');
|
||||||
|
expect(summary.player.width).toBe(500);
|
||||||
|
expect(summary.player.height).toBe(300);
|
||||||
|
});
|
||||||
|
|
||||||
|
test('type: video', async () => {
|
||||||
|
await setUpFastify('oembed-video.json');
|
||||||
|
const summary = await summaly(host);
|
||||||
|
expect(summary.player.url).toBe('https://example.com/');
|
||||||
|
expect(summary.player.width).toBe(500);
|
||||||
|
expect(summary.player.height).toBe(300);
|
||||||
|
});
|
||||||
|
|
||||||
|
test('max height', async () => {
|
||||||
|
await setUpFastify('oembed-too-tall.json');
|
||||||
|
const summary = await summaly(host);
|
||||||
|
expect(summary.player.height).toBe(1024);
|
||||||
|
});
|
||||||
|
|
||||||
|
test('children are ignored', async () => {
|
||||||
|
await setUpFastify('oembed-iframe-child.json');
|
||||||
|
const summary = await summaly(host);
|
||||||
|
expect(summary.player.url).toBe('https://example.com/');
|
||||||
|
});
|
||||||
|
|
||||||
|
test('allows fullscreen', async () => {
|
||||||
|
await setUpFastify('oembed-allow-fullscreen.json');
|
||||||
|
const summary = await summaly(host);
|
||||||
|
expect(summary.player.url).toBe('https://example.com/');
|
||||||
|
expect(summary.player.allow).toStrictEqual(['fullscreen']);
|
||||||
|
});
|
||||||
|
|
||||||
|
test('allows legacy allowfullscreen', async () => {
|
||||||
|
await setUpFastify('oembed-allow-fullscreen-legacy.json');
|
||||||
|
const summary = await summaly(host);
|
||||||
|
expect(summary.player.url).toBe('https://example.com/');
|
||||||
|
expect(summary.player.allow).toStrictEqual(['fullscreen']);
|
||||||
|
});
|
||||||
|
|
||||||
|
test('allows safelisted permissions', async () => {
|
||||||
|
await setUpFastify('oembed-allow-safelisted-permissions.json');
|
||||||
|
const summary = await summaly(host);
|
||||||
|
expect(summary.player.url).toBe('https://example.com/');
|
||||||
|
expect(summary.player.allow).toStrictEqual([
|
||||||
|
'autoplay', 'clipboard-write', 'fullscreen',
|
||||||
|
'encrypted-media', 'picture-in-picture', 'web-share',
|
||||||
|
]);
|
||||||
|
});
|
||||||
|
|
||||||
|
test('ignores rare permissions', async () => {
|
||||||
|
await setUpFastify('oembed-ignore-rare-permissions.json');
|
||||||
|
const summary = await summaly(host);
|
||||||
|
expect(summary.player.url).toBe('https://example.com/');
|
||||||
|
expect(summary.player.allow).toStrictEqual(['autoplay']);
|
||||||
|
});
|
||||||
|
|
||||||
|
test('oEmbed with relative path', async () => {
|
||||||
|
await setUpFastify('oembed.json', 'htmls/oembed-relative.html');
|
||||||
|
const summary = await summaly(host);
|
||||||
|
expect(summary.player.url).toBe('https://example.com/');
|
||||||
|
});
|
||||||
|
|
||||||
|
test('oEmbed with nonexistent path', async () => {
|
||||||
|
await setUpFastify('oembed.json', 'htmls/oembed-nonexistent-path.html');
|
||||||
|
const summary = await summaly(host);
|
||||||
|
expect(summary.player.url).toBe(null);
|
||||||
|
expect(summary.description).toBe('nonexistent');
|
||||||
|
});
|
||||||
|
|
||||||
|
test('oEmbed with wrong path', async () => {
|
||||||
|
await setUpFastify('oembed.json', 'htmls/oembed-wrong-path.html');
|
||||||
|
const summary = await summaly(host);
|
||||||
|
expect(summary.player.url).toBe(null);
|
||||||
|
expect(summary.description).toBe('wrong url');
|
||||||
|
});
|
||||||
|
|
||||||
|
test('oEmbed with OpenGraph', async () => {
|
||||||
|
await setUpFastify('oembed.json', 'htmls/oembed-and-og.html');
|
||||||
|
const summary = await summaly(host);
|
||||||
|
expect(summary.player.url).toBe('https://example.com/');
|
||||||
|
expect(summary.description).toBe('blobcats rule the world');
|
||||||
|
});
|
||||||
|
|
||||||
|
test('Invalid oEmbed with valid OpenGraph', async () => {
|
||||||
|
await setUpFastify('invalid/oembed-insecure.json', 'htmls/oembed-and-og.html');
|
||||||
|
const summary = await summaly(host);
|
||||||
|
expect(summary.player.url).toBe(null);
|
||||||
|
expect(summary.description).toBe('blobcats rule the world');
|
||||||
|
});
|
||||||
|
|
||||||
|
test('oEmbed with og:video', async () => {
|
||||||
|
await setUpFastify('oembed.json', 'htmls/oembed-and-og-video.html');
|
||||||
|
const summary = await summaly(host);
|
||||||
|
expect(summary.player.url).toBe('https://example.com/');
|
||||||
|
expect(summary.player.allow).toStrictEqual([]);
|
||||||
|
});
|
||||||
|
|
||||||
|
test('width: 100%', async () => {
|
||||||
|
await setUpFastify('oembed-percentage-width.json');
|
||||||
|
const summary = await summaly(host);
|
||||||
|
expect(summary.player.width).toBe(null);
|
||||||
|
expect(summary.player.height).toBe(300);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('ActivityPub', () => {
|
||||||
|
test('Basic', async () => {
|
||||||
|
app = fastify();
|
||||||
|
app.get('*', (request, reply) => {
|
||||||
|
return reply.send(fs.createReadStream(_dirname + '/htmls/activitypub.html'));
|
||||||
|
});
|
||||||
|
await app.listen({ port });
|
||||||
|
|
||||||
|
const summary = await summaly(host);
|
||||||
|
expect(summary.activityPub).toBe('https://misskey.test/notes/abcdefg');
|
||||||
|
});
|
||||||
|
|
||||||
|
test('Null', async () => {
|
||||||
|
app = fastify();
|
||||||
|
app.get('*', (request, reply) => {
|
||||||
|
return reply.send(fs.createReadStream(_dirname + '/htmls/basic.html'));
|
||||||
|
});
|
||||||
|
await app.listen({ port });
|
||||||
|
|
||||||
|
const summary = await summaly(host);
|
||||||
|
expect(summary.activityPub).toBe(null);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('sensitive', () => {
|
||||||
|
test('default', async () => {
|
||||||
|
app = fastify();
|
||||||
|
app.get('/', (request, reply) => {
|
||||||
|
return reply.send(fs.createReadStream(_dirname + '/htmls/basic.html'));
|
||||||
|
});
|
||||||
|
await app.listen({ port });
|
||||||
|
expect((await summaly(host)).sensitive).toBe(false);
|
||||||
|
});
|
||||||
|
|
||||||
|
test('mixi:content-rating 1', async () => {
|
||||||
|
app = fastify();
|
||||||
|
app.get('/', (request, reply) => {
|
||||||
|
return reply.send(fs.createReadStream(_dirname + '/htmls/mixi-sensitive.html'));
|
||||||
|
});
|
||||||
|
await app.listen({ port });
|
||||||
|
expect((await summaly(host)).sensitive).toBe(true);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
7
test/oembed/invalid/oembed-child-iframe.json
Normal file
7
test/oembed/invalid/oembed-child-iframe.json
Normal file
@ -0,0 +1,7 @@
|
|||||||
|
{
|
||||||
|
"version": "1.0",
|
||||||
|
"type": "rich",
|
||||||
|
"html": "<div><iframe src='https://example.com/'></iframe>",
|
||||||
|
"width": 500,
|
||||||
|
"height": 300
|
||||||
|
}
|
7
test/oembed/invalid/oembed-double-iframes.json
Normal file
7
test/oembed/invalid/oembed-double-iframes.json
Normal file
@ -0,0 +1,7 @@
|
|||||||
|
{
|
||||||
|
"version": "1.0",
|
||||||
|
"type": "rich",
|
||||||
|
"html": "<iframe src='https://example.com/'></iframe><iframe src='https://example.com/'></iframe>",
|
||||||
|
"width": 500,
|
||||||
|
"height": 300
|
||||||
|
}
|
7
test/oembed/invalid/oembed-future.json
Normal file
7
test/oembed/invalid/oembed-future.json
Normal file
@ -0,0 +1,7 @@
|
|||||||
|
{
|
||||||
|
"version": "11.0",
|
||||||
|
"type": "rich",
|
||||||
|
"html": "<iframe src='https://example.com/'></iframe>",
|
||||||
|
"width": 500,
|
||||||
|
"height": 300
|
||||||
|
}
|
7
test/oembed/invalid/oembed-insecure.json
Normal file
7
test/oembed/invalid/oembed-insecure.json
Normal file
@ -0,0 +1,7 @@
|
|||||||
|
{
|
||||||
|
"version": "1.0",
|
||||||
|
"type": "rich",
|
||||||
|
"html": "<iframe src='http://example.com/'></iframe>",
|
||||||
|
"width": 500,
|
||||||
|
"height": 300
|
||||||
|
}
|
7
test/oembed/invalid/oembed-invalid-height.json
Normal file
7
test/oembed/invalid/oembed-invalid-height.json
Normal file
@ -0,0 +1,7 @@
|
|||||||
|
{
|
||||||
|
"version": "1.0",
|
||||||
|
"type": "rich",
|
||||||
|
"html": "<iframe src='https://example.com/'></iframe>",
|
||||||
|
"width": 500,
|
||||||
|
"height": "blobcat"
|
||||||
|
}
|
6
test/oembed/invalid/oembed-no-height.json
Normal file
6
test/oembed/invalid/oembed-no-height.json
Normal file
@ -0,0 +1,6 @@
|
|||||||
|
{
|
||||||
|
"version": "1.0",
|
||||||
|
"type": "rich",
|
||||||
|
"html": "<iframe src='https://example.com/'></iframe>",
|
||||||
|
"width": 500
|
||||||
|
}
|
6
test/oembed/invalid/oembed-no-version.json
Normal file
6
test/oembed/invalid/oembed-no-version.json
Normal file
@ -0,0 +1,6 @@
|
|||||||
|
{
|
||||||
|
"type": "rich",
|
||||||
|
"html": "<iframe src='https://example.com/'></iframe>",
|
||||||
|
"width": 500,
|
||||||
|
"height": 300
|
||||||
|
}
|
7
test/oembed/invalid/oembed-old.json
Normal file
7
test/oembed/invalid/oembed-old.json
Normal file
@ -0,0 +1,7 @@
|
|||||||
|
{
|
||||||
|
"version": "0.1",
|
||||||
|
"type": "rich",
|
||||||
|
"html": "<iframe src='https://example.com/'></iframe>",
|
||||||
|
"width": 500,
|
||||||
|
"height": 300
|
||||||
|
}
|
7
test/oembed/invalid/oembed-photo.json
Normal file
7
test/oembed/invalid/oembed-photo.json
Normal file
@ -0,0 +1,7 @@
|
|||||||
|
{
|
||||||
|
"version": "1.0",
|
||||||
|
"type": "photo",
|
||||||
|
"url": "https://example.com/example.avif",
|
||||||
|
"width": 300,
|
||||||
|
"height": 300
|
||||||
|
}
|
7
test/oembed/invalid/oembed-too-powerful.json
Normal file
7
test/oembed/invalid/oembed-too-powerful.json
Normal file
@ -0,0 +1,7 @@
|
|||||||
|
{
|
||||||
|
"version": "1.0",
|
||||||
|
"type": "rich",
|
||||||
|
"html": "<iframe src='https://example.com/' allow='camera'></iframe>",
|
||||||
|
"width": 500,
|
||||||
|
"height": 300
|
||||||
|
}
|
7
test/oembed/invalid/oembed-too-powerful2.json
Normal file
7
test/oembed/invalid/oembed-too-powerful2.json
Normal file
@ -0,0 +1,7 @@
|
|||||||
|
{
|
||||||
|
"version": "1.0",
|
||||||
|
"type": "rich",
|
||||||
|
"html": "<iframe src='https://example.com/' allow='fullscreen;camera'></iframe>",
|
||||||
|
"width": 500,
|
||||||
|
"height": 300
|
||||||
|
}
|
7
test/oembed/oembed-allow-fullscreen-legacy.json
Normal file
7
test/oembed/oembed-allow-fullscreen-legacy.json
Normal file
@ -0,0 +1,7 @@
|
|||||||
|
{
|
||||||
|
"version": "1.0",
|
||||||
|
"type": "rich",
|
||||||
|
"html": "<iframe src='https://example.com/' allowfullscreen></iframe>",
|
||||||
|
"width": 500,
|
||||||
|
"height": 300
|
||||||
|
}
|
7
test/oembed/oembed-allow-fullscreen.json
Normal file
7
test/oembed/oembed-allow-fullscreen.json
Normal file
@ -0,0 +1,7 @@
|
|||||||
|
{
|
||||||
|
"version": "1.0",
|
||||||
|
"type": "rich",
|
||||||
|
"html": "<iframe src='https://example.com/' allow='fullscreen'></iframe>",
|
||||||
|
"width": 500,
|
||||||
|
"height": 300
|
||||||
|
}
|
7
test/oembed/oembed-allow-safelisted-permissions.json
Normal file
7
test/oembed/oembed-allow-safelisted-permissions.json
Normal file
@ -0,0 +1,7 @@
|
|||||||
|
{
|
||||||
|
"version": "1.0",
|
||||||
|
"type": "rich",
|
||||||
|
"html": "<iframe src='https://example.com/' allow='autoplay;clipboard-write;fullscreen;encrypted-media;picture-in-picture;web-share'></iframe>",
|
||||||
|
"width": 500,
|
||||||
|
"height": 300
|
||||||
|
}
|
7
test/oembed/oembed-iframe-child.json
Normal file
7
test/oembed/oembed-iframe-child.json
Normal file
@ -0,0 +1,7 @@
|
|||||||
|
{
|
||||||
|
"version": "1.0",
|
||||||
|
"type": "rich",
|
||||||
|
"html": "<iframe src='https://example.com/'><script>alert('Hahaha I take this world')</script></iframe>",
|
||||||
|
"width": 500,
|
||||||
|
"height": 300
|
||||||
|
}
|
7
test/oembed/oembed-ignore-rare-permissions.json
Normal file
7
test/oembed/oembed-ignore-rare-permissions.json
Normal file
@ -0,0 +1,7 @@
|
|||||||
|
{
|
||||||
|
"version": "1.0",
|
||||||
|
"type": "rich",
|
||||||
|
"html": "<iframe src='https://example.com/' allow='autoplay;gyroscope;accelerometer'></iframe>",
|
||||||
|
"width": 500,
|
||||||
|
"height": 300
|
||||||
|
}
|
7
test/oembed/oembed-percentage-width.json
Normal file
7
test/oembed/oembed-percentage-width.json
Normal file
@ -0,0 +1,7 @@
|
|||||||
|
{
|
||||||
|
"version": "1.0",
|
||||||
|
"type": "rich",
|
||||||
|
"html": "<iframe src='https://example.com/'></iframe>",
|
||||||
|
"width": "100%",
|
||||||
|
"height": 300
|
||||||
|
}
|
6
test/oembed/oembed-too-tall.json
Normal file
6
test/oembed/oembed-too-tall.json
Normal file
@ -0,0 +1,6 @@
|
|||||||
|
{
|
||||||
|
"version": "1.0",
|
||||||
|
"type": "rich",
|
||||||
|
"html": "<iframe src='https://example.com/'></iframe>",
|
||||||
|
"height": 3000
|
||||||
|
}
|
7
test/oembed/oembed-video.json
Normal file
7
test/oembed/oembed-video.json
Normal file
@ -0,0 +1,7 @@
|
|||||||
|
{
|
||||||
|
"version": "1.0",
|
||||||
|
"type": "video",
|
||||||
|
"html": "<iframe src='https://example.com/'></iframe>",
|
||||||
|
"width": 500,
|
||||||
|
"height": 300
|
||||||
|
}
|
7
test/oembed/oembed.json
Normal file
7
test/oembed/oembed.json
Normal file
@ -0,0 +1,7 @@
|
|||||||
|
{
|
||||||
|
"version": "1.0",
|
||||||
|
"type": "rich",
|
||||||
|
"html": "<iframe src='https://example.com/'></iframe>",
|
||||||
|
"width": 500,
|
||||||
|
"height": 300
|
||||||
|
}
|
86
tslint.json
86
tslint.json
@ -1,86 +0,0 @@
|
|||||||
{
|
|
||||||
"rules": {
|
|
||||||
"align": [true,
|
|
||||||
"parameters",
|
|
||||||
"statements"
|
|
||||||
],
|
|
||||||
"ban": false,
|
|
||||||
"class-name": true,
|
|
||||||
"comment-format": [true,
|
|
||||||
"check-upper-case"
|
|
||||||
],
|
|
||||||
"curly": true,
|
|
||||||
"eofline": true,
|
|
||||||
"forin": false,
|
|
||||||
"indent": [true, "tabs"],
|
|
||||||
"interface-name": false,
|
|
||||||
"jsdoc-format": true,
|
|
||||||
"label-position": true,
|
|
||||||
"label-undefined": true,
|
|
||||||
"max-line-length": false,
|
|
||||||
"member-access": false,
|
|
||||||
"member-ordering": [true,
|
|
||||||
"static-before-instance",
|
|
||||||
"variables-before-functions"
|
|
||||||
],
|
|
||||||
"no-any": false,
|
|
||||||
"no-arg": true,
|
|
||||||
"no-bitwise": true,
|
|
||||||
"no-console": [true,
|
|
||||||
"debug",
|
|
||||||
"info",
|
|
||||||
"time",
|
|
||||||
"timeEnd",
|
|
||||||
"trace"
|
|
||||||
],
|
|
||||||
"no-consecutive-blank-lines": true,
|
|
||||||
"no-construct": true,
|
|
||||||
"no-constructor-vars": true,
|
|
||||||
"no-debugger": true,
|
|
||||||
"no-duplicate-key": true,
|
|
||||||
"no-shadowed-variable": false,
|
|
||||||
"no-duplicate-variable": true,
|
|
||||||
"no-empty": true,
|
|
||||||
"no-eval": true,
|
|
||||||
"no-internal-module": true,
|
|
||||||
"no-require-imports": false,
|
|
||||||
"no-string-literal": false,
|
|
||||||
"no-switch-case-fall-through": true,
|
|
||||||
"no-trailing-whitespace": true,
|
|
||||||
"no-unreachable": true,
|
|
||||||
"no-unused-expression": true,
|
|
||||||
"no-unused-variable": true,
|
|
||||||
"no-use-before-declare": true,
|
|
||||||
"no-var-keyword": true,
|
|
||||||
"no-var-requires": false,
|
|
||||||
"one-line": [true,
|
|
||||||
"check-catch",
|
|
||||||
"check-whitespace"
|
|
||||||
],
|
|
||||||
"quotemark": false,
|
|
||||||
"radix": true,
|
|
||||||
"semicolon": true,
|
|
||||||
"switch-default": false,
|
|
||||||
"triple-equals": false,
|
|
||||||
"typedef": [true,
|
|
||||||
"call-signature",
|
|
||||||
"property-declaration"
|
|
||||||
],
|
|
||||||
"typedef-whitespace": [true, {
|
|
||||||
"call-signature": "nospace",
|
|
||||||
"index-signature": "nospace",
|
|
||||||
"parameter": "nospace",
|
|
||||||
"property-declaration": "nospace",
|
|
||||||
"variable-declaration": "nospace"
|
|
||||||
}],
|
|
||||||
"use-strict": false,
|
|
||||||
"variable-name": false,
|
|
||||||
"whitespace": [true,
|
|
||||||
"check-branch",
|
|
||||||
"check-decl",
|
|
||||||
"check-operator",
|
|
||||||
"check-separator",
|
|
||||||
"check-type"
|
|
||||||
]
|
|
||||||
}
|
|
||||||
}
|
|
Reference in New Issue
Block a user