Compare commits
No commits in common. "pristine-tar" and "master" have entirely different histories.
pristine-t
...
master
20
.bzrignore
Normal file
20
.bzrignore
Normal file
@ -0,0 +1,20 @@
|
||||
mydumper
|
||||
myloader
|
||||
*~
|
||||
*.dSYM
|
||||
export
|
||||
CMakeCache.txt
|
||||
CMakeFiles/
|
||||
Makefile
|
||||
cmake_install.cmake
|
||||
config.h
|
||||
docs/CMakeFiles/
|
||||
docs/Makefile
|
||||
docs/_doctrees/
|
||||
docs/_sources/
|
||||
docs/cmake_install.cmake
|
||||
docs/html/
|
||||
docs/man/
|
||||
docs/_build/conf.py
|
||||
docs/_build/sources.cmake
|
||||
.project
|
73
CMakeLists.txt
Normal file
73
CMakeLists.txt
Normal file
@ -0,0 +1,73 @@
|
||||
cmake_minimum_required(VERSION 2.6)
|
||||
project(mydumper)
|
||||
set(VERSION 0.9.5)
|
||||
set(ARCHIVE_NAME "${CMAKE_PROJECT_NAME}-${VERSION}")
|
||||
|
||||
#Required packages
|
||||
set(CMAKE_MODULE_PATH ${CMAKE_SOURCE_DIR}/cmake/modules)
|
||||
find_package(MySQL)
|
||||
find_package(ZLIB)
|
||||
find_package(GLIB2)
|
||||
find_package(PCRE)
|
||||
|
||||
option(BUILD_DOCS "Build the documentation" ON)
|
||||
|
||||
if (BUILD_DOCS)
|
||||
add_subdirectory(docs)
|
||||
endif (BUILD_DOCS)
|
||||
|
||||
option(WITH_BINLOG "Build binlog dump options" OFF)
|
||||
option(WITH_SSL "Build SSL support" ON)
|
||||
|
||||
set(CMAKE_C_FLAGS "-Wall -Wno-deprecated-declarations -Wunused -Wwrite-strings -Wno-strict-aliasing -Wextra -Wshadow -Werror -O3 -g ${MYSQL_CFLAGS}")
|
||||
|
||||
include_directories(${MYDUMPER_SOURCE_DIR} ${MYSQL_INCLUDE_DIR} ${GLIB2_INCLUDE_DIR} ${PCRE_INCLUDE_DIR} ${ZLIB_INCLUDE_DIRS})
|
||||
|
||||
if (NOT CMAKE_INSTALL_PREFIX)
|
||||
SET(CMAKE_INSTALL_PREFIX "/usr/local" CACHE STRING "Install path" FORCE)
|
||||
endif (NOT CMAKE_INSTALL_PREFIX)
|
||||
MARK_AS_ADVANCED(CMAKE)
|
||||
|
||||
CONFIGURE_FILE(${CMAKE_CURRENT_SOURCE_DIR}/config.h.in ${CMAKE_CURRENT_SOURCE_DIR}/config.h)
|
||||
|
||||
|
||||
if (WITH_BINLOG)
|
||||
|
||||
add_executable(mydumper mydumper.c binlog.c server_detect.c g_unix_signal.c connection.c getPassword.c)
|
||||
else (WITH_BINLOG)
|
||||
add_executable(mydumper mydumper.c server_detect.c g_unix_signal.c connection.c getPassword.c)
|
||||
endif (WITH_BINLOG)
|
||||
target_link_libraries(mydumper ${MYSQL_LIBRARIES} ${GLIB2_LIBRARIES} ${GTHREAD2_LIBRARIES} ${PCRE_PCRE_LIBRARY} ${ZLIB_LIBRARIES} stdc++)
|
||||
|
||||
|
||||
add_executable(myloader myloader.c connection.c getPassword.c)
|
||||
target_link_libraries(myloader ${MYSQL_LIBRARIES} ${GLIB2_LIBRARIES} ${GTHREAD2_LIBRARIES} ${PCRE_PCRE_LIBRARY} ${ZLIB_LIBRARIES} stdc++)
|
||||
|
||||
INSTALL(TARGETS mydumper myloader
|
||||
RUNTIME DESTINATION bin
|
||||
)
|
||||
|
||||
add_custom_target(dist
|
||||
COMMAND bzr export --root=${ARCHIVE_NAME}
|
||||
${CMAKE_BINARY_DIR}/${ARCHIVE_NAME}.tar.gz
|
||||
WORKING_DIRECTORY ${CMAKE_SOURCE_DIR})
|
||||
|
||||
OPTION(RUN_CPPCHECK "Run cppcheck" OFF)
|
||||
|
||||
IF(RUN_CPPCHECK)
|
||||
include(CppcheckTargets)
|
||||
add_cppcheck(mydumper)
|
||||
add_cppcheck(myloader)
|
||||
ENDIF(RUN_CPPCHECK)
|
||||
|
||||
|
||||
MESSAGE(STATUS "------------------------------------------------")
|
||||
MESSAGE(STATUS "MYSQL_CONFIG = ${MYSQL_CONFIG}")
|
||||
MESSAGE(STATUS "CMAKE_INSTALL_PREFIX = ${CMAKE_INSTALL_PREFIX}")
|
||||
MESSAGE(STATUS "BUILD_DOCS = ${BUILD_DOCS}")
|
||||
MESSAGE(STATUS "WITH_BINLOG = ${WITH_BINLOG}")
|
||||
MESSAGE(STATUS "RUN_CPPCHECK = ${RUN_CPPCHECK}")
|
||||
MESSAGE(STATUS "Change a values with: cmake -D<Variable>=<Value>")
|
||||
MESSAGE(STATUS "------------------------------------------------")
|
||||
MESSAGE(STATUS)
|
||||
|
674
LICENSE
Normal file
674
LICENSE
Normal file
@ -0,0 +1,674 @@
|
||||
GNU GENERAL PUBLIC LICENSE
|
||||
Version 3, 29 June 2007
|
||||
|
||||
Copyright (C) 2007 Free Software Foundation, Inc. <http://fsf.org/>
|
||||
Everyone is permitted to copy and distribute verbatim copies
|
||||
of this license document, but changing it is not allowed.
|
||||
|
||||
Preamble
|
||||
|
||||
The GNU General Public License is a free, copyleft license for
|
||||
software and other kinds of works.
|
||||
|
||||
The licenses for most software and other practical works are designed
|
||||
to take away your freedom to share and change the works. By contrast,
|
||||
the GNU General Public License is intended to guarantee your freedom to
|
||||
share and change all versions of a program--to make sure it remains free
|
||||
software for all its users. We, the Free Software Foundation, use the
|
||||
GNU General Public License for most of our software; it applies also to
|
||||
any other work released this way by its authors. You can apply it to
|
||||
your programs, too.
|
||||
|
||||
When we speak of free software, we are referring to freedom, not
|
||||
price. Our General Public Licenses are designed to make sure that you
|
||||
have the freedom to distribute copies of free software (and charge for
|
||||
them if you wish), that you receive source code or can get it if you
|
||||
want it, that you can change the software or use pieces of it in new
|
||||
free programs, and that you know you can do these things.
|
||||
|
||||
To protect your rights, we need to prevent others from denying you
|
||||
these rights or asking you to surrender the rights. Therefore, you have
|
||||
certain responsibilities if you distribute copies of the software, or if
|
||||
you modify it: responsibilities to respect the freedom of others.
|
||||
|
||||
For example, if you distribute copies of such a program, whether
|
||||
gratis or for a fee, you must pass on to the recipients the same
|
||||
freedoms that you received. You must make sure that they, too, receive
|
||||
or can get the source code. And you must show them these terms so they
|
||||
know their rights.
|
||||
|
||||
Developers that use the GNU GPL protect your rights with two steps:
|
||||
(1) assert copyright on the software, and (2) offer you this License
|
||||
giving you legal permission to copy, distribute and/or modify it.
|
||||
|
||||
For the developers' and authors' protection, the GPL clearly explains
|
||||
that there is no warranty for this free software. For both users' and
|
||||
authors' sake, the GPL requires that modified versions be marked as
|
||||
changed, so that their problems will not be attributed erroneously to
|
||||
authors of previous versions.
|
||||
|
||||
Some devices are designed to deny users access to install or run
|
||||
modified versions of the software inside them, although the manufacturer
|
||||
can do so. This is fundamentally incompatible with the aim of
|
||||
protecting users' freedom to change the software. The systematic
|
||||
pattern of such abuse occurs in the area of products for individuals to
|
||||
use, which is precisely where it is most unacceptable. Therefore, we
|
||||
have designed this version of the GPL to prohibit the practice for those
|
||||
products. If such problems arise substantially in other domains, we
|
||||
stand ready to extend this provision to those domains in future versions
|
||||
of the GPL, as needed to protect the freedom of users.
|
||||
|
||||
Finally, every program is threatened constantly by software patents.
|
||||
States should not allow patents to restrict development and use of
|
||||
software on general-purpose computers, but in those that do, we wish to
|
||||
avoid the special danger that patents applied to a free program could
|
||||
make it effectively proprietary. To prevent this, the GPL assures that
|
||||
patents cannot be used to render the program non-free.
|
||||
|
||||
The precise terms and conditions for copying, distribution and
|
||||
modification follow.
|
||||
|
||||
TERMS AND CONDITIONS
|
||||
|
||||
0. Definitions.
|
||||
|
||||
"This License" refers to version 3 of the GNU General Public License.
|
||||
|
||||
"Copyright" also means copyright-like laws that apply to other kinds of
|
||||
works, such as semiconductor masks.
|
||||
|
||||
"The Program" refers to any copyrightable work licensed under this
|
||||
License. Each licensee is addressed as "you". "Licensees" and
|
||||
"recipients" may be individuals or organizations.
|
||||
|
||||
To "modify" a work means to copy from or adapt all or part of the work
|
||||
in a fashion requiring copyright permission, other than the making of an
|
||||
exact copy. The resulting work is called a "modified version" of the
|
||||
earlier work or a work "based on" the earlier work.
|
||||
|
||||
A "covered work" means either the unmodified Program or a work based
|
||||
on the Program.
|
||||
|
||||
To "propagate" a work means to do anything with it that, without
|
||||
permission, would make you directly or secondarily liable for
|
||||
infringement under applicable copyright law, except executing it on a
|
||||
computer or modifying a private copy. Propagation includes copying,
|
||||
distribution (with or without modification), making available to the
|
||||
public, and in some countries other activities as well.
|
||||
|
||||
To "convey" a work means any kind of propagation that enables other
|
||||
parties to make or receive copies. Mere interaction with a user through
|
||||
a computer network, with no transfer of a copy, is not conveying.
|
||||
|
||||
An interactive user interface displays "Appropriate Legal Notices"
|
||||
to the extent that it includes a convenient and prominently visible
|
||||
feature that (1) displays an appropriate copyright notice, and (2)
|
||||
tells the user that there is no warranty for the work (except to the
|
||||
extent that warranties are provided), that licensees may convey the
|
||||
work under this License, and how to view a copy of this License. If
|
||||
the interface presents a list of user commands or options, such as a
|
||||
menu, a prominent item in the list meets this criterion.
|
||||
|
||||
1. Source Code.
|
||||
|
||||
The "source code" for a work means the preferred form of the work
|
||||
for making modifications to it. "Object code" means any non-source
|
||||
form of a work.
|
||||
|
||||
A "Standard Interface" means an interface that either is an official
|
||||
standard defined by a recognized standards body, or, in the case of
|
||||
interfaces specified for a particular programming language, one that
|
||||
is widely used among developers working in that language.
|
||||
|
||||
The "System Libraries" of an executable work include anything, other
|
||||
than the work as a whole, that (a) is included in the normal form of
|
||||
packaging a Major Component, but which is not part of that Major
|
||||
Component, and (b) serves only to enable use of the work with that
|
||||
Major Component, or to implement a Standard Interface for which an
|
||||
implementation is available to the public in source code form. A
|
||||
"Major Component", in this context, means a major essential component
|
||||
(kernel, window system, and so on) of the specific operating system
|
||||
(if any) on which the executable work runs, or a compiler used to
|
||||
produce the work, or an object code interpreter used to run it.
|
||||
|
||||
The "Corresponding Source" for a work in object code form means all
|
||||
the source code needed to generate, install, and (for an executable
|
||||
work) run the object code and to modify the work, including scripts to
|
||||
control those activities. However, it does not include the work's
|
||||
System Libraries, or general-purpose tools or generally available free
|
||||
programs which are used unmodified in performing those activities but
|
||||
which are not part of the work. For example, Corresponding Source
|
||||
includes interface definition files associated with source files for
|
||||
the work, and the source code for shared libraries and dynamically
|
||||
linked subprograms that the work is specifically designed to require,
|
||||
such as by intimate data communication or control flow between those
|
||||
subprograms and other parts of the work.
|
||||
|
||||
The Corresponding Source need not include anything that users
|
||||
can regenerate automatically from other parts of the Corresponding
|
||||
Source.
|
||||
|
||||
The Corresponding Source for a work in source code form is that
|
||||
same work.
|
||||
|
||||
2. Basic Permissions.
|
||||
|
||||
All rights granted under this License are granted for the term of
|
||||
copyright on the Program, and are irrevocable provided the stated
|
||||
conditions are met. This License explicitly affirms your unlimited
|
||||
permission to run the unmodified Program. The output from running a
|
||||
covered work is covered by this License only if the output, given its
|
||||
content, constitutes a covered work. This License acknowledges your
|
||||
rights of fair use or other equivalent, as provided by copyright law.
|
||||
|
||||
You may make, run and propagate covered works that you do not
|
||||
convey, without conditions so long as your license otherwise remains
|
||||
in force. You may convey covered works to others for the sole purpose
|
||||
of having them make modifications exclusively for you, or provide you
|
||||
with facilities for running those works, provided that you comply with
|
||||
the terms of this License in conveying all material for which you do
|
||||
not control copyright. Those thus making or running the covered works
|
||||
for you must do so exclusively on your behalf, under your direction
|
||||
and control, on terms that prohibit them from making any copies of
|
||||
your copyrighted material outside their relationship with you.
|
||||
|
||||
Conveying under any other circumstances is permitted solely under
|
||||
the conditions stated below. Sublicensing is not allowed; section 10
|
||||
makes it unnecessary.
|
||||
|
||||
3. Protecting Users' Legal Rights From Anti-Circumvention Law.
|
||||
|
||||
No covered work shall be deemed part of an effective technological
|
||||
measure under any applicable law fulfilling obligations under article
|
||||
11 of the WIPO copyright treaty adopted on 20 December 1996, or
|
||||
similar laws prohibiting or restricting circumvention of such
|
||||
measures.
|
||||
|
||||
When you convey a covered work, you waive any legal power to forbid
|
||||
circumvention of technological measures to the extent such circumvention
|
||||
is effected by exercising rights under this License with respect to
|
||||
the covered work, and you disclaim any intention to limit operation or
|
||||
modification of the work as a means of enforcing, against the work's
|
||||
users, your or third parties' legal rights to forbid circumvention of
|
||||
technological measures.
|
||||
|
||||
4. Conveying Verbatim Copies.
|
||||
|
||||
You may convey verbatim copies of the Program's source code as you
|
||||
receive it, in any medium, provided that you conspicuously and
|
||||
appropriately publish on each copy an appropriate copyright notice;
|
||||
keep intact all notices stating that this License and any
|
||||
non-permissive terms added in accord with section 7 apply to the code;
|
||||
keep intact all notices of the absence of any warranty; and give all
|
||||
recipients a copy of this License along with the Program.
|
||||
|
||||
You may charge any price or no price for each copy that you convey,
|
||||
and you may offer support or warranty protection for a fee.
|
||||
|
||||
5. Conveying Modified Source Versions.
|
||||
|
||||
You may convey a work based on the Program, or the modifications to
|
||||
produce it from the Program, in the form of source code under the
|
||||
terms of section 4, provided that you also meet all of these conditions:
|
||||
|
||||
a) The work must carry prominent notices stating that you modified
|
||||
it, and giving a relevant date.
|
||||
|
||||
b) The work must carry prominent notices stating that it is
|
||||
released under this License and any conditions added under section
|
||||
7. This requirement modifies the requirement in section 4 to
|
||||
"keep intact all notices".
|
||||
|
||||
c) You must license the entire work, as a whole, under this
|
||||
License to anyone who comes into possession of a copy. This
|
||||
License will therefore apply, along with any applicable section 7
|
||||
additional terms, to the whole of the work, and all its parts,
|
||||
regardless of how they are packaged. This License gives no
|
||||
permission to license the work in any other way, but it does not
|
||||
invalidate such permission if you have separately received it.
|
||||
|
||||
d) If the work has interactive user interfaces, each must display
|
||||
Appropriate Legal Notices; however, if the Program has interactive
|
||||
interfaces that do not display Appropriate Legal Notices, your
|
||||
work need not make them do so.
|
||||
|
||||
A compilation of a covered work with other separate and independent
|
||||
works, which are not by their nature extensions of the covered work,
|
||||
and which are not combined with it such as to form a larger program,
|
||||
in or on a volume of a storage or distribution medium, is called an
|
||||
"aggregate" if the compilation and its resulting copyright are not
|
||||
used to limit the access or legal rights of the compilation's users
|
||||
beyond what the individual works permit. Inclusion of a covered work
|
||||
in an aggregate does not cause this License to apply to the other
|
||||
parts of the aggregate.
|
||||
|
||||
6. Conveying Non-Source Forms.
|
||||
|
||||
You may convey a covered work in object code form under the terms
|
||||
of sections 4 and 5, provided that you also convey the
|
||||
machine-readable Corresponding Source under the terms of this License,
|
||||
in one of these ways:
|
||||
|
||||
a) Convey the object code in, or embodied in, a physical product
|
||||
(including a physical distribution medium), accompanied by the
|
||||
Corresponding Source fixed on a durable physical medium
|
||||
customarily used for software interchange.
|
||||
|
||||
b) Convey the object code in, or embodied in, a physical product
|
||||
(including a physical distribution medium), accompanied by a
|
||||
written offer, valid for at least three years and valid for as
|
||||
long as you offer spare parts or customer support for that product
|
||||
model, to give anyone who possesses the object code either (1) a
|
||||
copy of the Corresponding Source for all the software in the
|
||||
product that is covered by this License, on a durable physical
|
||||
medium customarily used for software interchange, for a price no
|
||||
more than your reasonable cost of physically performing this
|
||||
conveying of source, or (2) access to copy the
|
||||
Corresponding Source from a network server at no charge.
|
||||
|
||||
c) Convey individual copies of the object code with a copy of the
|
||||
written offer to provide the Corresponding Source. This
|
||||
alternative is allowed only occasionally and noncommercially, and
|
||||
only if you received the object code with such an offer, in accord
|
||||
with subsection 6b.
|
||||
|
||||
d) Convey the object code by offering access from a designated
|
||||
place (gratis or for a charge), and offer equivalent access to the
|
||||
Corresponding Source in the same way through the same place at no
|
||||
further charge. You need not require recipients to copy the
|
||||
Corresponding Source along with the object code. If the place to
|
||||
copy the object code is a network server, the Corresponding Source
|
||||
may be on a different server (operated by you or a third party)
|
||||
that supports equivalent copying facilities, provided you maintain
|
||||
clear directions next to the object code saying where to find the
|
||||
Corresponding Source. Regardless of what server hosts the
|
||||
Corresponding Source, you remain obligated to ensure that it is
|
||||
available for as long as needed to satisfy these requirements.
|
||||
|
||||
e) Convey the object code using peer-to-peer transmission, provided
|
||||
you inform other peers where the object code and Corresponding
|
||||
Source of the work are being offered to the general public at no
|
||||
charge under subsection 6d.
|
||||
|
||||
A separable portion of the object code, whose source code is excluded
|
||||
from the Corresponding Source as a System Library, need not be
|
||||
included in conveying the object code work.
|
||||
|
||||
A "User Product" is either (1) a "consumer product", which means any
|
||||
tangible personal property which is normally used for personal, family,
|
||||
or household purposes, or (2) anything designed or sold for incorporation
|
||||
into a dwelling. In determining whether a product is a consumer product,
|
||||
doubtful cases shall be resolved in favor of coverage. For a particular
|
||||
product received by a particular user, "normally used" refers to a
|
||||
typical or common use of that class of product, regardless of the status
|
||||
of the particular user or of the way in which the particular user
|
||||
actually uses, or expects or is expected to use, the product. A product
|
||||
is a consumer product regardless of whether the product has substantial
|
||||
commercial, industrial or non-consumer uses, unless such uses represent
|
||||
the only significant mode of use of the product.
|
||||
|
||||
"Installation Information" for a User Product means any methods,
|
||||
procedures, authorization keys, or other information required to install
|
||||
and execute modified versions of a covered work in that User Product from
|
||||
a modified version of its Corresponding Source. The information must
|
||||
suffice to ensure that the continued functioning of the modified object
|
||||
code is in no case prevented or interfered with solely because
|
||||
modification has been made.
|
||||
|
||||
If you convey an object code work under this section in, or with, or
|
||||
specifically for use in, a User Product, and the conveying occurs as
|
||||
part of a transaction in which the right of possession and use of the
|
||||
User Product is transferred to the recipient in perpetuity or for a
|
||||
fixed term (regardless of how the transaction is characterized), the
|
||||
Corresponding Source conveyed under this section must be accompanied
|
||||
by the Installation Information. But this requirement does not apply
|
||||
if neither you nor any third party retains the ability to install
|
||||
modified object code on the User Product (for example, the work has
|
||||
been installed in ROM).
|
||||
|
||||
The requirement to provide Installation Information does not include a
|
||||
requirement to continue to provide support service, warranty, or updates
|
||||
for a work that has been modified or installed by the recipient, or for
|
||||
the User Product in which it has been modified or installed. Access to a
|
||||
network may be denied when the modification itself materially and
|
||||
adversely affects the operation of the network or violates the rules and
|
||||
protocols for communication across the network.
|
||||
|
||||
Corresponding Source conveyed, and Installation Information provided,
|
||||
in accord with this section must be in a format that is publicly
|
||||
documented (and with an implementation available to the public in
|
||||
source code form), and must require no special password or key for
|
||||
unpacking, reading or copying.
|
||||
|
||||
7. Additional Terms.
|
||||
|
||||
"Additional permissions" are terms that supplement the terms of this
|
||||
License by making exceptions from one or more of its conditions.
|
||||
Additional permissions that are applicable to the entire Program shall
|
||||
be treated as though they were included in this License, to the extent
|
||||
that they are valid under applicable law. If additional permissions
|
||||
apply only to part of the Program, that part may be used separately
|
||||
under those permissions, but the entire Program remains governed by
|
||||
this License without regard to the additional permissions.
|
||||
|
||||
When you convey a copy of a covered work, you may at your option
|
||||
remove any additional permissions from that copy, or from any part of
|
||||
it. (Additional permissions may be written to require their own
|
||||
removal in certain cases when you modify the work.) You may place
|
||||
additional permissions on material, added by you to a covered work,
|
||||
for which you have or can give appropriate copyright permission.
|
||||
|
||||
Notwithstanding any other provision of this License, for material you
|
||||
add to a covered work, you may (if authorized by the copyright holders of
|
||||
that material) supplement the terms of this License with terms:
|
||||
|
||||
a) Disclaiming warranty or limiting liability differently from the
|
||||
terms of sections 15 and 16 of this License; or
|
||||
|
||||
b) Requiring preservation of specified reasonable legal notices or
|
||||
author attributions in that material or in the Appropriate Legal
|
||||
Notices displayed by works containing it; or
|
||||
|
||||
c) Prohibiting misrepresentation of the origin of that material, or
|
||||
requiring that modified versions of such material be marked in
|
||||
reasonable ways as different from the original version; or
|
||||
|
||||
d) Limiting the use for publicity purposes of names of licensors or
|
||||
authors of the material; or
|
||||
|
||||
e) Declining to grant rights under trademark law for use of some
|
||||
trade names, trademarks, or service marks; or
|
||||
|
||||
f) Requiring indemnification of licensors and authors of that
|
||||
material by anyone who conveys the material (or modified versions of
|
||||
it) with contractual assumptions of liability to the recipient, for
|
||||
any liability that these contractual assumptions directly impose on
|
||||
those licensors and authors.
|
||||
|
||||
All other non-permissive additional terms are considered "further
|
||||
restrictions" within the meaning of section 10. If the Program as you
|
||||
received it, or any part of it, contains a notice stating that it is
|
||||
governed by this License along with a term that is a further
|
||||
restriction, you may remove that term. If a license document contains
|
||||
a further restriction but permits relicensing or conveying under this
|
||||
License, you may add to a covered work material governed by the terms
|
||||
of that license document, provided that the further restriction does
|
||||
not survive such relicensing or conveying.
|
||||
|
||||
If you add terms to a covered work in accord with this section, you
|
||||
must place, in the relevant source files, a statement of the
|
||||
additional terms that apply to those files, or a notice indicating
|
||||
where to find the applicable terms.
|
||||
|
||||
Additional terms, permissive or non-permissive, may be stated in the
|
||||
form of a separately written license, or stated as exceptions;
|
||||
the above requirements apply either way.
|
||||
|
||||
8. Termination.
|
||||
|
||||
You may not propagate or modify a covered work except as expressly
|
||||
provided under this License. Any attempt otherwise to propagate or
|
||||
modify it is void, and will automatically terminate your rights under
|
||||
this License (including any patent licenses granted under the third
|
||||
paragraph of section 11).
|
||||
|
||||
However, if you cease all violation of this License, then your
|
||||
license from a particular copyright holder is reinstated (a)
|
||||
provisionally, unless and until the copyright holder explicitly and
|
||||
finally terminates your license, and (b) permanently, if the copyright
|
||||
holder fails to notify you of the violation by some reasonable means
|
||||
prior to 60 days after the cessation.
|
||||
|
||||
Moreover, your license from a particular copyright holder is
|
||||
reinstated permanently if the copyright holder notifies you of the
|
||||
violation by some reasonable means, this is the first time you have
|
||||
received notice of violation of this License (for any work) from that
|
||||
copyright holder, and you cure the violation prior to 30 days after
|
||||
your receipt of the notice.
|
||||
|
||||
Termination of your rights under this section does not terminate the
|
||||
licenses of parties who have received copies or rights from you under
|
||||
this License. If your rights have been terminated and not permanently
|
||||
reinstated, you do not qualify to receive new licenses for the same
|
||||
material under section 10.
|
||||
|
||||
9. Acceptance Not Required for Having Copies.
|
||||
|
||||
You are not required to accept this License in order to receive or
|
||||
run a copy of the Program. Ancillary propagation of a covered work
|
||||
occurring solely as a consequence of using peer-to-peer transmission
|
||||
to receive a copy likewise does not require acceptance. However,
|
||||
nothing other than this License grants you permission to propagate or
|
||||
modify any covered work. These actions infringe copyright if you do
|
||||
not accept this License. Therefore, by modifying or propagating a
|
||||
covered work, you indicate your acceptance of this License to do so.
|
||||
|
||||
10. Automatic Licensing of Downstream Recipients.
|
||||
|
||||
Each time you convey a covered work, the recipient automatically
|
||||
receives a license from the original licensors, to run, modify and
|
||||
propagate that work, subject to this License. You are not responsible
|
||||
for enforcing compliance by third parties with this License.
|
||||
|
||||
An "entity transaction" is a transaction transferring control of an
|
||||
organization, or substantially all assets of one, or subdividing an
|
||||
organization, or merging organizations. If propagation of a covered
|
||||
work results from an entity transaction, each party to that
|
||||
transaction who receives a copy of the work also receives whatever
|
||||
licenses to the work the party's predecessor in interest had or could
|
||||
give under the previous paragraph, plus a right to possession of the
|
||||
Corresponding Source of the work from the predecessor in interest, if
|
||||
the predecessor has it or can get it with reasonable efforts.
|
||||
|
||||
You may not impose any further restrictions on the exercise of the
|
||||
rights granted or affirmed under this License. For example, you may
|
||||
not impose a license fee, royalty, or other charge for exercise of
|
||||
rights granted under this License, and you may not initiate litigation
|
||||
(including a cross-claim or counterclaim in a lawsuit) alleging that
|
||||
any patent claim is infringed by making, using, selling, offering for
|
||||
sale, or importing the Program or any portion of it.
|
||||
|
||||
11. Patents.
|
||||
|
||||
A "contributor" is a copyright holder who authorizes use under this
|
||||
License of the Program or a work on which the Program is based. The
|
||||
work thus licensed is called the contributor's "contributor version".
|
||||
|
||||
A contributor's "essential patent claims" are all patent claims
|
||||
owned or controlled by the contributor, whether already acquired or
|
||||
hereafter acquired, that would be infringed by some manner, permitted
|
||||
by this License, of making, using, or selling its contributor version,
|
||||
but do not include claims that would be infringed only as a
|
||||
consequence of further modification of the contributor version. For
|
||||
purposes of this definition, "control" includes the right to grant
|
||||
patent sublicenses in a manner consistent with the requirements of
|
||||
this License.
|
||||
|
||||
Each contributor grants you a non-exclusive, worldwide, royalty-free
|
||||
patent license under the contributor's essential patent claims, to
|
||||
make, use, sell, offer for sale, import and otherwise run, modify and
|
||||
propagate the contents of its contributor version.
|
||||
|
||||
In the following three paragraphs, a "patent license" is any express
|
||||
agreement or commitment, however denominated, not to enforce a patent
|
||||
(such as an express permission to practice a patent or covenant not to
|
||||
sue for patent infringement). To "grant" such a patent license to a
|
||||
party means to make such an agreement or commitment not to enforce a
|
||||
patent against the party.
|
||||
|
||||
If you convey a covered work, knowingly relying on a patent license,
|
||||
and the Corresponding Source of the work is not available for anyone
|
||||
to copy, free of charge and under the terms of this License, through a
|
||||
publicly available network server or other readily accessible means,
|
||||
then you must either (1) cause the Corresponding Source to be so
|
||||
available, or (2) arrange to deprive yourself of the benefit of the
|
||||
patent license for this particular work, or (3) arrange, in a manner
|
||||
consistent with the requirements of this License, to extend the patent
|
||||
license to downstream recipients. "Knowingly relying" means you have
|
||||
actual knowledge that, but for the patent license, your conveying the
|
||||
covered work in a country, or your recipient's use of the covered work
|
||||
in a country, would infringe one or more identifiable patents in that
|
||||
country that you have reason to believe are valid.
|
||||
|
||||
If, pursuant to or in connection with a single transaction or
|
||||
arrangement, you convey, or propagate by procuring conveyance of, a
|
||||
covered work, and grant a patent license to some of the parties
|
||||
receiving the covered work authorizing them to use, propagate, modify
|
||||
or convey a specific copy of the covered work, then the patent license
|
||||
you grant is automatically extended to all recipients of the covered
|
||||
work and works based on it.
|
||||
|
||||
A patent license is "discriminatory" if it does not include within
|
||||
the scope of its coverage, prohibits the exercise of, or is
|
||||
conditioned on the non-exercise of one or more of the rights that are
|
||||
specifically granted under this License. You may not convey a covered
|
||||
work if you are a party to an arrangement with a third party that is
|
||||
in the business of distributing software, under which you make payment
|
||||
to the third party based on the extent of your activity of conveying
|
||||
the work, and under which the third party grants, to any of the
|
||||
parties who would receive the covered work from you, a discriminatory
|
||||
patent license (a) in connection with copies of the covered work
|
||||
conveyed by you (or copies made from those copies), or (b) primarily
|
||||
for and in connection with specific products or compilations that
|
||||
contain the covered work, unless you entered into that arrangement,
|
||||
or that patent license was granted, prior to 28 March 2007.
|
||||
|
||||
Nothing in this License shall be construed as excluding or limiting
|
||||
any implied license or other defenses to infringement that may
|
||||
otherwise be available to you under applicable patent law.
|
||||
|
||||
12. No Surrender of Others' Freedom.
|
||||
|
||||
If conditions are imposed on you (whether by court order, agreement or
|
||||
otherwise) that contradict the conditions of this License, they do not
|
||||
excuse you from the conditions of this License. If you cannot convey a
|
||||
covered work so as to satisfy simultaneously your obligations under this
|
||||
License and any other pertinent obligations, then as a consequence you may
|
||||
not convey it at all. For example, if you agree to terms that obligate you
|
||||
to collect a royalty for further conveying from those to whom you convey
|
||||
the Program, the only way you could satisfy both those terms and this
|
||||
License would be to refrain entirely from conveying the Program.
|
||||
|
||||
13. Use with the GNU Affero General Public License.
|
||||
|
||||
Notwithstanding any other provision of this License, you have
|
||||
permission to link or combine any covered work with a work licensed
|
||||
under version 3 of the GNU Affero General Public License into a single
|
||||
combined work, and to convey the resulting work. The terms of this
|
||||
License will continue to apply to the part which is the covered work,
|
||||
but the special requirements of the GNU Affero General Public License,
|
||||
section 13, concerning interaction through a network will apply to the
|
||||
combination as such.
|
||||
|
||||
14. Revised Versions of this License.
|
||||
|
||||
The Free Software Foundation may publish revised and/or new versions of
|
||||
the GNU General Public License from time to time. Such new versions will
|
||||
be similar in spirit to the present version, but may differ in detail to
|
||||
address new problems or concerns.
|
||||
|
||||
Each version is given a distinguishing version number. If the
|
||||
Program specifies that a certain numbered version of the GNU General
|
||||
Public License "or any later version" applies to it, you have the
|
||||
option of following the terms and conditions either of that numbered
|
||||
version or of any later version published by the Free Software
|
||||
Foundation. If the Program does not specify a version number of the
|
||||
GNU General Public License, you may choose any version ever published
|
||||
by the Free Software Foundation.
|
||||
|
||||
If the Program specifies that a proxy can decide which future
|
||||
versions of the GNU General Public License can be used, that proxy's
|
||||
public statement of acceptance of a version permanently authorizes you
|
||||
to choose that version for the Program.
|
||||
|
||||
Later license versions may give you additional or different
|
||||
permissions. However, no additional obligations are imposed on any
|
||||
author or copyright holder as a result of your choosing to follow a
|
||||
later version.
|
||||
|
||||
15. Disclaimer of Warranty.
|
||||
|
||||
THERE IS NO WARRANTY FOR THE PROGRAM, TO THE EXTENT PERMITTED BY
|
||||
APPLICABLE LAW. EXCEPT WHEN OTHERWISE STATED IN WRITING THE COPYRIGHT
|
||||
HOLDERS AND/OR OTHER PARTIES PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY
|
||||
OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING, BUT NOT LIMITED TO,
|
||||
THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
|
||||
PURPOSE. THE ENTIRE RISK AS TO THE QUALITY AND PERFORMANCE OF THE PROGRAM
|
||||
IS WITH YOU. SHOULD THE PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF
|
||||
ALL NECESSARY SERVICING, REPAIR OR CORRECTION.
|
||||
|
||||
16. Limitation of Liability.
|
||||
|
||||
IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING
|
||||
WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MODIFIES AND/OR CONVEYS
|
||||
THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES, INCLUDING ANY
|
||||
GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING OUT OF THE
|
||||
USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED TO LOSS OF
|
||||
DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY YOU OR THIRD
|
||||
PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER PROGRAMS),
|
||||
EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE POSSIBILITY OF
|
||||
SUCH DAMAGES.
|
||||
|
||||
17. Interpretation of Sections 15 and 16.
|
||||
|
||||
If the disclaimer of warranty and limitation of liability provided
|
||||
above cannot be given local legal effect according to their terms,
|
||||
reviewing courts shall apply local law that most closely approximates
|
||||
an absolute waiver of all civil liability in connection with the
|
||||
Program, unless a warranty or assumption of liability accompanies a
|
||||
copy of the Program in return for a fee.
|
||||
|
||||
END OF TERMS AND CONDITIONS
|
||||
|
||||
How to Apply These Terms to Your New Programs
|
||||
|
||||
If you develop a new program, and you want it to be of the greatest
|
||||
possible use to the public, the best way to achieve this is to make it
|
||||
free software which everyone can redistribute and change under these terms.
|
||||
|
||||
To do so, attach the following notices to the program. It is safest
|
||||
to attach them to the start of each source file to most effectively
|
||||
state the exclusion of warranty; and each file should have at least
|
||||
the "copyright" line and a pointer to where the full notice is found.
|
||||
|
||||
{one line to give the program's name and a brief idea of what it does.}
|
||||
Copyright (C) {year} {name of author}
|
||||
|
||||
This program is free software: you can redistribute it and/or modify
|
||||
it under the terms of the GNU General Public License as published by
|
||||
the Free Software Foundation, either version 3 of the License, or
|
||||
(at your option) any later version.
|
||||
|
||||
This program is distributed in the hope that it will be useful,
|
||||
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
GNU General Public License for more details.
|
||||
|
||||
You should have received a copy of the GNU General Public License
|
||||
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||
|
||||
Also add information on how to contact you by electronic and paper mail.
|
||||
|
||||
If the program does terminal interaction, make it output a short
|
||||
notice like this when it starts in an interactive mode:
|
||||
|
||||
{project} Copyright (C) {year} {fullname}
|
||||
This program comes with ABSOLUTELY NO WARRANTY; for details type `show w'.
|
||||
This is free software, and you are welcome to redistribute it
|
||||
under certain conditions; type `show c' for details.
|
||||
|
||||
The hypothetical commands `show w' and `show c' should show the appropriate
|
||||
parts of the General Public License. Of course, your program's commands
|
||||
might be different; for a GUI interface, you would use an "about box".
|
||||
|
||||
You should also get your employer (if you work as a programmer) or school,
|
||||
if any, to sign a "copyright disclaimer" for the program, if necessary.
|
||||
For more information on this, and how to apply and follow the GNU GPL, see
|
||||
<http://www.gnu.org/licenses/>.
|
||||
|
||||
The GNU General Public License does not permit incorporating your program
|
||||
into proprietary programs. If your program is a subroutine library, you
|
||||
may consider it more useful to permit linking proprietary applications with
|
||||
the library. If this is what you want to do, use the GNU Lesser General
|
||||
Public License instead of this License. But first, please read
|
||||
<http://www.gnu.org/philosophy/why-not-lgpl.html>.
|
84
README.md
Normal file
84
README.md
Normal file
@ -0,0 +1,84 @@
|
||||
# What is mydumper? Why?
|
||||
|
||||
* Parallelism (hence, speed) and performance (avoids expensive character set conversion routines, efficient code overall)
|
||||
* Easier to manage output (separate files for tables, dump metadata, etc, easy to view/parse data)
|
||||
* Consistency - maintains snapshot across all threads, provides accurate master and slave log positions, etc
|
||||
* Manageability - supports PCRE for specifying database and tables inclusions and exclusions
|
||||
|
||||
## How to install mydumper/myloader?
|
||||
|
||||
First get the correct url from the [releases section](https://github.com/maxbube/mydumper/releases) then:
|
||||
|
||||
### RedHat / Centos
|
||||
|
||||
```bash
|
||||
yum install https://github.com/maxbube/mydumper/releases/download/v0.9.5/mydumper-0.9.5-1.el7.x86_64.rpm
|
||||
```
|
||||
|
||||
### Ubuntu / Debian
|
||||
|
||||
```bash
|
||||
wget https://github.com/maxbube/mydumper/releases/download/v0.9.5/mydumper_0.9.5-1.xenial_amd64.deb
|
||||
dpkg -i mydumper_0.9.5-1.xenial_amd64.deb
|
||||
```
|
||||
|
||||
## How to build it?
|
||||
|
||||
Run:
|
||||
|
||||
```bash
|
||||
cmake .
|
||||
make
|
||||
```
|
||||
|
||||
One needs to install development versions of required libaries (MySQL, GLib, ZLib, PCRE):
|
||||
NOTE: you must use the correspondent mysql devel package.
|
||||
|
||||
* Ubuntu or Debian: apt-get install libglib2.0-dev libmysqlclient15-dev zlib1g-dev libpcre3-dev libssl-dev
|
||||
* Fedora, RedHat and CentOS: yum install glib2-devel mysql-devel zlib-devel pcre-devel openssl-devel
|
||||
* openSUSE: zypper install glib2-devel libmysqlclient-devel pcre-devel zlib-devel
|
||||
* MacOSX: port install glib2 mysql5 pcre pkgconfig cmake
|
||||
(You may want to run 'port select mysql mysql5' afterwards)
|
||||
|
||||
One has to make sure, that pkg-config, mysql_config, pcre-config are all in $PATH
|
||||
|
||||
Binlog dump is disabled by default to compile with it you need to add -DWITH_BINLOG=ON to cmake options
|
||||
|
||||
To build against mysql libs < 5.7 you need to disable SSL adding -DWITH_SSL=OFF
|
||||
|
||||
## How does consistent snapshot work?
|
||||
|
||||
This is all done following best MySQL practices and traditions:
|
||||
|
||||
* As a precaution, slow running queries on the server either abort the dump, or get killed
|
||||
* Global write lock is acquired ("FLUSH TABLES WITH READ LOCK")
|
||||
* Various metadata is read ("SHOW SLAVE STATUS","SHOW MASTER STATUS")
|
||||
* Other threads connect and establish snapshots ("START TRANSACTION WITH CONSISTENT SNAPSHOT")
|
||||
** On pre-4.1.8 it creates dummy InnoDB table, and reads from it.
|
||||
* Once all worker threads announce the snapshot establishment, master executes "UNLOCK TABLES" and starts queueing jobs.
|
||||
|
||||
This for now does not provide consistent snapshots for non-transactional engines - support for that is expected in 0.2 :)
|
||||
|
||||
## How to exclude (or include) databases?
|
||||
|
||||
Once can use --regex functionality, for example not to dump mysql and test databases:
|
||||
|
||||
```bash
|
||||
mydumper --regex '^(?!(mysql\.|test\.))'
|
||||
```
|
||||
|
||||
To dump only mysql and test databases:
|
||||
|
||||
```bash
|
||||
mydumper --regex '^(mysql\.|test\.)'
|
||||
```
|
||||
|
||||
To not dump all databases starting with test:
|
||||
|
||||
```bash
|
||||
mydumper --regex '^(?!(test))'
|
||||
```
|
||||
|
||||
Of course, regex functionality can be used to describe pretty much any list of tables.
|
||||
|
||||
|
284
binlog.c
Normal file
284
binlog.c
Normal file
@ -0,0 +1,284 @@
|
||||
/*
|
||||
This program is free software: you can redistribute it and/or modify
|
||||
it under the terms of the GNU General Public License as published by
|
||||
the Free Software Foundation, either version 3 of the License, or
|
||||
(at your option) any later version.
|
||||
|
||||
This program is distributed in the hope that it will be useful,
|
||||
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
GNU General Public License for more details.
|
||||
|
||||
You should have received a copy of the GNU General Public License
|
||||
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||
|
||||
Authors: Domas Mituzas, Facebook ( domas at fb dot com )
|
||||
Mark Leith, Oracle Corporation (mark dot leith at oracle dot com)
|
||||
Andrew Hutchings, SkySQL (andrew at skysql dot com)
|
||||
|
||||
*/
|
||||
|
||||
#include <glib.h>
|
||||
#include <glib/gstdio.h>
|
||||
#include <my_global.h>
|
||||
#include <mysql.h>
|
||||
#include <my_sys.h>
|
||||
#include <mysqld_error.h>
|
||||
#include <sql_common.h>
|
||||
#include <string.h>
|
||||
#include <zlib.h>
|
||||
#include "mydumper.h"
|
||||
#include "binlog.h"
|
||||
|
||||
#define BINLOG_MAGIC "\xfe\x62\x69\x6e"
|
||||
|
||||
#define EVENT_HEADER_LENGTH 19
|
||||
#define EVENT_ROTATE_FIXED_LENGTH 8
|
||||
|
||||
enum event_postions {
|
||||
EVENT_TIMESTAMP_POSITION= 0,
|
||||
EVENT_TYPE_POSITION= 4,
|
||||
EVENT_SERVERID_POSITION= 5,
|
||||
EVENT_LENGTH_POSITION= 9,
|
||||
EVENT_NEXT_POSITION= 13,
|
||||
EVENT_FLAGS_POSITION= 17,
|
||||
EVENT_EXTRA_FLAGS_POSITION= 19 // currently unused in v4 binlogs, but a good marker for end of header
|
||||
};
|
||||
|
||||
enum event_type {
|
||||
ROTATE_EVENT= 4,
|
||||
FORMAT_DESCRIPTION_EVENT= 15,
|
||||
EVENT_TOO_SHORT= 254 // arbitrary high number, in 5.1 the max event type number is 27 so this should be fine for a while
|
||||
};
|
||||
|
||||
extern int compress_output;
|
||||
extern gboolean daemon_mode;
|
||||
extern gboolean shutdown_triggered;
|
||||
|
||||
FILE *new_binlog_file(char *binlog_file, const char *binlog_dir);
|
||||
void close_binlog_file(FILE *outfile);
|
||||
char *rotate_file_name(const char *buf);
|
||||
|
||||
void get_binlogs(MYSQL *conn, struct configuration *conf) {
|
||||
// TODO: find logs we already have, use start position based on position of last log.
|
||||
MYSQL_RES *result;
|
||||
MYSQL_ROW row;
|
||||
char* last_filename = NULL;
|
||||
guint64 last_position;
|
||||
|
||||
// Only snapshot dump the binlogs once in daemon mode
|
||||
static gboolean got_binlogs= FALSE;
|
||||
if (got_binlogs)
|
||||
return;
|
||||
else
|
||||
got_binlogs= TRUE;
|
||||
|
||||
if (mysql_query(conn, "SHOW MASTER STATUS")) {
|
||||
g_critical("Error: Could not execute query: %s", mysql_error(conn));
|
||||
return;
|
||||
}
|
||||
|
||||
result = mysql_store_result(conn);
|
||||
if ((row = mysql_fetch_row(result))) {
|
||||
last_filename= g_strdup(row[0]);
|
||||
last_position= strtoll(row[1], NULL, 10);
|
||||
} else {
|
||||
g_critical("Error: Could not obtain binary log stop position");
|
||||
if (last_filename != NULL)
|
||||
g_free(last_filename);
|
||||
return;
|
||||
}
|
||||
mysql_free_result(result);
|
||||
|
||||
if (mysql_query(conn, "SHOW BINARY LOGS")) {
|
||||
g_critical("Error: Could not execute query: %s", mysql_error(conn));
|
||||
if (last_filename != NULL)
|
||||
g_free(last_filename);
|
||||
return;
|
||||
}
|
||||
|
||||
|
||||
result = mysql_store_result(conn);
|
||||
while ((row = mysql_fetch_row(result))) {
|
||||
struct job *j = g_new0(struct job,1);
|
||||
struct binlog_job *bj = g_new0(struct binlog_job,1);
|
||||
j->job_data=(void*) bj;
|
||||
bj->filename=g_strdup(row[0]);
|
||||
bj->start_position=4;
|
||||
bj->stop_position= (!strcasecmp(row[0], last_filename)) ? last_position : 0;
|
||||
j->conf=conf;
|
||||
j->type=JOB_BINLOG;
|
||||
g_async_queue_push(conf->queue,j);
|
||||
}
|
||||
mysql_free_result(result);
|
||||
if (last_filename != NULL)
|
||||
g_free(last_filename);
|
||||
}
|
||||
|
||||
void get_binlog_file(MYSQL *conn, char *binlog_file, const char *binlog_directory, guint64 start_position, guint64 stop_position, gboolean continuous) {
|
||||
// set serverID = max serverID - threadID to try an eliminate conflicts,
|
||||
// 0 is bad because mysqld will disconnect at the end of the last log
|
||||
// dupes aren't too bad since it is up to the client to check for them
|
||||
uchar buf[128];
|
||||
// We need to read the raw network packets
|
||||
NET* net;
|
||||
net= &conn->net;
|
||||
unsigned long len;
|
||||
FILE* outfile;
|
||||
guint32 event_type;
|
||||
gboolean read_error= FALSE;
|
||||
gboolean read_end= FALSE;
|
||||
gboolean rotated= FALSE;
|
||||
guint32 server_id= G_MAXUINT32 - mysql_thread_id(conn);
|
||||
guint64 pos_counter= 0;
|
||||
|
||||
int4store(buf, (guint32)start_position);
|
||||
// Binlog flags (2 byte int)
|
||||
int2store(buf + 4, 0);
|
||||
// ServerID
|
||||
int4store(buf + 6, server_id);
|
||||
memcpy(buf + 10, binlog_file, strlen(binlog_file));
|
||||
#if MYSQL_VERSION_ID < 50100
|
||||
if (simple_command(conn, COM_BINLOG_DUMP, (const char *)buf,
|
||||
#else
|
||||
if (simple_command(conn, COM_BINLOG_DUMP, buf,
|
||||
#endif
|
||||
strlen(binlog_file) + 10, 1)) {
|
||||
g_critical("Error: binlog: Critical error whilst requesting binary log");
|
||||
}
|
||||
|
||||
while(1) {
|
||||
outfile= new_binlog_file(binlog_file, binlog_directory);
|
||||
if (outfile == NULL) {
|
||||
g_critical("Error: binlog: Could not create binlog file '%s', %d", binlog_file, errno);
|
||||
return;
|
||||
}
|
||||
|
||||
write_binlog(outfile, BINLOG_MAGIC, 4);
|
||||
while(1) {
|
||||
len = 0;
|
||||
if (net->vio != 0) len=my_net_read(net);
|
||||
if ((len == 0) || (len == ~(unsigned long) 0)) {
|
||||
// Net timeout (set to 1 second)
|
||||
if (mysql_errno(conn) == ER_NET_READ_INTERRUPTED) {
|
||||
if (shutdown_triggered) {
|
||||
close_binlog_file(outfile);
|
||||
return;
|
||||
} else {
|
||||
continue;
|
||||
}
|
||||
// A real error
|
||||
} else {
|
||||
g_critical("Error: binlog: Network packet read error getting binlog file: %s", binlog_file);
|
||||
close_binlog_file(outfile);
|
||||
return;
|
||||
}
|
||||
}
|
||||
if (len < 8 && net->read_pos[0]) {
|
||||
// end of data
|
||||
break;
|
||||
}
|
||||
pos_counter += len;
|
||||
event_type= get_event((const char*)net->read_pos + 1, len -1);
|
||||
switch (event_type) {
|
||||
case EVENT_TOO_SHORT:
|
||||
g_critical("Error: binlog: Event too short in binlog file: %s", binlog_file);
|
||||
read_error= TRUE;
|
||||
break;
|
||||
case ROTATE_EVENT:
|
||||
if (rotated) {
|
||||
read_end= TRUE;
|
||||
} else {
|
||||
len= 1;
|
||||
rotated= TRUE;
|
||||
}
|
||||
break;
|
||||
default:
|
||||
// if we get this far this is a normal event to record
|
||||
break;
|
||||
}
|
||||
if (read_error) break;
|
||||
write_binlog(outfile, (const char*)net->read_pos + 1, len - 1);
|
||||
if (read_end) {
|
||||
if (!continuous) {
|
||||
break;
|
||||
} else {
|
||||
g_free(binlog_file);
|
||||
binlog_file= rotate_file_name((const char*)net->read_pos + 1);
|
||||
break;
|
||||
}
|
||||
}
|
||||
// stop if we are at requested end of last log
|
||||
if ((stop_position > 0) && (pos_counter >= stop_position)) break;
|
||||
}
|
||||
close_binlog_file(outfile);
|
||||
if ((!continuous) || (!read_end)) break;
|
||||
|
||||
if (continuous && read_end) {
|
||||
read_end= FALSE;
|
||||
rotated= FALSE;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
char *rotate_file_name(const char *buf) {
|
||||
guint32 event_length= 0;
|
||||
|
||||
// event length is 4 bytes at position 9
|
||||
event_length= uint4korr(&buf[EVENT_LENGTH_POSITION]);
|
||||
// event length includes the header, plus a rotate event has a fixed 8byte part we don't need
|
||||
event_length= event_length - EVENT_HEADER_LENGTH - EVENT_ROTATE_FIXED_LENGTH;
|
||||
|
||||
return g_strndup(&buf[EVENT_HEADER_LENGTH + EVENT_ROTATE_FIXED_LENGTH], event_length);
|
||||
}
|
||||
|
||||
FILE *new_binlog_file(char *binlog_file, const char *binlog_dir) {
|
||||
FILE *outfile;
|
||||
char* filename;
|
||||
|
||||
if (!compress_output) {
|
||||
filename= g_strdup_printf("%s/%s", binlog_dir, binlog_file);
|
||||
outfile= g_fopen(filename, "w");
|
||||
} else {
|
||||
filename= g_strdup_printf("%s/%s.gz", binlog_dir, binlog_file);
|
||||
outfile= (void*) gzopen(filename, "w");
|
||||
}
|
||||
g_free(filename);
|
||||
|
||||
return outfile;
|
||||
}
|
||||
|
||||
void close_binlog_file(FILE *outfile) {
|
||||
if (!compress_output)
|
||||
fclose(outfile);
|
||||
else
|
||||
gzclose((gzFile) outfile);
|
||||
}
|
||||
|
||||
unsigned int get_event(const char *buf, unsigned int len) {
|
||||
if (len < EVENT_TYPE_POSITION)
|
||||
return EVENT_TOO_SHORT;
|
||||
return buf[EVENT_TYPE_POSITION];
|
||||
|
||||
// TODO: Would be good if we can check for valid event type, unfortunately this check can change from version to version
|
||||
}
|
||||
|
||||
void write_binlog(FILE* file, const char* data, guint64 len) {
|
||||
int err;
|
||||
|
||||
if (len > 0) {
|
||||
int write_result;
|
||||
|
||||
if (!compress_output)
|
||||
write_result= write(fileno(file), data, len);
|
||||
else
|
||||
write_result= gzwrite((gzFile)file, data, len);
|
||||
|
||||
if (write_result <= 0) {
|
||||
if (!compress_output)
|
||||
g_critical("Error: binlog: Error writing binary log: %s", strerror(errno));
|
||||
else
|
||||
g_critical("Error: binlog: Error writing compressed binary log: %s", gzerror((gzFile)file, &err));
|
||||
}
|
||||
}
|
||||
}
|
30
binlog.h
Normal file
30
binlog.h
Normal file
@ -0,0 +1,30 @@
|
||||
/*
|
||||
This program is free software: you can redistribute it and/or modify
|
||||
it under the terms of the GNU General Public License as published by
|
||||
the Free Software Foundation, either version 3 of the License, or
|
||||
(at your option) any later version.
|
||||
|
||||
This program is distributed in the hope that it will be useful,
|
||||
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
GNU General Public License for more details.
|
||||
|
||||
You should have received a copy of the GNU General Public License
|
||||
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||
|
||||
Authors: Domas Mituzas, Facebook ( domas at fb dot com )
|
||||
Mark Leith, Oracle Corporation (mark dot leith at oracle dot com)
|
||||
Andrew Hutchings, SkySQL (andrew at skysql dot com)
|
||||
|
||||
*/
|
||||
|
||||
#ifndef _binlog_h
|
||||
#define _binlog_h
|
||||
#include "mydumper.h"
|
||||
|
||||
void get_binlogs(MYSQL *conn, struct configuration *conf);
|
||||
void get_binlog_file(MYSQL *conn, char *binlog_file, const char *binlog_directory, guint64 start_position, guint64 stop_position, gboolean continuous);
|
||||
unsigned int get_event(const char *buf, unsigned int len);
|
||||
void write_binlog(FILE* file, const char* data, guint64 len);
|
||||
|
||||
#endif
|
214
cmake/modules/CppcheckTargets.cmake
Normal file
214
cmake/modules/CppcheckTargets.cmake
Normal file
@ -0,0 +1,214 @@
|
||||
# - Run cppcheck on c++ source files as a custom target and a test
|
||||
#
|
||||
# include(CppcheckTargets)
|
||||
# add_cppcheck(<target-name> [UNUSED_FUNCTIONS] [STYLE] [POSSIBLE_ERROR] [FAIL_ON_WARNINGS]) -
|
||||
# Create a target to check a target's sources with cppcheck and the indicated options
|
||||
# add_cppcheck_sources(<target-name> [UNUSED_FUNCTIONS] [STYLE] [POSSIBLE_ERROR] [FAIL_ON_WARNINGS]) -
|
||||
# Create a target to check standalone sources with cppcheck and the indicated options
|
||||
#
|
||||
# Requires these CMake modules:
|
||||
# Findcppcheck
|
||||
#
|
||||
# Requires CMake 2.6 or newer (uses the 'function' command)
|
||||
#
|
||||
# Original Author:
|
||||
# 2009-2010 Ryan Pavlik <rpavlik@iastate.edu> <abiryan@ryand.net>
|
||||
# http://academic.cleardefinition.com
|
||||
# Iowa State University HCI Graduate Program/VRAC
|
||||
#
|
||||
# Copyright Iowa State University 2009-2010.
|
||||
# Distributed under the Boost Software License, Version 1.0.
|
||||
# (See accompanying file LICENSE_1_0.txt or copy at
|
||||
# http://www.boost.org/LICENSE_1_0.txt)
|
||||
|
||||
if(__add_cppcheck)
|
||||
return()
|
||||
endif()
|
||||
set(__add_cppcheck YES)
|
||||
|
||||
if(NOT CPPCHECK_FOUND)
|
||||
find_package(cppcheck QUIET)
|
||||
endif()
|
||||
|
||||
if(CPPCHECK_FOUND)
|
||||
if(NOT TARGET all_cppcheck)
|
||||
add_custom_target(all_cppcheck)
|
||||
set_target_properties(all_cppcheck PROPERTIES EXCLUDE_FROM_ALL TRUE)
|
||||
endif()
|
||||
endif()
|
||||
|
||||
function(add_cppcheck_sources _targetname)
|
||||
if(CPPCHECK_FOUND)
|
||||
set(_cppcheck_args)
|
||||
set(_input ${ARGN})
|
||||
list(FIND _input UNUSED_FUNCTIONS _unused_func)
|
||||
if("${_unused_func}" GREATER "-1")
|
||||
list(APPEND _cppcheck_args ${CPPCHECK_UNUSEDFUNC_ARG})
|
||||
list(REMOVE_AT _input ${_unused_func})
|
||||
endif()
|
||||
|
||||
list(FIND _input STYLE _style)
|
||||
if("${_style}" GREATER "-1")
|
||||
list(APPEND _cppcheck_args ${CPPCHECK_STYLE_ARG})
|
||||
list(REMOVE_AT _input ${_style})
|
||||
endif()
|
||||
|
||||
list(FIND _input POSSIBLE_ERROR _poss_err)
|
||||
if("${_poss_err}" GREATER "-1")
|
||||
list(APPEND _cppcheck_args ${CPPCHECK_POSSIBLEERROR_ARG})
|
||||
list(REMOVE_AT _input ${_poss_err})
|
||||
endif()
|
||||
|
||||
list(FIND _input FAIL_ON_WARNINGS _fail_on_warn)
|
||||
if("${_fail_on_warn}" GREATER "-1")
|
||||
list(APPEND
|
||||
CPPCHECK_FAIL_REGULAR_EXPRESSION
|
||||
${CPPCHECK_WARN_REGULAR_EXPRESSION})
|
||||
list(REMOVE_AT _input ${_fail_on_warn})
|
||||
endif()
|
||||
|
||||
set(_files)
|
||||
foreach(_source ${_input})
|
||||
get_source_file_property(_cppcheck_loc "${_source}" LOCATION)
|
||||
if(_cppcheck_loc)
|
||||
# This file has a source file property, carry on.
|
||||
get_source_file_property(_cppcheck_lang "${_source}" LANGUAGE)
|
||||
if("${_cppcheck_lang}" MATCHES "C")
|
||||
list(APPEND _files "${_cppcheck_loc}")
|
||||
endif()
|
||||
else()
|
||||
# This file doesn't have source file properties - figure it out.
|
||||
get_filename_component(_cppcheck_loc "${_source}" ABSOLUTE)
|
||||
if(EXISTS "${_cppcheck_loc}")
|
||||
list(APPEND _files "${_cppcheck_loc}")
|
||||
else()
|
||||
message(FATAL_ERROR
|
||||
"Adding CPPCHECK for file target ${_targetname}: "
|
||||
"File ${_source} does not exist or needs a corrected path location "
|
||||
"since we think its absolute path is ${_cppcheck_loc}")
|
||||
endif()
|
||||
endif()
|
||||
endforeach()
|
||||
|
||||
if("1.${CMAKE_VERSION}" VERSION_LESS "1.2.8.0")
|
||||
# Older than CMake 2.8.0
|
||||
add_test(${_targetname}_cppcheck_test
|
||||
"${CPPCHECK_EXECUTABLE}"
|
||||
${CPPCHECK_TEMPLATE_ARG}
|
||||
${_cppcheck_args}
|
||||
${_files})
|
||||
else()
|
||||
# CMake 2.8.0 and newer
|
||||
add_test(NAME
|
||||
${_targetname}_cppcheck_test
|
||||
COMMAND
|
||||
"${CPPCHECK_EXECUTABLE}"
|
||||
${CPPCHECK_TEMPLATE_ARG}
|
||||
${_cppcheck_args}
|
||||
${_files})
|
||||
endif()
|
||||
|
||||
set_tests_properties(${_targetname}_cppcheck_test
|
||||
PROPERTIES
|
||||
FAIL_REGULAR_EXPRESSION
|
||||
"${CPPCHECK_FAIL_REGULAR_EXPRESSION}")
|
||||
|
||||
add_custom_command(TARGET
|
||||
all_cppcheck
|
||||
PRE_BUILD
|
||||
COMMAND
|
||||
${CPPCHECK_EXECUTABLE}
|
||||
${CPPCHECK_QUIET_ARG}
|
||||
${CPPCHECK_TEMPLATE_ARG}
|
||||
${_cppcheck_args}
|
||||
${_files}
|
||||
WORKING_DIRECTORY
|
||||
"${CMAKE_CURRENT_SOURCE_DIR}"
|
||||
COMMENT
|
||||
"${_targetname}_cppcheck: Running cppcheck on target ${_targetname}..."
|
||||
VERBATIM)
|
||||
endif()
|
||||
endfunction()
|
||||
|
||||
function(add_cppcheck _name)
|
||||
if(NOT TARGET ${_name})
|
||||
message(FATAL_ERROR
|
||||
"add_cppcheck given a target name that does not exist: '${_name}' !")
|
||||
endif()
|
||||
if(CPPCHECK_FOUND)
|
||||
set(_cppcheck_args)
|
||||
|
||||
list(FIND ARGN UNUSED_FUNCTIONS _unused_func)
|
||||
if("${_unused_func}" GREATER "-1")
|
||||
list(APPEND _cppcheck_args ${CPPCHECK_UNUSEDFUNC_ARG})
|
||||
endif()
|
||||
|
||||
list(FIND ARGN STYLE _style)
|
||||
if("${_style}" GREATER "-1")
|
||||
list(APPEND _cppcheck_args ${CPPCHECK_STYLE_ARG})
|
||||
endif()
|
||||
|
||||
list(FIND ARGN POSSIBLE_ERROR _poss_err)
|
||||
if("${_poss_err}" GREATER "-1")
|
||||
list(APPEND _cppcheck_args ${CPPCHECK_POSSIBLEERROR_ARG})
|
||||
endif()
|
||||
|
||||
list(FIND _input FAIL_ON_WARNINGS _fail_on_warn)
|
||||
if("${_fail_on_warn}" GREATER "-1")
|
||||
list(APPEND
|
||||
CPPCHECK_FAIL_REGULAR_EXPRESSION
|
||||
${CPPCHECK_WARN_REGULAR_EXPRESSION})
|
||||
list(REMOVE_AT _input ${_unused_func})
|
||||
endif()
|
||||
|
||||
get_target_property(_cppcheck_sources "${_name}" SOURCES)
|
||||
set(_files)
|
||||
foreach(_source ${_cppcheck_sources})
|
||||
get_source_file_property(_cppcheck_lang "${_source}" LANGUAGE)
|
||||
get_source_file_property(_cppcheck_loc "${_source}" LOCATION)
|
||||
if("${_cppcheck_lang}" MATCHES "C")
|
||||
list(APPEND _files "${_cppcheck_loc}")
|
||||
endif()
|
||||
endforeach()
|
||||
|
||||
if("1.${CMAKE_VERSION}" VERSION_LESS "1.2.8.0")
|
||||
# Older than CMake 2.8.0
|
||||
add_test(${_name}_cppcheck_test
|
||||
"${CPPCHECK_EXECUTABLE}"
|
||||
${CPPCHECK_TEMPLATE_ARG}
|
||||
${_cppcheck_args}
|
||||
${_files})
|
||||
else()
|
||||
# CMake 2.8.0 and newer
|
||||
add_test(NAME
|
||||
${_name}_cppcheck_test
|
||||
COMMAND
|
||||
"${CPPCHECK_EXECUTABLE}"
|
||||
${CPPCHECK_TEMPLATE_ARG}
|
||||
${_cppcheck_args}
|
||||
${_files})
|
||||
endif()
|
||||
|
||||
set_tests_properties(${_name}_cppcheck_test
|
||||
PROPERTIES
|
||||
FAIL_REGULAR_EXPRESSION
|
||||
"${CPPCHECK_FAIL_REGULAR_EXPRESSION}")
|
||||
|
||||
add_custom_command(TARGET
|
||||
all_cppcheck
|
||||
PRE_BUILD
|
||||
COMMAND
|
||||
${CPPCHECK_EXECUTABLE}
|
||||
${CPPCHECK_QUIET_ARG}
|
||||
${CPPCHECK_TEMPLATE_ARG}
|
||||
"--enable=style,information,unusedFunction"
|
||||
${_cppcheck_args}
|
||||
${_files}
|
||||
WORKING_DIRECTORY
|
||||
"${CMAKE_CURRENT_SOURCE_DIR}"
|
||||
COMMENT
|
||||
"${_name}_cppcheck: Running cppcheck on target ${_name}..."
|
||||
VERBATIM)
|
||||
endif()
|
||||
|
||||
endfunction()
|
22
cmake/modules/FindGLIB2.cmake
Normal file
22
cmake/modules/FindGLIB2.cmake
Normal file
@ -0,0 +1,22 @@
|
||||
# - Try to find the GLIB2 libraries
|
||||
|
||||
if(GLIB2_INCLUDE_DIR AND GLIB2_LIBRARIES AND GTHREAD2_LIBRARIES)
|
||||
# Already in cache, be silent
|
||||
set(GLIB2_FIND_QUIETLY TRUE)
|
||||
endif(GLIB2_INCLUDE_DIR AND GLIB2_LIBRARIES AND GTHREAD2_LIBRARIES)
|
||||
|
||||
if (NOT WIN32)
|
||||
include(FindPkgConfig)
|
||||
pkg_search_module(PC_GLIB2 REQUIRED glib-2.0)
|
||||
pkg_search_module(PC_GTHREAD2 REQUIRED gthread-2.0)
|
||||
endif(NOT WIN32)
|
||||
|
||||
set(GLIB2_INCLUDE_DIR ${PC_GLIB2_INCLUDE_DIRS})
|
||||
|
||||
find_library(GLIB2_LIBRARIES NAMES glib-2.0 HINTS ${PC_GLIB2_LIBDIR} ${PC_GLIB2_LIBRARY_DIRS})
|
||||
|
||||
find_library(GTHREAD2_LIBRARIES NAMES gthread-2.0 HINTS ${PC_GTHREAD2_LIBDIR} ${PC_GTHREAD2_LIBRARY_DIRS})
|
||||
|
||||
|
||||
mark_as_advanced(GLIB2_INCLUDE_DIR GLIB2_LIBRARIES GTHREAD2_LIBRARIES)
|
||||
|
111
cmake/modules/FindMySQL.cmake
Normal file
111
cmake/modules/FindMySQL.cmake
Normal file
@ -0,0 +1,111 @@
|
||||
# - Find MySQL
|
||||
# Find the MySQL includes and client library
|
||||
# This module defines
|
||||
# MYSQL_INCLUDE_DIR, where to find mysql.h
|
||||
# MYSQL_LIBRARIES, the libraries needed to use MySQL.
|
||||
# MYSQL_FOUND, If false, do not try to use MySQL.
|
||||
#
|
||||
# Copyright (c) 2006, Jaroslaw Staniek, <js@iidea.pl>
|
||||
# Lot of adustmens by Michal Cihar <michal@cihar.com>
|
||||
#
|
||||
# vim: expandtab sw=4 ts=4 sts=4:
|
||||
#
|
||||
# Redistribution and use is allowed according to the terms of the BSD license.
|
||||
|
||||
if(UNIX)
|
||||
set(MYSQL_CONFIG_PREFER_PATH "$ENV{MYSQL_HOME}/bin" CACHE FILEPATH
|
||||
"preferred path to MySQL (mysql_config)")
|
||||
find_program(MYSQL_CONFIG mysql_config
|
||||
${MYSQL_CONFIG_PREFER_PATH}
|
||||
/usr/local/mysql/bin/
|
||||
/usr/local/bin/
|
||||
/usr/bin/
|
||||
)
|
||||
|
||||
if(MYSQL_CONFIG)
|
||||
message(STATUS "Using mysql-config: ${MYSQL_CONFIG}")
|
||||
# set CFLAGS
|
||||
exec_program(${MYSQL_CONFIG}
|
||||
ARGS --cflags
|
||||
OUTPUT_VARIABLE MY_TMP)
|
||||
|
||||
set(MYSQL_CFLAGS ${MY_TMP} CACHE STRING INTERNAL)
|
||||
|
||||
# set INCLUDE_DIR
|
||||
exec_program(${MYSQL_CONFIG}
|
||||
ARGS --include
|
||||
OUTPUT_VARIABLE MY_TMP)
|
||||
|
||||
string(REGEX REPLACE "-I([^ ]*)( .*)?" "\\1" MY_TMP "${MY_TMP}")
|
||||
|
||||
set(MYSQL_ADD_INCLUDE_DIR ${MY_TMP} CACHE FILEPATH INTERNAL)
|
||||
|
||||
# set LIBRARY_DIR
|
||||
exec_program(${MYSQL_CONFIG}
|
||||
ARGS --libs_r
|
||||
OUTPUT_VARIABLE MY_TMP)
|
||||
|
||||
set(MYSQL_ADD_LIBRARIES "")
|
||||
|
||||
# prepend space in order to match separate words only (e.g. rather
|
||||
# than "-linux" from within "-L/usr/lib/i386-linux-gnu")
|
||||
string(REGEX MATCHALL " +-l[^ ]*" MYSQL_LIB_LIST " ${MY_TMP}")
|
||||
foreach(MY_LIB ${MYSQL_LIB_LIST})
|
||||
string(REGEX REPLACE "[ ]*-l([^ ]*)" "\\1" MY_LIB "${MY_LIB}")
|
||||
list(APPEND MYSQL_ADD_LIBRARIES "${MY_LIB}")
|
||||
endforeach(MY_LIB ${MYSQL_LIBS})
|
||||
|
||||
set(MYSQL_ADD_LIBRARY_PATH "")
|
||||
|
||||
string(REGEX MATCHALL " +-L[^ ]*" MYSQL_LIBDIR_LIST " ${MY_TMP}")
|
||||
foreach(MY_LIB ${MYSQL_LIBDIR_LIST})
|
||||
string(REGEX REPLACE "[ ]*-L([^ ]*)" "\\1" MY_LIB "${MY_LIB}")
|
||||
list(APPEND MYSQL_ADD_LIBRARY_PATH "${MY_LIB}")
|
||||
endforeach(MY_LIB ${MYSQL_LIBS})
|
||||
|
||||
else(MYSQL_CONFIG)
|
||||
set(MYSQL_ADD_LIBRARIES "")
|
||||
list(APPEND MYSQL_ADD_LIBRARIES "mysqlclient")
|
||||
endif(MYSQL_CONFIG)
|
||||
else(UNIX)
|
||||
set(MYSQL_ADD_INCLUDE_DIR "c:/msys/local/include" CACHE FILEPATH INTERNAL)
|
||||
set(MYSQL_ADD_LIBRARY_PATH "c:/msys/local/lib" CACHE FILEPATH INTERNAL)
|
||||
ENDIF(UNIX)
|
||||
|
||||
find_path(MYSQL_INCLUDE_DIR mysql.h
|
||||
${MYSQL_ADD_INCLUDE_DIR}
|
||||
/usr/local/include
|
||||
/usr/local/include/mysql
|
||||
/usr/local/mysql/include
|
||||
/usr/local/mysql/include/mysql
|
||||
/usr/include
|
||||
/usr/include/mysql
|
||||
/usr/include/mysql/private
|
||||
)
|
||||
|
||||
set(TMP_MYSQL_LIBRARIES "")
|
||||
set(CMAKE_FIND_LIBRARY_SUFFIXES .so .a .lib .so.1)
|
||||
foreach(MY_LIB ${MYSQL_ADD_LIBRARIES})
|
||||
find_library("MYSQL_LIBRARIES_${MY_LIB}" NAMES ${MY_LIB}
|
||||
HINTS
|
||||
${MYSQL_ADD_LIBRARY_PATH}
|
||||
/usr/lib/mysql
|
||||
/usr/lib
|
||||
/usr/local/lib
|
||||
/usr/local/lib/mysql
|
||||
/usr/local/mysql/lib
|
||||
)
|
||||
list(APPEND TMP_MYSQL_LIBRARIES "${MYSQL_LIBRARIES_${MY_LIB}}")
|
||||
endforeach(MY_LIB ${MYSQL_ADD_LIBRARIES})
|
||||
|
||||
set(MYSQL_LIBRARIES ${TMP_MYSQL_LIBRARIES} CACHE FILEPATH INTERNAL)
|
||||
|
||||
if(MYSQL_INCLUDE_DIR AND MYSQL_LIBRARIES)
|
||||
set(MYSQL_FOUND TRUE CACHE INTERNAL "MySQL found")
|
||||
message(STATUS "Found MySQL: ${MYSQL_INCLUDE_DIR}, ${MYSQL_LIBRARIES}")
|
||||
else(MYSQL_INCLUDE_DIR AND MYSQL_LIBRARIES)
|
||||
set(MYSQL_FOUND FALSE CACHE INTERNAL "MySQL found")
|
||||
message(STATUS "MySQL not found.")
|
||||
endif(MYSQL_INCLUDE_DIR AND MYSQL_LIBRARIES)
|
||||
|
||||
mark_as_advanced(MYSQL_INCLUDE_DIR MYSQL_LIBRARIES MYSQL_CFLAGS)
|
45
cmake/modules/FindPCRE.cmake
Normal file
45
cmake/modules/FindPCRE.cmake
Normal file
@ -0,0 +1,45 @@
|
||||
# - Try to find the PCRE regular expression library
|
||||
# Once done this will define
|
||||
#
|
||||
# PCRE_FOUND - system has the PCRE library
|
||||
# PCRE_INCLUDE_DIR - the PCRE include directory
|
||||
# PCRE_LIBRARIES - The libraries needed to use PCRE
|
||||
|
||||
# Copyright (c) 2006, Alexander Neundorf, <neundorf@kde.org>
|
||||
#
|
||||
# Redistribution and use is allowed according to the terms of the BSD license.
|
||||
# For details see the accompanying COPYING-CMAKE-SCRIPTS file.
|
||||
|
||||
|
||||
if (PCRE_INCLUDE_DIR AND PCRE_PCREPOSIX_LIBRARY AND PCRE_PCRE_LIBRARY)
|
||||
# Already in cache, be silent
|
||||
set(PCRE_FIND_QUIETLY TRUE)
|
||||
endif (PCRE_INCLUDE_DIR AND PCRE_PCREPOSIX_LIBRARY AND PCRE_PCRE_LIBRARY)
|
||||
|
||||
|
||||
if (NOT WIN32)
|
||||
# use pkg-config to get the directories and then use these values
|
||||
# in the FIND_PATH() and FIND_LIBRARY() calls
|
||||
find_package(PkgConfig)
|
||||
|
||||
pkg_check_modules(PC_PCRE REQUIRED libpcre)
|
||||
|
||||
set(PCRE_DEFINITIONS ${PC_PCRE_CFLAGS_OTHER})
|
||||
|
||||
endif (NOT WIN32)
|
||||
|
||||
find_path(PCRE_INCLUDE_DIR pcre.h
|
||||
HINTS ${PC_PCRE_INCLUDEDIR} ${PC_PCRE_INCLUDE_DIRS}
|
||||
PATH_SUFFIXES pcre)
|
||||
|
||||
find_library(PCRE_PCRE_LIBRARY NAMES pcre HINTS ${PC_PCRE_LIBDIR} ${PC_PCRE_LIBRARY_DIRS})
|
||||
|
||||
find_library(PCRE_PCREPOSIX_LIBRARY NAMES pcreposix HINTS ${PC_PCRE_LIBDIR} ${PC_PCRE_LIBRARY_DIRS})
|
||||
|
||||
include(FindPackageHandleStandardArgs)
|
||||
find_package_handle_standard_args(PCRE DEFAULT_MSG PCRE_INCLUDE_DIR PCRE_PCRE_LIBRARY PCRE_PCREPOSIX_LIBRARY )
|
||||
|
||||
set(PCRE_LIBRARIES ${PCRE_PCRE_LIBRARY} ${PCRE_PCREPOSIX_LIBRARY})
|
||||
|
||||
mark_as_advanced(PCRE_INCLUDE_DIR PCRE_LIBRARIES PCRE_PCREPOSIX_LIBRARY PCRE_PCRE_LIBRARY)
|
||||
|
57
cmake/modules/FindSphinx.cmake
Normal file
57
cmake/modules/FindSphinx.cmake
Normal file
@ -0,0 +1,57 @@
|
||||
# - This module looks for Sphinx
|
||||
# Find the Sphinx documentation generator
|
||||
#
|
||||
# This modules defines
|
||||
# SPHINX_EXECUTABLE
|
||||
# SPHINX_FOUND
|
||||
# SPHINX_MAJOR_VERSION
|
||||
# SPHINX_MINOR_VERSION
|
||||
# SPHINX_VERSION
|
||||
|
||||
#=============================================================================
|
||||
# Copyright 2002-2009 Kitware, Inc.
|
||||
# Copyright 2009-2011 Peter Colberg
|
||||
#
|
||||
# Distributed under the OSI-approved BSD License (the "License");
|
||||
# see accompanying file COPYING-CMAKE-SCRIPTS for details.
|
||||
#
|
||||
# This software is distributed WITHOUT ANY WARRANTY; without even the
|
||||
# implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
|
||||
# See the License for more information.
|
||||
#=============================================================================
|
||||
# (To distribute this file outside of CMake, substitute the full
|
||||
# License text for the above reference.)
|
||||
|
||||
find_program(SPHINX_EXECUTABLE NAMES sphinx-build
|
||||
HINTS
|
||||
$ENV{SPHINX_DIR}
|
||||
PATH_SUFFIXES bin
|
||||
DOC "Sphinx documentation generator"
|
||||
)
|
||||
|
||||
include(FindPackageHandleStandardArgs)
|
||||
|
||||
find_package_handle_standard_args(Sphinx DEFAULT_MSG
|
||||
SPHINX_EXECUTABLE
|
||||
)
|
||||
|
||||
if (SPHINX_EXECUTABLE)
|
||||
execute_process (
|
||||
COMMAND "${SPHINX_EXECUTABLE}" -h
|
||||
OUTPUT_VARIABLE _SPHINX_VERSION_OUTPUT
|
||||
ERROR_VARIABLE _SPHINX_VERSION_OUTPUT
|
||||
)
|
||||
if (_SPHINX_VERSION_OUTPUT MATCHES "Sphinx v([0-9]+\\.[0-9]+\\.[0-9]+)")
|
||||
set (SPHINX_VERSION "${CMAKE_MATCH_1}")
|
||||
string (REPLACE "." ";" _SPHINX_VERSION_LIST "${SPHINX_VERSION}")
|
||||
list (GET _SPHINX_VERSION_LIST 0 SPHINX_MAJOR_VERSION)
|
||||
list (GET _SPHINX_VERSION_LIST 1 SPHINX_MINOR_VERSION)
|
||||
# patch version meh :)
|
||||
endif()
|
||||
endif()
|
||||
|
||||
message("${SPHINX_MAJOR_VERSION}")
|
||||
|
||||
mark_as_advanced(
|
||||
SPHINX_EXECUTABLE
|
||||
)
|
142
cmake/modules/Findcppcheck.cmake
Normal file
142
cmake/modules/Findcppcheck.cmake
Normal file
@ -0,0 +1,142 @@
|
||||
# - try to find cppcheck tool
|
||||
#
|
||||
# Cache Variables:
|
||||
# CPPCHECK_EXECUTABLE
|
||||
#
|
||||
# Non-cache variables you might use in your CMakeLists.txt:
|
||||
# CPPCHECK_FOUND
|
||||
# CPPCHECK_POSSIBLEERROR_ARG
|
||||
# CPPCHECK_UNUSEDFUNC_ARG
|
||||
# CPPCHECK_STYLE_ARG
|
||||
# CPPCHECK_QUIET_ARG
|
||||
# CPPCHECK_INCLUDEPATH_ARG
|
||||
# CPPCHECK_FAIL_REGULAR_EXPRESSION
|
||||
# CPPCHECK_WARN_REGULAR_EXPRESSION
|
||||
# CPPCHECK_MARK_AS_ADVANCED - whether to mark our vars as advanced even
|
||||
# if we don't find this program.
|
||||
#
|
||||
# Requires these CMake modules:
|
||||
# FindPackageHandleStandardArgs (known included with CMake >=2.6.2)
|
||||
#
|
||||
# Original Author:
|
||||
# 2009-2010 Ryan Pavlik <rpavlik@iastate.edu> <abiryan@ryand.net>
|
||||
# http://academic.cleardefinition.com
|
||||
# Iowa State University HCI Graduate Program/VRAC
|
||||
#
|
||||
# Copyright Iowa State University 2009-2010.
|
||||
# Distributed under the Boost Software License, Version 1.0.
|
||||
# (See accompanying file LICENSE_1_0.txt or copy at
|
||||
# http://www.boost.org/LICENSE_1_0.txt)
|
||||
|
||||
file(TO_CMAKE_PATH "${CPPCHECK_ROOT_DIR}" CPPCHECK_ROOT_DIR)
|
||||
set(CPPCHECK_ROOT_DIR
|
||||
"${CPPCHECK_ROOT_DIR}"
|
||||
CACHE
|
||||
PATH
|
||||
"Path to search for cppcheck")
|
||||
|
||||
# cppcheck app bundles on Mac OS X are GUI, we want command line only
|
||||
set(_oldappbundlesetting ${CMAKE_FIND_APPBUNDLE})
|
||||
set(CMAKE_FIND_APPBUNDLE NEVER)
|
||||
|
||||
# If we have a custom path, look there first.
|
||||
if(CPPCHECK_ROOT_DIR)
|
||||
find_program(CPPCHECK_EXECUTABLE
|
||||
NAMES
|
||||
cppcheck
|
||||
cli
|
||||
PATHS
|
||||
"${CPPCHECK_ROOT_DIR}"
|
||||
PATH_SUFFIXES
|
||||
cli
|
||||
NO_DEFAULT_PATH)
|
||||
endif()
|
||||
|
||||
find_program(CPPCHECK_EXECUTABLE NAMES cppcheck)
|
||||
|
||||
# Restore original setting for appbundle finding
|
||||
set(CMAKE_FIND_APPBUNDLE ${_oldappbundlesetting})
|
||||
|
||||
if(CPPCHECK_EXECUTABLE)
|
||||
# Find out where our test file is
|
||||
get_filename_component(_cppcheckmoddir ${CMAKE_CURRENT_LIST_FILE} PATH)
|
||||
set(_cppcheckdummyfile "${_cppcheckmoddir}/Findcppcheck.cpp")
|
||||
|
||||
# Check for the two types of command line arguments by just trying them
|
||||
execute_process(COMMAND
|
||||
"${CPPCHECK_EXECUTABLE}"
|
||||
"--enable=style"
|
||||
"--quiet"
|
||||
"${_cppcheckdummyfile}"
|
||||
RESULT_VARIABLE
|
||||
_cppcheck_new_result
|
||||
OUTPUT_QUIET
|
||||
ERROR_QUIET)
|
||||
execute_process(COMMAND
|
||||
"${CPPCHECK_EXECUTABLE}"
|
||||
"--style"
|
||||
"--quiet"
|
||||
"${_cppcheckdummyfile}"
|
||||
RESULT_VARIABLE
|
||||
_cppcheck_old_result
|
||||
OUTPUT_QUIET
|
||||
ERROR_QUIET)
|
||||
if("${_cppcheck_new_result}" EQUAL 0)
|
||||
# New arguments
|
||||
set(CPPCHECK_UNUSEDFUNC_ARG "--enable=unusedFunctions")
|
||||
set(CPPCHECK_POSSIBLEERROR_ARG "--enable=possibleError")
|
||||
set(CPPCHECK_STYLE_ARG "--enable=style")
|
||||
set(CPPCHECK_QUIET_ARG "--quiet")
|
||||
set(CPPCHECK_INCLUDEPATH_ARG "-I")
|
||||
if(MSVC)
|
||||
set(CPPCHECK_TEMPLATE_ARG --template vs)
|
||||
set(CPPCHECK_FAIL_REGULAR_EXPRESSION "[(]error[)]")
|
||||
set(CPPCHECK_WARN_REGULAR_EXPRESSION "[(]style[)]")
|
||||
elseif(CMAKE_COMPILER_IS_GNUCXX)
|
||||
set(CPPCHECK_TEMPLATE_ARG --template gcc)
|
||||
set(CPPCHECK_FAIL_REGULAR_EXPRESSION " error: ")
|
||||
set(CPPCHECK_WARN_REGULAR_EXPRESSION " style: ")
|
||||
else()
|
||||
message(STATUS
|
||||
"Warning: FindCppcheck doesn't know how to format error messages for your compiler!")
|
||||
set(CPPCHECK_TEMPLATE_ARG --template gcc)
|
||||
set(CPPCHECK_FAIL_REGULAR_EXPRESSION " error: ")
|
||||
set(CPPCHECK_WARN_REGULAR_EXPRESSION " style: ")
|
||||
endif()
|
||||
elseif("${_cppcheck_old_result}" EQUAL 0)
|
||||
# Old arguments
|
||||
set(CPPCHECK_UNUSEDFUNC_ARG "--unused-functions")
|
||||
set(CPPCHECK_POSSIBLEERROR_ARG "--all")
|
||||
set(CPPCHECK_STYLE_ARG "--style")
|
||||
set(CPPCHECK_QUIET_ARG "--quiet")
|
||||
set(CPPCHECK_INCLUDEPATH_ARG "-I")
|
||||
set(CPPCHECK_FAIL_REGULAR_EXPRESSION "error:")
|
||||
set(CPPCHECK_WARN_REGULAR_EXPRESSION "[(]style[)]")
|
||||
else()
|
||||
# No idea - some other issue must be getting in the way
|
||||
message(STATUS
|
||||
"WARNING: Can't detect whether CPPCHECK wants new or old-style arguments!")
|
||||
endif()
|
||||
|
||||
|
||||
endif()
|
||||
|
||||
set(CPPCHECK_ALL
|
||||
"${CPPCHECK_EXECUTABLE} ${CPPCHECK_POSSIBLEERROR_ARG} ${CPPCHECK_UNUSEDFUNC_ARG} ${CPPCHECK_STYLE_ARG} ${CPPCHECK_QUIET_ARG} ${CPPCHECK_INCLUDEPATH_ARG} some/include/path")
|
||||
|
||||
include(FindPackageHandleStandardArgs)
|
||||
find_package_handle_standard_args(cppcheck
|
||||
DEFAULT_MSG
|
||||
CPPCHECK_ALL
|
||||
CPPCHECK_EXECUTABLE
|
||||
CPPCHECK_POSSIBLEERROR_ARG
|
||||
CPPCHECK_UNUSEDFUNC_ARG
|
||||
CPPCHECK_STYLE_ARG
|
||||
CPPCHECK_INCLUDEPATH_ARG
|
||||
CPPCHECK_QUIET_ARG)
|
||||
|
||||
if(CPPCHECK_FOUND OR CPPCHECK_MARK_AS_ADVANCED)
|
||||
mark_as_advanced(CPPCHECK_ROOT_DIR)
|
||||
endif()
|
||||
|
||||
mark_as_advanced(CPPCHECK_EXECUTABLE)
|
16
cmake/modules/Findcppcheck.cpp
Normal file
16
cmake/modules/Findcppcheck.cpp
Normal file
@ -0,0 +1,16 @@
|
||||
/**
|
||||
* \file Findcppcheck.cpp
|
||||
* \brief Dummy C++ source file used by CMake module Findcppcheck.cmake
|
||||
*
|
||||
* \author
|
||||
* Ryan Pavlik, 2009-2010
|
||||
* <rpavlik@iastate.edu>
|
||||
* http://academic.cleardefinition.com/
|
||||
*
|
||||
*/
|
||||
|
||||
|
||||
|
||||
int main(int argc, char* argv[]) {
|
||||
return 0;
|
||||
}
|
66
common.h
Normal file
66
common.h
Normal file
@ -0,0 +1,66 @@
|
||||
/*
|
||||
This program is free software: you can redistribute it and/or modify
|
||||
it under the terms of the GNU General Public License as published by
|
||||
the Free Software Foundation, either version 3 of the License, or
|
||||
(at your option) any later version.
|
||||
|
||||
This program is distributed in the hope that it will be useful,
|
||||
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
GNU General Public License for more details.
|
||||
|
||||
You should have received a copy of the GNU General Public License
|
||||
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||
|
||||
Authors: Andrew Hutchings, SkySQL (andrew at skysql dot com)
|
||||
*/
|
||||
#ifndef _common_h
|
||||
#define _common_h
|
||||
|
||||
char *hostname=NULL;
|
||||
char *username=NULL;
|
||||
char *password=NULL;
|
||||
char *socket_path=NULL;
|
||||
char *db=NULL;
|
||||
char *defaults_file=NULL;
|
||||
#ifdef WITH_SSL
|
||||
char *key=NULL;
|
||||
char *cert=NULL;
|
||||
char *ca=NULL;
|
||||
char *capath=NULL;
|
||||
char *cipher=NULL;
|
||||
#endif
|
||||
|
||||
gboolean askPassword=FALSE;
|
||||
guint port=0;
|
||||
guint num_threads= 4;
|
||||
guint verbose=2;
|
||||
gboolean ssl= FALSE;
|
||||
gboolean compress_protocol= FALSE;
|
||||
gboolean program_version= FALSE;
|
||||
|
||||
GOptionEntry common_entries[] =
|
||||
{
|
||||
{ "host", 'h', 0, G_OPTION_ARG_STRING, &hostname, "The host to connect to", NULL },
|
||||
{ "user", 'u', 0, G_OPTION_ARG_STRING, &username, "Username with the necessary privileges", NULL },
|
||||
{ "password", 'p', 0, G_OPTION_ARG_STRING, &password, "User password", NULL },
|
||||
{ "ask-password", 'a', 0, G_OPTION_ARG_NONE, &askPassword, "Prompt For User password", NULL },
|
||||
{ "port", 'P', 0, G_OPTION_ARG_INT, &port, "TCP/IP port to connect to", NULL },
|
||||
{ "socket", 'S', 0, G_OPTION_ARG_STRING, &socket_path, "UNIX domain socket file to use for connection", NULL },
|
||||
{ "threads", 't', 0, G_OPTION_ARG_INT, &num_threads, "Number of threads to use, default 4", NULL },
|
||||
{ "compress-protocol", 'C', 0, G_OPTION_ARG_NONE, &compress_protocol, "Use compression on the MySQL connection", NULL },
|
||||
{ "version", 'V', 0, G_OPTION_ARG_NONE, &program_version, "Show the program version and exit", NULL },
|
||||
{ "verbose", 'v', 0, G_OPTION_ARG_INT, &verbose, "Verbosity of output, 0 = silent, 1 = errors, 2 = warnings, 3 = info, default 2", NULL },
|
||||
{ "defaults-file", 0, 0, G_OPTION_ARG_FILENAME, &defaults_file, "Use a specific defaults file", NULL },
|
||||
#ifdef WITH_SSL
|
||||
{ "ssl", 0, 0, G_OPTION_ARG_NONE, &ssl, "Connect using SSL", NULL},
|
||||
{ "key", 0, 0, G_OPTION_ARG_STRING, &key, "The path name to the key file", NULL },
|
||||
{ "cert", 0, 0, G_OPTION_ARG_STRING, &cert, "The path name to the certificate file", NULL },
|
||||
{ "ca", 0, 0, G_OPTION_ARG_STRING, &ca, "The path name to the certificate authority file", NULL },
|
||||
{ "capath", 0, 0, G_OPTION_ARG_STRING, &capath, "The path name to a directory that contains trusted SSL CA certificates in PEM format", NULL },
|
||||
{ "cipher", 0, 0, G_OPTION_ARG_STRING, &cipher, "A list of permissible ciphers to use for SSL encryption", NULL },
|
||||
#endif
|
||||
{ NULL, 0, 0, G_OPTION_ARG_NONE, NULL, NULL, NULL }
|
||||
};
|
||||
|
||||
#endif
|
17
config.h.in
Normal file
17
config.h.in
Normal file
@ -0,0 +1,17 @@
|
||||
#ifndef CONFIG_H
|
||||
#define CONFIG_H
|
||||
|
||||
#cmakedefine VERSION "@VERSION@"
|
||||
#cmakedefine WITH_BINLOG
|
||||
|
||||
#if defined(LIBMYSQL_VERSION)
|
||||
#define MYSQL_VERSION_STR LIBMYSQL_VERSION
|
||||
#elif defined(MARIADB_CLIENT_VERSION_STR)
|
||||
#define MYSQL_VERSION_STR MARIADB_CLIENT_VERSION_STR
|
||||
#elif defined(MYSQL_VERSION_NUMBER)
|
||||
#define MYSQL_VERSION_STR MYSQL_VERSION_NUMBER
|
||||
#else
|
||||
#define MYSQL_VERSION_STR MYSQL_SERVER_VERSION
|
||||
#endif
|
||||
|
||||
#endif
|
54
connection.c
Normal file
54
connection.c
Normal file
@ -0,0 +1,54 @@
|
||||
/*
|
||||
This program is free software: you can redistribute it and/or modify
|
||||
it under the terms of the GNU General Public License as published by
|
||||
the Free Software Foundation, either version 3 of the License, or
|
||||
(at your option) any later version.
|
||||
|
||||
This program is distributed in the hope that it will be useful,
|
||||
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
GNU General Public License for more details.
|
||||
|
||||
You should have received a copy of the GNU General Public License
|
||||
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||
|
||||
Authors: Aaron Brady, Shopify (insom)
|
||||
*/
|
||||
|
||||
#include <pcre.h>
|
||||
#include <glib.h>
|
||||
#include <string.h>
|
||||
#include "connection.h"
|
||||
|
||||
extern char *defaults_file;
|
||||
#ifdef WITH_SSL
|
||||
extern char *key;
|
||||
extern char *cert;
|
||||
extern char *ca;
|
||||
extern char *capath;
|
||||
extern char *cipher;
|
||||
extern gboolean ssl;
|
||||
#endif
|
||||
extern guint compress_protocol;
|
||||
|
||||
void configure_connection(MYSQL *conn, const char *name) {
|
||||
if (defaults_file != NULL) {
|
||||
mysql_options(conn,MYSQL_READ_DEFAULT_FILE,defaults_file);
|
||||
}
|
||||
mysql_options(conn, MYSQL_READ_DEFAULT_GROUP, name);
|
||||
|
||||
if (compress_protocol)
|
||||
mysql_options(conn, MYSQL_OPT_COMPRESS, NULL);
|
||||
|
||||
#ifdef WITH_SSL
|
||||
unsigned int i;
|
||||
if (ssl) {
|
||||
i = SSL_MODE_REQUIRED;
|
||||
} else {
|
||||
i = SSL_MODE_DISABLED;
|
||||
}
|
||||
|
||||
mysql_ssl_set(conn,key,cert,ca,capath,cipher);
|
||||
mysql_options(conn,MYSQL_OPT_SSL_MODE,&i);
|
||||
#endif
|
||||
}
|
22
connection.h
Normal file
22
connection.h
Normal file
@ -0,0 +1,22 @@
|
||||
/*
|
||||
This program is free software: you can redistribute it and/or modify
|
||||
it under the terms of the GNU General Public License as published by
|
||||
the Free Software Foundation, either version 3 of the License, or
|
||||
(at your option) any later version.
|
||||
|
||||
This program is distributed in the hope that it will be useful,
|
||||
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
GNU General Public License for more details.
|
||||
|
||||
You should have received a copy of the GNU General Public License
|
||||
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||
|
||||
Authors: Aaron Brady, Shopify (insom)
|
||||
*/
|
||||
#ifndef _connection_h
|
||||
#define _connection_h
|
||||
#include <mysql.h>
|
||||
|
||||
void configure_connection(MYSQL *conn, const char* name);
|
||||
#endif
|
93
debian/changelog
vendored
Normal file
93
debian/changelog
vendored
Normal file
@ -0,0 +1,93 @@
|
||||
mydumper (0.9.5-1) unstable; urgency=medium
|
||||
|
||||
* add patch to FindMySQL cmake module to find libatomic when it's needed
|
||||
|
||||
-- Mateusz Kijowski <mateusz.kijowski@gmail.com> Wed, 18 Jan 2017 01:09:19 +0100
|
||||
|
||||
mydumper (0.9.1-4) unstable; urgency=medium
|
||||
|
||||
* manpages are generated by Sphinx, so moving back python-sphinx
|
||||
to B-D (Closes: #851433)
|
||||
* Add a Breaks+Replaces relation to mydumper-doc since it replaces
|
||||
doc-base files from main package (Closes: #851221)
|
||||
|
||||
-- Mateusz Kijowski <mateusz.kijowski@gmail.com> Sun, 15 Jan 2017 16:57:52 +0100
|
||||
|
||||
mydumper (0.9.1-3) unstable; urgency=medium
|
||||
|
||||
* use sphinxdoc dh helper only in appropriate targets (Closes: #850847,
|
||||
#850969)
|
||||
|
||||
-- Mateusz Kijowski <mateusz.kijowski@gmail.com> Wed, 11 Jan 2017 11:44:34 +0100
|
||||
|
||||
mydumper (0.9.1-2) unstable; urgency=medium
|
||||
|
||||
* Depend on default-libmysqlclient-dev (Closes: #845877, #841607)
|
||||
* bump standards version to 3.9.8
|
||||
* Separate doc package to comply with standards 3.9.7
|
||||
* dh compat level bumped to 9
|
||||
* changed homepage and debian/watch to github
|
||||
|
||||
-- Mateusz Kijowski <mateusz.kijowski@gmail.com> Mon, 02 Jan 2017 15:40:34 +0100
|
||||
|
||||
mydumper (0.9.1-1) unstable; urgency=medium
|
||||
|
||||
* Imported Upstream version 0.9.1
|
||||
|
||||
-- Mateusz Kijowski <mateusz.kijowski@gmail.com> Tue, 10 Nov 2015 17:59:57 +0100
|
||||
|
||||
mydumper (0.6.2-1) unstable; urgency=medium
|
||||
|
||||
* Imported Upstream version 0.6.2 (Closes: #790640)
|
||||
* fix manpage inconsistencies (Closes: #773990) (LP: #1402381)
|
||||
* bump standards version to 3.9.6 (no changes needed)
|
||||
* allow uscan to verify upstream tarball
|
||||
|
||||
-- Mateusz Kijowski <mateusz.kijowski@gmail.com> Wed, 01 Jul 2015 13:54:15 +0200
|
||||
|
||||
mydumper (0.6.1-1) unstable; urgency=medium
|
||||
|
||||
* updated standards version
|
||||
* move sphinxdoc depends to recommends (Closes: #730443)
|
||||
* homepage updated (Closes: #714921)
|
||||
* explicitly tell CMAKE to link libm
|
||||
patch 0003-explicitly-link-libm.patch
|
||||
* Imported Upstream version 0.6.1 (Closes: #735851) (LP: #1270330)
|
||||
|
||||
-- Mateusz Kijowski <mateusz.kijowski@gmail.com> Thu, 20 Mar 2014 01:17:43 +0100
|
||||
|
||||
mydumper (0.5.2-1) unstable; urgency=low
|
||||
|
||||
* Imported Upstream version 0.5.2 (Closes: #707458)
|
||||
* removed patches merged by upstream
|
||||
|
||||
-- Mateusz Kijowski <mateusz.kijowski@gmail.com> Thu, 09 May 2013 12:08:05 +0200
|
||||
|
||||
mydumper (0.5.1-3) unstable; urgency=low
|
||||
|
||||
* fix FTBFS in multiarch env (Closes: #673262) (LP: #1002291)
|
||||
patch 0006-fix-runaway-regexp-in-in-cmake.patch
|
||||
|
||||
-- Mateusz Kijowski <mateusz.kijowski@gmail.com> Mon, 04 Jun 2012 17:35:01 +0200
|
||||
|
||||
mydumper (0.5.1-2) unstable; urgency=low
|
||||
|
||||
* fixed watch file
|
||||
* updated standards version
|
||||
* patch 0005-zlib-file-type-change-fixes.patch added
|
||||
fix FTBFS from zlib > 1.2.4 (Closes: #662582) (LP: #948122)
|
||||
|
||||
-- Mateusz Kijowski <mateusz.kijowski@gmail.com> Fri, 23 Mar 2012 17:27:57 +0100
|
||||
|
||||
mydumper (0.5.1-1) unstable; urgency=low
|
||||
|
||||
* Initial release (Closes: #648711)
|
||||
* This is my first Debian package, be gentle
|
||||
* Patched CMakeLists.txt so that generated config.h file is put in the
|
||||
sources dir, where it is expected during the build
|
||||
* Patched documenation CMakeLists.txt so that it won't install
|
||||
documentation source files
|
||||
* Patch added to fix missing description in the manpages
|
||||
* Fixed typo in myloader
|
||||
|
||||
-- Mateusz Kijowski <mateusz.kijowski@gmail.com> Thu, 15 Nov 2011 11:26:38 +0100
|
1
debian/compat
vendored
Normal file
1
debian/compat
vendored
Normal file
@ -0,0 +1 @@
|
||||
9
|
48
debian/control
vendored
Normal file
48
debian/control
vendored
Normal file
@ -0,0 +1,48 @@
|
||||
Source: mydumper
|
||||
Section: database
|
||||
Priority: extra
|
||||
Maintainer: Mateusz Kijowski <mateusz.kijowski@gmail.com>
|
||||
Build-Depends: debhelper (>= 9.0.0), cmake, quilt, default-libmysqlclient-dev | libmysqlclient-dev, libglib2.0-dev, libpcre3-dev, zlib1g-dev, python-sphinx (>= 1.0.7+dfsg), python-docutils, libatomic1 | libatomic-ops-dev
|
||||
Standards-Version: 3.9.8
|
||||
Homepage: https://github.com/maxbube/mydumper
|
||||
#Vcs-Git: git://git.debian.org/collab-maint/mydumper.git
|
||||
#Vcs-Browser: http://git.debian.org/?p=collab-maint/mydumper.git;a=summary
|
||||
|
||||
Package: mydumper
|
||||
Architecture: any
|
||||
Depends: ${shlibs:Depends}, ${misc:Depends}
|
||||
Suggests: mydumper-doc
|
||||
Description: High-performance MySQL backup tool
|
||||
Mydumper (aka. MySQL Data Dumper) is a high-performance multi-threaded
|
||||
backup (and restore) toolset for MySQL and Drizzle. The main developers
|
||||
originally worked as Support Engineers at MySQL (one has moved to
|
||||
Facebook and another to SkySQL) and this is how they would envisage
|
||||
mysqldump based on years of user feedback.
|
||||
.
|
||||
Mydumper features
|
||||
.
|
||||
Lightweight C source
|
||||
Up to 10x faster dumps compared to mysqldump
|
||||
Consistent snapshots for transactional and non-transactional tables
|
||||
File compression on-the-fly
|
||||
Binary log dumps
|
||||
Multi-threaded restore utility
|
||||
Daemon mode for timed snapshots and continuous binary logs
|
||||
.
|
||||
Mydumper is still under active development but is well tested/used in
|
||||
production on some large installations.
|
||||
|
||||
Package: mydumper-doc
|
||||
Architecture: all
|
||||
Depends: ${misc:Depends}, ${sphinxdoc:Depends}
|
||||
Breaks: mydumper (<< 0.9.1-2)
|
||||
Replaces: mydumper (<< 0.9.1-2)
|
||||
Section: doc
|
||||
Description: High-performance MySQL backup tool - documentation
|
||||
Mydumper (aka. MySQL Data Dumper) is a high-performance multi-threaded
|
||||
backup (and restore) toolset for MySQL and Drizzle. The main developers
|
||||
originally worked as Support Engineers at MySQL (one has moved to
|
||||
Facebook and another to SkySQL) and this is how they would envisage
|
||||
mysqldump based on years of user feedback.
|
||||
.
|
||||
This package contains documentation for mydumper generated by sphinx
|
30
debian/copyright
vendored
Normal file
30
debian/copyright
vendored
Normal file
@ -0,0 +1,30 @@
|
||||
Format: http://anonscm.debian.org/viewvc/dep/web/deps/dep5.mdwn?revision=174
|
||||
Upstream-Name: mydumper
|
||||
Source: https://launchpad.net/mydumper/+download
|
||||
|
||||
Files: *
|
||||
Copyright: 2009-2011 Domas Mituzas ( domas at fb dot com )
|
||||
2009-2011 Andrew Hutchings ( andrew at skysql dot com )
|
||||
2009-2011 Mark Leith ( mark dot leith at oracle dot com )
|
||||
License: GPL-3.0+
|
||||
|
||||
Files: debian/*
|
||||
Copyright: 2011 Mateusz Kijowski <mateusz.kijowski@gmail.com>
|
||||
License: GPL-3.0+
|
||||
|
||||
License: GPL-3.0+
|
||||
This program is free software: you can redistribute it and/or modify
|
||||
it under the terms of the GNU General Public License as published by
|
||||
the Free Software Foundation, either version 3 of the License, or
|
||||
(at your option) any later version.
|
||||
.
|
||||
This package is distributed in the hope that it will be useful,
|
||||
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
GNU General Public License for more details.
|
||||
.
|
||||
You should have received a copy of the GNU General Public License
|
||||
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||
.
|
||||
On Debian systems, the complete text of the GNU General
|
||||
Public License version 3 can be found in "/usr/share/common-licenses/GPL-3".
|
1
debian/docs
vendored
Normal file
1
debian/docs
vendored
Normal file
@ -0,0 +1 @@
|
||||
README.md
|
1
debian/install
vendored
Normal file
1
debian/install
vendored
Normal file
@ -0,0 +1 @@
|
||||
usr/bin/*
|
2
debian/manpages
vendored
Normal file
2
debian/manpages
vendored
Normal file
@ -0,0 +1,2 @@
|
||||
debian/tmp/usr/share/man/man1/mydumper.1
|
||||
debian/tmp/usr/share/man/man1/myloader.1
|
11
debian/mydumper-doc.doc-base
vendored
Normal file
11
debian/mydumper-doc.doc-base
vendored
Normal file
@ -0,0 +1,11 @@
|
||||
Document: mydumper
|
||||
Title: MySQL Data Dumper documentation
|
||||
Author: Domas Mituzas, Andrew Hutchings, Mark Leith
|
||||
Max Bubenick
|
||||
Abstract: This is the documentation shipped with
|
||||
mydumper, describing usage and prvoding examples
|
||||
Section: Data Management
|
||||
|
||||
Format: HTML
|
||||
Index: /usr/share/doc/mydumper-doc/html/index.html
|
||||
Files: /usr/share/doc/mydumper-doc/html/*.html
|
1
debian/mydumper-doc.docs
vendored
Normal file
1
debian/mydumper-doc.docs
vendored
Normal file
@ -0,0 +1 @@
|
||||
debian/tmp/usr/share/doc/mydumper/html
|
39
debian/patches/0001-manpage-whatis-description.patch
vendored
Normal file
39
debian/patches/0001-manpage-whatis-description.patch
vendored
Normal file
@ -0,0 +1,39 @@
|
||||
From: Mateusz Kijowski <mkijowski@trojmiasto.pl>
|
||||
Date: Tue, 15 Nov 2011 11:19:01 +0100
|
||||
Subject: manpage whatis description
|
||||
|
||||
---
|
||||
docs/CMakeLists.txt | 4 ++++
|
||||
docs/_build/conf.py.in | 4 ++--
|
||||
2 files changed, 6 insertions(+), 2 deletions(-)
|
||||
|
||||
diff --git a/docs/CMakeLists.txt b/docs/CMakeLists.txt
|
||||
index 8fdb6b4..c43d246 100644
|
||||
--- a/docs/CMakeLists.txt
|
||||
+++ b/docs/CMakeLists.txt
|
||||
@@ -27,6 +27,10 @@ if(GENERATE_DOC)
|
||||
# master document with modules index
|
||||
set(REF_MASTER_DOC "modules")
|
||||
|
||||
+ # vars necessary for manpages generation
|
||||
+ set(MYDUMPER_PROGRAM_DESC "multi-threaded MySQL dumping")
|
||||
+ set(MYLOADER_PROGRAM_DESC "multi-threaded MySQL loader")
|
||||
+
|
||||
# substitute variables in configuration and scripts
|
||||
foreach(file
|
||||
conf.py
|
||||
diff --git a/docs/_build/conf.py.in b/docs/_build/conf.py.in
|
||||
index 9985c74..4844933 100644
|
||||
--- a/docs/_build/conf.py.in
|
||||
+++ b/docs/_build/conf.py.in
|
||||
@@ -211,8 +211,8 @@ latex_documents = [
|
||||
# One entry per manual page. List of tuples
|
||||
# (source start file, name, description, authors, manual section).
|
||||
man_pages = [
|
||||
- ('mydumper_usage', 'mydumper', u'@PROGRAM_DESC@',
|
||||
+ ('mydumper_usage', 'mydumper', u'@MYDUMPER_PROGRAM_DESC@',
|
||||
[u'Andrew Hutchings'], 1),
|
||||
- ('myloader_usage', 'myloader', u'@PROGRAM_DESC@',
|
||||
+ ('myloader_usage', 'myloader', u'@MYLOADER_PROGRAM_DESC@',
|
||||
[u'Andrew Hutchings'], 1)
|
||||
]
|
24
debian/patches/0002-dont-install-documentation-source.patch
vendored
Normal file
24
debian/patches/0002-dont-install-documentation-source.patch
vendored
Normal file
@ -0,0 +1,24 @@
|
||||
From: Mateusz Kijowski <mkijowski@trojmiasto.pl>
|
||||
Date: Tue, 22 Nov 2011 17:44:04 +0100
|
||||
Subject: dont install documentation source
|
||||
|
||||
---
|
||||
docs/CMakeLists.txt | 5 -----
|
||||
1 file changed, 5 deletions(-)
|
||||
|
||||
diff --git a/docs/CMakeLists.txt b/docs/CMakeLists.txt
|
||||
index c43d246..7c75c99 100644
|
||||
--- a/docs/CMakeLists.txt
|
||||
+++ b/docs/CMakeLists.txt
|
||||
@@ -61,11 +61,6 @@ if(GENERATE_DOC)
|
||||
"${SOURCES_DIR}"
|
||||
)
|
||||
|
||||
- # note the trailing slash to exclude directory name
|
||||
- install(DIRECTORY "${SOURCES_DIR}/"
|
||||
- DESTINATION "share/doc/mydumper"
|
||||
- )
|
||||
-
|
||||
# Sphinx cache with pickled ReST documents
|
||||
set(SPHINX_CACHE_DIR "${CMAKE_CURRENT_BINARY_DIR}/_doctrees")
|
||||
# HTML output directory
|
34
debian/patches/0003-explicitly-link-libm.patch
vendored
Normal file
34
debian/patches/0003-explicitly-link-libm.patch
vendored
Normal file
@ -0,0 +1,34 @@
|
||||
From: Mateusz Kijowski <mateusz.kijowski@gmail.com>
|
||||
Date: Mon, 20 Jan 2014 21:42:40 +0100
|
||||
Subject: explicitly link libm
|
||||
|
||||
---
|
||||
CMakeLists.txt | 7 +++----
|
||||
1 file changed, 3 insertions(+), 4 deletions(-)
|
||||
|
||||
diff --git a/CMakeLists.txt b/CMakeLists.txt
|
||||
index 5da3828..0a6f7e7 100644
|
||||
--- a/CMakeLists.txt
|
||||
+++ b/CMakeLists.txt
|
||||
@@ -9,6 +9,7 @@
|
||||
find_package(ZLIB)
|
||||
find_package(GLIB2)
|
||||
find_package(PCRE)
|
||||
+find_library(M_LIB m)
|
||||
|
||||
option(BUILD_DOCS "Build the documentation" ON)
|
||||
|
||||
@@ -37,11 +38,11 @@
|
||||
else (WITH_BINLOG)
|
||||
add_executable(mydumper mydumper.c server_detect.c g_unix_signal.c connection.c getPassword.c)
|
||||
endif (WITH_BINLOG)
|
||||
-target_link_libraries(mydumper ${MYSQL_LIBRARIES} ${GLIB2_LIBRARIES} ${GTHREAD2_LIBRARIES} ${PCRE_PCRE_LIBRARY} ${ZLIB_LIBRARIES} stdc++)
|
||||
+target_link_libraries(mydumper ${MYSQL_LIBRARIES} ${GLIB2_LIBRARIES} ${GTHREAD2_LIBRARIES} ${PCRE_PCRE_LIBRARY} ${ZLIB_LIBRARIES} ${M_LIB} stdc++)
|
||||
|
||||
|
||||
add_executable(myloader myloader.c connection.c getPassword.c)
|
||||
-target_link_libraries(myloader ${MYSQL_LIBRARIES} ${GLIB2_LIBRARIES} ${GTHREAD2_LIBRARIES} ${PCRE_PCRE_LIBRARY} ${ZLIB_LIBRARIES} stdc++)
|
||||
+target_link_libraries(myloader ${MYSQL_LIBRARIES} ${GLIB2_LIBRARIES} ${GTHREAD2_LIBRARIES} ${PCRE_PCRE_LIBRARY} ${ZLIB_LIBRARIES} ${M_LIB} stdc++)
|
||||
|
||||
INSTALL(TARGETS mydumper myloader
|
||||
RUNTIME DESTINATION bin
|
3
debian/patches/series
vendored
Normal file
3
debian/patches/series
vendored
Normal file
@ -0,0 +1,3 @@
|
||||
0001-manpage-whatis-description.patch
|
||||
0002-dont-install-documentation-source.patch
|
||||
0003-explicitly-link-libm.patch
|
13
debian/rules
vendored
Executable file
13
debian/rules
vendored
Executable file
@ -0,0 +1,13 @@
|
||||
#!/usr/bin/make -f
|
||||
# -*- makefile -*-
|
||||
# Sample debian/rules that uses debhelper.
|
||||
# This file was originally written by Joey Hess and Craig Small.
|
||||
# As a special exception, when this file is copied by dh-make into a
|
||||
# dh-make output file, you may use that output file without restriction.
|
||||
# This special exception was added by Craig Small in version 0.37 of dh-make.
|
||||
|
||||
# Uncomment this to turn on verbose mode.
|
||||
export DH_VERBOSE=1
|
||||
|
||||
%:
|
||||
dh $@ --with=sphinxdoc
|
1
debian/source/format
vendored
Normal file
1
debian/source/format
vendored
Normal file
@ -0,0 +1 @@
|
||||
3.0 (quilt)
|
39
debian/upstream/signing-key.asc
vendored
Normal file
39
debian/upstream/signing-key.asc
vendored
Normal file
@ -0,0 +1,39 @@
|
||||
-----BEGIN PGP PUBLIC KEY BLOCK-----
|
||||
Version: GnuPG v1
|
||||
|
||||
mQMuBFGKRv4RCAC4Kooe3nlePJ+y9+sYkLK0uM69HZ1VkgiOn02jxJSe43TQt0pd
|
||||
Co6LZjaURef9r0pJFZhFnnll+HwwjkYkFZDSViegCoYB1rmhqio7JOe2c3AIx69b
|
||||
Nm94u0SVJejXhbpguEAx2pmJiUa2b8Fbpzib/7bj+xGixI+AZC/IeKCvXPw4s3c0
|
||||
G8OvmFXHGjVJth4kNbjSB90QFK2NMMtGtHFaak8rahWWvN2JPMCjHfpSc4QhOZ17
|
||||
dCaz580QCRrmlnWN4q/p8vrBBpFzMkOU6+ww27JRw/uHEt49Q9YOy19XjXf6W99Q
|
||||
pXiMsO5hZo3wwFxWCBGnTqP4cMA/0wxyqFbnAQDj56SEGk5vGitBlxjGw1U5mQJS
|
||||
9rZvUejxsgpR3Fjonwf8CsVSh3B+w9HTQu56L3OtWVgfhY72OyQ04Gbt57BDCX7B
|
||||
k5EejeeRYJwdcDpmLDV+rjONnpHCtWT91v7Lxex9McOiwE01c85yrCbydYcPOYuy
|
||||
q5aWs3kIJX8/7O5YRc1Q3mOd8/JcBHvmVQtvKKQFOpjlpRb4uoBz+yaqvgiiWedC
|
||||
rG/NOyn2ZiDafBclI4+u1xt0J+0ebZzzNcRxDMa1C8uBGedDStc1TGsZfA2vHXnO
|
||||
cEKXUwA4b03alcj7eHc3zYCsaU+DDbY1DxUL8tWM3Kax3vsJk6Lu/D5ez/f8wo4P
|
||||
l/PexQZn0gByUjorxGq8dOlwduqJ+yyOmxOaUA/MjAgAgHnUHBneTNr6LXV6C+mX
|
||||
9hSCnb6uGNUH/m2SCoWsE8XSM/HRdYjyBTOXeYhA+v+TkHuH9GRzY3Jrh417jhbB
|
||||
NbWPSqbxaZwY7hF4RIPPxXn70nROgQ58hxUPEsLtGTTfY97RhrVqYWp6LfiEUFEl
|
||||
i0afQJSSYFTr6bpALhoRqd3qBd+ZGu8G+JQ44zYbiEIxdvFXISbhDminmZmR0n0O
|
||||
iE24WsKqvHFPBztKd/yXaA0L32Rwl48/BWxUBiKNPDjmXu9BVDmZlYzEg/9HzDvW
|
||||
lk0NSe9pN6JirmHmwpeO2e2X586XyjSM6PVTHSQQQibhf5iYzAILx3G5GGxZfVQ2
|
||||
tbQnTWF4IEJ1YmVuaWNrIDxtYXguYnViZW5pY2tAcGVyY29uYS5jb20+iHoEExEI
|
||||
ACIFAlGKRv4CGwMGCwkIBwMCBhUIAgkKCwQWAgMBAh4BAheAAAoJEEQzg2b5a/CS
|
||||
cw4A/1D494VgJAZFB2P+e+tLWtRYszDy6V3fDeX+v/0vv0+EAQCeP0TXA6Q92pE7
|
||||
3szZh7u7ddRkn494f52CJEDp8lb4XrkCDQRRikb+EAgAwP3I6mrBeDDLh2QbIS5G
|
||||
9U8hi1L7CAaFK7sLS2n/Uzy3PAQzyLTkMY4C0BmdnY+nEHFB99CiEiuQw2a8/NsM
|
||||
JCLPgOlkzklfAyR2Dl3EPbwbTpglpjUpjFf5IehWBFNh1VwYMSNw4jom362B+E1b
|
||||
uhtGSQLAtpTl/h2m6udAaVvjmSS6UMjH9nuX5gVfUUiJI2m6kSLfrENEVACmqXmF
|
||||
Js2LqD+GVGZh5nHnWC8ldfGPBYGNArwefzG4w9X4YpKOtOGfvjb3nqA3tBnypnXA
|
||||
KIUol96JdA7Bq2bDtROG2fJWXuTlqdOebo7IiBcKNi2QsIgH84DmUYaWssQCUbEI
|
||||
vwAEDQf9Gw1BgAIrpFS7Gxzt+iedIW0uk4m1/LZEcvgcoY/g+c4u+439kfZFMjIR
|
||||
h6ReGGT21fRkKt2l0hoMebpgR1idPIIP3AmIZ2tf/lsXe74oK3FTK5/Stq9wKd4r
|
||||
VGTVHQg10jZLBN6ksHMz+Kz3wnswfnfaABnvqzLcYrWrQjgusNCu6gKnoLv+X4bi
|
||||
5sQPxFFFTXacGQ+MLXqY2f7WKb2OeXW8ViMXNoHwiUh1+7CEuW+4+8yhykC1r7lu
|
||||
0taS++XuEeixoz149HlZcwuO7PIe92OZFaKCKJ+TfdegAGZlsEDdWn6JcDMDdjqw
|
||||
TYLUg98u32VZB2KwgzEDZL8O/DD/SohhBBgRCAAJBQJRikb+AhsMAAoJEEQzg2b5
|
||||
a/CSST8A/3qV/GXE0ABPxlAEF8yI5htqJHnSQDIEpr3QWt1ti+rDAQCRVFO7UHOS
|
||||
XmBM/2ByCbTTKokM+QivLrDB9Paf+xRmag==
|
||||
=zxkn
|
||||
-----END PGP PUBLIC KEY BLOCK-----
|
3
debian/watch
vendored
Normal file
3
debian/watch
vendored
Normal file
@ -0,0 +1,3 @@
|
||||
version=3
|
||||
opts=filenamemangle=s/.+\/v?(\d\S+)\.tar\.gz/mydumper-$1\.tar\.gz/ \
|
||||
https://github.com/maxbube/mydumper/releases .*/v?(\d\S+)\.tar\.gz
|
156
docs/CMakeLists.txt
Normal file
156
docs/CMakeLists.txt
Normal file
@ -0,0 +1,156 @@
|
||||
# Generate documentation in HTML and PDF format using Sphinx.
|
||||
|
||||
set(GENERATE_DOC TRUE)
|
||||
|
||||
# We use the Sphinx documentation generator to render HTML and manual
|
||||
# pages from the user and reference documentation in ReST format.
|
||||
find_package(Sphinx QUIET)
|
||||
if(NOT SPHINX_FOUND)
|
||||
message(WARNING "Unable to find Sphinx documentation generator")
|
||||
set(GENERATE_DOC FALSE)
|
||||
endif(NOT SPHINX_FOUND)
|
||||
|
||||
if(SPHINX_MAJOR_VERSION LESS 1)
|
||||
message(WARNING "Sphinx is older than v1.0, not building docs")
|
||||
set(GENERATE_DOC FALSE)
|
||||
endif(SPHINX_MAJOR_VERSION LESS 1)
|
||||
|
||||
if(GENERATE_DOC)
|
||||
# documentation tools
|
||||
set(SOURCE_BUILD_DIR "${CMAKE_CURRENT_SOURCE_DIR}/_build")
|
||||
# configured documentation tools and intermediate build results
|
||||
set(BINARY_BUILD_DIR "${CMAKE_CURRENT_BINARY_DIR}/_build")
|
||||
# static ReST documentation sources
|
||||
set(SOURCES_DIR "${CMAKE_CURRENT_BINARY_DIR}/_sources")
|
||||
# generated ReST documentation sources
|
||||
set(REF_SOURCES_DIR "${SOURCES_DIR}/reference")
|
||||
# master document with modules index
|
||||
set(REF_MASTER_DOC "modules")
|
||||
|
||||
# substitute variables in configuration and scripts
|
||||
foreach(file
|
||||
conf.py
|
||||
sources.cmake
|
||||
)
|
||||
configure_file(
|
||||
"${SOURCE_BUILD_DIR}/${file}.in"
|
||||
"${BINARY_BUILD_DIR}/${file}"
|
||||
@ONLY
|
||||
)
|
||||
endforeach(file)
|
||||
|
||||
set(CLEAN_FILES
|
||||
"${BINARY_BUILD_DIR}/html"
|
||||
)
|
||||
|
||||
add_custom_target(ALL
|
||||
DEPENDS "${REF_SOURCES_DIR}/${REF_MASTER_DOC}.rst"
|
||||
)
|
||||
|
||||
# Sphinx requires all sources in the same directory tree. As we wish
|
||||
# to include generated reference documention from the build tree, we
|
||||
# copy static ReST documents to the build tree before calling Sphinx.
|
||||
add_custom_target(doc_sources ALL
|
||||
"${CMAKE_COMMAND}" -P "${BINARY_BUILD_DIR}/sources.cmake"
|
||||
)
|
||||
list(APPEND CLEAN_FILES
|
||||
"${SOURCES_DIR}"
|
||||
)
|
||||
|
||||
# note the trailing slash to exclude directory name
|
||||
install(DIRECTORY "${SOURCES_DIR}/"
|
||||
DESTINATION "share/doc/mydumper"
|
||||
)
|
||||
|
||||
# Sphinx cache with pickled ReST documents
|
||||
set(SPHINX_CACHE_DIR "${CMAKE_CURRENT_BINARY_DIR}/_doctrees")
|
||||
# HTML output directory
|
||||
set(SPHINX_HTML_DIR "${CMAKE_CURRENT_BINARY_DIR}/html")
|
||||
|
||||
# This target builds HTML documentation using Sphinx.
|
||||
add_custom_target(doc_html ALL
|
||||
${SPHINX_EXECUTABLE}
|
||||
-q -b html
|
||||
-c "${BINARY_BUILD_DIR}"
|
||||
-d "${SPHINX_CACHE_DIR}"
|
||||
"${SOURCES_DIR}"
|
||||
"${SPHINX_HTML_DIR}"
|
||||
COMMENT "Building HTML documentation with Sphinx"
|
||||
)
|
||||
list(APPEND CLEAN_FILES
|
||||
"${SPHINX_CACHE_DIR}"
|
||||
"${SPHINX_HTML_DIR}"
|
||||
)
|
||||
add_dependencies(doc_html
|
||||
doc_sources
|
||||
)
|
||||
install(DIRECTORY "${SPHINX_HTML_DIR}"
|
||||
DESTINATION "share/doc/mydumper"
|
||||
)
|
||||
|
||||
# HTML output directory
|
||||
set(SPHINX_MAN_DIR "${CMAKE_CURRENT_BINARY_DIR}/man")
|
||||
# This target builds a manual page using Sphinx.
|
||||
|
||||
add_custom_target(doc_man ALL
|
||||
${SPHINX_EXECUTABLE}
|
||||
-q -b man
|
||||
-c "${BINARY_BUILD_DIR}"
|
||||
-d "${SPHINX_CACHE_DIR}"
|
||||
"${SOURCES_DIR}"
|
||||
"${SPHINX_MAN_DIR}"
|
||||
COMMENT "Building manual page with Sphinx"
|
||||
)
|
||||
list(APPEND CLEAN_FILES
|
||||
"${SPHINX_MAN_DIR}"
|
||||
)
|
||||
add_dependencies(doc_man
|
||||
doc_sources
|
||||
)
|
||||
# serialize Sphinx targets to avoid cache conflicts in parallel builds
|
||||
add_dependencies(doc_man
|
||||
doc_html
|
||||
)
|
||||
install(FILES "${SPHINX_MAN_DIR}/mydumper.1" "${SPHINX_MAN_DIR}/myloader.1"
|
||||
DESTINATION "share/man/man1"
|
||||
)
|
||||
|
||||
# This target builds PDF documentation using Sphinx and LaTeX.
|
||||
if(PDFLATEX_COMPILER)
|
||||
# PDF output directory
|
||||
set(SPHINX_PDF_DIR "${CMAKE_CURRENT_BINARY_DIR}/pdf")
|
||||
|
||||
add_custom_target(doc_pdf ALL
|
||||
${SPHINX_EXECUTABLE}
|
||||
-q -b latex
|
||||
-c "${BINARY_BUILD_DIR}"
|
||||
-d "${SPHINX_CACHE_DIR}"
|
||||
"${SOURCES_DIR}"
|
||||
"${SPHINX_PDF_DIR}"
|
||||
COMMENT "Building PDF documentation with Sphinx"
|
||||
)
|
||||
add_custom_command(TARGET doc_pdf POST_BUILD
|
||||
COMMAND ${CMAKE_MAKE_PROGRAM} LATEXOPTS=-interaction=batchmode
|
||||
WORKING_DIRECTORY "${SPHINX_PDF_DIR}"
|
||||
)
|
||||
list(APPEND CLEAN_FILES
|
||||
"${SPHINX_PDF_DIR}"
|
||||
)
|
||||
add_dependencies(doc_pdf
|
||||
doc_sources
|
||||
)
|
||||
# serialize Sphinx targets to avoid cache conflicts in parallel builds
|
||||
add_dependencies(doc_pdf
|
||||
doc_man
|
||||
)
|
||||
install(FILES "${SPHINX_PDF_DIR}/mydumper.pdf"
|
||||
DESTINATION "share/doc/mydumper"
|
||||
)
|
||||
endif(PDFLATEX_COMPILER)
|
||||
|
||||
# Add output directories to clean target.
|
||||
set_directory_properties(PROPERTIES
|
||||
ADDITIONAL_MAKE_CLEAN_FILES "${CLEAN_FILES}"
|
||||
)
|
||||
|
||||
endif(GENERATE_DOC)
|
218
docs/_build/conf.py.in
vendored
Normal file
218
docs/_build/conf.py.in
vendored
Normal file
@ -0,0 +1,218 @@
|
||||
# -*- coding: utf-8 -*-
|
||||
#
|
||||
# MySQL Data Dumper documentation build configuration file, created by
|
||||
# sphinx-quickstart on Tue Apr 26 11:44:25 2011.
|
||||
#
|
||||
# This file is execfile()d with the current directory set to its containing dir.
|
||||
#
|
||||
# Note that not all possible configuration values are present in this
|
||||
# autogenerated file.
|
||||
#
|
||||
# All configuration values have a default; values that are commented out
|
||||
# serve to show the default.
|
||||
|
||||
import sys, os
|
||||
|
||||
# If extensions (or modules to document with autodoc) are in another directory,
|
||||
# add these directories to sys.path here. If the directory is relative to the
|
||||
# documentation root, use os.path.abspath to make it absolute, like shown here.
|
||||
#sys.path.insert(0, os.path.abspath('.'))
|
||||
|
||||
# -- General configuration -----------------------------------------------------
|
||||
|
||||
# If your documentation needs a minimal Sphinx version, state it here.
|
||||
#needs_sphinx = '1.0'
|
||||
|
||||
# Add any Sphinx extension module names here, as strings. They can be extensions
|
||||
# coming with Sphinx (named 'sphinx.ext.*') or your custom ones.
|
||||
extensions = ['sphinx.ext.todo']
|
||||
|
||||
# Add any paths that contain templates here, relative to this directory.
|
||||
templates_path = ['@CMAKE_CURRENT_SOURCE_DIR@/_templates']
|
||||
|
||||
# The suffix of source filenames.
|
||||
source_suffix = '.rst'
|
||||
|
||||
# The encoding of source files.
|
||||
#source_encoding = 'utf-8-sig'
|
||||
|
||||
# The master toctree document.
|
||||
master_doc = 'index'
|
||||
|
||||
# General information about the project.
|
||||
project = u'@PROJECT_NAME@'
|
||||
copyright = u'2011, Andrew Hutchings'
|
||||
|
||||
# The version info for the project you're documenting, acts as replacement for
|
||||
# |version| and |release|, also used in various other places throughout the
|
||||
# built documents.
|
||||
#
|
||||
# The short X.Y version.
|
||||
version = '@VERSION@'
|
||||
# The full version, including alpha/beta/rc tags.
|
||||
release = '@VERSION@'
|
||||
|
||||
# The language for content autogenerated by Sphinx. Refer to documentation
|
||||
# for a list of supported languages.
|
||||
#language = None
|
||||
|
||||
# There are two options for replacing |today|: either, you set today to some
|
||||
# non-false value, then it is used:
|
||||
#today = ''
|
||||
# Else, today_fmt is used as the format for a strftime call.
|
||||
#today_fmt = '%B %d, %Y'
|
||||
|
||||
# List of patterns, relative to source directory, that match files and
|
||||
# directories to ignore when looking for source files.
|
||||
exclude_patterns = ['_build']
|
||||
|
||||
# The reST default role (used for this markup: `text`) to use for all documents.
|
||||
#default_role = None
|
||||
|
||||
# If true, '()' will be appended to :func: etc. cross-reference text.
|
||||
#add_function_parentheses = True
|
||||
|
||||
# If true, the current module name will be prepended to all description
|
||||
# unit titles (such as .. function::).
|
||||
#add_module_names = True
|
||||
|
||||
# If true, sectionauthor and moduleauthor directives will be shown in the
|
||||
# output. They are ignored by default.
|
||||
#show_authors = False
|
||||
|
||||
# The name of the Pygments (syntax highlighting) style to use.
|
||||
pygments_style = 'sphinx'
|
||||
|
||||
# A list of ignored prefixes for module index sorting.
|
||||
#modindex_common_prefix = []
|
||||
|
||||
|
||||
# -- Options for HTML output ---------------------------------------------------
|
||||
|
||||
# The theme to use for HTML and HTML Help pages. See the documentation for
|
||||
# a list of builtin themes.
|
||||
html_theme = 'default'
|
||||
|
||||
# Theme options are theme-specific and customize the look and feel of a theme
|
||||
# further. For a list of options available for each theme, see the
|
||||
# documentation.
|
||||
#html_theme_options = {}
|
||||
|
||||
# Add any paths that contain custom themes here, relative to this directory.
|
||||
#html_theme_path = []
|
||||
|
||||
# The name for this set of Sphinx documents. If None, it defaults to
|
||||
# "<project> v<release> documentation".
|
||||
#html_title = None
|
||||
|
||||
# A shorter title for the navigation bar. Default is the same as html_title.
|
||||
#html_short_title = None
|
||||
|
||||
# The name of an image file (relative to this directory) to place at the top
|
||||
# of the sidebar.
|
||||
#html_logo = None
|
||||
|
||||
# The name of an image file (within the static path) to use as favicon of the
|
||||
# docs. This file should be a Windows icon file (.ico) being 16x16 or 32x32
|
||||
# pixels large.
|
||||
#html_favicon = None
|
||||
|
||||
# Add any paths that contain custom static files (such as style sheets) here,
|
||||
# relative to this directory. They are copied after the builtin static files,
|
||||
# so a file named "default.css" will overwrite the builtin "default.css".
|
||||
html_static_path = ['@CMAKE_CURRENT_SOURCE_DIR@/_static']
|
||||
|
||||
# If not '', a 'Last updated on:' timestamp is inserted at every page bottom,
|
||||
# using the given strftime format.
|
||||
#html_last_updated_fmt = '%b %d, %Y'
|
||||
|
||||
# If true, SmartyPants will be used to convert quotes and dashes to
|
||||
# typographically correct entities.
|
||||
#html_use_smartypants = True
|
||||
|
||||
# Custom sidebar templates, maps document names to template names.
|
||||
#html_sidebars = {}
|
||||
|
||||
# Additional templates that should be rendered to pages, maps page names to
|
||||
# template names.
|
||||
#html_additional_pages = {}
|
||||
|
||||
# If false, no module index is generated.
|
||||
#html_domain_indices = True
|
||||
|
||||
# If false, no index is generated.
|
||||
#html_use_index = True
|
||||
|
||||
# If true, the index is split into individual pages for each letter.
|
||||
#html_split_index = False
|
||||
|
||||
# If true, links to the reST sources are added to the pages.
|
||||
#html_show_sourcelink = True
|
||||
|
||||
# If true, "Created using Sphinx" is shown in the HTML footer. Default is True.
|
||||
#html_show_sphinx = True
|
||||
|
||||
# If true, "(C) Copyright ..." is shown in the HTML footer. Default is True.
|
||||
#html_show_copyright = True
|
||||
|
||||
# If true, an OpenSearch description file will be output, and all pages will
|
||||
# contain a <link> tag referring to it. The value of this option must be the
|
||||
# base URL from which the finished HTML is served.
|
||||
#html_use_opensearch = ''
|
||||
|
||||
# This is the file name suffix for HTML files (e.g. ".xhtml").
|
||||
#html_file_suffix = None
|
||||
|
||||
# Output file base name for HTML help builder.
|
||||
htmlhelp_basename = 'MySQLDataDumperdoc'
|
||||
|
||||
|
||||
# -- Options for LaTeX output --------------------------------------------------
|
||||
|
||||
# The paper size ('letter' or 'a4').
|
||||
#latex_paper_size = 'letter'
|
||||
|
||||
# The font size ('10pt', '11pt' or '12pt').
|
||||
#latex_font_size = '10pt'
|
||||
|
||||
# Grouping the document tree into LaTeX files. List of tuples
|
||||
# (source start file, target name, title, author, documentclass [howto/manual]).
|
||||
latex_documents = [
|
||||
('index', 'MySQLDataDumper.tex', u'@PROJECT_NAME@ Documentation',
|
||||
u'Andrew Hutchings', 'manual'),
|
||||
]
|
||||
|
||||
# The name of an image file (relative to this directory) to place at the top of
|
||||
# the title page.
|
||||
#latex_logo = None
|
||||
|
||||
# For "manual" documents, if this is true, then toplevel headings are parts,
|
||||
# not chapters.
|
||||
#latex_use_parts = False
|
||||
|
||||
# If true, show page references after internal links.
|
||||
#latex_show_pagerefs = False
|
||||
|
||||
# If true, show URL addresses after external links.
|
||||
#latex_show_urls = False
|
||||
|
||||
# Additional stuff for the LaTeX preamble.
|
||||
#latex_preamble = ''
|
||||
|
||||
# Documents to append as an appendix to all manuals.
|
||||
#latex_appendices = []
|
||||
|
||||
# If false, no module index is generated.
|
||||
#latex_domain_indices = True
|
||||
|
||||
|
||||
# -- Options for manual page output --------------------------------------------
|
||||
|
||||
# One entry per manual page. List of tuples
|
||||
# (source start file, name, description, authors, manual section).
|
||||
man_pages = [
|
||||
('mydumper_usage', 'mydumper', u'@PROGRAM_DESC@',
|
||||
[u'Andrew Hutchings'], 1),
|
||||
('myloader_usage', 'myloader', u'@PROGRAM_DESC@',
|
||||
[u'Andrew Hutchings'], 1)
|
||||
]
|
16
docs/_build/sources.cmake.in
vendored
Normal file
16
docs/_build/sources.cmake.in
vendored
Normal file
@ -0,0 +1,16 @@
|
||||
# This script recursively copies all ReST documents from the source directory to
|
||||
# the binary directory. CMAKE_CURRENT_SOURCE_DIR and SOURCES_DIR are substituted
|
||||
# upon the cmake stage. The script is executed upon the make stage to ensure
|
||||
# that the binary sources directory is always up to date.
|
||||
|
||||
file(GLOB SOURCES
|
||||
RELATIVE "@CMAKE_CURRENT_SOURCE_DIR@"
|
||||
"@CMAKE_CURRENT_SOURCE_DIR@/*.rst"
|
||||
)
|
||||
foreach(source ${SOURCES})
|
||||
configure_file(
|
||||
"@CMAKE_CURRENT_SOURCE_DIR@/${source}"
|
||||
"@SOURCES_DIR@/${source}"
|
||||
COPYONLY
|
||||
)
|
||||
endforeach(source)
|
9
docs/authors.rst
Normal file
9
docs/authors.rst
Normal file
@ -0,0 +1,9 @@
|
||||
Authors
|
||||
=======
|
||||
|
||||
The code for mydumper has been written by the following people:
|
||||
|
||||
* `Domas Mituzas <http://dom.as/>`_, Facebook ( domas at fb dot com )
|
||||
* `Andrew Hutchings <http://www.linuxjedi.co.uk>`_, SkySQL ( andrew at skysql dot com )
|
||||
* `Mark Leith <http://www.markleith.co.uk/>`_, Oracle Corporation ( mark dot leith at oracle dot com )
|
||||
* `Max Bubenick <http://www.bube.com.ar>`_, Percona RDBA ( max dot bubenick at percona dot com )
|
70
docs/compiling.rst
Normal file
70
docs/compiling.rst
Normal file
@ -0,0 +1,70 @@
|
||||
Compiling
|
||||
=========
|
||||
|
||||
Requirements
|
||||
------------
|
||||
|
||||
mydumper requires the following before it can be compiled:
|
||||
|
||||
* `CMake <http://www.cmake.org/>`_
|
||||
* `Glib2 <http://www.gtk.org/index.php>`_ (with development packages)
|
||||
* `PCRE <http://www.pcre.org/>`_ (with development packages)
|
||||
* `MySQL <http://www.mysql.com/>`_ client libraries (with development packages)
|
||||
|
||||
Additionally the following packages are optional:
|
||||
|
||||
* `python-sphinx <http://sphinx.pocoo.org/>`_ (for documentation)
|
||||
|
||||
Ubuntu/Debian
|
||||
^^^^^^^^^^^^^
|
||||
|
||||
.. code-block:: bash
|
||||
|
||||
apt-get install libglib2.0-dev libmysqlclient15-dev zlib1g-dev libpcre3-dev
|
||||
|
||||
Fedora/Redhat/CentOS
|
||||
^^^^^^^^^^^^^^^^^^^^
|
||||
|
||||
.. code-block:: bash
|
||||
|
||||
yum install glib2-devel mysql-devel zlib-devel pcre-devel
|
||||
|
||||
OpenSUSE
|
||||
^^^^^^^^
|
||||
|
||||
.. code-block:: bash
|
||||
|
||||
zypper install glib2-devel libmysqlclient-devel pcre-devel zlib-devel
|
||||
|
||||
Mac OSX
|
||||
^^^^^^^
|
||||
|
||||
.. code-block:: bash
|
||||
|
||||
port install glib2 mysql5 pcre
|
||||
|
||||
CMake
|
||||
-----
|
||||
|
||||
CMake is used for mydumper's build system and is executed as follows::
|
||||
|
||||
cmake .
|
||||
make
|
||||
|
||||
You can optionally provide parameters for CMake, the possible options are:
|
||||
|
||||
* ``-DMYSQL_CONFIG=/path/to/mysql_config`` - The path and filename for the mysql_config executable
|
||||
* ``-DCMAKE_INSTALL_PREFIX=/install/path`` - The path where mydumper should be installed
|
||||
|
||||
Documentation
|
||||
-------------
|
||||
|
||||
If you wish to just compile the documentation you can do so with::
|
||||
|
||||
cmake .
|
||||
make doc_html
|
||||
|
||||
or for a man page output::
|
||||
|
||||
cmake .
|
||||
make doc_man
|
37
docs/examples.rst
Normal file
37
docs/examples.rst
Normal file
@ -0,0 +1,37 @@
|
||||
Examples
|
||||
========
|
||||
|
||||
Simple Usage
|
||||
------------
|
||||
Just running :program:`mydumper` without any options will try to connect to a
|
||||
server using the default socket path. It will then dump the tables from all
|
||||
databases using 4 worker threads.
|
||||
|
||||
Regex
|
||||
-----
|
||||
To use :program:`mydumper`'s regex feature simply use the
|
||||
:option:`--regex <mydumper --regex>` option. In the following example mydumper
|
||||
will ignore the ``test`` and ``mysql`` databases::
|
||||
|
||||
mydumper --regex '^(?!(mysql\.|test\.))'
|
||||
|
||||
Restoring a dump
|
||||
----------------
|
||||
Mydumper now include myloader which is a multi-threaded restoration tool. To
|
||||
use myloader with a mydumper dump you simply need to pass it the directory of
|
||||
the dump along with a user capable of restoring the schemas and data. As an
|
||||
example the following will restore a dump overwriting any existing tables::
|
||||
|
||||
myloader --directory=export-20110614-094953 --overwrite-tables --user=root
|
||||
|
||||
Daemon mode
|
||||
-----------
|
||||
Mydumper has a daemon mode which will snapshot the dump data every so often
|
||||
whilst continuously retreiving the binary log files. This gives a continuous
|
||||
consistent backup right up to the point where the database server fails. To use
|
||||
this you simply need to use the :option:`--daemon <mydumper --daemon>` option.
|
||||
|
||||
In the following example mydumper will use daemon mode, creating a snapshot
|
||||
every half an hour and log to an output file::
|
||||
|
||||
mydumper --daemon --snapshot-interval=30 --logfile=dump.log
|
61
docs/files.rst
Normal file
61
docs/files.rst
Normal file
@ -0,0 +1,61 @@
|
||||
Output Files
|
||||
============
|
||||
|
||||
mydumper generates several files during the generation of the dump. Many of
|
||||
these are for the table data itself since every table has at least one file.
|
||||
|
||||
Metadata
|
||||
--------
|
||||
When a dump is executed a file called ``metadata.partial`` is created in the output
|
||||
directory and is renamed to ``metadata`` when mydumper finish without error.
|
||||
This contains the start and end time of the dump as well as the
|
||||
master binary log positions if applicable.
|
||||
|
||||
This is an example of the content of this file::
|
||||
|
||||
Started dump at: 2011-05-05 13:57:17
|
||||
SHOW MASTER STATUS:
|
||||
Log: linuxjedi-laptop-bin.000001
|
||||
Pos: 106
|
||||
|
||||
Finished dump at: 2011-05-05 13:57:17
|
||||
|
||||
Table Data
|
||||
----------
|
||||
The data from every table is written into a separate file, also if the
|
||||
:option:`--rows <mydumper --rows>` option is used then each chunk of table will
|
||||
be in a separate file. The file names for this are in the format::
|
||||
|
||||
database.table.sql(.gz)
|
||||
|
||||
or if chunked::
|
||||
|
||||
database.table.chunk.sql(.gz)
|
||||
|
||||
Where 'chunk' is a number padded with up to 5 zeros.
|
||||
|
||||
Table Schemas
|
||||
-------------
|
||||
When the :option:`--schemas <mydumper --schemas>` option is used mydumper will
|
||||
create a file for the schema of every table it is writing data for. The files
|
||||
for this are in the following format::
|
||||
|
||||
database.table-schema.sql(.gz)
|
||||
|
||||
Binary Logs
|
||||
-----------
|
||||
Binary logs are retrieved when :option:`--binlogs <mydumper --binlogs>` option
|
||||
has been set. This will store them in the ``binlog_snapshot/`` sub-directory
|
||||
inside the dump directory.
|
||||
|
||||
The binary log files have the same filename as the MySQL server that supplies them and will also have a .gz on the end if they are compressed.
|
||||
|
||||
Daemon mode
|
||||
-----------
|
||||
Daemon mode does things a little differently. There are the directories ``0``
|
||||
and ``1`` inside the dump directory. These alternate when dumping so that if
|
||||
mydumper fails for any reason there is still a good snapshot. When a snapshot
|
||||
dump is complete the ``last_dump`` symlink is updated to point to that dump.
|
||||
|
||||
If binary logging is enabled mydumper will connect as if it is a slave server
|
||||
and constantly retreives the binary logs into the ``binlogs`` subdirectory.
|
25
docs/index.rst
Normal file
25
docs/index.rst
Normal file
@ -0,0 +1,25 @@
|
||||
.. MySQL Data Dumper documentation master file
|
||||
You can adapt this file completely to your liking, but it should at least
|
||||
contain the root `toctree` directive.
|
||||
|
||||
Welcome to MySQL Data Dumper's documentation!
|
||||
=============================================
|
||||
|
||||
Contents:
|
||||
|
||||
.. toctree::
|
||||
:maxdepth: 2
|
||||
|
||||
authors
|
||||
compiling
|
||||
mydumper_usage
|
||||
myloader_usage
|
||||
files
|
||||
examples
|
||||
|
||||
Indices and tables
|
||||
==================
|
||||
|
||||
* :ref:`genindex`
|
||||
* :ref:`search`
|
||||
|
214
docs/mydumper_usage.rst
Normal file
214
docs/mydumper_usage.rst
Normal file
@ -0,0 +1,214 @@
|
||||
Mydumper Usage
|
||||
==============
|
||||
|
||||
Synopsis
|
||||
--------
|
||||
|
||||
:program:`mydumper` [:ref:`OPTIONS <mydumper-options-label>`]
|
||||
|
||||
Description
|
||||
-----------
|
||||
|
||||
:program:`mydumper` is a tool used for backing up MySQL database servers much
|
||||
faster than the mysqldump tool distributed with MySQL. It also has the
|
||||
capability to retrieve the binary logs from the remote server at the same time
|
||||
as the dump itself. The advantages of mydumper are:
|
||||
|
||||
* Parallelism (hence, speed) and performance (avoids expensive character set conversion routines, efficient code overall)
|
||||
* Easier to manage output (separate files for tables, dump metadata, etc, easy to view/parse data)
|
||||
* Consistency - maintains snapshot across all threads, provides accurate master and slave log positions, etc
|
||||
* Manageability - supports PCRE for specifying database and tables inclusions and exclusions
|
||||
|
||||
.. _mydumper-options-label:
|
||||
|
||||
Options
|
||||
-------
|
||||
|
||||
The :program:`mydumper` tool has several available options:
|
||||
|
||||
.. program:: mydumper
|
||||
|
||||
.. option:: --help, -?
|
||||
|
||||
Show help text
|
||||
|
||||
.. option:: --defaults-file
|
||||
|
||||
Use the given option file. If the file does not exist or is otherwise inaccessible, no failure occurs
|
||||
|
||||
.. option:: --host, -h
|
||||
|
||||
Hostname of MySQL server to connect to (default localhost)
|
||||
|
||||
.. option:: --user, -u
|
||||
|
||||
MySQL username with the correct privileges to execute the dump
|
||||
|
||||
.. option:: --password, -p
|
||||
|
||||
The corresponding password for the MySQL user
|
||||
|
||||
.. option:: --port, -P
|
||||
|
||||
The port for the MySQL connection.
|
||||
|
||||
.. note::
|
||||
|
||||
For localhost TCP connections use 127.0.0.1 for :option:`--host`.
|
||||
|
||||
.. option:: --socket, -S
|
||||
|
||||
The UNIX domain socket file to use for the connection
|
||||
|
||||
.. option:: --database, -B
|
||||
|
||||
Database to dump
|
||||
|
||||
.. option:: --tables-list, -T
|
||||
|
||||
A comma separated list of tables to dump
|
||||
|
||||
.. option:: --threads, -t
|
||||
|
||||
The number of threads to use for dumping data, default is 4
|
||||
|
||||
.. note::
|
||||
|
||||
Other threads are used in mydumper, this option does not control these
|
||||
|
||||
.. option:: --outputdir, -o
|
||||
|
||||
Output directory name, default is export-YYYYMMDD-HHMMSS
|
||||
|
||||
.. option:: --statement-size, -s
|
||||
|
||||
The maximum size for an insert statement before breaking into a new
|
||||
statement, default 1,000,000 bytes
|
||||
|
||||
.. option:: --rows, -r
|
||||
|
||||
Split table into chunks of this many rows, default unlimited
|
||||
|
||||
.. option:: --compress, -c
|
||||
|
||||
Compress the output files
|
||||
|
||||
.. option:: --compress-input, -C
|
||||
|
||||
Use client protocol compression for connections to the MySQL server
|
||||
|
||||
.. option:: --build-empty-files, -e
|
||||
|
||||
Create empty dump files if there is no data to dump
|
||||
|
||||
.. option:: --regex, -x
|
||||
|
||||
A regular expression to match against database and table
|
||||
|
||||
.. option:: --omit-from-file, -O
|
||||
|
||||
File containing a list of database.table entries to skip, one per line; the
|
||||
skipped entries have precedence over patterns specified by the regex option
|
||||
|
||||
.. option:: --ignore-engines, -i
|
||||
|
||||
Comma separated list of storage engines to ignore
|
||||
|
||||
.. option:: --no-schemas, -m
|
||||
|
||||
Do not dump schemas with the data
|
||||
|
||||
.. option:: --no-data, -d
|
||||
|
||||
Do not dump table data
|
||||
|
||||
.. option:: --triggers, -G
|
||||
|
||||
Dump triggers
|
||||
|
||||
.. option:: --events, -E
|
||||
|
||||
Dump events
|
||||
|
||||
.. option:: --routines, -R
|
||||
|
||||
Dump stored procedures and functions
|
||||
|
||||
.. option:: --no-views, -W
|
||||
|
||||
Do not dump views
|
||||
|
||||
.. option:: --long-query-guard, -l
|
||||
|
||||
Timeout for long query execution in seconds, default 60
|
||||
|
||||
.. option:: --kill-long-queries, -K
|
||||
|
||||
Kill long running queries instead of aborting the dump
|
||||
|
||||
.. option:: --version, -V
|
||||
|
||||
Show the program version and exit
|
||||
|
||||
.. option:: --verbose, -v
|
||||
|
||||
The verbosity of messages. 0 = silent, 1 = errors, 2 = warnings, 3 = info.
|
||||
Default is 2.
|
||||
|
||||
.. option:: --binlogs, -b
|
||||
|
||||
Get the binlogs from the server as well as the dump files (You need to compile with -DWITH_BINLOG=ON)
|
||||
|
||||
.. option:: --daemon, -D
|
||||
|
||||
Enable daemon mode
|
||||
|
||||
.. option:: --snapshot-interval, -I
|
||||
|
||||
Interval between each dump snapshot (in minutes), requires
|
||||
:option:`--daemon`, default 60 (minutes)
|
||||
|
||||
.. option:: --logfile, -L
|
||||
|
||||
A file to log mydumper output to instead of console output. Useful for
|
||||
daemon mode.
|
||||
|
||||
.. option:: --no-locks, -k
|
||||
|
||||
Do not execute the temporary shared read lock.
|
||||
|
||||
.. warning::
|
||||
|
||||
This will cause inconsistent backups.
|
||||
|
||||
.. option:: --no-backup-locks
|
||||
|
||||
Do not use Percona Backup Locks
|
||||
|
||||
.. option:: --[skip-]tz-utc
|
||||
|
||||
SET TIME_ZONE='+00:00' at top of dump to allow dumping of TIMESTAMP data
|
||||
when a server has data in different time zones or data is being moved
|
||||
between servers with different time zones, defaults to on use --skip-tz-utc
|
||||
to disable.
|
||||
|
||||
.. option:: --less-locking
|
||||
|
||||
Minimize locking time on InnoDB tables grabbing a LOCK TABLE ... READ
|
||||
on all non-innodb tables.
|
||||
|
||||
.. option:: --chunk-filesize -F
|
||||
|
||||
Split tables into chunks of this output file size. This value is in MB
|
||||
|
||||
.. option:: --success-on-1146
|
||||
|
||||
Not increment error count and Warning instead of Critical in case of table doesn't exist
|
||||
|
||||
.. option:: --use-savepoints
|
||||
|
||||
Use savepoints to reduce metadata locking issues, needs SUPER privilege
|
||||
|
||||
.. option:: --complete-insert
|
||||
|
||||
Use complete INSERT statements that include column names.
|
103
docs/myloader_usage.rst
Normal file
103
docs/myloader_usage.rst
Normal file
@ -0,0 +1,103 @@
|
||||
Myloader Usage
|
||||
==============
|
||||
|
||||
Synopsis
|
||||
--------
|
||||
|
||||
:program:`myloader` :option:`--directory <myloader --directory>` = /path/to/mydumper/backup [:ref:`OPTIONS <myloader-options-label>`]
|
||||
|
||||
Description
|
||||
-----------
|
||||
|
||||
:program:`myloader` is a tool used for multi-threaded restoration of mydumper
|
||||
backups.
|
||||
|
||||
.. _myloader-options-label:
|
||||
|
||||
Options
|
||||
-------
|
||||
|
||||
The :program:`myloader` tool has several available options:
|
||||
|
||||
.. program:: myloader
|
||||
|
||||
.. option:: --help, -?
|
||||
|
||||
Show help text
|
||||
|
||||
.. option:: --defaults-file
|
||||
|
||||
Use the given option file. If the file does not exist or is otherwise inaccessible, no failure occurs
|
||||
|
||||
.. option:: --host, -h
|
||||
|
||||
Hostname of MySQL server to connect to (default localhost)
|
||||
|
||||
.. option:: --user, -u
|
||||
|
||||
MySQL username with the correct privileges to execute the restoration
|
||||
|
||||
.. option:: --password, -p
|
||||
|
||||
The corresponding password for the MySQL user
|
||||
|
||||
.. option:: --port, -P
|
||||
|
||||
The port for the MySQL connection.
|
||||
|
||||
.. note::
|
||||
|
||||
For localhost TCP connections use 127.0.0.1 for :option:`--host`.
|
||||
|
||||
.. option:: --socket, -S
|
||||
|
||||
The UNIX domain socket file to use for the connection
|
||||
|
||||
.. option:: --threads, -t
|
||||
|
||||
The number of threads to use for restoring data, default is 4
|
||||
|
||||
.. option:: --version, -V
|
||||
|
||||
Show the program version and exit
|
||||
|
||||
.. option:: --compress-protocol, -C
|
||||
|
||||
Use client protocol compression for connections to the MySQL server
|
||||
|
||||
.. option:: --directory, -d
|
||||
|
||||
The directory of the mydumper backup to restore
|
||||
|
||||
.. option:: --database, -B
|
||||
|
||||
An alternative database to load the dump into
|
||||
|
||||
.. note::
|
||||
|
||||
For use with single database dumps. When using with multi-database dumps
|
||||
that have duplicate table names in more than one database it may cause
|
||||
errors. Alternatively this scenario may give unpredictable results with
|
||||
:option:`--overwrite-tables`.
|
||||
|
||||
.. option:: --source-db, -s
|
||||
|
||||
Database to restore, useful in combination with --database
|
||||
|
||||
.. option:: --queries-per-transaction, -q
|
||||
|
||||
Number of INSERT queries to execute per transaction during restore, default
|
||||
is 1000.
|
||||
|
||||
.. option:: --overwrite-tables, -o
|
||||
|
||||
Drop any existing tables when restoring schemas
|
||||
|
||||
.. option:: --enable-binlog, -e
|
||||
|
||||
Log the data loading in the MySQL binary log if enabled (off by default)
|
||||
|
||||
.. option:: --verbose, -v
|
||||
|
||||
The verbosity of messages. 0 = silent, 1 = errors, 2 = warnings, 3 = info.
|
||||
Default is 2.
|
128
g_unix_signal.c
Normal file
128
g_unix_signal.c
Normal file
@ -0,0 +1,128 @@
|
||||
#define _POSIX_SOURCE
|
||||
#include <signal.h>
|
||||
#include <glib.h>
|
||||
|
||||
static GPtrArray *signal_data = NULL;
|
||||
|
||||
typedef struct _GUnixSignalData {
|
||||
guint source_id;
|
||||
GMainContext *context;
|
||||
gboolean triggered;
|
||||
gint signum;
|
||||
} GUnixSignalData;
|
||||
|
||||
typedef struct _GUnixSignalSource {
|
||||
GSource source;
|
||||
GUnixSignalData *data;
|
||||
} GUnixSignalSource;
|
||||
|
||||
static inline GUnixSignalData* get_signal_data(guint index)
|
||||
{
|
||||
return (GUnixSignalData*)g_ptr_array_index(signal_data, index);
|
||||
}
|
||||
|
||||
static void handler(gint signum) {
|
||||
g_assert(signal_data != NULL);
|
||||
guint i;
|
||||
for (i = 0; i < signal_data->len; ++i)
|
||||
if (get_signal_data(i)->signum == signum)
|
||||
get_signal_data(i)->triggered = TRUE;
|
||||
|
||||
struct sigaction action;
|
||||
action.sa_handler= handler;
|
||||
sigemptyset (&action.sa_mask);
|
||||
action.sa_flags = 0;
|
||||
sigaction(signum, &action, NULL);
|
||||
}
|
||||
|
||||
static gboolean check(GSource *source)
|
||||
{
|
||||
GUnixSignalSource *signal_source = (GUnixSignalSource*) source;
|
||||
return signal_source->data->triggered;
|
||||
}
|
||||
|
||||
static gboolean prepare(GSource *source, gint *timeout_)
|
||||
{
|
||||
GUnixSignalSource *signal_source = (GUnixSignalSource*) source;
|
||||
if (signal_source->data->context == NULL) {
|
||||
g_main_context_ref(signal_source->data->context = g_source_get_context(source));
|
||||
signal_source->data->source_id = g_source_get_id(source);
|
||||
}
|
||||
|
||||
*timeout_ = -1;
|
||||
return signal_source->data->triggered;
|
||||
}
|
||||
|
||||
static gboolean dispatch(GSource *source, GSourceFunc callback, gpointer user_data)
|
||||
{
|
||||
GUnixSignalSource *signal_source = (GUnixSignalSource*) source;
|
||||
signal_source->data->triggered = FALSE;
|
||||
return callback(user_data) ? TRUE : FALSE;
|
||||
}
|
||||
static void finalize(GSource *source)
|
||||
{
|
||||
GUnixSignalSource *signal_source = (GUnixSignalSource*) source;
|
||||
|
||||
struct sigaction action;
|
||||
action.sa_handler= NULL;
|
||||
sigemptyset (&action.sa_mask);
|
||||
action.sa_flags = 0;
|
||||
|
||||
sigaction(signal_source->data->signum, &action, NULL);
|
||||
g_main_context_unref(signal_source->data->context);
|
||||
g_ptr_array_remove_fast(signal_data, signal_source->data);
|
||||
if (signal_data->len == 0)
|
||||
signal_data = (GPtrArray*) g_ptr_array_free(signal_data, TRUE);
|
||||
g_free(signal_source->data);
|
||||
|
||||
}
|
||||
static GSourceFuncs SourceFuncs =
|
||||
{
|
||||
.prepare = prepare,
|
||||
.check = check,
|
||||
.dispatch = dispatch,
|
||||
.finalize = finalize,
|
||||
.closure_callback = NULL, .closure_marshal = NULL
|
||||
};
|
||||
|
||||
static void g_unix_signal_source_init(GSource *source, gint signum)
|
||||
{
|
||||
GUnixSignalSource *signal_source = (GUnixSignalSource *) source;
|
||||
signal_source->data = g_new(GUnixSignalData, 1);
|
||||
signal_source->data->triggered = FALSE;
|
||||
signal_source->data->signum = signum;
|
||||
signal_source->data->context = NULL;
|
||||
|
||||
if (signal_data == NULL)
|
||||
signal_data = g_ptr_array_new();
|
||||
g_ptr_array_add(signal_data, signal_source->data);
|
||||
}
|
||||
|
||||
GSource *g_unix_signal_source_new(gint signum)
|
||||
{
|
||||
GSource *source = g_source_new(&SourceFuncs, sizeof(GUnixSignalSource));
|
||||
g_unix_signal_source_init(source, signum);
|
||||
struct sigaction action;
|
||||
action.sa_handler= handler;
|
||||
sigemptyset (&action.sa_mask);
|
||||
action.sa_flags = 0;
|
||||
sigaction(signum, &action, NULL);
|
||||
return source;
|
||||
}
|
||||
|
||||
guint g_unix_signal_add_full(gint priority, gint signum, GSourceFunc function, gpointer data, GDestroyNotify notify)
|
||||
{
|
||||
g_return_val_if_fail(function != NULL, 0);
|
||||
GSource *source = g_unix_signal_source_new(signum);
|
||||
if (priority != G_PRIORITY_DEFAULT)
|
||||
g_source_set_priority (source, priority);
|
||||
g_source_set_callback(source, function, data, notify);
|
||||
guint id = g_source_attach(source, NULL);
|
||||
g_source_unref(source);
|
||||
return id;
|
||||
}
|
||||
|
||||
guint g_unix_signal_add(gint signum, GSourceFunc function, gpointer data)
|
||||
{
|
||||
return g_unix_signal_add_full(G_PRIORITY_DEFAULT, signum, function, data, NULL);
|
||||
}
|
10
g_unix_signal.h
Normal file
10
g_unix_signal.h
Normal file
@ -0,0 +1,10 @@
|
||||
#ifndef G_UNIX_SIGNAL_H
|
||||
#define G_UNIX_SIGNAL_H
|
||||
|
||||
#include <glib.h>
|
||||
|
||||
GSource *g_unix_signal_source_new(gint signum);
|
||||
guint g_unix_signal_add(gint signum, GSourceFunc function, gpointer data);
|
||||
guint g_unix_signal_add_full(gint priority, gint signum, GSourceFunc function, gpointer data, GDestroyNotify notify);
|
||||
|
||||
#endif /* G_UNIX_SIGNAL_H */
|
10
getPassword.c
Normal file
10
getPassword.c
Normal file
@ -0,0 +1,10 @@
|
||||
#include <stdio.h>
|
||||
#include <unistd.h>
|
||||
#include "getPassword.h"
|
||||
|
||||
char* passwordPrompt(void) {
|
||||
char *password;
|
||||
password = getpass("Enter MySQL Password: ");
|
||||
|
||||
return password;
|
||||
}
|
4
getPassword.h
Normal file
4
getPassword.h
Normal file
@ -0,0 +1,4 @@
|
||||
#ifndef GET_PASSWORD_PROMPT
|
||||
#define GET_PASSWORD_PROMPT
|
||||
char* passwordPrompt(void);
|
||||
#endif
|
3153
mydumper.c
Normal file
3153
mydumper.c
Normal file
File diff suppressed because it is too large
Load Diff
100
mydumper.h
Normal file
100
mydumper.h
Normal file
@ -0,0 +1,100 @@
|
||||
/*
|
||||
This program is free software: you can redistribute it and/or modify
|
||||
it under the terms of the GNU General Public License as published by
|
||||
the Free Software Foundation, either version 3 of the License, or
|
||||
(at your option) any later version.
|
||||
|
||||
This program is distributed in the hope that it will be useful,
|
||||
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
GNU General Public License for more details.
|
||||
|
||||
You should have received a copy of the GNU General Public License
|
||||
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||
|
||||
Authors: Domas Mituzas, Facebook ( domas at fb dot com )
|
||||
Mark Leith, Oracle Corporation (mark dot leith at oracle dot com)
|
||||
Andrew Hutchings, SkySQL (andrew at skysql dot com)
|
||||
Max Bubenick, Percona RDBA (max dot bubenick at percona dot com)
|
||||
|
||||
*/
|
||||
|
||||
#ifndef _mydumper_h
|
||||
#define _mydumper_h
|
||||
|
||||
enum job_type { JOB_SHUTDOWN, JOB_RESTORE, JOB_DUMP, JOB_DUMP_NON_INNODB, JOB_SCHEMA, JOB_VIEW, JOB_TRIGGERS, JOB_SCHEMA_POST, JOB_BINLOG, JOB_LOCK_DUMP_NON_INNODB };
|
||||
|
||||
struct configuration {
|
||||
char use_any_index;
|
||||
GAsyncQueue* queue;
|
||||
GAsyncQueue* queue_less_locking;
|
||||
GAsyncQueue* ready;
|
||||
GAsyncQueue* ready_less_locking;
|
||||
GAsyncQueue* unlock_tables;
|
||||
GMutex* mutex;
|
||||
int done;
|
||||
};
|
||||
|
||||
struct thread_data {
|
||||
struct configuration *conf;
|
||||
guint thread_id;
|
||||
};
|
||||
|
||||
struct job {
|
||||
enum job_type type;
|
||||
void *job_data;
|
||||
struct configuration *conf;
|
||||
};
|
||||
|
||||
struct table_job {
|
||||
char *database;
|
||||
char *table;
|
||||
char *filename;
|
||||
char *where;
|
||||
};
|
||||
|
||||
struct tables_job {
|
||||
GList* table_job_list;
|
||||
};
|
||||
|
||||
struct schema_job {
|
||||
char *database;
|
||||
char *table;
|
||||
char *filename;
|
||||
};
|
||||
|
||||
struct view_job {
|
||||
char *database;
|
||||
char *table;
|
||||
char *filename;
|
||||
char *filename2;
|
||||
};
|
||||
|
||||
struct schema_post_job {
|
||||
char *database;
|
||||
char *filename;
|
||||
};
|
||||
|
||||
struct restore_job {
|
||||
char *database;
|
||||
char *table;
|
||||
char *filename;
|
||||
};
|
||||
|
||||
struct binlog_job {
|
||||
char *filename;
|
||||
guint64 start_position;
|
||||
guint64 stop_position;
|
||||
};
|
||||
|
||||
struct db_table {
|
||||
char* database;
|
||||
char* table;
|
||||
guint64 datalength;
|
||||
};
|
||||
|
||||
struct schema_post {
|
||||
char* database;
|
||||
};
|
||||
|
||||
#endif
|
Binary file not shown.
@ -1 +0,0 @@
|
||||
a9d7dbd013ae57389085202f035ea0d7d309fe0d
|
Binary file not shown.
@ -1 +0,0 @@
|
||||
6094bf22505140f4019b56f70a009ed7d5bb7fee
|
587
myloader.c
Normal file
587
myloader.c
Normal file
@ -0,0 +1,587 @@
|
||||
/*
|
||||
This program is free software: you can redistribute it and/or modify
|
||||
it under the terms of the GNU General Public License as published by
|
||||
the Free Software Foundation, either version 3 of the License, or
|
||||
(at your option) any later version.
|
||||
|
||||
This program is distributed in the hope that it will be useful,
|
||||
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
GNU General Public License for more details.
|
||||
|
||||
You should have received a copy of the GNU General Public License
|
||||
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||
|
||||
Authors: Andrew Hutchings, SkySQL (andrew at skysql dot com)
|
||||
*/
|
||||
|
||||
#define _LARGEFILE64_SOURCE
|
||||
#define _FILE_OFFSET_BITS 64
|
||||
|
||||
#include <mysql.h>
|
||||
|
||||
#if defined MARIADB_CLIENT_VERSION_STR && !defined MYSQL_SERVER_VERSION
|
||||
#define MYSQL_SERVER_VERSION MARIADB_CLIENT_VERSION_STR
|
||||
#endif
|
||||
|
||||
#include <unistd.h>
|
||||
#include <stdio.h>
|
||||
#include <string.h>
|
||||
#include <glib.h>
|
||||
#include <glib/gstdio.h>
|
||||
#include <stdlib.h>
|
||||
#include <stdarg.h>
|
||||
#include <errno.h>
|
||||
#include <zlib.h>
|
||||
#include "common.h"
|
||||
#include "myloader.h"
|
||||
#include "connection.h"
|
||||
#include "config.h"
|
||||
#include "getPassword.h"
|
||||
|
||||
guint commit_count= 1000;
|
||||
gchar *directory= NULL;
|
||||
gboolean overwrite_tables= FALSE;
|
||||
gboolean enable_binlog= FALSE;
|
||||
gchar *source_db= NULL;
|
||||
static GMutex *init_mutex= NULL;
|
||||
|
||||
guint errors= 0;
|
||||
|
||||
gboolean read_data(FILE *file, gboolean is_compressed, GString *data, gboolean *eof);
|
||||
void restore_data(MYSQL *conn, char *database, char *table, const char *filename, gboolean is_schema, gboolean need_use);
|
||||
void *process_queue(struct thread_data *td);
|
||||
void add_table(const gchar* filename, struct configuration *conf);
|
||||
void add_schema(const gchar* filename, MYSQL *conn);
|
||||
void restore_databases(struct configuration *conf, MYSQL *conn);
|
||||
void restore_schema_view(MYSQL *conn);
|
||||
void restore_schema_triggers(MYSQL *conn);
|
||||
void restore_schema_post(MYSQL *conn);
|
||||
void no_log(const gchar *log_domain, GLogLevelFlags log_level, const gchar *message, gpointer user_data);
|
||||
void set_verbose(guint verbosity);
|
||||
void create_database(MYSQL *conn, gchar *database);
|
||||
|
||||
static GOptionEntry entries[] =
|
||||
{
|
||||
{ "directory", 'd', 0, G_OPTION_ARG_STRING, &directory, "Directory of the dump to import", NULL },
|
||||
{ "queries-per-transaction", 'q', 0, G_OPTION_ARG_INT, &commit_count, "Number of queries per transaction, default 1000", NULL },
|
||||
{ "overwrite-tables", 'o', 0, G_OPTION_ARG_NONE, &overwrite_tables, "Drop tables if they already exist", NULL },
|
||||
{ "database", 'B', 0, G_OPTION_ARG_STRING, &db, "An alternative database to restore into", NULL },
|
||||
{ "source-db", 's', 0, G_OPTION_ARG_STRING, &source_db, "Database to restore", NULL },
|
||||
{ "enable-binlog", 'e', 0, G_OPTION_ARG_NONE, &enable_binlog, "Enable binary logging of the restore data", NULL },
|
||||
{ NULL, 0, 0, G_OPTION_ARG_NONE, NULL, NULL, NULL }
|
||||
};
|
||||
|
||||
void no_log(const gchar *log_domain, GLogLevelFlags log_level, const gchar *message, gpointer user_data) {
|
||||
(void) log_domain;
|
||||
(void) log_level;
|
||||
(void) message;
|
||||
(void) user_data;
|
||||
}
|
||||
|
||||
void set_verbose(guint verbosity) {
|
||||
switch (verbosity) {
|
||||
case 0:
|
||||
g_log_set_handler(NULL, (GLogLevelFlags)(G_LOG_LEVEL_MASK), no_log, NULL);
|
||||
break;
|
||||
case 1:
|
||||
g_log_set_handler(NULL, (GLogLevelFlags)(G_LOG_LEVEL_WARNING | G_LOG_LEVEL_MESSAGE), no_log, NULL);
|
||||
break;
|
||||
case 2:
|
||||
g_log_set_handler(NULL, (GLogLevelFlags)(G_LOG_LEVEL_MESSAGE), no_log, NULL);
|
||||
break;
|
||||
default:
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
int main(int argc, char *argv[]) {
|
||||
struct configuration conf= { NULL, NULL, NULL, 0 };
|
||||
|
||||
GError *error= NULL;
|
||||
GOptionContext *context;
|
||||
|
||||
g_thread_init(NULL);
|
||||
|
||||
init_mutex= g_mutex_new();
|
||||
|
||||
if(db == NULL && source_db != NULL){
|
||||
db = g_strdup(source_db);
|
||||
}
|
||||
|
||||
context= g_option_context_new("multi-threaded MySQL loader");
|
||||
GOptionGroup *main_group= g_option_group_new("main", "Main Options", "Main Options", NULL, NULL);
|
||||
g_option_group_add_entries(main_group, entries);
|
||||
g_option_group_add_entries(main_group, common_entries);
|
||||
g_option_context_set_main_group(context, main_group);
|
||||
if (!g_option_context_parse(context, &argc, &argv, &error)) {
|
||||
g_print("option parsing failed: %s, try --help\n", error->message);
|
||||
exit(EXIT_FAILURE);
|
||||
}
|
||||
g_option_context_free(context);
|
||||
|
||||
//prompt for password if it's NULL
|
||||
if ( sizeof(password) == 0 || ( password == NULL && askPassword ) ){
|
||||
password = passwordPrompt();
|
||||
}
|
||||
|
||||
if (program_version) {
|
||||
g_print("myloader %s, built against MySQL %s\n", VERSION, MYSQL_VERSION_STR);
|
||||
exit(EXIT_SUCCESS);
|
||||
}
|
||||
|
||||
set_verbose(verbose);
|
||||
|
||||
if (!directory) {
|
||||
g_critical("a directory needs to be specified, see --help\n");
|
||||
exit(EXIT_FAILURE);
|
||||
} else {
|
||||
char *p= g_strdup_printf("%s/metadata", directory);
|
||||
if (!g_file_test(p, G_FILE_TEST_EXISTS)) {
|
||||
g_critical("the specified directory is not a mydumper backup\n");
|
||||
exit(EXIT_FAILURE);
|
||||
}
|
||||
}
|
||||
MYSQL *conn;
|
||||
conn= mysql_init(NULL);
|
||||
|
||||
configure_connection(conn,"myloader");
|
||||
if (!mysql_real_connect(conn, hostname, username, password, NULL, port, socket_path, 0)) {
|
||||
g_critical("Error connection to database: %s", mysql_error(conn));
|
||||
exit(EXIT_FAILURE);
|
||||
}
|
||||
|
||||
if (mysql_query(conn, "SET SESSION wait_timeout = 2147483")){
|
||||
g_warning("Failed to increase wait_timeout: %s", mysql_error(conn));
|
||||
}
|
||||
|
||||
if (!enable_binlog)
|
||||
mysql_query(conn, "SET SQL_LOG_BIN=0");
|
||||
|
||||
mysql_query(conn, "/*!40014 SET FOREIGN_KEY_CHECKS=0*/");
|
||||
conf.queue= g_async_queue_new();
|
||||
conf.ready= g_async_queue_new();
|
||||
|
||||
guint n;
|
||||
GThread **threads= g_new(GThread*, num_threads);
|
||||
struct thread_data *td= g_new(struct thread_data, num_threads);
|
||||
for (n= 0; n < num_threads; n++) {
|
||||
td[n].conf= &conf;
|
||||
td[n].thread_id= n+1;
|
||||
threads[n]= g_thread_create((GThreadFunc)process_queue, &td[n], TRUE, NULL);
|
||||
g_async_queue_pop(conf.ready);
|
||||
}
|
||||
g_async_queue_unref(conf.ready);
|
||||
|
||||
g_message("%d threads created", num_threads);
|
||||
|
||||
restore_databases(&conf, conn);
|
||||
|
||||
for (n= 0; n < num_threads; n++) {
|
||||
struct job *j= g_new0(struct job, 1);
|
||||
j->type = JOB_SHUTDOWN;
|
||||
g_async_queue_push(conf.queue, j);
|
||||
}
|
||||
|
||||
for (n= 0; n < num_threads; n++) {
|
||||
g_thread_join(threads[n]);
|
||||
}
|
||||
|
||||
restore_schema_post(conn);
|
||||
|
||||
restore_schema_view(conn);
|
||||
|
||||
restore_schema_triggers(conn);
|
||||
|
||||
g_async_queue_unref(conf.queue);
|
||||
mysql_close(conn);
|
||||
mysql_thread_end();
|
||||
mysql_library_end();
|
||||
g_free(directory);
|
||||
g_free(td);
|
||||
g_free(threads);
|
||||
|
||||
return errors ? EXIT_FAILURE : EXIT_SUCCESS;
|
||||
}
|
||||
|
||||
void restore_databases(struct configuration *conf, MYSQL *conn) {
|
||||
GError *error= NULL;
|
||||
GDir* dir= g_dir_open(directory, 0, &error);
|
||||
|
||||
if (error) {
|
||||
g_critical("cannot open directory %s, %s\n", directory, error->message);
|
||||
errors++;
|
||||
return;
|
||||
}
|
||||
|
||||
const gchar* filename= NULL;
|
||||
|
||||
while((filename= g_dir_read_name(dir))) {
|
||||
if (!source_db || g_str_has_prefix(filename, g_strdup_printf("%s.", source_db))){
|
||||
if (g_strrstr(filename, "-schema.sql")) {
|
||||
add_schema(filename, conn);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
g_dir_rewind(dir);
|
||||
|
||||
while((filename= g_dir_read_name(dir))) {
|
||||
if (!source_db || g_str_has_prefix(filename, g_strdup_printf("%s.", source_db))){
|
||||
if (!g_strrstr(filename, "-schema.sql")
|
||||
&& !g_strrstr(filename, "-schema-view.sql")
|
||||
&& !g_strrstr(filename, "-schema-triggers.sql")
|
||||
&& !g_strrstr(filename, "-schema-post.sql")
|
||||
&& !g_strrstr(filename, "-schema-create.sql")
|
||||
&& g_strrstr(filename, ".sql")) {
|
||||
add_table(filename, conf);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
g_dir_close(dir);
|
||||
}
|
||||
|
||||
|
||||
void restore_schema_view(MYSQL *conn){
|
||||
GError *error= NULL;
|
||||
GDir* dir= g_dir_open(directory, 0, &error);
|
||||
|
||||
if (error) {
|
||||
g_critical("cannot open directory %s, %s\n", directory, error->message);
|
||||
errors++;
|
||||
return;
|
||||
}
|
||||
|
||||
const gchar* filename= NULL;
|
||||
|
||||
while((filename= g_dir_read_name(dir))) {
|
||||
if (!source_db || g_str_has_prefix(filename, source_db)){
|
||||
if (g_strrstr(filename, "-schema-view.sql")) {
|
||||
add_schema(filename, conn);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
g_dir_close(dir);
|
||||
}
|
||||
|
||||
void restore_schema_triggers(MYSQL *conn){
|
||||
GError *error= NULL;
|
||||
GDir* dir= g_dir_open(directory, 0, &error);
|
||||
gchar** split_file= NULL;
|
||||
gchar* database=NULL;
|
||||
gchar** split_table= NULL;
|
||||
gchar* table= NULL;
|
||||
|
||||
if (error) {
|
||||
g_critical("cannot open directory %s, %s\n", directory, error->message);
|
||||
errors++;
|
||||
return;
|
||||
}
|
||||
|
||||
const gchar* filename= NULL;
|
||||
|
||||
while((filename= g_dir_read_name(dir))) {
|
||||
if (!source_db || g_str_has_prefix(filename, source_db)){
|
||||
if (g_strrstr(filename, "-schema-triggers.sql")) {
|
||||
split_file= g_strsplit(filename, ".", 0);
|
||||
database= split_file[0];
|
||||
split_table= g_strsplit(split_file[1], "-schema", 0);
|
||||
table= split_table[0];
|
||||
g_message("Restoring triggers for `%s`.`%s`", db ? db : database, table);
|
||||
restore_data(conn, database, table, filename, TRUE, TRUE);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
g_strfreev(split_table);
|
||||
g_strfreev(split_file);
|
||||
g_dir_close(dir);
|
||||
}
|
||||
|
||||
void restore_schema_post(MYSQL *conn){
|
||||
GError *error= NULL;
|
||||
GDir* dir= g_dir_open(directory, 0, &error);
|
||||
gchar** split_file= NULL;
|
||||
gchar* database=NULL;
|
||||
//gchar* table=NULL;
|
||||
|
||||
|
||||
if (error) {
|
||||
g_critical("cannot open directory %s, %s\n", directory, error->message);
|
||||
errors++;
|
||||
return;
|
||||
}
|
||||
|
||||
const gchar* filename= NULL;
|
||||
|
||||
while((filename= g_dir_read_name(dir))) {
|
||||
if (!source_db || g_str_has_prefix(filename, source_db)){
|
||||
if (g_strrstr(filename, "-schema-post.sql")) {
|
||||
split_file= g_strsplit(filename, "-schema-post.sql", 0);
|
||||
database= split_file[0];
|
||||
//table= split_file[0]; //NULL
|
||||
g_message("Restoring routines and events for `%s`", db ? db : database);
|
||||
restore_data(conn, database, NULL, filename, TRUE, TRUE);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
g_strfreev(split_file);
|
||||
g_dir_close(dir);
|
||||
}
|
||||
|
||||
void create_database(MYSQL *conn, gchar *database){
|
||||
|
||||
gchar* query = NULL;
|
||||
|
||||
if((db == NULL && source_db == NULL) || (db != NULL && source_db != NULL && !g_ascii_strcasecmp(db, source_db))){
|
||||
const gchar* filename= g_strdup_printf("%s-schema-create.sql", db ? db : database);
|
||||
const gchar* filenamegz= g_strdup_printf("%s-schema-create.sql.gz", db ? db : database);
|
||||
const gchar* filepath= g_strdup_printf("%s/%s-schema-create.sql", directory, db ? db : database);
|
||||
const gchar* filepathgz= g_strdup_printf("%s/%s-schema-create.sql.gz", directory, db ? db : database);
|
||||
|
||||
if (g_file_test (filepath, G_FILE_TEST_EXISTS)){
|
||||
restore_data(conn, database, NULL, filename, TRUE, FALSE);
|
||||
}else if (g_file_test (filepathgz, G_FILE_TEST_EXISTS)){
|
||||
restore_data(conn, database, NULL, filenamegz, TRUE, FALSE);
|
||||
}else{
|
||||
query= g_strdup_printf("CREATE DATABASE `%s`", db ? db : database);
|
||||
mysql_query(conn, query);
|
||||
}
|
||||
}else{
|
||||
query= g_strdup_printf("CREATE DATABASE `%s`", db ? db : database);
|
||||
mysql_query(conn, query);
|
||||
}
|
||||
|
||||
g_free(query);
|
||||
return;
|
||||
}
|
||||
|
||||
void add_schema(const gchar* filename, MYSQL *conn) {
|
||||
// 0 is database, 1 is table with -schema on the end
|
||||
gchar** split_file= g_strsplit(filename, ".", 0);
|
||||
gchar* database= split_file[0];
|
||||
// Remove the -schema from the table name
|
||||
gchar** split_table= g_strsplit(split_file[1], "-schema", 0);
|
||||
gchar* table= split_table[0];
|
||||
|
||||
gchar* query= g_strdup_printf("SHOW CREATE DATABASE `%s`", db ? db : database);
|
||||
if (mysql_query(conn, query)) {
|
||||
g_message("Creating database `%s`", db ? db : database);
|
||||
create_database(conn, database);
|
||||
} else {
|
||||
MYSQL_RES *result= mysql_store_result(conn);
|
||||
// In drizzle the query succeeds with no rows
|
||||
my_ulonglong row_count= mysql_num_rows(result);
|
||||
mysql_free_result(result);
|
||||
if (row_count == 0) {
|
||||
create_database(conn, database);
|
||||
}
|
||||
}
|
||||
|
||||
if (overwrite_tables) {
|
||||
g_message("Dropping table or view (if exists) `%s`.`%s`", db ? db : database, table);
|
||||
query= g_strdup_printf("DROP TABLE IF EXISTS `%s`.`%s`", db ? db : database, table);
|
||||
mysql_query(conn, query);
|
||||
query= g_strdup_printf("DROP VIEW IF EXISTS `%s`.`%s`", db ? db : database, table);
|
||||
mysql_query(conn, query);
|
||||
}
|
||||
|
||||
g_free(query);
|
||||
|
||||
g_message("Creating table `%s`.`%s`", db ? db : database, table);
|
||||
restore_data(conn, database, table, filename, TRUE, TRUE);
|
||||
g_strfreev(split_table);
|
||||
g_strfreev(split_file);
|
||||
return;
|
||||
}
|
||||
|
||||
void add_table(const gchar* filename, struct configuration *conf) {
|
||||
struct job *j= g_new0(struct job, 1);
|
||||
struct restore_job *rj= g_new(struct restore_job, 1);
|
||||
j->job_data= (void*) rj;
|
||||
rj->filename= g_strdup(filename);
|
||||
j->type= JOB_RESTORE;
|
||||
gchar** split_file= g_strsplit(filename, ".", 0);
|
||||
rj->database= g_strdup(split_file[0]);
|
||||
rj->table= g_strdup(split_file[1]);
|
||||
rj->part= g_ascii_strtoull(split_file[2], NULL, 10);
|
||||
g_async_queue_push(conf->queue, j);
|
||||
return;
|
||||
}
|
||||
|
||||
void *process_queue(struct thread_data *td) {
|
||||
struct configuration *conf= td->conf;
|
||||
g_mutex_lock(init_mutex);
|
||||
MYSQL *thrconn= mysql_init(NULL);
|
||||
g_mutex_unlock(init_mutex);
|
||||
|
||||
configure_connection(thrconn,"myloader");
|
||||
|
||||
if (!mysql_real_connect(thrconn, hostname, username, password, NULL, port, socket_path, 0)) {
|
||||
g_critical("Failed to connect to MySQL server: %s", mysql_error(thrconn));
|
||||
exit(EXIT_FAILURE);
|
||||
}
|
||||
|
||||
if (mysql_query(thrconn, "SET SESSION wait_timeout = 2147483")){
|
||||
g_warning("Failed to increase wait_timeout: %s", mysql_error(thrconn));
|
||||
}
|
||||
|
||||
if (!enable_binlog)
|
||||
mysql_query(thrconn, "SET SQL_LOG_BIN=0");
|
||||
|
||||
mysql_query(thrconn, "/*!40101 SET NAMES binary*/");
|
||||
mysql_query(thrconn, "/*!40101 SET SQL_MODE='NO_AUTO_VALUE_ON_ZERO' */");
|
||||
mysql_query(thrconn, "/*!40014 SET UNIQUE_CHECKS=0 */");
|
||||
mysql_query(thrconn, "SET autocommit=0");
|
||||
|
||||
g_async_queue_push(conf->ready, GINT_TO_POINTER(1));
|
||||
|
||||
struct job* job= NULL;
|
||||
struct restore_job* rj= NULL;
|
||||
for(;;) {
|
||||
job= (struct job*)g_async_queue_pop(conf->queue);
|
||||
|
||||
switch (job->type) {
|
||||
case JOB_RESTORE:
|
||||
rj= (struct restore_job *)job->job_data;
|
||||
g_message("Thread %d restoring `%s`.`%s` part %d", td->thread_id, rj->database, rj->table, rj->part);
|
||||
restore_data(thrconn, rj->database, rj->table, rj->filename, FALSE, TRUE);
|
||||
if (rj->database) g_free(rj->database);
|
||||
if (rj->table) g_free(rj->table);
|
||||
if (rj->filename) g_free(rj->filename);
|
||||
g_free(rj);
|
||||
g_free(job);
|
||||
break;
|
||||
case JOB_SHUTDOWN:
|
||||
g_message("Thread %d shutting down", td->thread_id);
|
||||
if (thrconn)
|
||||
mysql_close(thrconn);
|
||||
g_free(job);
|
||||
mysql_thread_end();
|
||||
return NULL;
|
||||
break;
|
||||
default:
|
||||
g_critical("Something very bad happened!");
|
||||
exit(EXIT_FAILURE);
|
||||
}
|
||||
}
|
||||
if (thrconn)
|
||||
mysql_close(thrconn);
|
||||
mysql_thread_end();
|
||||
return NULL;
|
||||
}
|
||||
|
||||
void restore_data(MYSQL *conn, char *database, char *table, const char *filename, gboolean is_schema, gboolean need_use) {
|
||||
void *infile;
|
||||
gboolean is_compressed= FALSE;
|
||||
gboolean eof= FALSE;
|
||||
guint query_counter= 0;
|
||||
GString *data= g_string_sized_new(512);
|
||||
|
||||
gchar* path= g_build_filename(directory, filename, NULL);
|
||||
|
||||
if (!g_str_has_suffix(path, ".gz")) {
|
||||
infile= g_fopen(path, "r");
|
||||
is_compressed= FALSE;
|
||||
} else {
|
||||
infile= (void*) gzopen(path, "r");
|
||||
is_compressed= TRUE;
|
||||
}
|
||||
|
||||
if (!infile) {
|
||||
g_critical("cannot open file %s (%d)", filename, errno);
|
||||
errors++;
|
||||
return;
|
||||
}
|
||||
|
||||
|
||||
if(need_use){
|
||||
gchar *query= g_strdup_printf("USE `%s`", db ? db : database);
|
||||
|
||||
if (mysql_query(conn, query)) {
|
||||
g_critical("Error switching to database %s whilst restoring table %s", db ? db : database, table);
|
||||
g_free(query);
|
||||
errors++;
|
||||
return;
|
||||
}
|
||||
|
||||
g_free(query);
|
||||
}
|
||||
|
||||
|
||||
if (!is_schema)
|
||||
mysql_query(conn, "START TRANSACTION");
|
||||
|
||||
while (eof == FALSE) {
|
||||
if (read_data(infile, is_compressed, data, &eof)) {
|
||||
// Search for ; in last 5 chars of line
|
||||
if (g_strrstr(&data->str[data->len >= 5 ? data->len - 5 : 0], ";\n")) {
|
||||
if (mysql_real_query(conn, data->str, data->len)) {
|
||||
g_critical("Error restoring %s.%s from file %s: %s", db ? db : database, table, filename, mysql_error(conn));
|
||||
errors++;
|
||||
return;
|
||||
}
|
||||
query_counter++;
|
||||
if (!is_schema &&(query_counter == commit_count)) {
|
||||
query_counter= 0;
|
||||
if (mysql_query(conn, "COMMIT")) {
|
||||
g_critical("Error committing data for %s.%s: %s", db ? db : database, table, mysql_error(conn));
|
||||
errors++;
|
||||
return;
|
||||
}
|
||||
mysql_query(conn, "START TRANSACTION");
|
||||
}
|
||||
|
||||
g_string_set_size(data, 0);
|
||||
}
|
||||
} else {
|
||||
g_critical("error reading file %s (%d)", filename, errno);
|
||||
errors++;
|
||||
return;
|
||||
}
|
||||
}
|
||||
if (!is_schema && mysql_query(conn, "COMMIT")) {
|
||||
g_critical("Error committing data for %s.%s from file %s: %s", db ? db : database, table, filename, mysql_error(conn));
|
||||
errors++;
|
||||
}
|
||||
g_string_free(data, TRUE);
|
||||
g_free(path);
|
||||
if (!is_compressed) {
|
||||
fclose(infile);
|
||||
} else {
|
||||
gzclose((gzFile)infile);
|
||||
}
|
||||
return;
|
||||
}
|
||||
|
||||
gboolean read_data(FILE *file, gboolean is_compressed, GString *data, gboolean *eof) {
|
||||
char buffer[256];
|
||||
|
||||
do {
|
||||
if (!is_compressed) {
|
||||
if (fgets(buffer, 256, file) == NULL) {
|
||||
if (feof(file)) {
|
||||
*eof= TRUE;
|
||||
buffer[0]= '\0';
|
||||
} else {
|
||||
return FALSE;
|
||||
}
|
||||
}
|
||||
} else {
|
||||
if (!gzgets((gzFile)file, buffer, 256)) {
|
||||
if (gzeof((gzFile)file)) {
|
||||
*eof= TRUE;
|
||||
buffer[0]= '\0';
|
||||
} else {
|
||||
return FALSE;
|
||||
}
|
||||
}
|
||||
}
|
||||
g_string_append(data, buffer);
|
||||
} while ((buffer[strlen(buffer)] != '\0') && *eof == FALSE);
|
||||
|
||||
return TRUE;
|
||||
}
|
51
myloader.h
Normal file
51
myloader.h
Normal file
@ -0,0 +1,51 @@
|
||||
/*
|
||||
This program is free software: you can redistribute it and/or modify
|
||||
it under the terms of the GNU General Public License as published by
|
||||
the Free Software Foundation, either version 3 of the License, or
|
||||
(at your option) any later version.
|
||||
|
||||
This program is distributed in the hope that it will be useful,
|
||||
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
GNU General Public License for more details.
|
||||
|
||||
You should have received a copy of the GNU General Public License
|
||||
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||
|
||||
Authors: Domas Mituzas, Facebook ( domas at fb dot com )
|
||||
Mark Leith, Oracle Corporation (mark dot leith at oracle dot com)
|
||||
Andrew Hutchings, SkySQL (andrew at skysql dot com)
|
||||
|
||||
*/
|
||||
|
||||
#ifndef _myloader_h
|
||||
#define _myloader_h
|
||||
|
||||
enum job_type { JOB_SHUTDOWN, JOB_RESTORE };
|
||||
|
||||
struct configuration {
|
||||
GAsyncQueue* queue;
|
||||
GAsyncQueue* ready;
|
||||
GMutex* mutex;
|
||||
int done;
|
||||
};
|
||||
|
||||
struct thread_data {
|
||||
struct configuration *conf;
|
||||
guint thread_id;
|
||||
};
|
||||
|
||||
struct job {
|
||||
enum job_type type;
|
||||
void *job_data;
|
||||
struct configuration *conf;
|
||||
};
|
||||
|
||||
struct restore_job {
|
||||
char *database;
|
||||
char *table;
|
||||
char *filename;
|
||||
guint part;
|
||||
};
|
||||
|
||||
#endif
|
71
server_detect.c
Normal file
71
server_detect.c
Normal file
@ -0,0 +1,71 @@
|
||||
/*
|
||||
This program is free software: you can redistribute it and/or modify
|
||||
it under the terms of the GNU General Public License as published by
|
||||
the Free Software Foundation, either version 3 of the License, or
|
||||
(at your option) any later version.
|
||||
|
||||
This program is distributed in the hope that it will be useful,
|
||||
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
GNU General Public License for more details.
|
||||
|
||||
You should have received a copy of the GNU General Public License
|
||||
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||
|
||||
Authors: Andrew Hutchings, SkySQL (andrew at skysql dot com)
|
||||
*/
|
||||
|
||||
#include <pcre.h>
|
||||
#include <glib.h>
|
||||
#include <string.h>
|
||||
#include "server_detect.h"
|
||||
|
||||
int detect_server(MYSQL *conn) {
|
||||
pcre *re= NULL;
|
||||
const char *error;
|
||||
int erroroffset;
|
||||
int ovector[9]= {0};
|
||||
int rc;
|
||||
const char* db_version= mysql_get_server_info(conn);
|
||||
|
||||
re= pcre_compile(DETECT_MYSQL_REGEX, 0, &error, &erroroffset, NULL);
|
||||
if (!re) {
|
||||
g_critical("Regular expression fail: %s", error);
|
||||
exit(EXIT_FAILURE);
|
||||
}
|
||||
|
||||
rc = pcre_exec(re, NULL, db_version, strlen(db_version), 0, 0, ovector, 9);
|
||||
pcre_free(re);
|
||||
|
||||
if (rc > 0) {
|
||||
return SERVER_TYPE_MYSQL;
|
||||
}
|
||||
|
||||
re= pcre_compile(DETECT_DRIZZLE_REGEX, 0, &error, &erroroffset, NULL);
|
||||
if (!re) {
|
||||
g_critical("Regular expression fail: %s", error);
|
||||
exit(EXIT_FAILURE);
|
||||
}
|
||||
|
||||
rc = pcre_exec(re, NULL, db_version, strlen(db_version), 0, 0, ovector, 9);
|
||||
pcre_free(re);
|
||||
|
||||
if (rc > 0) {
|
||||
return SERVER_TYPE_DRIZZLE;
|
||||
}
|
||||
|
||||
re= pcre_compile(DETECT_MARIADB_REGEX, 0, &error, &erroroffset, NULL);
|
||||
if (!re) {
|
||||
g_critical("Regular expression fail: %s", error);
|
||||
exit(EXIT_FAILURE);
|
||||
}
|
||||
|
||||
rc = pcre_exec(re, NULL, db_version, strlen(db_version), 0, 0, ovector, 9);
|
||||
pcre_free(re);
|
||||
|
||||
if (rc > 0) {
|
||||
return SERVER_TYPE_MYSQL;
|
||||
}
|
||||
|
||||
return SERVER_TYPE_UNKNOWN;
|
||||
}
|
28
server_detect.h
Normal file
28
server_detect.h
Normal file
@ -0,0 +1,28 @@
|
||||
/*
|
||||
This program is free software: you can redistribute it and/or modify
|
||||
it under the terms of the GNU General Public License as published by
|
||||
the Free Software Foundation, either version 3 of the License, or
|
||||
(at your option) any later version.
|
||||
|
||||
This program is distributed in the hope that it will be useful,
|
||||
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
GNU General Public License for more details.
|
||||
|
||||
You should have received a copy of the GNU General Public License
|
||||
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||
|
||||
Authors: Andrew Hutchings, SkySQL (andrew at skysql dot com)
|
||||
*/
|
||||
#ifndef _server_detect_h
|
||||
#define _server_detect_h
|
||||
|
||||
#include <mysql.h>
|
||||
|
||||
#define DETECT_MYSQL_REGEX "^([3-9]\\.[0-9]+\\.[0-9]+)"
|
||||
#define DETECT_DRIZZLE_REGEX "^(20[0-9]{2}\\.(0[1-9]|1[012])\\.[0-9]+)"
|
||||
#define DETECT_MARIADB_REGEX "^([0-9]{1,2}\\.[0-9]+\\.[0-9]+)"
|
||||
|
||||
enum server_type { SERVER_TYPE_UNKNOWN, SERVER_TYPE_MYSQL, SERVER_TYPE_DRIZZLE };
|
||||
int detect_server(MYSQL *conn);
|
||||
#endif
|
Loading…
Reference in New Issue
Block a user