node.js – pedidos http fáceis com compactação gzip / deflate

Eu estou tentando descobrir como a melhor maneira de enviar facilmente solicitações HTTP / HTTPS e lidar com respostas compactadas gzip / deflate junto com cookies.

O melhor que encontrei foi https://github.com/mikeal/request, que lida com tudo, exceto a compactação. Existe um módulo ou método que fará tudo o que eu pedir?

Se não, posso combinar a solicitação e o zlib de alguma maneira? Eu tentei combinar o zlib e o http.ServerRequest, e ele falhou miseravelmente.

Obrigado!

Para qualquer um que se deparasse com isso nos últimos tempos, a biblioteca de solicitações suporta a descompactação gzip fora da checkbox agora. Use da seguinte maneira:

 request( { method: 'GET' , uri: 'http://www.google.com' , gzip: true } , function (error, response, body) { // body is the decompressed response body console.log('server encoded the data as: ' + (response.headers['content-encoding'] || 'identity')) console.log('the decoded data is: ' + body) } ) 

Do leia-me do github https://github.com/request/request

gzip – Se for verdade, adicione um header Accept-Encoding para solicitar codificações de conteúdo compactado do servidor (se ainda não estiver presente) e decodificar codificações de conteúdo suportadas na resposta. Nota: A decodificação automática do conteúdo da resposta é executada nos dados do corpo retornados por solicitação (através do stream de solicitação e transmitidos para a function de retorno de chamada), mas não é executada no stream de resposta (disponível no evento de resposta) que é o http não modificado .IncomingMessage object que pode conter dados compactados. Veja o exemplo abaixo.

Aqui está um exemplo de trabalho que gunzips a resposta

 function gunzipJSON(response){ var gunzip = zlib.createGunzip(); var json = ""; gunzip.on('data', function(data){ json += data.toString(); }); gunzip.on('end', function(){ parseJSON(json); }); response.pipe(gunzip); } 

Código completo: https://gist.github.com/0xPr0xy/5002984

Confira os exemplos em http://nodejs.org/docs/v0.6.0/api/zlib.html#examples

O zlib agora está embutido no nó.

 //functions.js: var ce=require('cloneextend'); //console.log({aa:'bb',dd:new Date('10/10/2011')}); //console.log(ce.clone({aa:'bb',dd:new Date('10/10/2011')})); exports.cloneextend=ce; exports.clone=ce.clone; exports.extend=ce.extend; ////////////request var request1=require('request'); var Iconv=require('iconv').Iconv; var iconv_utf8_to_latin = new Iconv('utf-8','iso-8859-1'); var iconv_iso8859_8i_to_utf8 = new Iconv('iso-8859-8','utf-8'); var iconv_utf8_to_iso8859_8i = new Iconv('utf-8','iso-8859-8'); exports.iconv_iso8859_8i_to_utf8=iconv_iso8859_8i_to_utf8; exports.iconv_utf8_to_iso8859_8i=iconv_utf8_to_iso8859_8i; var zlib=require('zlib'); function request_unzip(options,cb) { var enc=options.encoding; options.encoding=null; var r=request1(options) .on('response',function(response) { var bufarr=[]; var errored=false; switch (response.headers['content-encoding']) { // or, just use zlib.createUnzip() to handle both cases case 'gzip': case 'deflate': if(response.headers['content-encoding']=='gzip') var zpipe=zlib.createGunzip(); else var zpipe=zlib.createInflate(); zpipe .on('data', function(d){bufarr.push(d);}) .on('end', function(){ if(errored) return;errored=true; cb(null,response, enc?Buffer.concat(bufarr).toString(enc):Buffer.concat(bufarr) ); }) .on('error', function(err){ if(errored) return;errored=true; cb(err,response,null);}); response.pipe(zpipe); response .on('error', function(err){ if(errored) return;errored=true; cb(err,response,null);}); break; default: response .on('data', function(d){bufarr.push(d);}) .on('end', function(){ if(errored) return;errored=true; cb(null,response, enc?Buffer.concat(bufarr).toString(enc):Buffer.concat(bufarr) ); }) .on('error', function(err){ if(errored) return;errored=true; cb(err,response,null);}); break; } }); return r; } function request(options,cb)// a request that fixes encoding { if(options.encoding=='iso-8859-8') { options.encoding='binary'; return request_unzip(options, function(error,request,data){ if(data===undefined||data===null) { data2=null; cb(error,request,data2); } else { try{ cb(error,request, iconv_iso8859_8i_to_utf8.convert(iconv_utf8_to_latin.convert(data)).toString('utf8') //conver buffer to string ); } catch(e) { data2=null; error=e; cb(error,request,data2); } } }); } else return request_unzip(options,cb); } request.__proto__=request1; exports.request=request; ie9headers= // no var goes to global { followAllRedirects:true, headers: { "Accept": "text/html,application/xhtml+xml,application/xml;q=0.9,*/*;q=0.8", "Accept-Language": "he-IL,he;q=0.8,en-US;q=0.6,en;q=0.4", "User-Agent": "Mozilla/5.0 (Windows NT 6.1) AppleWebKit/537.22 (KHTML, like Gecko) Chrome/25.0.1364.172 Safari/537.22",//"Mozilla/5.0 (compatible; MSIE 9.0; Windows NT 6.0; Trident/5.0)", "Accept-Charset": "windows-1255,utf-8;q=0.7,*;q=0.3", "Accept-Encoding": "gzip,deflate,sdch" } } /// example: f=require('./function.js'); //goes global function getsomething(cb){ function getit(){ f.request(f.extend({jar:j,url:myurl, headers:{Referer:url}, encoding:'UTF-8' },ie9headers),function(error,request,data) { if(error) setTimeout(getit,1000); //console.log("data",data); var parsed=myparse(data); cb(parsed); });} getit(); } 

Olhando dentro do código-fonte – você deve definir o parâmetro gzip no próprio comando lib para que o gzip funcione. Não tenho certeza se isso foi intencional ou não, mas esta é a implementação atual. Nenhum header extra é necessário.

 var request = require('request'); request.gzip = true; request({url: 'https://...'}, // use encoding:null for buffer instead of UTF8 function(error, response, body) { ... } );