Jump to content
Giovani Silva

PHP NFSe com SOAP envio de XML RPS para Betha Fly-enota

Recommended Posts

Boa noite, estou tentando enviar com PHP SOAP o XML de um lote RPS assinado para o Web Service da Betha também conhecido como Fly e-nota. 

No entanto, não tenho tido nenhum retorno quanto a se deu algum erro ou algo do tipo.

 

Documentação https://e-gov.betha.com.br/e-nota-test/ambienteteste.faces 

 

Ambiente Teste Webservice 

https://e-gov.betha.com.br/e-nota-contribuinte-test-ws/recepcionarLoteRps?wsdl

https://e-gov.betha.com.br/e-nota-contribuinte-test-ws/consultarLoteRps?wsdl 

 

Layout https://e-gov.betha.com.br/e-nota/manuais/layout.pdf 
Modelo Conceitual http://www.abrasf.org.br/arquivos/files/NFSE-NACIONAL_Modelo_Conceitual versao 2-02.pdf

Manual de Integração ABRASF http://www.abrasf.org.br/arquivos/files/NFSE-NACIONAL_Manual_De_Integracao versao 2-02.pdf 

 

Tem uma opção na área administrativa Fly e-nota para solicitar a prefeitura autorização para impressão de RPS tanto manual como eletrônica.
Não sei se isso é obrigatório ou se é necessário para o meu objetivo.

  

Parte do código para o envio:

------------------------------

function transmiteRps($xmlAssinado,$certificadoPem,$senha){
    $wsdl            = 'https://e-gov.betha.com.br/e-nota-contribuinte-test-ws/recepcionarLoteRps?wsdl';
    $endpoint        = 'https://e-gov.betha.com.br/e-nota-contribuinte-test-ws/recepcionarLoteRps';
    $certificate    = $certificadoPem;
    $password        = $senha;
    
    $options = array(
        'location'        => $endpoint,
        'keep_alive'    => true,
        'trace'            => true,
        'local_cert'    => $certificate,
        'passphrase'    => $password,
        'cache_wsdl'    => WSDL_CACHE_NONE
    );
    
    try {
        
        $client = new SoapClient($wsdl, $options);



        $function = 'EnviarLoteRpsEnvio';
        $arguments = ['EnviarLoteRpsEnvio' => ['xml'=>$xmlAssinado]];
        
        $options = [];
        $result = $client -> __soapCall($function, $arguments, $options);



    } catch(Exception $e){
        $result = false;
    }
      
    if($result!==false){
        return xml2array($result->return);
    }else{
        return false;
    }
    
}//transmiteRps





----------------------------------------------------------


function xml2array($xmlstring){
    $xml = simplexml_load_string($xmlstring);
    $json = json_encode($xml);
    $array = json_decode($json, TRUE);
    return $array;
    
}



-----------------------------------------------



function consultaRps($cnpj, $inscricao, $protocolo, $certificadoPem, $senha){
    
    $wsdl            = 'https://e-gov.betha.com.br/e-nota-contribuinte-test-ws/consultarLoteRps?wsdl';
    $endpoint        = 'https://e-gov.betha.com.br/e-nota-contribuinte-test-ws/consultarLoteRps';
    $certificate    = $certificadoPem;
    $password        = $senha;

    

$xml = '<?xml version="1.0" encoding="utf-8"?>
    <ConsultarLoteRpsEnvio xmlns="http://www.betha.com.br/e-nota-contribuinte-test-ws">
    <Prestador>
        <CpfCnpj>
            <Cnpj>'.$cnpj.'</Cnpj>
        </CpfCnpj>
        <InscricaoMunicipal>'.$inscricao.'</InscricaoMunicipal>
    </Prestador>
    <Protocolo>'.$protocolo.'</Protocolo>
    </ConsultarLoteRpsEnvio>';
    
    
    $options = array(
        'location'        => $endpoint,
        'keep_alive'    => true,
        'trace'            => true,
        'local_cert'    => $certificate,
        'passphrase'    => $password,
        'cache_wsdl'    => WSDL_CACHE_NONE
    );
    
    try {
        
        $client = new SoapClient($wsdl, $options);
        $function = 'consultarLoteRps';
        $arguments = ['consultarLoteRps' => ['xml'=>$xml]];
        
        $options = [];
        $result = $client -> __soapCall($function, $arguments, $options);
    } catch(Exception $e){
        $result = false;
    }
    
    if($result!==false){
        return $result->return;
    }else{
        return false;
    }
    
} //consultaRps



--------------------------------



$certificadoPem=dirname(__FILE__).'/certificado.pem';
$senha='xxxxx'; //senha do certificado



----------------------------------



//assina RPS
$xmlAssinado=assinaRps(dirname(__FILE__).'/only_rps.xml'); //usei para assinar só a RPS



file_put_contents(dirname(__FILE__).'/rps_temp/rps1.xml','<?xml version="1.0" encoding="utf-8"?>
<EnviarLoteRpsEnvio xmlns="https://e-gov.betha.com.br/e-nota-contribuinte-test-ws"><LoteRps Id="LOTE1"><NumeroLote>1</NumeroLote><Cnpj>xxxxxxxxxxxxxx</Cnpj><InscricaoMunicipal>xxxx</InscricaoMunicipal><QuantidadeRps>1</QuantidadeRps>
<ListaRps>'.str_replace('<?xml version="1.0" encoding="utf-8"?>','',$xmlAssinado).'</ListaRps></LoteRps></EnviarLoteRpsEnvio>');

$_SESSION['nome_uri']='LOTE1';



$xmlAssinado=assinaRps(dirname(__FILE__).'/rps_temp/rps1.xml'); //usei para assinar o Lote RPS






$resultado=transmiteRps($xmlAssinado,
                       $certificadoPem,
                       $senha);


if(!empty($resultado["Protocolo"])){//se tem protocolo e sucesso
    //$resultado["NumeroLote"]
    //$resultado["Protocolo"]
    //consulta lote e pega nfse
    
    $nfseResposta = consultaRps('xxxxxxxxxxxxx', //cnpj
                               'xxxx', //inscrição municipal
                               $resultado["Protocolo"],
                               $certificadoPem,
                               $senha);
    
    if($nfseResposta!==false){
        echo '<textarea cols="120" rows="70">';
        echo $nfseResposta;
        echo '</textarea>';
    }else{
        echo 'Erro ao consultar RPS';
    }
    
}else{
    echo "erro no envio";
}


 

Vídeo Tutorial do código NFSe acima porém para outro webservice que não é da Betha.

https://www.youtube.com/watch?v=v1rByVX1_FY

 

Desde já agradeço a atenção e colaboração dos colegas.
Funcionando postarei aqui a solução com o código completo para quem precisar.

Edited by Gabriel Heming
adicionar marcação de código

Share this post


Link to post
Share on other sites

Um dos pontos que analizei, é que você está enviando os dados de forma errada. Apesar do WSDL definir os tipos como um XML, não é um XML que deve ser enviado. Você deve criar um objeto que corresponda ao tipo definido na mensageria.

 

Veja o seguinte XML:

<ConsultarLoteRpsEnvio xmlns="http://www.betha.com.br/e-nota-contribuinte-test-ws">
    <Prestador>
        <CpfCnpj>
            <Cnpj>?</Cnpj>
        </CpfCnpj>
        <InscricaoMunicipal>?</InscricaoMunicipal>
    </Prestador>
    <Protocolo>?</Protocolo>
</ConsultarLoteRpsEnvio>

Sendo ConsultarLoteRpsEnvio o nome do método, o que está abaixo são as propriedades.

 

Com anos de experiência, eu compreendi que o Soap trabalha melhor quando lhe é enviado um objeto do tipo stdClass ao invés de um array. Logo, quando for montar o objeto, ele deve ser similar a este:

//Cria o objeto nó raiz
$consultarLoteRpsEnvio = new stdClass();

//cria o objeto do nó Prestador
$prestador = new stdClass();

//Cria o objeto do nó CpfCnpj
$cpfCnpj = new stdClass();
$cpfCnpj->Cnpj = 'xxxxxxxxxxxx';

//Adiciona o nó CpfCnpj ao nó Prestador
$prestador->CpfCnpj = $cpfCnpj;
$prestador->InstricaoMunicipal = 'xxxxxx';

//Adiciona o nó Prestador ao nó Raiz
$consultarLoteRpsEnvio->Prestador = $crestador;
$consultarLoteRpsEnvio->Protocolo = 'xxxxx';

E, na hora de enviar, utilize apenas o objeto:

$result = $client->ConsultarLoteRpsEnvio($consultarLoteRpsEnvio);

Você pode encontrar outro exemplo aqui:

 

Share this post


Link to post
Share on other sites

Bom dia Gabriel, muito obrigado pela dica.
Gostei do conceito de criar e enviar o objeto.

 

Vou tentar fazer como você mencionou.

 

Bem, pelo que entendi o mesmo conceito deve ser aplicado a função transmiteRps.

 

Mas a parte que diz no manual que tanto a RPS como o Lote devem ser Assinados
a assinatura também deve ser colocada no objeto da forma que mencionasse?
Em nenhuma parte enviarei um arquivo XML do Lote RPS, apenas o objeto? quanto a senha, assinatura,... e outras coisas para ter um minimo de segurança?

 

O certificado eu converti para .pem Nesse caso o processo de assinar o objeto seria o mesmo praticamente que eu havia usado?

 

Só lembrando a parte que receberá o RPS https://e-gov.betha.com.br/e-nota-contribuinte-test-ws/recepcionarLoteRps?wsdl 

 

Obs. No caso que fiz eu gravei o XML assinado, mas na hora de enviar não tive o Protocolo para a parte da consulta. Ou seja, algo errado mas também não tive resposta de erro do webservice.

Share this post


Link to post
Share on other sites

Quanto a essas situações do certificado, terei de pesquisar, nunca precisei enviar nada assinado digitalmente via SOAP.

Share this post


Link to post
Share on other sites

Obrigado, vou dar uma olhada.

Aproveitando, você sabe interpretar o WSDL? 
https://e-gov.betha.com.br/e-nota-contribuinte-test-ws/recepcionarLoteRps?wsdl 
é que na chamada
__soapCall($function, $arguments, $options); 

 

Tenho duvida quanto a função e os argumentos usados para o envio dos dados.
         $function = 'EnviarLoteRpsEnvio';
        $arguments = ['EnviarLoteRpsEnvio' => ['xml'=>$xmlAssinado]];

 

--- Dúvida: EnviarLoteRpsEnvioResponse ou EnviarLoteRpsEnvio em ambos os casos (função e nos argumentos) ?

 

Outra questão: ao usar stdClass na real não estaria criando a mesma estrutura do XML manualmente e salvando em um objeto para enviá-la ao webservice?

 

Share this post


Link to post
Share on other sites

Caso você não conheça todas as partes do WSDL, use o SoapUI para interpretar o WSDL, ele lhe ajudar bastante.

 

6 minutos atrás, didio20ss disse:

[...] ao usar stdClass na real não estaria criando a mesma estrutura do XML manualmente e salvando em um objeto para enviá-la ao webservice?

 

 

Sim e não. A ideia é criar exatamente a estrutura do XML, entretanto, a sua estrutura de XML é apenas uma string, por isso elas possuem comportamentos diferentes.

Share this post


Link to post
Share on other sites

Oi, tive mais uma dúvida, tenho dois modelos de lote RPS onde um baixei da Betha e outro fornecido por uma empresa de automação que usa o sistema da Betha.
No cabeçalho dessa outra empresa tem o "e:" e no na Betha não.

Empresa Midia
<e:EnviarLoteRpsEnvio xmlns:e="http://www.betha.com.br/e-nota-contribuinte-ws">
Betha 
<EnviarLoteRpsEnvio xmlns = "http://www.betha.com.br/e-nota-contribuinte-ws">

Esse "e:" ou as vezes já notei outras expressões no lugar,
como se chama e para que serve? 
Faz diferença usar ?

 

----------------
Outra dúvida

Nos exemplos da Betha tem:
envioSoapUi.xml que tem o comentário "Quando utilizar, por exemplo, o Soap-UI para envio do XML deve ser utilizada a estrutura abaixo" que teria um cabeçalho <soapenv 
E tem exemplos sem isso. 
Na hora de montar a estrutura do XML devo usar isso no meu código php ou não?

Outros exemplos que tem:
RecepcionarLoteRps.xml

RecepcionarLoteRpsSincrono.xml
GerarNfse.xml <GerarNfseEnvio xmlns = "http://www.betha.com.br/e-nota-contribuinte-ws">

 

Qual a diferença do Sincrono e do outro? Qual a melhor opção?

 

E eu estava a usar o RecepcionarLoteRps.
Me passaram a informação que gerando e enviando o XML do lote RPS para o Webservice da Betha, 
o sistema deles gera a DANFE e até envia a NFSe para o e-mail do cliente se o mesmo for informado no XML
.

 

Então devo continuar a usar apenas o RecepecionarLoteRps ou também devo usar o GerarNfse?

 

Obrigado mais uma vez

Share this post


Link to post
Share on other sites
2 horas atrás, didio20ss disse:

Esse "e:" ou as vezes já notei outras expressões no lugar,
como se chama e para que serve? 
Faz diferença usar ?

 

É um prefixo de namespace. Quando não utilizado, tudo que é global, cai no namespace sem prefixo. Já, caso você utilizar, você pode usar o prefixo para informar o namespace utilizado.

 

Para saber mais, de uma olhada aqui: https://msdn.microsoft.com/en-us/library/aa468565.aspx

 

2 horas atrás, didio20ss disse:

Nos exemplos da Betha tem:
envioSoapUi.xml que tem o comentário "Quando utilizar, por exemplo, o Soap-UI para envio do XML deve ser utilizada a estrutura abaixo" que teria um cabeçalho <soapenv 
E tem exemplos sem isso. 
Na hora de montar a estrutura do XML devo usar isso no meu código php ou não?

 

essa estrutura deve ser usada quando você quiser manipular completamente a chamada do SOAP (cabeçalho e corpo da requisição). Isso normalmente acontece quando utiliza-se a biblioteca cURL. Com a biblioteca SOAP do PHP, ele já faz isso por ti.

 

2 horas atrás, didio20ss disse:

Outros exemplos que tem:
RecepcionarLoteRps.xml

RecepcionarLoteRpsSincrono.xml
GerarNfse.xml <GerarNfseEnvio xmlns = "http://www.betha.com.br/e-nota-contribuinte-ws">

 

Qual a diferença do Sincrono e do outro? Qual a melhor opção?

 

Sincrono será feita a requisição e o sistema irá esperar o retorno dessa requisição. Assíncrono é o contrário, a requisição é efetuada, o script não esperar o retorno e continua executando. Qual é melhor? Nenhum. Cada um tem seu propósito.

 

2 horas atrás, didio20ss disse:

E eu estava a usar o RecepcionarLoteRps.
Me passaram a informação que gerando e enviando o XML do lote RPS para o Webservice da Betha, 
o sistema deles gera a DANFE e até envia a NFSe para o e-mail do cliente se o mesmo for informado no XML
.

 

Então devo continuar a usar apenas o RecepecionarLoteRps ou também devo usar o GerarNfse?

 

Não sei, isso foge da área de programação e entra na área de negócio da empresa Betha. Entre em contato com eles e verifique o que for melhor.

Share this post


Link to post
Share on other sites

Agradeço a ajuda do Gabriel Heming e outras fontes sobre o assunto.

Para quem precisar assinar e enviar lote RPS para a Betha emitindo assim a NFSe seguem algumas dicas que me ajudaram a resolver o problema:

 

1º - Ao gerar o XML da RPS, deve-se assinar primeiro a RPS e depois o Lote.
2º - para assinar o XML converte-se o arquivo .pfx do certificado para .pem para assim poder trabalhar com assinatura no servidor PHP

Para a conversão basta procurar na internet por "convert pfx to pem" e terá como exemplo https://www.sslshopper.com/ssl-converter.html 

O meu certificado baixei da Betha e ao converter o arquivo continua contendo tanto a chave privada quanto a publica.
Por isso, após a conversão para "pem" deve-se abrir o arquivo com Bloco de Notas, Notepad++ ou outro, 
Verá vários blocos de código. O primeiro é a chave primaria. Deve-se copiar o segundo bloco que é a chave publica e salvar em um novo arquivo chavepublica.pem 
Irá precisar dos dois arquivos .pem para a assinatura, o completo e o com apenas a chave pública.

 

Biblioteca para assinar o XML https://github.com/robrichards/xmlseclibs 

Vídeo bem útil do Bill Barsch da Geranet https://www.youtube.com/watch?v=v1rByVX1_FY

Usei apenas a parte para assinar o documento.

 

* Importante: Usar o "force_uri". Não sei se porque usei o certificado da Betha no meu caso o atributo URI não apareceu no XML como no exemplo do vídeo, mas teve efeito na assinatura. Sem essa opção ela não é válida.

 

3º - Para o envio usei o cURL adicionando o cabeçalho SOAP por ele.
Segundo algumas fontes, usando o somente o SOAP pode ocorre de não trazer retorno em caso de erros e ai você não sabe o que fazer. Já com o cURL você recebe sempre as mensagens. Quanto usei só o SOAP tinha erro mas a página ficava em branco, de mogo que eu não sabia o que estava acontecendo.


Agradeço ao POST do Luiz Paulo Franz aqui do fórum sobre o envio por cURL.

 

* Tive que fazer algumas modificações no código dele pois tava dando erro de assinatura inválida.

Tirei toda a parte do código dele que fazia alterações no XML, onde ele usava str_replace
Motivo: após você assinar um documento, ele não deve ser alterado, pois assim a assinatura se tornaria inválida.

 

Observações finais:

Fazer os testes no ambiente de homologação.

cuidar de acentos ou caracteres especiais no XML.
Também é bom remover os espaços em branco e quebras de linha antes de se assinar o documento.

Nunca repetir o número do lote e usar em sequência.

 

Espero que estas informações possam ajudar a outros.

 

Um forte abraço a todos
Giovani Silva
Virthuz Anunciante Virtual

Share this post


Link to post
Share on other sites

Boa tarde Amigo! como você assinou o xml, digo as tags, não consegui fazer isso, só assina tudo junto. você poderia passar sua função de assinatura?

Edited by Gabriel Heming
remover o quote

Share this post


Link to post
Share on other sites

Boa tarde Josemir Costa,  você conseguiu resolver sua questão? 
Chegou a olhar os links que mencionei? 

Biblioteca para assinar o XML https://github.com/robrichards/xmlseclibs 

Vídeo bem útil do Bill Barsch da Geranet https://www.youtube.com/watch?v=v1rByVX1_FY 

 

Para assinar individual, colocamos a primeira parte do XML em uma string e então chamamos a função de assinar enviando o código. Após pegamos o retorno e concatenamos com o restante do XML de deveria estar antes e após a primeira string já assinada. Daí chamamos novamente a função de assinar passando a segunda string (XML completo) a ser assinada.

 

Não sei se ficou claro.

 

O único cuidado que tem que ter é não modificar nada no primeiro bloco após assinado, pois remover espaços após a assinatura por exemplo, a invalidaria.

 

Att.

Giovani Silva
Virthuz Anunciante Virtual

Share this post


Link to post
Share on other sites
4 minutos atrás, Giovani Silva disse:

Boa tarde Josemir Costa,  você conseguiu resolver sua questão? 
Chegou a olhar os links que mencionei? 

Biblioteca para assinar o XML https://github.com/robrichards/xmlseclibs 

Vídeo bem útil do Bill Barsch da Geranet https://www.youtube.com/watch?v=v1rByVX1_FY 

 

Para assinar individual, colocamos a primeira parte do XML em uma string e então chamamos a função de assinar enviando o código. Após pegamos o retorno e concatenamos com o restante do XML de deveria estar antes e após a primeira string já assinada. Daí chamamos novamente a função de assinar passando a segunda string (XML completo) a ser assinada.

 

Não sei se ficou claro.

 

O único cuidado que tem que ter é não modificar nada no primeiro bloco após assinado, pois remover espaços após a assinatura por exemplo, a invalidaria.

 

Att.

Giovani Silva
Virthuz Anunciante Virtual

Então! A questão é que quando assino o xml usando a biblioteca ele só assina a primeira tag, como faço p assinar a tag correta? tipo assinar a tag <rps> depois a tag <LoteRps> que são as que devem conter os ids e serem assinadas. Deu pra entender?

Share this post


Link to post
Share on other sites

$rps="<Rps><InfRps Id="rps1"> ... </InfRps></Rps";

$RpsAssinado=assinaRps($rps,$chave_publica);

 

$Lote='<e:EnviarLoteRpsEnvio xmlns:e="http://www.betha.com.br/e-nota-contribuinte-ws"><LoteRps Id="LOTE1'"><NumeroLote>1</NumeroLote><Cnpj>xxxxxxx</Cnpj><InscricaoMunicipal>xxxx</InscricaoMunicipal><QuantidadeRps>1</QuantidadeRps><ListaRps>'.$RpsAssinado.'</ListaRps></LoteRps></e:EnviarLoteRpsEnvio>';

$LoteAssinado=assinaRps($Lote , $chave_publica);

 

Share this post


Link to post
Share on other sites

Mais um detalhe Josemir, na função assinaRps não sei se você estava enviando um arquivo com o XML.
Caso era um arquivo, na função teria a opção $doc -> load($arquivo)

Nessa caso tem que substituir por 
$doc -> loadXML($arquivo) 

para funcionar com a string 
$rps="<Rps><InfRps Id="rps1"> ... </InfRps></Rps";

Share this post


Link to post
Share on other sites

Veja meu xml:

<Rps>
 <InfDeclaracaoPrestacaoServico>
  <Rps Id="1">
   <IdentificacaoRps>
    <Numero>254122</Numero>
    <Serie>serie</Serie>
    <Tipo>1</Tipo>
   </IdentificacaoRps>
   <DataEmissao>2018-04-10</DataEmissao>
   <Status>1</Status>
  </Rps>
  <Competencia>2018-04-10</Competencia>
  <Servico>
   <Valores>
    <ValorServicos>400.00</ValorServicos>
    <ValorIss>8.00</ValorIss>
    <Aliquota>0.0200</Aliquota>
   </Valores>
   <IssRetido>1</IssRetido>
   <ItemListaServico>1401</ItemListaServico>
   <Discriminacao>Descriminação do serviço</Discriminacao>
   <CodigoMunicipio>3100401</CodigoMunicipio>
   <ExigibilidadeISS>3</ExigibilidadeISS>
  </Servico>
  <Prestador>
   <CpfCnpj>
    <Cnpj>01139388000179</Cnpj>
   </CpfCnpj>
   <InscricaoMunicipal>1234</InscricaoMunicipal>
  </Prestador>
  <Tomador>
   <IdentificacaoTomador>
    <CpfCnpj>
     <Cnpj>12345678912345</Cnpj>
    </CpfCnpj>
    <InscricaoMunicipal>1234</InscricaoMunicipal>
   </IdentificacaoTomador>
   <RazaoSocial>TOMADOR</RazaoSocial>
   <Endereco>
    <Endereco>teste</Endereco>
    <Numero>123</Numero>
    <Bairro>teste</Bairro>
    <CodigoMunicipio>1234</CodigoMunicipio>
    <Uf>SE</Uf>
    <Cep>35438000</Cep>
   </Endereco>
   <Contato>
    <Telefone>32554411</Telefone>
    <Email>teste@dominiodoemail.com</Email>
   </Contato>
  </Tomador>
  <RegimeEspecialTributacao>1</RegimeEspecialTributacao>
  <OptanteSimplesNacional>2</OptanteSimplesNacional>
  <IncentivoFiscal>2</IncentivoFiscal>
 </InfDeclaracaoPrestacaoServico>
</Rps>';

quando eu coloco a opção do force_uri não aparece na assinatura, na verdade aparece um código aleatório, quando coloco o id na tag pai <rps> aí sim aparece mas na estrutura correta o id aparece na tag rps filho. Não consegui chegar a uma solução ainda. 

 

Muito obrigado pela atenção e pasciência!

Edited by Gabriel Heming
adicionar marcação de código

Share this post


Link to post
Share on other sites

Boa noite, em nossos testes não lembro se tentamos colocar um ID na tag Rps e sim na InfRps.

Conforme havia comentado na solução aqui neste post, é necessário deixar o force_uri como true mesmo que aparentemente não apareça na estrutura do seu xml ou que fique em branco. Funciona dessa maneira mesmo. Agora se desativar o force_uri da erro na assinatura.

 

Então mesmo que não apareça o uri na assinatura de seu xml, ele fez alguma coisa nela tornando-a válida.

Share this post


Link to post
Share on other sites

Create an account or sign in to comment

You need to be a member in order to leave a comment

Create an account

Sign up for a new account in our community. It's easy!

Register a new account

Sign in

Already have an account? Sign in here.

Sign In Now

  • Similar Content

    • By Jeovane Carvalho
      Olá pessoal, estou precisando de um help !
       
      Pois eu tenho uma classe assim:
       
      DadosComboClasse.php
       
      class DadosComboCidades { private $conecta_banco; public function __construct(){ $this->conecta_banco= new Conexao(); $this->conecta_banco->Conectar(); } public $array = array(); public $imoveis_tipo; public $cidade; public $bairro; public $bairro_id2; public $idcidade; public $negocios; public $qtd; public $bairro_id; public $id_cidade; public $id_tipo; public $id_bairro; public $nome_empresa_2; public $nome_empresa_3; public $nome_empresa_4; public $nome_empresa_5; public $titulo_cadastro; public $creci; public $facebook; public $twitter; public $linha_b; public $sloga; public $site; public $sqlT_1; public $sqlT_2; public $sqlT_3; public $sqlT_4; public $row; public $id='1'; function BuscarCidades(){ $result=$this->conecta_banco->banco; $pega_cidade=new ComboCidades();=> instancio os ids ou arrays quando são requisitados tipo assim : finalidade ID:1 Tipo ID:3,9 (array) $pega_cidade->PegarCidadeId(); $ativo='1'; ARRAYS ABAIXO: $this->id_negocio=$pega_cidade->id_negocio; $this->id_tipo=$this->id_cidade=$pega_cidade->id_tipo; $this->id_cidade=$pega_cidade->id_cidade; Acima eu pego os dados do array vindo da pagina combo_cidades.php if($pega_cidade->id_negocio): $negocio_array = explode(",",$pega_cidade->id_negocio); if($pega_cidade->id_tipo): $tipos_array = explode(",", $pega_cidade->id_tipo); if($pega_cidade->id_cidade): $cidades_array = explode(",", $pega_cidade->id_cidade); foreach($negocio_array as $negocio): foreach($tipos_array as $tipos): foreach($cidades_array as $cidades): $sqlT_1 = $result->prepare("SELECT i.id,t.tipo_nome,c.cidade,b.bairro,i.ativo FROM imoveis i LEFT JOIN imoveis_tipo t ON (t.id = i.id_tipo_imovel) LEFT JOIN cidades c ON (c.id = i.id_cidade)LEFT JOIN bairros b ON (b.id = i.bairro) WHERE i.id_tipo_imovel IN ('".$tipos."') AND i.id_cidade IN ('".$cidades."') AND i.ativo=? ORDER BY i.bairro DESC"); $sqlT_1->bind_param('s',$ativo); $sqlT_1->execute(); $this->sqlT_1=$sqlT_1->get_result(); Acima com get_result eu gero os dados que preciso do array abaixo são outras chamadas do sql que vou implementar na solução acima $sqlT_2=$result->prepare("SELECT id, bairro,ativo FROM bairros WHERE id AND ativo=? ORDER BY bairro"); $sqlT_2->bind_param('s',$ativo); $sqlT_2->execute(); $this->sqlT_2=$sqlT_2->get_result(); "'%".$negocio."%'"; $where = " i.ativo =? "; if( $negocio) { $where .="AND i.id_negocio_tipo LIKE ?"; }if( $tipos ) { $where .=" AND i.id_tipo_imovel IN ('".$tipos."')"; } if( $cidades ) { $where .=" AND i.id_cidade IN ('".$cidades."')"; } if($this->bairro_id2) { $where .=" AND i.bairro IN ('".$this->bairro_id2."')"; } $sqlT_3 = $result->prepare("SELECT i.id,i.ativo,n.tipo,b.bairro,t.tipo_nome,c.cidade,i.id_negocio_tipo, COUNT(b.bairro) AS Qtd FROM imoveis i LEFT JOIN negocio_tipo n ON (n.id= i.id_negocio_tipo) LEFT JOIN imoveis_tipo t ON (t.id = i.id_tipo_imovel) LEFT JOIN cidades c ON (c.id = i.id_cidade) LEFT JOIN bairros b ON (b.id = i.bairro) WHERE ".$where." GROUP BY b.bairro HAVING COUNT( b.bairro) > 0 ORDER BY b.bairro "); //EXECUTA A QUERY $conta=$sqlT_3->num_rows(); $sqlT_3->bind_param('ss',$ativo,$negocio); $sqlT_3->execute(); $this->sqlT_3=$sqlT_3->get_result(); endforeach; endforeach; endforeach; endif; endif; endif; } }  
       
      Abaixo a classe combocidades.php
       
      require_once("tags2.php"); //RECEBE PARÃMETRO class ComboCidades{ private $conecta_banco; public function __construct(){ $this->conecta_banco= new Conexao(); $this->conecta_banco->Conectar(); } public $id_negocio; public $id_tipo; public $busca_cidade; public $id_cidade; public function PegarCidadeId(){ function filter( $str ){ return addslashes( $str ); } function getPost( $key ){ return isset( $_GET[ $key ] ) ? filter( $_GET[ $key ] ) : null; } $this->id_negocio = getPost('id_negocio'); $this->id_tipo = getPost('id_tipo'); $this->id_cidade=getPost('id_cidade'); } } class ExibirBairros extends DadosComboCidades{ public function Mostrar(){ extract( $_GET, EXTR_OVERWRITE); $busca_bairro=new DadosComboCidades(); $busca_bairro->BuscarCidades(); $busca_bairro->id_cidade; if($busca_bairro->id_cidade==0): echo "<div class='atencao'></div><div class='atencao_text'>Selecione uma Cidade para pesquisar por Bairros !</div> "."<br />"; endif; echo "<div class='sel'></div>"."<br />"; /* MONTA CRITERIOS DE BUSCA */ //QUERY echo '<br /><div class="botao-modal"><button type="button" class="botao_marcar" title="Todos" id="todos" onclick="marcardesmarcar();">Selecionar Todos</button> </div>'; Aqui onde chamo os array já convertidos pelo foreach na classe DadosComboCidades.php echo'<br /><br /><div style=" font-weight:bold; margin-left:30px;">'.$busca_bairro->imoveis_tipo.' (s) </div>'; echo'<br /><br /><div style=" font-weight:bold; margin-left:30px;">'.$busca_bairro->cidade.'</div>'; echo'<div class="form_modal"><form name="form1">'.'<br /><br />'; echo '<input type="checkbox" id="c'.$bairro.'" class="marcar" value="'.$bairro_id.'" name="bairro[]"/><label for="c'.$bairro_id.'"><span></span></label> '.$bairro.' - '.$cidade.' ('.$qtd.') <br />'; //FECHA IF (row) //PRINTA O RESULTADO if($bairro_id): echo FALSE; else: echo "<div class='atencao'></div><div class='atencao_text'>Não existem imóveis para essa Busca !</div> "."<br />"; echo'</form></div></b><br />'; endif; } } Abaixo onde istancio e chamo a classe DadosComboCidades $f=new ExibirBairros(); $f->Mostrar(); RESUMINDO:
       
      O que acontece é que eu não consigo pegar o array no seu índice, ou seja , por exemplo se o id tipo for ID: 1 rola beleza a busca  e a impressão dos dados mas se eu for buscar 2 informações no mesmo id tipo , assim => ID: array(5,9) , na outra página só consigo pegar o último id em vez de buscar 5,9 só busca o 9, por eu estar acessando o array de outra página , e pelo que entendi , acessando o array de outra página ele só percorre o último em vez do índice.
      Mas se eu jogar todo código dentro da classe ai roda beleza , mas não quero misturar objeto com html (isso é má prática na programação), por isso estou remodelando meu código em POO
      Se alguém tiver uma solução eu agradeço  .  :)
    • By Keber_Crato
      Boa noite galera!!  estou com dificuldades em conseguir o somatório de de dois campos de uma tabela e o somatório de um campo de outra tabela,
      segue assim:
      Tenho uma tabela chamada CID_TITULAR  e outra CID_COMPOSICAO. a tabela CID_TITULAR recebe os cadastro de um programa social e a tabela CID_COMPOSICAO o cadastros dos dependentes (filhos, parentes) TITULAR_COD = COMPOSICAO_COD_TITULAR (chave de ligação)
      tenho dois campos na CID_TITULAR de valores, são eles: TITULAR_RENDA e TITULAR_RENDA_MENSAL e na outra tabela CID_COMPOSICAO tenho um campo de valor COMPOSICAO_RENDA
      estão povoado assim:
       
      Na tabela CID_TITULAR nos campos TITULAR_RENDA (R$ 10.00) e TITULAR_RENDA_MENSAL (R$ 10.00) e na tabela CID_COMPOSICAO no campo COMPOSICAO_RENDA tenho 3 registros com os valores R$ 300.00, R$ 200.00 e R$ 100.00
      GOSTARIA QUE A SELEÇÃO FOSSE O VALOR DOS DOIS CAMPOS DA TABELA CID_TITULAR + O CAMPO DA OUTRA TABELA CID_COMPOSICAO
       
      NO CASO O SOMATÓRIO CORRETO SERIA R$ 620,00
      10,00 no campo TITULAR_RENDA
      10,00 no campo TITULAR_RENDA_MENSAL
      300,00 NO 1º REGISTRO no campo COMPOSICAO_RENDA
      200,00 NO 2º REGISTRO no campo COMPOSICAO_RENDA
      100,00 NO 3º REGISTRO no campo COMPOSICAO_RENDA
       
      TOTALIZANDO R$ 620,00
      O PROBLEMA É:
      Quando faço a seleção com inner join a resposta é R$ 660,00 e não 620,00. a seleção triplica o valor da primeira tabela devido ter três registros na tabela CID_COMPOSICAO
       
      TABELA CID_TITULAR
      CREATE TABLE `cid_titular` (
        `titular_cod` int(11) NOT NULL AUTO_INCREMENT,
        `titular_renda` float(10,2) DEFAULT NULL,
        `titular_renda_mensal` float(10,2) DEFAULT NULL,
        PRIMARY KEY (`titular_cod`)
      ) ENGINE=InnoDB DEFAULT CHARSET=utf8 COLLATE=utf8_unicode_ci
       
      TABELA CID_COMPOSICAO
      CREATE TABLE `cid_composicao` (
        `composicao_cod` int(11) NOT NULL AUTO_INCREMENT,
        `composicao_cod_titular` int(11) DEFAULT NULL,
        `composicao_renda` float(10,2) DEFAULT NULL,
        PRIMARY KEY (`composicao_cod`)
      ) ENGINE=InnoDB DEFAULT CHARSET=utf8 COLLATE=utf8_unicode_ci
       
       
      GOSTARIA MUITO DE AJUDA GALERA. OBRIGADO!!!!
       

    • By malufla79
      Boa tarde, estou com probleminha no phpmailer, o código abaixo funciona corretamente quando consigo enviar para um único e-mail, mas quando coloco a opção todos, ele só manda apenas para primeiro e-mail e não apresenta nenhum erro. Poderia me dar uma ajudinha? Obrigada
       
      Controle
      if ($_POST['filtro'] == 'todos') { // essa consulta pego a lista de e-mails cadastrado no meu banco de dados $usuarios->consultar("select * from mala_email"); $linha = $usuarios->Linha; $rs = $usuarios->Result; } if ($_POST['filtro'] == 'todos') { for ($i=0; $i<$linha; $i++) { $email = $rs[$i]['email']; $mensagem = $_POST['editor']; $util->EnviarEmaiLivre($email, utf8_decode($_POST['assunto']), $mensagem); } } else { $mensagem = $_POST['editor']; $util->EnviarEmaiLivre($_POST['email'], utf8_decode($_POST['assunto']), $mensagem); } Função
      function EnviarEmaiLivre($email, $assunto, $mensagem) { date_default_timezone_set('Etc/UTC'); require 'phpmailer/PHPMailerAutoload.php'; $hostSmtp = "mail.domino.com.br"; $smtpUser = "email@email.com.br"; $senhaSmtp = "********"; $mensagem = ' <p><img src="http://www.dominio.com.br/images/logodolugar.png" alt="Logo do Lugar" width="550"></p> '.$mensagem; //Create a new PHPMailer instance $mail = new PHPMailer; //Tell PHPMailer to use SMTP $mail->isSMTP(); //Enable SMTP debugging // 0 = off (for production use) // 1 = client messages // 2 = client and server messages $mail->SMTPDebug = 0; //Ask for HTML-friendly debug output $mail->Debugoutput = 'html'; //Set the hostname of the mail server $mail->Host = $hostSmtp; // use // if your network does not support SMTP over IPv6 //Set the SMTP port number - 587 for authenticated TLS, a.k.a. RFC4409 SMTP submission $mail->Port = 587; //Set the encryption system to use - ssl (deprecated) or tls $mail->SMTPSecure = 'tsl'; //Whether to use SMTP authentication $mail->SMTPAuth = true; //Username to use for SMTP authentication - use full email address for gmail $mail->Username = $smtpUser; //Password to use for SMTP authentication $mail->Password = $senhaSmtp; //Set who the message is to be sent from $mail->setFrom('email@email.com.br', 'E-mail do Lugar'); //Set who the message is to be sent to $mail->addAddress($email); //Set the subject line $mail->Subject = $assunto; //Replace the plain text body with one created manually $mail->Body = $mensagem; $mail->isHTML(true); if ($conteudoArquivo != '' && $arquivo != '') { //Attach an image file $mail->addAttachment('phpmailer/images/phpmailer_mini.png'); //Read an HTML message body from an external file, convert referenced images to embedded, //convert HTML into a basic plain-text alternative body $mail->msgHTML(file_get_contents('contents.html'), dirname(__FILE__)); } if (!$mail->send()) { $retorno = false; } else { $retorno = true; } }  
       
    • By Leandro Volanick
      Boa tarde pessoal.
       
      Gostaria de uma ajuda em uma consulta sql, onde quero agrupar os todos produtos comprados pelo mesmo CPF, dentro de uma array. 
       
      Segue o que estou usando:
      $data=mysqli_query($mysqli, "SELECT xml.id, xml.xProd as produto, prod.id as iddoproduto, GROUP_CONCAT( prod.xProd ) FROM produtos AS prod LEFT JOIN xml as xml ON xml.xProd = prod.xProd GROUP BY xml.CPFdest"); while($info=mysqli_fetch_array($data)) echo '['.$info['iddoproduto'].'],'; No momento, o codigo está trazendo apenas 1 produto, quero que ele printe todos produtos que o CPFdest já comprou.
       
       
    • By Vítor Gonçalves
      Boa Tarde, gostaria se alguém poderia me ajudar!
      tenho um sistema desktop que é validado a cada 6 meses, quando termina essa validade ele pedi para renovar!
      - desejo criar um script em PHP  com a seguinte função:
      -pegar a senha gerada no sistema e enviar para o servidor web( hostinger server-sql) via php
      - o servidor gera e mostra uma contra senha  para ser copiada de volta.
      - ao colar a contra senha no campo "nova licença" no meu programa, ele compara com a que tá no banco de dados na hostinger se bate a mesma e libera o uso.
       
      - já vi isso em outro sistema mas não tive acesso aos scripts para ver como funciona.
×

Important Information

Ao usar o fórum, você concorda com nossos Terms of Use.