mirror of
https://github.com/yang991178/fluent-reader.git
synced 2025-04-03 21:21:17 +02:00
use fetch api instead of node http
This commit is contained in:
parent
8263e1095c
commit
eecf8b57a5
1
dist/styles.css
vendored
1
dist/styles.css
vendored
@ -560,6 +560,7 @@ img.favicon {
|
|||||||
align-items: center;
|
align-items: center;
|
||||||
color: var(--neutralSecondary);
|
color: var(--neutralSecondary);
|
||||||
font-size: 14px;
|
font-size: 14px;
|
||||||
|
user-select: none;
|
||||||
}
|
}
|
||||||
|
|
||||||
.info {
|
.info {
|
||||||
|
13
package.json
13
package.json
@ -22,7 +22,10 @@
|
|||||||
"output": "./bin/${platform}/${arch}/"
|
"output": "./bin/${platform}/${arch}/"
|
||||||
},
|
},
|
||||||
"win": {
|
"win": {
|
||||||
"target": [ "nsis", "appx" ]
|
"target": [
|
||||||
|
"nsis",
|
||||||
|
"appx"
|
||||||
|
]
|
||||||
},
|
},
|
||||||
"appx": {
|
"appx": {
|
||||||
"applicationId": "FluentReader",
|
"applicationId": "FluentReader",
|
||||||
@ -43,7 +46,9 @@
|
|||||||
"deleteAppDataOnUninstall": true
|
"deleteAppDataOnUninstall": true
|
||||||
},
|
},
|
||||||
"mac": {
|
"mac": {
|
||||||
"target": [ "dmg" ]
|
"target": [
|
||||||
|
"dmg"
|
||||||
|
]
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
@ -55,17 +60,14 @@
|
|||||||
"@types/redux": "^3.6.0",
|
"@types/redux": "^3.6.0",
|
||||||
"@types/redux-thunk": "^2.1.0",
|
"@types/redux-thunk": "^2.1.0",
|
||||||
"@types/reselect": "^2.2.0",
|
"@types/reselect": "^2.2.0",
|
||||||
"@yang991178/electron-proxy-agent": "^1.2.1",
|
|
||||||
"@yang991178/rss-parser": "^3.8.1",
|
"@yang991178/rss-parser": "^3.8.1",
|
||||||
"electron": "^9.0.4",
|
"electron": "^9.0.4",
|
||||||
"electron-builder": "^22.7.0",
|
"electron-builder": "^22.7.0",
|
||||||
"electron-react-devtools": "^0.5.3",
|
"electron-react-devtools": "^0.5.3",
|
||||||
"electron-store": "^5.2.0",
|
"electron-store": "^5.2.0",
|
||||||
"electron-window-state": "^5.0.3",
|
"electron-window-state": "^5.0.3",
|
||||||
"favicon": "0.0.2",
|
|
||||||
"html-webpack-plugin": "^4.3.0",
|
"html-webpack-plugin": "^4.3.0",
|
||||||
"nedb": "^1.8.0",
|
"nedb": "^1.8.0",
|
||||||
"pac-proxy-agent": "^4.1.0",
|
|
||||||
"react": "^16.13.1",
|
"react": "^16.13.1",
|
||||||
"react-dom": "^16.13.1",
|
"react-dom": "^16.13.1",
|
||||||
"react-intl-universal": "^2.2.5",
|
"react-intl-universal": "^2.2.5",
|
||||||
@ -74,7 +76,6 @@
|
|||||||
"redux-devtools": "^3.5.0",
|
"redux-devtools": "^3.5.0",
|
||||||
"redux-thunk": "^2.3.0",
|
"redux-thunk": "^2.3.0",
|
||||||
"reselect": "^4.0.0",
|
"reselect": "^4.0.0",
|
||||||
"simplebar-react": "^2.2.0",
|
|
||||||
"ts-loader": "^7.0.4",
|
"ts-loader": "^7.0.4",
|
||||||
"typescript": "^3.9.2",
|
"typescript": "^3.9.2",
|
||||||
"webpack": "^4.43.0",
|
"webpack": "^4.43.0",
|
||||||
|
@ -164,6 +164,9 @@ class AppTab extends React.Component<AppTabProps, AppTabState> {
|
|||||||
text={intl.get("app.setPac")} />
|
text={intl.get("app.setPac")} />
|
||||||
</Stack.Item>
|
</Stack.Item>
|
||||||
</Stack>
|
</Stack>
|
||||||
|
<span className="settings-hint up">
|
||||||
|
{intl.get("app.pacHint")}
|
||||||
|
</span>
|
||||||
</form>}
|
</form>}
|
||||||
|
|
||||||
<Label>{intl.get("app.cleanup")}</Label>
|
<Label>{intl.get("app.cleanup")}</Label>
|
||||||
|
@ -66,6 +66,9 @@ app.on("second-instance", () => {
|
|||||||
})
|
})
|
||||||
|
|
||||||
app.on("window-all-closed", function () {
|
app.on("window-all-closed", function () {
|
||||||
|
if (mainWindow) {
|
||||||
|
mainWindow.webContents.session.clearStorageData({ storages: ["cookies"] })
|
||||||
|
}
|
||||||
mainWindow = null
|
mainWindow = null
|
||||||
if (restarting) {
|
if (restarting) {
|
||||||
init()
|
init()
|
||||||
|
@ -141,6 +141,7 @@
|
|||||||
"enableProxy": "Enable Proxy",
|
"enableProxy": "Enable Proxy",
|
||||||
"badUrl": "Invalid URL",
|
"badUrl": "Invalid URL",
|
||||||
"pac": "PAC Address",
|
"pac": "PAC Address",
|
||||||
"setPac": "Set PAC"
|
"setPac": "Set PAC",
|
||||||
|
"pacHint": "For Socks proxies, it is recommended for PAC to return \"SOCKS5\" for proxy-side DNS. Turning off proxy requires restart."
|
||||||
}
|
}
|
||||||
}
|
}
|
@ -141,6 +141,7 @@
|
|||||||
"enableProxy": "启用代理",
|
"enableProxy": "启用代理",
|
||||||
"badUrl": "请正确输入URL",
|
"badUrl": "请正确输入URL",
|
||||||
"pac": "PAC地址",
|
"pac": "PAC地址",
|
||||||
"setPac": "设置PAC"
|
"setPac": "设置PAC",
|
||||||
|
"pacHint": "对于Socks代理建议PAC返回“SOCKS5”以启用代理端解析。关闭代理需重启应用后生效。"
|
||||||
}
|
}
|
||||||
}
|
}
|
@ -163,7 +163,7 @@ export function fetchItems(): AppThunk<Promise<void>> {
|
|||||||
if (r.status === "fulfilled") items.push(...r.value)
|
if (r.status === "fulfilled") items.push(...r.value)
|
||||||
else {
|
else {
|
||||||
console.log(r.reason)
|
console.log(r.reason)
|
||||||
dispatch(fetchItemsFailure(getState().sources[i], r.reason))
|
dispatch(fetchItemsFailure(sources[i], r.reason))
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
insertItems(items)
|
insertItems(items)
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
import Parser = require("@yang991178/rss-parser")
|
import Parser = require("@yang991178/rss-parser")
|
||||||
import intl = require("react-intl-universal")
|
import intl = require("react-intl-universal")
|
||||||
import * as db from "../db"
|
import * as db from "../db"
|
||||||
import { rssParser, faviconPromise, ActionStatus, AppThunk } from "../utils"
|
import { rssParser, fetchFavicon, ActionStatus, AppThunk, parseRSS } from "../utils"
|
||||||
import { RSSItem, insertItems, ItemActionTypes, FETCH_ITEMS, MARK_READ, MARK_UNREAD, MARK_ALL_READ } from "./item"
|
import { RSSItem, insertItems, ItemActionTypes, FETCH_ITEMS, MARK_READ, MARK_UNREAD, MARK_ALL_READ } from "./item"
|
||||||
import { SourceGroup } from "./group"
|
import { SourceGroup } from "./group"
|
||||||
import { saveSettings } from "./app"
|
import { saveSettings } from "./app"
|
||||||
@ -29,21 +29,15 @@ export class RSSSource {
|
|||||||
}
|
}
|
||||||
|
|
||||||
async fetchMetaData(parser: Parser) {
|
async fetchMetaData(parser: Parser) {
|
||||||
let feed = await parser.parseURL(this.url)
|
let feed = await parseRSS(this.url)
|
||||||
if (!this.name) {
|
if (!this.name) {
|
||||||
if (feed.title) this.name = feed.title.trim()
|
if (feed.title) this.name = feed.title.trim()
|
||||||
this.name = this.name || intl.get("sources.untitled")
|
this.name = this.name || intl.get("sources.untitled")
|
||||||
}
|
}
|
||||||
let domain = this.url.split("/").slice(0, 3).join("/")
|
let domain = this.url.split("/").slice(0, 3).join("/")
|
||||||
let f: string = null
|
|
||||||
try {
|
try {
|
||||||
f = await faviconPromise(domain)
|
let f = await fetchFavicon(domain)
|
||||||
if (f === null) f = domain + "/favicon.ico"
|
if (f !== null) this.iconurl = f
|
||||||
let result = await fetch(f)
|
|
||||||
if (result.status == 200 && result.headers.has("Content-Type")
|
|
||||||
&& result.headers.get("Content-Type").startsWith("image")) {
|
|
||||||
this.iconurl = f
|
|
||||||
}
|
|
||||||
} finally {
|
} finally {
|
||||||
return feed
|
return feed
|
||||||
}
|
}
|
||||||
@ -82,7 +76,7 @@ export class RSSSource {
|
|||||||
}
|
}
|
||||||
|
|
||||||
static async fetchItems(source: RSSSource, parser: Parser) {
|
static async fetchItems(source: RSSSource, parser: Parser) {
|
||||||
let feed = await parser.parseURL(source.url)
|
let feed = await parseRSS(source.url)
|
||||||
db.sdb.update({ sid: source.sid }, { $set: { lastFetched: new Date() } })
|
db.sdb.update({ sid: source.sid }, { $set: { lastFetched: new Date() } })
|
||||||
return await this.checkItems(source, feed.items)
|
return await this.checkItems(source, feed.items)
|
||||||
}
|
}
|
||||||
|
@ -35,12 +35,11 @@ export function setProxy(address = null) {
|
|||||||
} else {
|
} else {
|
||||||
store.set(PAC_STORE_KEY, address)
|
store.set(PAC_STORE_KEY, address)
|
||||||
}
|
}
|
||||||
remote.getCurrentWebContents().session.setProxy({
|
if (getProxyStatus()) {
|
||||||
pacScript: getProxyStatus() ? address : ""
|
let rules = { pacScript: address }
|
||||||
})
|
remote.getCurrentWebContents().session.setProxy(rules)
|
||||||
remote.session.fromPartition("sandbox").setProxy({
|
remote.session.fromPartition("sandbox").setProxy(rules)
|
||||||
pacScript: getProxyStatus() ? address : ""
|
}
|
||||||
})
|
|
||||||
}
|
}
|
||||||
|
|
||||||
const VIEW_STORE_KEY = "view"
|
const VIEW_STORE_KEY = "view"
|
||||||
|
@ -17,41 +17,61 @@ export type AppThunk<ReturnType = void> = ThunkAction<
|
|||||||
export type AppDispatch = ThunkDispatch<RootState, undefined, AnyAction>
|
export type AppDispatch = ThunkDispatch<RootState, undefined, AnyAction>
|
||||||
|
|
||||||
import Parser = require("@yang991178/rss-parser")
|
import Parser = require("@yang991178/rss-parser")
|
||||||
const customFields = {
|
|
||||||
item: ["thumb", "image", ["content:encoded", "fullContent"]] as Parser.CustomFieldItem[]
|
|
||||||
}
|
|
||||||
|
|
||||||
import ElectronProxyAgent = require("@yang991178/electron-proxy-agent")
|
|
||||||
import { ViewType } from "./models/page"
|
|
||||||
import { IPartialTheme } from "@fluentui/react"
|
|
||||||
import { SourceGroup } from "./models/group"
|
|
||||||
let agent = new ElectronProxyAgent(remote.getCurrentWebContents().session)
|
|
||||||
export const rssParser = new Parser({
|
export const rssParser = new Parser({
|
||||||
customFields: customFields,
|
customFields: {
|
||||||
requestOptions: {
|
item: ["thumb", "image", ["content:encoded", "fullContent"]] as Parser.CustomFieldItem[]
|
||||||
agent: agent
|
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
|
export async function parseRSS(url: string) {
|
||||||
|
try {
|
||||||
|
let result = await fetch(url, { credentials: "omit" })
|
||||||
|
if (result.ok) {
|
||||||
|
return await rssParser.parseString(await result.text())
|
||||||
|
} else {
|
||||||
|
throw new Error(result.statusText)
|
||||||
|
}
|
||||||
|
} catch {
|
||||||
|
throw new Error("A network error has occured.")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
export const domParser = new DOMParser()
|
export const domParser = new DOMParser()
|
||||||
|
|
||||||
const favicon = require("favicon")
|
import Url = require("url")
|
||||||
export function faviconPromise(url: string): Promise<string> {
|
export async function fetchFavicon(url: string) {
|
||||||
return new Promise<string>((resolve, reject) => {
|
try {
|
||||||
favicon(url, (err, icon: string) => {
|
let result = await fetch(url, { credentials: "omit" })
|
||||||
if (err) reject(err)
|
if (result.ok) {
|
||||||
else if (!icon) resolve(icon)
|
let html = await result.text()
|
||||||
else {
|
let dom = domParser.parseFromString(html, "text/html")
|
||||||
let parts = icon.split("//")
|
let links = dom.getElementsByTagName("link")
|
||||||
resolve(parts[0] + "//" + parts[parts.length - 1])
|
for (let link of links) {
|
||||||
|
let rel = link.getAttribute("rel")
|
||||||
|
if ((rel === "icon" || rel === "shortcut icon") && link.hasAttribute("href")) {
|
||||||
|
let href = link.getAttribute("href")
|
||||||
|
let parsedUrl = Url.parse(url)
|
||||||
|
if (href.startsWith("//")) return parsedUrl.protocol + href
|
||||||
|
else if (href.startsWith("/")) return url + href
|
||||||
|
else return href
|
||||||
|
}
|
||||||
}
|
}
|
||||||
})
|
}
|
||||||
})
|
url = url + "/favicon.ico"
|
||||||
|
result = await fetch(url, { credentials: "omit" })
|
||||||
|
if (result.status == 200 && result.headers.has("Content-Type")
|
||||||
|
&& result.headers.get("Content-Type").startsWith("image")) {
|
||||||
|
return url
|
||||||
|
}
|
||||||
|
return null
|
||||||
|
} catch {
|
||||||
|
return null
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
export function htmlDecode(input: string) {
|
export function htmlDecode(input: string) {
|
||||||
var doc = domParser.parseFromString(input, "text/html");
|
var doc = domParser.parseFromString(input, "text/html")
|
||||||
return doc.documentElement.textContent;
|
return doc.documentElement.textContent
|
||||||
}
|
}
|
||||||
|
|
||||||
export function openExternal(url: string) {
|
export function openExternal(url: string) {
|
||||||
|
Loading…
x
Reference in New Issue
Block a user