Thank you for your interest in contributing to swblocks-baselib!
swblocks-baselib is built on and intended for open source and we fully intend to accept public contributions in the near future. Until then, feel free to file issues and open pull requests, but note that we won't be merging them until the necessary processes are in place.
If you are currently J.P. Morgan Chase employee and interested in contributing you can contact Lazar Ivanov about more details on the process and governance of contributions from current J.P. Morgan Chase employees.
The swblocks-baselib library comes with a comprehensive unit tests suite and virtually all parts of it are very extensively tested. It also comes currently with a couple of application binaries - 'bl-tool' and 'bl-messaging-broker'.
In order to be able to make changes to the library code and contribute these back you will need to be able to modify and add to the tests code suite and also of course be able to build and run the unit tests suite, etc. The way you can do this is by using the swblocks-baselib development environment and its own build system. The library comes also with its own flexible make files and build + test system which allows you to build these binaries and unit tests and to run the unit tests. To learn how to use the swblocks-baselib library development environment and build system look at the next section.
The library development environment is based on GNU make, but it has somewhat sophisticated structure and implementation that is declarative and build-by-convention and of course also avoids the pitfalls of recursive make. The development environment and the build system assumes certain structure of dependencies and how they are built. This structure and its assumptions are embedded in the make files and the steps to build and prepare the external dependencies are described in the various notes files (under notes folder in the root). As mentioned in the README.md file the swblocks-baselib library and its development environment and build system has very limited dependencies, namely C++11 compliant compiler and toolchain (e.g. recent versions of Clang, GCC or VC++), Boost, OpenSSL and lastly a small JSON parser library which is header files only and does not need to be built (JSON Spirit 4.08), so preparing the dependencies for the development environment is relatively easy, but it is not automated (you have to follow the instructions in the notes files).
The development environment and build system has a minimal notion of versioning to facilitate evolution without having to break the world, but it generally requires "forced upgrade" model where older versions are quickly depreciated and the users / clients of the library are expected to upgrade and move forward. That applies to both the development environment and the code itself. The code of course is changed carefully with backward compatibility in mind and older versions of the development environment or dependencies are not explicitly broken or not supported, but they might be, so if you are a user of the library and you can't upgrade easily at some point you might not be able to pick up the latest version of swblocks-baselib library and might have to stay on older version - btw, the model is very similar to the way Boost is versioned and how backward compatibility is maintained there.
A development environment version is a notion of collection of specific versions of compiler toolchains + a collection of compatible versions of the 3rd party dependencies. Currently swblocks-baselib library officially supports two development environment versions - devenv3 (latest) and devenv2 (old) with the devenv2 environment of course to be eventually depreciated in the somewhat near future (probably within a year or so). Here are the collections of the compiler toolchains versions and the 3rd party dependencies versions for each development environment:
- devenv2
- Operating Systems
- Linux (RedHat rhel5, rhel6, rhel7, Ubuntu 12.04, 14.04, 16.04)
- Windows (7, 8.x, 10 - both x86 and x64)
- Compilers
- GCC 4.9
- Clang 3.5
- vc12 (Visual C++ 2013)
- C++11 standard library implementations
- libstdc++ for GCC and Clang
- msvcrt for vc12
- Boost 1.58
- OpenSSL 1.0.2d
- JSON Spirit 4.08
- Operating Systems
- devenv3
- Operating Systems
- Darwin (15.x, 16.x - i.e. El Capitan and macOS Sierra)
- Linux (RedHat rhel5, rhel6, rhel7, Ubuntu 12.04, 14.04, 16.04)
- Windows (7, 8.x, 10 - both x86 and x64)
- Compilers
- GCC 6.3 (for all Linux operating systems)
- Clang 3.9 (for Ubuntu)
- Clang 7.3 (for Darwin)
- vc14 (Visual C++ 2015)
- C++11 standard library implementations
- libc++ (for Darwin)
- libstdc++ for GCC and Clang
- msvcrt for vc14
- Boost 1.63
- OpenSSL 1.1.0d
- JSON Spirit 4.08
- Operating Systems
In addition to the code dependencies of the library itself (compiler toolchain, boost, openssl, JSON spirit) the development environment also has few additional dependencies such as GNU make (e.g. via msys on Windows), git, python, etc, plus some optional such as Eclipse CDT (to use C++ indexer and IDE). The "binary blob" that contains the pre-built versions of the development environment with all code dependencies plus the additional and optional such is called "devenv distribution". Some popular combinations of devenv distributions were prebuilt and uploaded on Google drive, so they can be downloaded directly and facilitate the development and ease of use of the swblocks-baselib library. Here is the list of the currently supported devenv distributions with the link of where to download them from (more will be coming in the future):
- devenv3 for Windows (Win7 - Win10, including latest Win10 creator's edition, SDK 8.1 and 10), vc14 (VS 2015) can be downloaded from here; or from command line:
wget https://storage.googleapis.com/swblocks-dist/devenv/3/windows-x64/dist-devenv3-windows.zip - devenv3 for Windows both x64 and x86 (Win7 - Win10, including latest Win10 creator's edition, SDK 8.1 and 10), vc14 (VS 2015) can be downloaded from here; or from command line:
wget https://storage.googleapis.com/swblocks-dist/devenv/3/windows-full/dist-devenv3-windows-x64-and-x86.zip - devenv3 for Darwin (d156 - 15.x - 16.x, El Capitan and macOS Sierra), clang 7.3, libc++ can be downloaded here; or from command line:
wget https://storage.googleapis.com/swblocks-dist/devenv/3/darwin-elc/dist-devenv3-darwin-15.6.tar.gz - devenv3 for Ubuntu x64 16.04 LTS with GCC 6.3 and Clang 3.9 can be downloaded here; or from command line:
wget https://storage.googleapis.com/swblocks-dist/devenv/3/ub16-x64/dist-devenv3-ub16-gcc630-clang391.tar.gz - devenv3 for Ubuntu x86 16.04 LTS with GCC 6.3 can be downloaded here; or from command line:
wget https://storage.googleapis.com/swblocks-dist/devenv/3/ub16-x86/dist-devenv3-ub16-x86-gcc630.tar.gz - devenv3 for RedHat 6 / CentOS 6 with GCC 6.3 can be downloaded here; or from command line:
wget https://storage.googleapis.com/swblocks-dist/devenv/3/rhel6-x64/dist-devenv3-rhel6.tar.gz
All links above are either .zip file (for Windows) or a .gz tar file for Darwin and Linux. Once they are downloaded they can be extracted into some location (e.g. c:\swblocks for Windows or /Users/userid/swblocks for Darwin and Linux) and then once you clone the swblocks-baselib library from github in order to use it you simply need to create a small .mk file in projects/make folder called ci-init-env.mk and point the 3 DIST roots to the place where you have extracted the development environment blob.
E.g. if you have extracted the development environment into c:\swblocks then the projects/make/ci-init-env.mk can look the following way for Windows:
# initialize the important env roots
DIST_ROOT_DEPS1 = /c/swblocks/dist-devenv3-windows
DIST_ROOT_DEPS2 = /c/swblocks/dist-devenv3-windows
DIST_ROOT_DEPS3 = /c/swblocks/dist-devenv3-windows
Note that on Windows you should use UNIX style paths since we use GNU make (via msys)
E.g. projects/make/ci-init-env.mk can look the following way for Darwin (or you can adjust for Linux):
# initialize the important env roots
DIST_ROOT_DEPS1 = /Users/userid/swblocks/dist-devenv3-darwin-15.6
DIST_ROOT_DEPS2 = /Users/userid/swblocks/dist-devenv3-darwin-15.6
DIST_ROOT_DEPS3 = /Users/userid/swblocks/dist-devenv3-darwin-15.6
An additional step which applies only to Windows is to add few entries to the Path environment variable associated with your account permanently as these tools are necessary to build, run tests and use the development environment in general, but by default are not available on Windows. On Linux / UNIX / Darwin these are typically available, or can be made available, in the OS, so this step is not necessary on these platforms. Here is the list of entries which need to be added to the path assuming the location where you have extracted the development environment distribution is c:\swblocks (preferably in this order):
c:\swblocks\dist-devenv3-windows\msys2\latest\msys64\usr\bin
c:\swblocks\dist-devenv3-windows\git\latest\default\bin
c:\swblocks\dist-devenv3-windows\python\2.7-latest\default
If you are using WinDbg for debugging on Windows you can also optionally add the following entry to the Path associated with your account:
c:\swblocks\dist-devenv3-windows\winsdk\10\default\Debuggers\x64
Once you create unpack the development environment distribution and create the projects/make/ci-init-env.mk as per above you can now build the code run tests by specifying the targets, etc. The unit test targets start with utf_baselib prefix (e.g. utf_baselib, utf_baselib_data) and the respective test targets start with test_utf_baselib. The other targets are the respective binary names and there are special targets too, just type make help for more information.
Here are some examples e.g.:
make utf_baselib
make test_utf_baselib_data
make -k -j6
make -k -j6 && make -k -j8 test
make help
make -k -j6 && make -k -j8 test && make install
The Eclipse CDT indexer is an excellent tool and probably one of the best such that exist to vastly improve productivity of C++ developers. swblocks-baselib comes with a small python script that generates eclipse project files for the various targets, so Eclipse can be used with the CDT indexer for development. The script doesn't take any arguments hence can be simply executed like this:
$ python scripts/generate-eclipse-project-config.py
At this point the eclipse projects will be generated in bld/makeeclipse location and you can simply copy/move them in more permanent location (e.g. if you use the CI_ENV_ROOT environment variable to control the development environment configuration then the expected location to move them to would be $CI_ENV_ROOT/projects/makeeclipse) and then of course start using them by importing them in an Eclipse workspace.
Note also that each time you import or open the projects for the first time in an Eclipse workspace you may get something like the following errors on Windows:
Program "g++" not found in PATH <my-target> ...
Program "gcc" not found in PATH <my-target> ...
These errors can simply be ignored (deleted once). They won't show up again until you re-open and / or re-import the project again.
Note also that it is highly recommended (and actually some parts actually required - e.g. style and formatting, line ending, etc) that configure your Eclipse workspace according to the instructions in notes\eclipse_workspace.txt.
This section of course will not cover general information about how to use Git and GitHub (there is plenty of information on the internet and GitHub site itself), but if you are new to Git it is highly recommended to read the following link which is not the typical Git tutorial, but will help you understand Git conceptually. And of course for tutorials on the specifics, the likes of "getting started", "cheat sheets", etc, you can search on Google as there are plenty of those available on the internet.
Here are some details on how to use GitHub to open pull requests and contribute specifically to the swblocks-baselib library. First of course you will need to create an account on GitHub and then create your own private fork (on GitHub) of the repository off the master copy located in the JP Morgan Chase account area here. The way you can create a private fork of the repository is by first going to the master copy link and then clicking the 'Fork' button in the top right corner. The reason you need to create a private fork is that the master copy is locked down for write access and you can't directly make changes to it via push Git commands, but only via pull requests (from branches in your own private fork).
How do you want to organize the branches in your own private fork is up to your own preference, but it is recommended to pull directly from the master copy into your private branches in your fork and then when you have changes ready you can create pull requests from branches in your private for to master branch in the master copy fork.
In order to pull easily changes from the master copy repository it is recommended to add it as an additional remote where fetch is allowed, but push is disabled. You can accomplish this with the following Git commands:
git remote add --mirror=fetch public_origin https://github.com/jpmorganchase/swblocks-baselib.git
git remote set-url --push public_origin disabled
Now every time you want to pull changes from the master repository you can do this with the following Git command:
git pull public_origin master
If you want to do pull / push for your own private branches in your own private fork you of course can do this in the usual way:
git checkout -b mybranch
git push --set-upstream origin mybranch
git push
git pull origin master
Note also that it is recommended to not use pull requests in your own private fork (but use Git push / pull directly) as this will create extra 'pollution' with unnecessary commits for the private pull requests and these will eventually leak in the master repository as part of your official / public pull requests.