File Coverage

blib/lib/WWW/Correios/SIGEP.pm
Criterion Covered Total %
statement 39 97 40.2
branch 9 78 11.5
condition 6 119 5.0
subroutine 7 15 46.6
pod 3 11 27.2
total 64 320 20.0


line stmt bran cond sub pod time code
1             package WWW::Correios::SIGEP;
2 2     2   198856 use strict;
  2         12  
  2         60  
3 2     2   9 use warnings;
  2         4  
  2         55  
4 2     2   967 use WWW::Correios::SIGEP::LogisticaReversa;
  2         6  
  2         75  
5 2     2   17 use WWW::Correios::SIGEP::Common;
  2         3  
  2         4271  
6              
7             our $VERSION = 0.06;
8              
9             sub new {
10 2     2 1 8 my ($class, $params) = @_;
11 2 100 66     14 $params = {} unless $params && ref $params eq 'HASH';
12              
13 2 100       9 if ($params->{sandbox}) {
14 1         3 $params->{target} = 'https://apphom.correios.com.br/SigepMasterJPA/AtendeClienteService/AtendeCliente?wsdl';
15             # na sandbox, Correios nos instruem a ignorar configurações do cliente e usar essas:
16 1         3 $params->{usuario} = 'sigep';
17 1         3 $params->{senha} = 'n5f9t8';
18 1         1 $params->{contrato} = '9992157880';
19 1         2 $params->{cartao} = '0067599079';
20              
21 1         3 $params->{wsdl_local_file} = 'sandbox/atende_cliente.wsdl';
22             }
23             else {
24 1         3 $params->{target} = 'https://apps.correios.com.br/SigepMasterJPA/AtendeClienteService/AtendeCliente?wsdl';
25 1         3 $params->{wsdl_local_file} = 'live/atende_cliente.wsdl';
26             }
27              
28 2         12 WWW::Correios::SIGEP::Common::build_transport($params);
29 2         23 return bless $params, $class;
30             }
31              
32             sub logistica_reversa {
33 3     3 1 12 my ($self, $params) = @_;
34 3 100 66     24 $params = {} unless $params && ref $params eq 'HASH';
35              
36 3 50 66     25 if (!$self->{scol_obj} || keys %$params) {
37             $self->{scol_obj} = WWW::Correios::SIGEP::LogisticaReversa->new(+{
38             debug => $self->{debug},
39             sandbox => $self->{sandbox},
40             usuario => $self->{usuario},
41             senha => $self->{senha},
42 3         38 %$params,
43             });
44             }
45 3         33 return $self->{scol_obj};
46             }
47              
48             sub busca_cliente {
49 0     0 1 0 my ($self, $params) = @_;
50 0 0 0     0 $params = {} unless $params && ref $params eq 'HASH';
51              
52             return WWW::Correios::SIGEP::Common::call($self, 'buscaCliente', {
53             idContrato => $self->{contrato} || $params->{idContrato},
54             idCartaoPostagem => $self->{cartao} || $params->{idCartaoPostagem},
55             usuario => $self->{usuario} || $params->{usuario},
56             senha => $self->{senha} || $params->{senha},
57 0   0     0 });
      0        
      0        
      0        
58             }
59              
60             sub consulta_cep {
61 0     0 0 0 my ($self, $cep) = @_;
62 0         0 $cep =~ s/\D//g;
63              
64 0         0 return WWW::Correios::SIGEP::Common::call($self, 'consultaCEP', {
65             cep => $cep
66             });
67             }
68              
69             sub cartao_valido {
70 0     0 0 0 my ($self, $params) = @_;
71 0 0 0     0 $params = {} unless $params && ref $params eq 'HASH';
72              
73             my $return = WWW::Correios::SIGEP::Common::call(
74             $self,
75             'getStatusCartaoPostagem',
76             {
77             numeroCartaoPostagem => $self->{cartao} || $params->{numeroCartaoPostagem},
78             usuario => $self->{usuario} || $params->{usuario},
79             senha => $self->{senha} || $params->{senha},
80             }
81 0   0     0 );
      0        
      0        
82 0         0 return $return eq 'Normal';
83             }
84              
85             sub servico_disponivel {
86 0     0 0 0 my ($self, $params) = @_;
87              
88 0         0 $params->{cep_origem} =~ s/\D+//g;
89 0         0 $params->{cep_destino} =~ s/\D+//g;
90              
91             return WWW::Correios::SIGEP::Common::call(
92             $self,
93             'verificaDisponibilidadeServico',
94             {
95             usuario => $self->{usuario} || $params->{usuario},
96             senha => $self->{senha} || $params->{senha},
97             codAdministrativo => $self->{codigo} || $params->{codigo},
98             numeroServico => $params->{codigo_servico},
99             cepOrigem => $params->{cep_origem},
100             cepDestino => $params->{cep_destino},
101             }
102 0   0     0 );
      0        
      0        
103             }
104              
105             sub solicita_etiquetas {
106 0     0 0 0 my ($self, $cnpj, $id, $n) = @_;
107              
108             my $return = WWW::Correios::SIGEP::Common::call($self, 'solicitaEtiquetas', {
109             tipoDestinatario => 'C',
110             identificador => $cnpj,
111             idServico => $id,
112             qtdEtiquetas => $n,
113             usuario => $self->{usuario},
114             senha => $self->{senha}
115 0         0 });
116 0 0 0     0 return () unless $return && !ref $return;
117              
118 0         0 my ($i, $f) = map { s/\D+//g; $_ } split /\s*,\s*/ => $return;
  0         0  
  0         0  
119 0         0 my $prefixo = substr $return, 0, 2;
120              
121 0         0 my @etiquetas;
122 0         0 foreach my $codigo ($i .. $f) {
123 0         0 push @etiquetas, $prefixo . $codigo . digito_verificador($codigo) . 'BR';
124             }
125 0         0 return @etiquetas;
126             }
127              
128             sub fecha_plp_varios_servicos {
129 0     0 0 0 my ($self, $params) = @_;
130 0 0       0 die "fecha_plp_varios_servicos: parametros exigidos"
131             unless ref $params eq 'HASH';
132              
133 0         0 my $xml;
134 0 0       0 if (exists $params->{xml}) {
135 0         0 $xml = $params->{xml};
136             }
137             else {
138 0         0 $xml = $self->gera_xml_plp($params);
139             }
140              
141             return WWW::Correios::SIGEP::Common::call(
142             $self,
143             'fechaPlpVariosServicos',
144             {
145             usuario => $self->{usuario} || $params->{usuario},
146             senha => $self->{senha} || $params->{senha},
147             cartaoPostagem => $self->{cartao} || $params->{cartao},
148             xml => $xml,
149             idPlpCliente => $params->{id},
150             listaEtiquetas => [
151             map {
152 0         0 my $etq = $_->{etiqueta};
153 0         0 substr($etq, 10, 1, '');
154 0         0 $etq;
155 0   0     0 } @{$params->{objetos}}
  0   0     0  
      0        
156             ],
157             }
158             );
159             }
160              
161             sub status_plp {
162 0     0 0 0 my ($self, $id) = @_;
163 0 0       0 die "status_plp: id da PLP exigido" unless defined $id;
164             return WWW::Correios::SIGEP::Common::call(
165             $self,
166             'solicitaXmlPlp',
167             {
168             usuario => $self->{usuario},
169             senha => $self->{senha},
170 0         0 idPlpMaster => $id,
171             }
172             );
173             }
174              
175             sub gera_xml_plp {
176 0     0 0 0 my ($self, $params) = @_;
177              
178             # I'm sorry, ubu.
179             my $xml = 'Postagem2.3'
180             . ($params->{cartao} || $self->{cartao} || die "cartao de postagem exigido")
181             . ''
182             . ($params->{contrato} || $self->{contrato} || die "contrato exigido")
183             . ''
184             . ($params->{diretoria} || die "diretoria exigido")
185             . ''
186             . ($params->{codigo_administrativo} || die "codigo_administrativo exigido")
187             . '
188             . ($params->{remetente}{nome} || die "remetente.nome exigido")
189             . ']]>
190             . ($params->{remetente}{logradouro} || die "remetente.logradouro exigido")
191             . ']]>'
192             . (defined $params->{remetente}{numero}
193             && $params->{remetente}{numero} =~ m{\A(?:s/n|\d+)\z} # digit or "s/n"
194             ? $params->{remetente}{numero} : die "remetente.numero exigido"
195             )
196             . ''
197             . (defined $params->{remetente}{complemento}
198             ? '
199             . $params->{remetente}{complemento}
200             . ']]>'
201             : ''
202             )
203             . '
204             . ($params->{remetente}{bairro} || die "remetente.bairro exigido")
205             . ']]>'
206             . ($params->{remetente}{cep} =~ /\A\d{8}\z/ ? $params->{remetente}{cep} : die "remetente.cep (somente numeros) exigido")
207             . '
208             . ($params->{remetente}{cidade} || die "remetente.cidade exigido")
209             . ']]>'
210             . uc($params->{remetente}{estado} || die "remetente.estado (sigla) exigido")
211             . ''
212             . (defined $params->{remetente}{telefone}
213             ? '
214             . $params->{remetente}{telefone}
215             . ']]>'
216             : ''
217             )
218             . (defined $params->{remetente}{fax}
219             ? '
220             . $params->{remetente}{fax}
221             . ']]>'
222             : ''
223             )
224             . (defined $params->{remetente}{email}
225             ? '
226             . $params->{remetente}{email}
227 0 0 0     0 . ']]>'
    0 0        
    0 0        
    0 0        
    0 0        
    0 0        
      0        
      0        
      0        
      0        
228             : ''
229             )
230             . ''
231             ;
232              
233 0 0       0 die "objetos exigidos (ao menos 1)" unless @{$params->{objetos}} > 0;
  0         0  
234 0         0 foreach my $obj (@{$params->{objetos}}) {
  0         0  
235 0 0       0 if (defined $obj->{valor_declarado}) {
236 0 0       0 if ($obj->{valor_declarado} =~ /\A(\d{1,9}),(\d{2})\z/) {
237 0         0 my $valor_declarado = $1 + $2/100;
238 0 0       0 if ($obj->{codigo_postagem_sigla} eq 'PAC') {
    0          
    0          
239 0 0 0     0 die "objetos[].valor_declarado (PAC) precisa ser entre 18,50 e 3000,00"
240             unless $valor_declarado >= 18.5 && $valor_declarado <= 3000;
241             }
242             elsif ($obj->{codigo_postagem_sigla} eq 'SEDEX') {
243 0 0 0     0 die "objetos[].valor_declarado (SEDEX) precisa ser entre 18,50 e 10000,00"
244             unless $valor_declarado >= 18.5 && $valor_declarado <= 10_000;
245             }
246             elsif ($obj->{codigo_postagem_sigla} eq 'MINI') {
247 0 0 0     0 die "objetos[].valor_declarado (MINI) precisa ser entre 1 e 100,00"
248             unless $valor_declarado > 1 && $valor_declarado <= 100;
249             }
250             else {
251             die "objetos[].codigo_postagem_sigla ($obj->{codigo_postagem_sigla}) precisa ser SEDEX, PAC, CARTA ou MINI"
252 0 0       0 unless $obj->{codigo_postagem_sigla} eq 'CARTA';
253             }
254             }
255             else {
256 0         0 die "objetos[].valor_declarado (formato NNNN,NN) invalido";
257             }
258             }
259             $xml .= ''
260             . ($obj->{etiqueta} || die "objetos[].etiqueta exigido")
261             . ''
262             . ($obj->{codigo_postagem} || die "objetos[].codigo_postagem exigido")
263             . '0,00'
264             . ($obj->{peso} || die "objetos[].peso em gramas exigido")
265             . '
266             . (substr($obj->{destinatario}{nome},0,50) || die "objetos[].destinatario.nome exigido")
267             . ']]>'
268             . (defined $obj->{destinatario}{telefone}
269             ? '
270             . $obj->{destinatario}{telefone}
271             . ']]>'
272             : ''
273             )
274             . (defined $obj->{destinatario}{celular}
275             ? '
276             . $obj->{destinatario}{celular}
277             . ']]>'
278             : ''
279             )
280             . (defined $obj->{destinatario}{email}
281             ? '
282             . $obj->{destinatario}{email}
283             . ']]>'
284             : ''
285             )
286             . '
287             . ($obj->{destinatario}{logradouro} || die "objetos[].destinatario.logradouro exigido")
288             . ']]>'
289             . (defined $obj->{destinatario}{complemento}
290             ? '
291             . $obj->{destinatario}{complemento}
292             . ']]>'
293             : ''
294             )
295             . ''
296             . (defined $obj->{destinatario}{numero}
297             && $obj->{destinatario}{numero} =~ m{\A(?:s/n|\d+)\z} # digit or "s/n"
298             ? $obj->{destinatario}{numero} : die "objetos[].destinatario.numero"
299             )
300             . '
301             . ($obj->{destinatario}{bairro} || die "objetos[].destinatario.bairro exigido")
302             . ']]>
303             . ($obj->{destinatario}{cidade} || die "objetos[].destinatario.cidade exigido")
304             . ']]>'
305             . (uc $obj->{destinatario}{uf} || die "objetos[].destinatario.uf exigido")
306             . '
307             . ($obj->{destinatario}{cep} =~ /\A\d{8}\z/ ? $obj->{destinatario}{cep} : die "objetos[].destinatario.cep (somente numeros) exigido")
308             . ']]>'
309             . ''
310             . ''
311             . ''
312             . '0,0'
313             . ''
314             # 'O serviço adicional “025”, referente ao registro, deve sempre ser informado.'
315             . '025'
316             . (exists $obj->{servicos_adicionais} && @{$obj->{servicos_adicionais}} > 0
317 0         0 ? join('' => map '' . $_ . '', @{$obj->{servicos_adicionais}})
318             : ''
319             )
320             . (defined $obj->{valor_declarado}
321             ? '' . ($obj->{codigo_postagem_sigla} eq 'PAC' ? '064'
322             : $obj->{codigo_postagem_sigla} eq 'SEDEX' ? '019'
323             : $obj->{codigo_postagem_sigla} eq 'MINI' ? '065'
324             : '035' # CARTA
325             )
326             . '' . $obj->{valor_declarado} . ''
327             : ''
328             )
329             . ''
330             . ''
331             . ($obj->{tipo} || die "objetos[].tipo (001, 002, 003) exigido")
332             . ''
333             . ($obj->{altura} || '0')
334             . ''
335             . ($obj->{largura} || '0')
336             . ''
337             . ($obj->{comprimento} || '0')
338             . ''
339 0 0 0     0 . ($obj->{diametro} || '0')
    0 0        
    0 0        
    0 0        
    0 0        
    0 0        
    0 0        
    0 0        
    0 0        
    0 0        
    0 0        
      0        
      0        
      0        
      0        
340             . '0'
341             ;
342             }
343 0         0 $xml .= '';
344 0         0 return $xml;
345             }
346              
347             sub digito_verificador {
348 2     2 0 23005 my ($codigo) = @_;
349 2         10 my @numeros = split // => $codigo;
350 2         5 my @magica = ( 8, 6, 4, 2, 3, 5, 9, 7 );
351              
352 2         3 my $soma = 0;
353 2         6 foreach ( 0 .. 7 ) {
354 16         23 $soma += ( $numeros[$_] * $magica[$_] );
355             }
356              
357 2         4 my $resto = $soma % 11;
358 2 50       8 my $dv = $resto == 0 ? 5
    50          
359             : $resto == 1 ? 0
360             : 11 - $resto
361             ;
362              
363 2         9 return $dv;
364             }
365              
366             1;
367             __END__