use fetch api instead of node http

This commit is contained in:
刘浩远 2020-06-18 13:42:14 +08:00
parent 8263e1095c
commit eecf8b57a5
10 changed files with 74 additions and 51 deletions

1
dist/styles.css vendored
View File

@ -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 {

View File

@ -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",

View File

@ -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>

View File

@ -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()

View File

@ -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."
} }
} }

View File

@ -141,6 +141,7 @@
"enableProxy": "启用代理", "enableProxy": "启用代理",
"badUrl": "请正确输入URL", "badUrl": "请正确输入URL",
"pac": "PAC地址", "pac": "PAC地址",
"setPac": "设置PAC" "setPac": "设置PAC",
"pacHint": "对于Socks代理建议PAC返回“SOCKS5”以启用代理端解析。关闭代理需重启应用后生效。"
} }
} }

View File

@ -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)

View File

@ -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)
} }

View File

@ -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"

View File

@ -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) {