481.1.9
by Monty Taylor
Added autoconf tests for location of cstdint and cinttypes. Use those in C++ programs now, so that we don't have to define _STDC_LIMIT_MACROS, etc by hand. Stop, in fact, defining those by hand. |
1 |
#include "config.h" |
2 |
||
324.1.1
by Mats Kindahl
Adding specification of a simple protobuf-based binary log format, |
3 |
#include "binlog_encoding.h" |
4 |
#include "binary_log.h" |
|
5 |
||
6 |
#include "ioutil.h" |
|
7 |
||
8 |
#include <google/protobuf/io/zero_copy_stream_impl.h> |
|
9 |
#include <google/protobuf/io/coded_stream.h> |
|
10 |
||
11 |
#include <iostream> |
|
12 |
#include <sstream> |
|
13 |
#include <string> |
|
14 |
#include <map> |
|
15 |
||
16 |
#include <getopt.h> |
|
17 |
#include <sys/stat.h> |
|
18 |
#include <fcntl.h> |
|
19 |
||
20 |
using namespace google::protobuf::io; |
|
21 |
||
22 |
typedef std::map<std::string,std::string> Assign; |
|
23 |
||
24 |
void print_usage_and_exit(char *prog) { |
|
25 |
using std::cerr; |
|
26 |
const char *name= strrchr(prog, '/'); |
|
27 |
||
28 |
if (name) |
|
29 |
++name; |
|
30 |
else
|
|
31 |
name= "binlog_writer"; |
|
32 |
cerr << "Usage: " << name << " <options> <query>\n" |
|
33 |
<< " --output name Append query to file <name> (default: 'log.bin')\n" |
|
34 |
<< " --set var=val Set value of user variable for query\n" |
|
35 |
<< " --trans-id <id> Set transaction id to <id>\n" |
|
36 |
<< std::flush; |
|
37 |
exit(1); |
|
38 |
}
|
|
39 |
||
40 |
||
41 |
void
|
|
42 |
write_query(CodedOutputStream* out, |
|
43 |
unsigned long trans_id, |
|
44 |
const std::string& query, |
|
45 |
const Assign& assign) |
|
46 |
{
|
|
47 |
BinaryLog::Query *message = new BinaryLog::Query; |
|
48 |
||
49 |
{
|
|
50 |
BinaryLog::Header *header= message->mutable_header(); |
|
51 |
header->set_seqno(time(NULL)); |
|
52 |
header->set_server_id(1); |
|
53 |
header->set_trans_id(trans_id); |
|
54 |
}
|
|
55 |
||
56 |
message->set_query(query); |
|
57 |
for (Assign::const_iterator ii= assign.begin() ; |
|
58 |
ii != assign.end() ; |
|
59 |
++ii ) |
|
60 |
{
|
|
61 |
BinaryLog::Query::Variable *var= message->add_variable(); |
|
62 |
var->set_name(ii->first); |
|
63 |
var->set_value(ii->second); |
|
64 |
}
|
|
65 |
||
66 |
BinaryLog::Event event(BinaryLog::Event::QUERY, message); |
|
67 |
event.write(out); |
|
68 |
}
|
|
69 |
||
70 |
||
71 |
int main(int argc, char *argv[]) |
|
72 |
{
|
|
73 |
using std::ios; |
|
74 |
GOOGLE_PROTOBUF_VERIFY_VERSION; |
|
75 |
||
76 |
static struct option options[] = { |
|
77 |
{ "set", 1 /* has_arg */, NULL, 0 }, |
|
78 |
{ "trans-id", 1 /* has_arg */, NULL, 0 }, |
|
79 |
{ "output", 1 /* has_arg */, NULL, 0 }, |
|
80 |
{ 0, 0, 0, 0 } |
|
81 |
};
|
|
82 |
||
83 |
Assign assign; |
|
84 |
unsigned long trans_id= 0; |
|
85 |
const char* file_name= "log.bin"; |
|
86 |
||
87 |
int ch, option_index; |
|
88 |
while ((ch= getopt_long(argc, argv, "", options, &option_index)) != -1) { |
|
89 |
if (ch == '?') |
|
90 |
print_usage_and_exit(argv[0]); |
|
91 |
||
92 |
switch (option_index) { |
|
93 |
case 0: // --set |
|
94 |
{
|
|
95 |
// Split the supplied string at the first '='
|
|
96 |
char *end= optarg + strlen(optarg); |
|
97 |
char *pos= strchr(optarg, '='); |
|
98 |
if (!pos) |
|
99 |
pos= end; |
|
100 |
const std::string key(optarg, pos); |
|
101 |
const std::string value(pos == end ? end : pos+1, end); |
|
102 |
assign[key]= value; |
|
103 |
}
|
|
104 |
||
105 |
case 1: // --trans-id |
|
106 |
trans_id= strtoul(optarg, NULL, 0); |
|
107 |
break; |
|
108 |
||
109 |
case 2: // --output |
|
110 |
file_name= optarg; |
|
111 |
break; |
|
112 |
}
|
|
113 |
}
|
|
114 |
||
115 |
if (optind >= argc) |
|
116 |
print_usage_and_exit(argv[0]); |
|
117 |
||
118 |
int outfd= open(file_name, O_WRONLY | O_CREAT | O_APPEND, 0644); |
|
119 |
if (outfd < 0) |
|
120 |
{
|
|
121 |
perror("open"); |
|
122 |
exit(1); |
|
123 |
}
|
|
124 |
||
125 |
ZeroCopyOutputStream* raw_output = new FileOutputStream(outfd); |
|
126 |
CodedOutputStream* coded_output = new CodedOutputStream(raw_output); |
|
127 |
||
128 |
std::stringstream sout; |
|
129 |
sout << ioutil::join(" ", &argv[optind], &argv[argc]); |
|
130 |
||
131 |
write_query(coded_output, trans_id, sout.str(), assign); |
|
132 |
||
133 |
delete coded_output; |
|
134 |
delete raw_output; |
|
135 |
close(outfd); |
|
136 |
exit(0); |
|
137 |
}
|