File Coverage

blib/lib/WWW/Correios/SIGEP.pm
Criterion Covered Total %
statement 39 96 40.6
branch 9 72 12.5
condition 6 116 5.1
subroutine 7 15 46.6
pod 3 11 27.2
total 64 310 20.6


line stmt bran cond sub pod time code
1             package WWW::Correios::SIGEP;
2 2     2   159102 use strict;
  2         11  
  2         49  
3 2     2   9 use warnings;
  2         4  
  2         40  
4 2     2   754 use WWW::Correios::SIGEP::LogisticaReversa;
  2         6  
  2         61  
5 2     2   14 use WWW::Correios::SIGEP::Common;
  2         4  
  2         3472  
6              
7             our $VERSION = 0.04;
8              
9             sub new {
10 2     2 1 6 my ($class, $params) = @_;
11 2 100 66     13 $params = {} unless $params && ref $params eq 'HASH';
12              
13 2 100       8 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         2 $params->{senha} = 'n5f9t8';
18 1         2 $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         4 $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         10 WWW::Correios::SIGEP::Common::build_transport($params);
29 2         22 return bless $params, $class;
30             }
31              
32             sub logistica_reversa {
33 3     3 1 11 my ($self, $params) = @_;
34 3 100 66     21 $params = {} unless $params && ref $params eq 'HASH';
35              
36 3 50 66     20 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         30 %$params,
43             });
44             }
45 3         29 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          
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             else {
247             die "objetos[].codigo_postagem_sigla precisa ser SEDEX, PAC ou CARTA"
248 0 0       0 unless $obj->{codigo_postagem_sigla} eq 'CARTA';
249             }
250             }
251             else {
252 0         0 die "objetos[].valor_declarado (formato NNNN,NN) invalido";
253             }
254             }
255             $xml .= ''
256             . ($obj->{etiqueta} || die "objetos[].etiqueta exigido")
257             . ''
258             . ($obj->{codigo_postagem} || die "objetos[].codigo_postagem exigido")
259             . '0,00'
260             . ($obj->{peso} || die "objetos[].peso em gramas exigido")
261             . '
262             . (substr($obj->{destinatario}{nome},0,50) || die "objetos[].destinatario.nome exigido")
263             . ']]>'
264             . (defined $obj->{destinatario}{telefone}
265             ? '
266             . $obj->{destinatario}{telefone}
267             . ']]>'
268             : ''
269             )
270             . (defined $obj->{destinatario}{celular}
271             ? '
272             . $obj->{destinatario}{celular}
273             . ']]>'
274             : ''
275             )
276             . (defined $obj->{destinatario}{email}
277             ? '
278             . $obj->{destinatario}{email}
279             . ']]>'
280             : ''
281             )
282             . '
283             . ($obj->{destinatario}{logradouro} || die "objetos[].destinatario.logradouro exigido")
284             . ']]>'
285             . (defined $obj->{destinatario}{complemento}
286             ? '
287             . $obj->{destinatario}{complemento}
288             . ']]>'
289             : ''
290             )
291             . ''
292             . (defined $obj->{destinatario}{numero}
293             && $obj->{destinatario}{numero} =~ m{\A(?:s/n|\d+)\z} # digit or "s/n"
294             ? $obj->{destinatario}{numero} : die "objetos[].destinatario.numero"
295             )
296             . '
297             . ($obj->{destinatario}{bairro} || die "objetos[].destinatario.bairro exigido")
298             . ']]>
299             . ($obj->{destinatario}{cidade} || die "objetos[].destinatario.cidade exigido")
300             . ']]>'
301             . (uc $obj->{destinatario}{uf} || die "objetos[].destinatario.uf exigido")
302             . '
303             . ($obj->{destinatario}{cep} =~ /\A\d{8}\z/ ? $obj->{destinatario}{cep} : die "objetos[].destinatario.cep (somente numeros) exigido")
304             . ']]>'
305             . ''
306             . ''
307             . ''
308             . '0,0'
309             . ''
310             # 'O serviço adicional “025”, referente ao registro, deve sempre ser informado.'
311             . '025'
312             . (exists $obj->{servicos_adicionais} && @{$obj->{servicos_adicionais}} > 0
313 0         0 ? join('' => map '' . $_ . '', @{$obj->{servicos_adicionais}})
314             : ''
315             )
316             . (defined $obj->{valor_declarado}
317             ? '' . ($obj->{codigo_postagem_sigla} eq 'PAC'
318             ? '064' : $obj->{codigo_postagem_sigla} eq 'SEDEX' ? '019' : '035'
319             ) . '' . $obj->{valor_declarado} . ''
320             : ''
321             )
322             . ''
323             . ''
324             . ($obj->{tipo} || die "objetos[].tipo (001, 002, 003) exigido")
325             . ''
326             . ($obj->{altura} || '0')
327             . ''
328             . ($obj->{largura} || '0')
329             . ''
330             . ($obj->{comprimento} || '0')
331             . ''
332 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        
333             . '0'
334             ;
335             }
336 0         0 $xml .= '';
337 0         0 return $xml;
338             }
339              
340             sub digito_verificador {
341 2     2 0 13733 my ($codigo) = @_;
342 2         10 my @numeros = split // => $codigo;
343 2         6 my @magica = ( 8, 6, 4, 2, 3, 5, 9, 7 );
344              
345 2         3 my $soma = 0;
346 2         5 foreach ( 0 .. 7 ) {
347 16         24 $soma += ( $numeros[$_] * $magica[$_] );
348             }
349              
350 2         4 my $resto = $soma % 11;
351 2 50       7 my $dv = $resto == 0 ? 5
    50          
352             : $resto == 1 ? 0
353             : 11 - $resto
354             ;
355              
356 2         10 return $dv;
357             }
358              
359             1;
360             __END__