Release 2017.1
[cascardo/rnetclient.git] / rnet_encode.c
1 /*
2  *  Copyright (C) 2013-2017  Thadeu Lima de Souza Cascardo <cascardo@minaslivre.org>
3  *  Copyright (C) 2014  Alexandre Oliva <lxoliva@fsfla.org>
4  *
5  *  This program is free software; you can redistribute it and/or modify
6  *  it under the terms of the GNU General Public License as published by
7  *  the Free Software Foundation; either version 3 of the License, or
8  *  (at your option) any later version.
9  *
10  *  This program is distributed in the hope that it will be useful,
11  *  but WITHOUT ANY WARRANTY; without even the implied warranty of
12  *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
13  *  GNU General Public License for more details.
14  *
15  *  You should have received a copy of the GNU General Public License along
16  *  with this program; if not, write to the Free Software Foundation, Inc.,
17  *  51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
18  */
19
20 #include "rnet_encode.h"
21 #include <unistd.h>
22 #include <stdio.h>
23 #include <stdlib.h>
24 #include <string.h>
25 #include <errno.h>
26 #include <fcntl.h>
27 #include "rnet_message.h"
28 #include "decfile.h"
29
30 int rnet_encode(struct rnet_decfile *decfile, struct rnet_message **msg, char *client)
31 {
32         int r = -EIO;
33
34         uint32_t tp_arq;
35         uint32_t id_dec;
36         char *cpf;
37         char *codigo_recnet;
38         char *ano;
39         char *exerc;
40         char *uf;
41         uint16_t versao_pgd;
42         uint64_t file_len;
43         char *hash;
44         char *header;
45         uint8_t ret;
46
47         size_t header_size, header_head, header_tail;
48
49         if (client == NULL)
50                 client = RNET_DEFAULT_VERSION;
51
52         *msg = rnet_message_new();
53         if (*msg == NULL) {
54                 return -ENOMEM;
55         }
56
57         file_len = rnet_decfile_get_file(decfile)->len;
58         hash = rnet_decfile_get_file_hash(decfile);
59         if (!hash)
60                 goto out;
61         header = rnet_decfile_get_header(decfile);
62         header_head = RNET_HEADER_HEAD_COMMON;
63         header_tail = RNET_HEADER_TAIL_COMMON;
64         /* This was already checked at parse time. */
65         header_size = strlen(header);
66
67         codigo_recnet = rnet_decfile_get_header_field(decfile, "codigo_recnet");
68         tp_arq = strtoul(codigo_recnet, NULL, 10);
69         id_dec = strtoul(rnet_decfile_get_header_field(decfile, "hash"), NULL, 10);
70         cpf = rnet_decfile_get_header_field(decfile, "cpf");
71         ano = rnet_decfile_get_header_field(decfile, "ano");
72         exerc = rnet_decfile_get_header_field(decfile, "exerc");
73         uf = rnet_decfile_get_header_field(decfile, "uf");
74         versao_pgd = strtoul(rnet_decfile_get_header_field(decfile, "nr_versao"), NULL, 10);
75         ret = strtoul(rnet_decfile_get_header_field(decfile, "in_ret"), NULL, 10);
76
77         (*msg)->buffer[0] = 0x40;
78         (*msg)->len = 1;
79         r = rnet_message_add_u32(msg, "a_comp", 0);
80         r = rnet_message_add_u32(msg, "tp_arq", tp_arq);
81         r = rnet_message_add_u32(msg, "id_dec", id_dec);
82         r = rnet_message_add_ascii(msg, "exercicio", ano);
83         r = rnet_message_add_ascii(msg, "exercicio_pgd", exerc);
84         r = rnet_message_add_buffer(msg, "hash_arq", hash, 16);
85         r = rnet_message_add_buffer(msg, "hash_trans", hash, 16);
86         r = rnet_message_add_ascii(msg, "ni", cpf);
87         r = rnet_message_add_ascii(msg, "tp_ni", "CPF");
88         r = rnet_message_add_u8(msg, "num_ass", 0);
89         r = rnet_message_add_u32(msg, "p_comp", 0);
90         r = rnet_message_add_u8(msg, "ret", ret);
91         r = rnet_message_add_u64(msg, "tam_arq", file_len);
92         r = rnet_message_add_u64(msg, "tam_assinado", file_len);
93         r = rnet_message_add_u64(msg, "tam_trans", file_len);
94         r = rnet_message_add_ascii(msg, "uf", uf);
95         r = rnet_message_add_u8(msg, "vrs_des_pa", 0);
96         r = rnet_message_add_u16(msg, "versao_pgd", versao_pgd);
97         r = rnet_message_add_u8(msg, "critica_validador", 0x06);
98         r = rnet_message_add_ascii(msg, "ip_loc", "127.0.0.1");
99         r = rnet_message_add_ascii(msg, "versao_java", "1.5.0-gij;Free Software rnetclient pretending to be GNU Interpreter for Java");
100         r = rnet_message_add_ascii(msg, "origem", "JA2R");
101         r = rnet_message_add_ascii(msg, "so", "GNU");
102         r = rnet_message_add_ascii(msg, "cliente", client);
103         r = rnet_message_add_buffer(msg, "dados_val",
104                                     header + header_head,
105                                     header_size - header_tail - header_head);
106         r = rnet_message_add_u32(msg, "tam_dados_val", 0);
107         r = rnet_message_add_u32(msg, "tam_dados_val_chave", 0);
108         r = rnet_message_add_u32(msg, "arquivos_restantes", 0);
109
110         free(hash);
111
112         if (r < 0)
113                 goto out;
114         return 0;
115
116 out:
117         rnet_message_del(*msg);
118         return r;
119 }