Skip to content

Commit dcef3ca

Browse files
**Automated nightly release 72**
commit a3ffbda Merge: ad945bd 0e3d765 Author: Emma <[email protected]> Merge branch 'upstream_development' into development commit 0e3d765 Merge: 651fd1f 64679ca Author: Emma <[email protected]> Merge branch 'development' of https://github.com/FreeTubeApp/FreeTube into upstream_development commit 64679ca Author: J. Lavoie <[email protected]> ... **Full Changelog**: 0.17.1-nightly-71...0.17.1-nightly-72
2 parents 38758ee + a3ffbda commit dcef3ca

40 files changed

+972
-348
lines changed

.github/workflows/flatpak.yml

+4-4
Original file line numberDiff line numberDiff line change
@@ -77,22 +77,22 @@ jobs:
7777
date +"%Y-%m-%d" >> $GITHUB_ENV
7878
echo 'EOF' >> $GITHUB_ENV
7979
- name: Update x64 File Location in yml File
80-
uses: mikefarah/[email protected].1
80+
uses: mikefarah/[email protected].2
8181
with:
8282
# The Command which should be run
8383
cmd: yq w -i io.freetubeapp.FreeTube.yml modules[0].sources[0].url 'https://github.com/FreeTubeApp/FreeTube/releases/download/v${{ steps.sub.outputs.result }}-beta/freetube-${{ steps.sub.outputs.result }}-linux-portable-x64.zip'
8484
- name: Update x64 Hash in yml File
85-
uses: mikefarah/[email protected].1
85+
uses: mikefarah/[email protected].2
8686
with:
8787
# The Command which should be run
8888
cmd: yq w -i io.freetubeapp.FreeTube.yml modules[0].sources[0].sha256 ${{ env.HASH_X64 }}
8989
- name: Update ARM File Location in yml File
90-
uses: mikefarah/[email protected].1
90+
uses: mikefarah/[email protected].2
9191
with:
9292
# The Command which should be run
9393
cmd: yq w -i io.freetubeapp.FreeTube.yml modules[0].sources[1].url 'https://github.com/FreeTubeApp/FreeTube/releases/download/v${{ steps.sub.outputs.result }}-beta/freetube-${{ steps.sub.outputs.result }}-linux-portable-arm64.zip'
9494
- name: Update ARM Hash in yml File
95-
uses: mikefarah/[email protected].1
95+
uses: mikefarah/[email protected].2
9696
with:
9797
# The Command which should be run
9898
cmd: yq w -i io.freetubeapp.FreeTube.yml modules[0].sources[1].sha256 ${{ env.HASH_ARM64 }}

.github/workflows/report.yml

+6-6
Original file line numberDiff line numberDiff line change
@@ -13,7 +13,7 @@ jobs:
1313

1414
# For bug reports
1515
- name: New bug issue
16-
uses: alex-page/[email protected].1
16+
uses: alex-page/[email protected].2
1717
if: contains(github.event.issue.labels.*.name, 'bug') && github.event.action == 'opened'
1818
with:
1919
project: Bug Reports
@@ -22,7 +22,7 @@ jobs:
2222
action: update
2323

2424
- name: Bug issue closed
25-
uses: alex-page/[email protected].1
25+
uses: alex-page/[email protected].2
2626
if: github.event.action == 'closed' || github.event.action == 'deleted'
2727
with:
2828
action: delete
@@ -31,7 +31,7 @@ jobs:
3131
repo-token: ${{ secrets.PUSH_TOKEN }}
3232

3333
- name: Bug issue reopened
34-
uses: alex-page/[email protected].1
34+
uses: alex-page/[email protected].2
3535
if: contains(github.event.issue.labels.*.name, 'bug') && github.event.action == 'reopened'
3636
with:
3737
project: Bug Reports
@@ -41,7 +41,7 @@ jobs:
4141

4242
# For feature requests
4343
- name: New feature issue
44-
uses: alex-page/[email protected].1
44+
uses: alex-page/[email protected].2
4545
if: contains(github.event.issue.labels.*.name, 'enhancement') && github.event.action == 'opened'
4646
with:
4747
project: Feature Requests
@@ -50,7 +50,7 @@ jobs:
5050
action: update
5151

5252
- name: Feature request issue closed
53-
uses: alex-page/[email protected].1
53+
uses: alex-page/[email protected].2
5454
if: github.event.action == 'closed' || github.event.action == 'deleted'
5555
with:
5656
action: delete
@@ -59,7 +59,7 @@ jobs:
5959
repo-token: ${{ secrets.PUSH_TOKEN }}
6060

6161
- name: Feature request issue reopened
62-
uses: alex-page/[email protected].1
62+
uses: alex-page/[email protected].2
6363
if: contains(github.event.issue.labels.*.name, 'enhancement') && github.event.action == 'reopened'
6464
with:
6565
project: Feature Requests

_scripts/dev-runner.js

+5-3
Original file line numberDiff line numberDiff line change
@@ -53,10 +53,12 @@ async function restartElectron() {
5353

5454
electronProcess = spawn(electron, [
5555
path.join(__dirname, '../dist/main.js'),
56-
// '--enable-logging', Enable to show logs from all electron processes
56+
// '--enable-logging', // Enable to show logs from all electron processes
5757
remoteDebugging ? '--inspect=9222' : '',
58-
remoteDebugging ? '--remote-debugging-port=9223' : '',
59-
])
58+
remoteDebugging ? '--remote-debugging-port=9223' : ''
59+
],
60+
// { stdio: 'inherit' } // required for logs to actually appear in the stdout
61+
)
6062

6163
electronProcess.on('exit', (code, _) => {
6264
if (code === relaunchExitCode) {

package.json

+5-4
Original file line numberDiff line numberDiff line change
@@ -63,17 +63,18 @@
6363
"https-proxy-agent": "^5.0.0",
6464
"lodash.debounce": "^4.0.8",
6565
"lodash.isequal": "^4.5.0",
66-
"marked": "^4.0.17",
66+
"marked": "^4.1.1",
6767
"nedb-promises": "^6.2.1",
6868
"opml-to-json": "^1.0.1",
6969
"process": "^0.11.10",
7070
"socks-proxy-agent": "^6.0.0",
7171
"video.js": "7.18.1",
7272
"videojs-contrib-quality-levels": "^2.1.0",
7373
"videojs-http-source-selector": "^1.1.6",
74+
"videojs-mobile-ui": "^0.8.0",
7475
"videojs-overlay": "^2.1.4",
7576
"videojs-vtt-thumbnails-freetube": "0.0.15",
76-
"vue": "^2.7.10",
77+
"vue": "^2.7.13",
7778
"vue-i18n": "^8.27.2",
7879
"vue-observe-visibility": "^1.0.0",
7980
"vue-router": "^3.6.5",
@@ -95,7 +96,7 @@
9596
"copy-webpack-plugin": "^11.0.0",
9697
"cordova": "^11.0.0",
9798
"css-loader": "^6.7.1",
98-
"css-minimizer-webpack-plugin": "^4.1.0",
99+
"css-minimizer-webpack-plugin": "^4.2.2",
99100
"electron": "^21.1.1",
100101
"electron-builder": "^23.6.0",
101102
"eslint": "^7.32.0",
@@ -113,7 +114,7 @@
113114
"lefthook": "^1.1.2",
114115
"mini-css-extract-plugin": "^2.6.1",
115116
"npm-run-all": "^4.1.5",
116-
"prettier": "^2.3.2",
117+
"prettier": "^2.7.1",
117118
"rimraf": "^3.0.2",
118119
"sass": "^1.54.9",
119120
"sass-loader": "^13.0.2",

src/main/ImageCache.js

+73
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,73 @@
1+
// cleanup expired images once every 5 mins
2+
const CLEANUP_INTERVAL = 300_000
3+
4+
// images expire after 2 hours if no expiry information is found in the http headers
5+
const FALLBACK_MAX_AGE = 7200
6+
7+
export class ImageCache {
8+
constructor() {
9+
this._cache = new Map()
10+
11+
setInterval(this._cleanup.bind(this), CLEANUP_INTERVAL)
12+
}
13+
14+
add(url, mimeType, data, expiry) {
15+
this._cache.set(url, { mimeType, data, expiry })
16+
}
17+
18+
has(url) {
19+
return this._cache.has(url)
20+
}
21+
22+
get(url) {
23+
const entry = this._cache.get(url)
24+
25+
if (!entry) {
26+
// this should never happen as the `has` method should be used to check for the existence first
27+
throw new Error(`No image cache entry for ${url}`)
28+
}
29+
30+
return {
31+
data: entry.data,
32+
mimeType: entry.mimeType
33+
}
34+
}
35+
36+
_cleanup() {
37+
// seconds since 1970-01-01 00:00:00
38+
const now = Math.trunc(Date.now() / 1000)
39+
40+
for (const [key, entry] of this._cache.entries()) {
41+
if (entry.expiry <= now) {
42+
this._cache.delete(key)
43+
}
44+
}
45+
}
46+
}
47+
48+
/**
49+
* Extracts the cache expiry timestamp of image from HTTP headers
50+
* @param {Record<string, string>} headers
51+
* @returns a timestamp in seconds
52+
*/
53+
export function extractExpiryTimestamp(headers) {
54+
const maxAgeRegex = /max-age=([0-9]+)/
55+
56+
const cacheControl = headers['cache-control']
57+
if (cacheControl && maxAgeRegex.test(cacheControl)) {
58+
let maxAge = parseInt(cacheControl.match(maxAgeRegex)[1])
59+
60+
if (headers.age) {
61+
maxAge -= parseInt(headers.age)
62+
}
63+
64+
// we don't need millisecond precision, so we can store it as seconds to use less memory
65+
return Math.trunc(Date.now() / 1000) + maxAge
66+
} else if (headers.expires) {
67+
// https://developer.mozilla.org/en-US/docs/Web/HTTP/Headers/Expires
68+
69+
return Math.trunc(Date.parse(headers.expires) / 1000)
70+
} else {
71+
return Math.trunc(Date.now() / 1000) + FALLBACK_MAX_AGE
72+
}
73+
}

src/main/index.js

+121-1
Original file line numberDiff line numberDiff line change
@@ -1,12 +1,14 @@
11
import {
22
app, BrowserWindow, dialog, Menu, ipcMain,
3-
powerSaveBlocker, screen, session, shell, nativeTheme
3+
powerSaveBlocker, screen, session, shell, nativeTheme, net, protocol
44
} from 'electron'
55
import path from 'path'
66
import cp from 'child_process'
77

88
import { IpcChannels, DBActions, SyncEvents } from '../constants'
99
import baseHandlers from '../datastores/handlers/base'
10+
import { extractExpiryTimestamp, ImageCache } from './ImageCache'
11+
import { existsSync } from 'fs'
1012

1113
if (process.argv.includes('--version')) {
1214
app.exit()
@@ -49,6 +51,17 @@ function runApp() {
4951
app.commandLine.appendSwitch('enable-file-cookies')
5052
app.commandLine.appendSwitch('ignore-gpu-blacklist')
5153

54+
// command line switches need to be added before the app ready event first
55+
// that means we can't use the normal settings system as that is asynchonous,
56+
// doing it synchronously ensures that we add it before the event fires
57+
const replaceHttpCache = existsSync(`${app.getPath('userData')}/experiment-replace-http-cache`)
58+
if (replaceHttpCache) {
59+
// the http cache causes excessive disk usage during video playback
60+
// we've got a custom image cache to make up for disabling the http cache
61+
// experimental as it increases RAM use in favour of reduced disk use
62+
app.commandLine.appendSwitch('disable-http-cache')
63+
}
64+
5265
// See: https://stackoverflow.com/questions/45570589/electron-protocol-handler-not-working-on-windows
5366
// remove so we can register each time as we run the app.
5467
app.removeAsDefaultProtocolClient('freetube')
@@ -149,6 +162,113 @@ function runApp() {
149162
})
150163
})
151164

165+
if (replaceHttpCache) {
166+
// in-memory image cache
167+
168+
const imageCache = new ImageCache()
169+
170+
protocol.registerBufferProtocol('imagecache', (request, callback) => {
171+
// Remove `imagecache://` prefix
172+
const url = decodeURIComponent(request.url.substring(13))
173+
if (imageCache.has(url)) {
174+
const cached = imageCache.get(url)
175+
176+
// eslint-disable-next-line node/no-callback-literal
177+
callback({
178+
mimeType: cached.mimeType,
179+
data: cached.data
180+
})
181+
return
182+
}
183+
184+
const newRequest = net.request({
185+
method: request.method,
186+
url
187+
})
188+
189+
// Electron doesn't allow certain headers to be set:
190+
// https://www.electronjs.org/docs/latest/api/client-request#requestsetheadername-value
191+
// also blacklist Origin and Referrer as we don't want to let YouTube know about them
192+
const blacklistedHeaders = ['content-length', 'host', 'trailer', 'te', 'upgrade', 'cookie2', 'keep-alive', 'transfer-encoding', 'origin', 'referrer']
193+
194+
for (const header of Object.keys(request.headers)) {
195+
if (!blacklistedHeaders.includes(header.toLowerCase())) {
196+
newRequest.setHeader(header, request.headers[header])
197+
}
198+
}
199+
200+
newRequest.on('response', (response) => {
201+
const chunks = []
202+
response.on('data', (chunk) => {
203+
chunks.push(chunk)
204+
})
205+
206+
response.on('end', () => {
207+
const data = Buffer.concat(chunks)
208+
209+
const expiryTimestamp = extractExpiryTimestamp(response.headers)
210+
const mimeType = response.headers['content-type']
211+
212+
imageCache.add(url, mimeType, data, expiryTimestamp)
213+
214+
// eslint-disable-next-line node/no-callback-literal
215+
callback({
216+
mimeType,
217+
data: data
218+
})
219+
})
220+
221+
response.on('error', (error) => {
222+
console.error('image cache error', error)
223+
224+
// error objects don't get serialised properly
225+
// https://stackoverflow.com/a/53624454
226+
227+
const errorJson = JSON.stringify(error, (key, value) => {
228+
if (value instanceof Error) {
229+
return {
230+
// Pull all enumerable properties, supporting properties on custom Errors
231+
...value,
232+
// Explicitly pull Error's non-enumerable properties
233+
name: value.name,
234+
message: value.message,
235+
stack: value.stack
236+
}
237+
}
238+
239+
return value
240+
})
241+
242+
// eslint-disable-next-line node/no-callback-literal
243+
callback({
244+
statusCode: response.statusCode ?? 400,
245+
mimeType: 'application/json',
246+
data: Buffer.from(errorJson)
247+
})
248+
})
249+
})
250+
251+
newRequest.end()
252+
})
253+
254+
const imageRequestFilter = { urls: ['https://*/*', 'http://*/*'] }
255+
session.defaultSession.webRequest.onBeforeRequest(imageRequestFilter, (details, callback) => {
256+
// the requests made by the imagecache:// handler to fetch the image,
257+
// are allowed through, as their resourceType is 'other'
258+
if (details.resourceType === 'image') {
259+
// eslint-disable-next-line node/no-callback-literal
260+
callback({
261+
redirectURL: `imagecache://${encodeURIComponent(details.url)}`
262+
})
263+
} else {
264+
// eslint-disable-next-line node/no-callback-literal
265+
callback({})
266+
}
267+
})
268+
269+
// --- end of `if experimentsDisableDiskCache` ---
270+
}
271+
152272
await createWindow()
153273

154274
if (isDev) {

0 commit comments

Comments
 (0)