Adding gc tests: fixing a memory leak with C++ URLRequest objects.

This commit is contained in:
ali.ibrahim 2016-10-17 19:02:25 +02:00
parent 9b4e9c642a
commit c198828e58
4 changed files with 95 additions and 27 deletions

View file

@ -215,6 +215,7 @@ URLRequest::BuildArgsArray(ArgTypes... args) const {
template <typename... ArgTypes>
void URLRequest::EmitRequestEvent(ArgTypes... args) {
v8::HandleScope handle_scope(isolate());
auto arguments = BuildArgsArray(args...);
v8::Local<v8::Function> _emitRequestEvent;
auto wrapper = GetWrapper();
@ -225,6 +226,7 @@ void URLRequest::EmitRequestEvent(ArgTypes... args) {
template <typename... ArgTypes>
void URLRequest::EmitResponseEvent(ArgTypes... args) {
v8::HandleScope handle_scope(isolate());
auto arguments = BuildArgsArray(args...);
v8::Local<v8::Function> _emitResponseEvent;
auto wrapper = GetWrapper();

View file

@ -95,7 +95,7 @@ URLRequest.prototype._emitRequestEvent = function (async, ...rest) {
URLRequest.prototype._emitResponseEvent = function (async, ...rest) {
if (async) {
process.nextTick(() => {
this.clientRequest.emit.apply(this._response, rest)
this._response.emit.apply(this._response, rest)
})
} else {
this._response.emit.apply(this._response, rest)

View file

@ -26,7 +26,7 @@ const kOneKiloByte = 1024
const kOneMegaByte = kOneKiloByte * kOneKiloByte
describe('net module', function () {
this.timeout(0)
// this.timeout(0)
describe('HTTP basics', function () {
let server
beforeEach(function (done) {
@ -1155,42 +1155,108 @@ describe('net module', function () {
urlRequest.end()
})
})
describe('Stability and performance', function(done) {
describe('Stability and performance', function (done) {
let server
beforeEach(function (done) {
server = http.createServer()
server.listen(0, '127.0.0.1', function () {
server.url = 'http://127.0.0.1:' + server.address().port
done()
})
server = http.createServer()
server.listen(0, '127.0.0.1', function () {
server.url = 'http://127.0.0.1:' + server.address().port
done()
})
})
afterEach(function () {
server.close()
server = null
server.close()
server = null
})
it.only('should free unreferenced, never-started request objects', function(done) {
it('should free unreferenced, never-started request objects', function (done) {
const requestUrl = '/requestUrl'
ipcRenderer.on('api-net-spec-done', function() {
ipcRenderer.once('api-net-spec-unused-done', function () {
done()
})
const testCode = `
// Load the net module in the main browser process.
const {net} = require('electron')
const urlRequest = net.request('${server.url}${requestUrl}')
process.nextTick(function() {
net._RequestGarbageCollectionForTesting()
event.sender.send('api-net-spec-done')
ipcRenderer.send('eval', `
const {net} = require('electron')
const urlRequest = net.request('${server.url}${requestUrl}')
process.nextTick(function () {
net._RequestGarbageCollectionForTesting()
event.sender.send('api-net-spec-unused-done')
})
`)
})
it('should not collect on-going requests', function (done) {
const requestUrl = '/requestUrl'
server.on('request', function (request, response) {
switch (request.url) {
case requestUrl:
response.statusCode = 200
response.statusMessage = 'OK'
response.write(randomString(kOneKiloByte))
ipcRenderer.once('api-net-spec-ongoing-resume-response', function () {
response.write(randomString(kOneKiloByte))
response.end()
})
break
default:
assert(false)
}
})
`
ipcRenderer.send('eval', testCode)
ipcRenderer.once('api-net-spec-ongoing-done', function () {
done()
})
// Execute below code directly within the browser context without
// using the remote module.
ipcRenderer.send('eval', `
const {net} = require('electron')
const urlRequest = net.request('${server.url}${requestUrl}')
urlRequest.on('response', function (response) {
response.on('data', function () {
})
response.on('end', function () {
event.sender.send('api-net-spec-ongoing-done')
})
process.nextTick(function () {
// Trigger a garbage collection.
net._RequestGarbageCollectionForTesting()
event.sender.send('api-net-spec-ongoing-resume-response')
})
})
urlRequest.end()
`)
})
it.skip('should not collect on-going requests', function(done) {
assert(false)
})
it.skip('should collect unreferenced, ended requests', function(done) {
assert(false)
it('should collect unreferenced, ended requests', function (done) {
const requestUrl = '/requestUrl'
server.on('request', function (request, response) {
switch (request.url) {
case requestUrl:
response.statusCode = 200
response.statusMessage = 'OK'
response.end()
break
default:
assert(false)
}
})
ipcRenderer.once('api-net-spec-done', function () {
done()
})
ipcRenderer.send('eval', `
const {net} = require('electron')
const urlRequest = net.request('${server.url}${requestUrl}')
urlRequest.on('response', function (response) {
response.on('data', function () {
})
response.on('end', function () {
})
})
urlRequest.on('close', function () {
process.nextTick(function () {
net._RequestGarbageCollectionForTesting()
event.sender.send('api-net-spec-done')
})
})
urlRequest.end()
`)
})
})
})

View file

@ -8,7 +8,7 @@ const ipcMain = electron.ipcMain
const dialog = electron.dialog
const BrowserWindow = electron.BrowserWindow
const protocol = electron.protocol
const v8 = require('v8');
const v8 = require('v8')
const Coverage = require('electabul').Coverage
const fs = require('fs')