/// <summary> /// Obtém a URL para uso no QR-Code /// </summary> /// <param name="infNFeSupl"></param> /// <param name="nfe"></param> /// <param name="cIdToken"></param> /// <param name="csc"></param> /// <returns></returns> public static string ObterUrlQrCode(this infNFeSupl infNFeSupl, Classes.NFe nfe, string cIdToken, string csc, VersaoServico versaoServico = VersaoServico.ve310) { //Passo 1: Converter o valor da Data e Hora de Emissão da NFC-e (dhEmi) para HEXA; var dhEmi = ObterHexDeString(nfe.infNFe.ide.ProxyDhEmi); //Passo 2: Converter o valor do Digest Value da NFC-e (digVal) para HEXA; //Ao se efetuar a assinatura digital da NFCe emitida em contingência off-line, o campo digest value constante da XMl Signature deve obrigatoriamente ser idêntico ao encontrado quando da geração do digest value para a montagem QR Code. //Ver página 18 do Manual de Padrões Padrões Técnicos do DANFE - NFC - e e QR Code, versão 3.2 if (nfe.Signature == null) { throw new Exception("Não é possível obter a URL do QR-Code de uma NFCe não assinada!"); } var digVal = ObterHexDeString(nfe.Signature.SignedInfo.Reference.DigestValue); //Na hipótese do consumidor não se identificar, não existirá o parâmetro cDest no QR Code; var cDest = ""; if (nfe.infNFe.dest != null) { cDest = "&cDest=" + nfe.infNFe.dest.CPF + nfe.infNFe.dest.CNPJ + nfe.infNFe.dest.idEstrangeiro; } //Passo 3: Substituir os valores (“dhEmi” e “digVal”) nos parâmetros; var dadosBase = "chNFe=" + nfe.infNFe.Id.Substring(3) + "&nVersao=100&tpAmb=" + ((int)nfe.infNFe.ide.tpAmb) + cDest + "&dhEmi=" + dhEmi + "&vNF=" + nfe.infNFe.total.ICMSTot.vNF.ToString("0.00").Replace(',', '.') + "&vICMS=" + nfe.infNFe.total.ICMSTot.vICMS.ToString("0.00").Replace(',', '.') + "&digVal=" + digVal + "&cIdToken=" + cIdToken; //Passo 4: Adicionar, ao final dos parâmetros, o CSC (CSC do contribuinte disponibilizado pela SEFAZ do Estado onde a empresa esta localizada): var dadosParaSh1 = dadosBase + csc; //Passo 5: Aplicar o algoritmo SHA-1 sobre todos os parâmetros concatenados. Asaída do algoritmo SHA-1 deve ser em HEXADECIMAL. var sha1ComCsc = ObterHexSha1DeString(dadosParaSh1); //Passo 6: Adicione o resultado sem o CSC e gere a imagem do QR Code: 1º parte (endereço da consulta) +2º parte (tabela 3 com indicação SIM na última coluna). return(ObterUrl(infNFeSupl, nfe.infNFe.ide.tpAmb, nfe.infNFe.ide.cUF, TipoUrlConsultaPublica.UrlQrCode, versaoServico) + "?" + dadosBase + "&cHashQRCode=" + sha1ComCsc); }
/// <summary> /// Obtém a URL para montagem do QR-Code /// </summary> public static string ObterUrlQrCode(this infNFeSupl infNFeSupl, Classes.NFe nfe, VersaoQrCode versaoQrCode, string cIdToken, string csc) { Func <string, string> msgErro = parametro => $"O {parametro} não foi informado!"; if (string.IsNullOrEmpty(cIdToken)) { throw new ArgumentNullException(nameof(cIdToken), msgErro("token")); } if (string.IsNullOrEmpty(csc)) { throw new ArgumentNullException(nameof(cIdToken), msgErro("CSC")); } var versaoServico = Conversao.StringParaVersaoServico(nfe.infNFe.versao); switch (versaoQrCode) { case VersaoQrCode.QrCodeVersao1: return(ObterUrlQrCode1(infNFeSupl, nfe, cIdToken, csc, versaoServico)); case VersaoQrCode.QrCodeVersao2: return(ObterUrlQrCode2(infNFeSupl, nfe, cIdToken, csc, versaoServico)); default: throw new ArgumentOutOfRangeException("versaoQrCode", versaoQrCode, null); } }
public void UrlQrCode2DeveTerminarComQueryPararametro(Estado estado, TipoAmbiente tipoAmbiente) { var infNFeSupl = new infNFeSupl(); var url = infNFeSupl.ObterUrlQrCode2ComParametro(tipoAmbiente, estado, VersaoServico.Versao400); Assert.EndsWith("?p=", url); }
/// <summary> /// Obtém a URL para uso no DANFE da NFCe /// </summary> /// <param name="infNFeSupl"></param> /// <param name="tipoAmbiente"></param> /// <param name="estado"></param> /// <param name="tipoUrlConsultaPublica"></param> /// <returns></returns> public static string ObterUrl(this infNFeSupl infNFeSupl, TipoAmbiente tipoAmbiente, Estado estado, TipoUrlConsultaPublica tipoUrlConsultaPublica, VersaoServico versaoServico = VersaoServico.ve310) {// todo endereço qrcode var query = from qr in EndQrCodeNfce where qr.TipoAmbiente == tipoAmbiente && qr.Estado == estado && qr.TipoUrlConsultaPublica == tipoUrlConsultaPublica && qr.VersaoServico == versaoServico select qr.Url; var listaRetorno = query as IList <string> ?? query.ToList(); var qtdeRetorno = listaRetorno.Count(); if (qtdeRetorno == 0) { throw new Exception(string.Format("Não foi possível obter o {0}, para o Estado {1}, ambiente: {2}", tipoUrlConsultaPublica.Descricao(), estado, tipoAmbiente.Descricao())); } if (qtdeRetorno > 1) { throw new Exception("A função ObterUrl obteve mais de um resultado!"); } return(listaRetorno.FirstOrDefault()); }
/// <summary> /// Obtém a URL para montagem do QR-Code /// </summary> public static string ObterUrlQrCode(this infNFeSupl infNFeSupl, Classes.NFe nfe, VersaoQrCode versaoQrCode, string cIdToken, string csc) { var versaoServico = Conversao.StringParaVersaoServico(nfe.infNFe.versao); switch (versaoQrCode) { case VersaoQrCode.QrCodeVersao1: return(ObterUrlQrCode1(infNFeSupl, nfe, cIdToken, csc, versaoServico)); case VersaoQrCode.QrCodeVersao2: return(ObterUrlQrCode2(infNFeSupl, nfe, cIdToken, csc, versaoServico)); default: throw new ArgumentOutOfRangeException("versaoQrCode", versaoQrCode, null); } }
/// <summary> /// Obtém a URL para uso no QR-Code, versão 2.0 - leiaute 4.00 /// </summary> private static string ObterUrlQrCode2(infNFeSupl infNFeSupl, Classes.NFe nfe, string cIdToken, string csc, VersaoServico versaoServico) { #region 1ª parte const string parametro = "?p="; var url = ObterUrl(infNFeSupl, nfe.infNFe.ide.tpAmb, nfe.infNFe.ide.cUF, TipoUrlConsultaPublica.UrlQrCode, versaoServico, VersaoQrCode.QrCodeVersao2); if (!url.EndsWith(parametro)) { url += parametro; } #endregion #region 2ª parte const string pipe = "|"; //Chave de Acesso da NFC-e var chave = nfe.infNFe.Id.Substring(3); //Identificação do Ambiente (1 – Produção, 2 – Homologação) var ambiente = (int)nfe.infNFe.ide.tpAmb; //Identificador do CSC (Código de Segurança do Contribuinte no Banco de Dados da SEFAZ). Informar sem os zeros não significativos var idCsc = Convert.ToInt16(cIdToken); string dadosBase; if (nfe.infNFe.ide.tpEmis == TipoEmissao.teOffLine) { var diaEmi = nfe.infNFe.ide.dhEmi.Day.ToString("D2"); var valorNfce = nfe.infNFe.total.ICMSTot.vNF.ToString("0.00").Replace(',', '.'); var digVal = Conversao.ObterHexDeString(nfe.Signature.SignedInfo.Reference.DigestValue); dadosBase = string.Concat(chave, pipe, (int)VersaoQrCode.QrCodeVersao2, pipe, ambiente, pipe, diaEmi, pipe, valorNfce, pipe, digVal, pipe, idCsc); } else { dadosBase = string.Concat(chave, pipe, (int)VersaoQrCode.QrCodeVersao2, pipe, ambiente, pipe, idCsc); } var dadosSha1 = string.Concat(dadosBase, csc); var sh1 = Conversao.ObterHexSha1DeString(dadosSha1); return(string.Concat(url, dadosBase, pipe, sh1)); #endregion }
/// <summary> /// Obtém a URL para o QR-Code versão 2.0 com o tratamento de parâmetro query no final da url /// </summary> /// <returns></returns> public static string ObterUrlQrCode2ComParametro(this infNFeSupl infNFeSupl, TipoAmbiente tipoAmbiente, Estado estado, VersaoServico versaoServico) { const string interrogacao = "?"; const string parametro = "p="; var url = ObterUrl(infNFeSupl, tipoAmbiente, estado, TipoUrlConsultaPublica.UrlQrCode, versaoServico, VersaoQrCode.QrCodeVersao2); if (!url.EndsWith(interrogacao)) { url += interrogacao; } if (!url.EndsWith(parametro)) { url += parametro; } return(url); }
/// <summary> /// Obtém a URL de consulta pela chave /// </summary> public static string ObterUrlConsulta(this infNFeSupl infNFeSupl, Classes.NFe nfe, VersaoQrCode versaoQrCode) { var versaoServico = Conversao.StringParaVersaoServico(nfe.infNFe.versao); return(ObterUrl(infNFeSupl, nfe.infNFe.ide.tpAmb, nfe.infNFe.ide.cUF, TipoUrlConsultaPublica.UrlConsulta, versaoServico, versaoQrCode)); }