Skip to content
Felipe Aburaya edited this page Jul 3, 2017 · 27 revisions

Why should I use this framework?

The main objective of 3FD is to help C++ programmers to code their applications faster writing less boiler plate code. These are the main features provided by the framework:

  • An error handling infrastructure with stack tracing
  • An easy to use command line parser
  • A simplified model for heterogeneous programming backed by OpenCL
  • A OOP+RAII model for relational data access backed by SQLite embedded engine
  • A OOP+RAII model for ISAM data access backed by Microsoft ESE
  • Helpers for quick setup and use of Microsoft SQL Server Service Broker queues
  • Helpers & wrappers around Microsoft WWS API, for quick development of SOAP web services
  • Helpers & wrappers around Microsoft RPC API, for quick development of RPC servers and clients
  • Garbage collection

How to use 3FD

Previous versions of the framework were available as compiled DLL's deployed by an msi installer, and that was part of the mentality when the framework was thought to behave like a COM component. However, after working with portions of the framework on the field, I came to the conclusion that C++ programmers are better served with a distributed source code of a highly modularized framework solution.

Basically, you download the source code where the main project is the framework. For Windows platform, Visual Studio 2013 (can be Express) or later is required. A port for POSIX is available and you can take a look here. Because this project is a static library, your solution can link to it, and only what is necessary is ever linked to your final build. Projects for unit and integration tests are also present in the source.

3FD makes use of both POCO and Boost, so you need to have them in your system. Also, in Windows Visual Leak Detector is used to control memory leaks, but you can rule this out of the dependencies if you want, by just erasing the #include <vld.h> in preprocessing.h.

System Requirements

Applications built using the framework require at least Windows Vista (or Server 2008). The main reason for such dependency is because these Windows versions introduced several improvements in the Win32 API that help this framework to be faster.

The project is currently being developed in Visual Studio 2015 Community Update 3 and it uses C++11 language features like lambdas and variadic macros. That's why the client application has to be compiled in a C++11 compliant compiler. Currently, the only compilers I know to be capable of it are the Microsoft Visual C++ 2013 and 2015 (including Express and Community editions) and Clang v3.3 (or later) with libc++.

The distributions of MinGW for Windows based on GCC 4.7 are still not compatible. They are not C++11 compliant enough, rather because of the Standard Library implementation which is a little outdated compared to MSVC. The same problem was verified in Linux. Because of that, Clang and libc++ were chosen in the POSIX port of 3FD.

Prepare to make integration tests work

This is optional, but you might want to run the integration tests in order to test how compliant to the framework requirements your environment is. The modules that require special attention are the Microsoft RPC and WWS API wrappers and MSSQL broker helpers.

The integration tests for RPC and WWS API wrappers are split into server (IntegrationTests project, where the tests for all other modules also reside) and client (TestRpcClient and TestsWwsClient projects) applications. All the applications must start simultaneously, but the client applications stay on hold until the user allows them to continue. That is because they must wait the servers to become available before firing their requests. The user has to observe IntegrationTests.exe running. When the RPC request-response test case starts at server side, then the execution of TestRpcClient.exe can be resumed. When the server side test case for WWS module starts, then the execution of TestWwsClient.exe can be resumed.

Microsoft RPC

The tests cover several scenarios with different security set-ups. The security set-up for NTLM and Kerberos SSP's is selected when the test applications are built, by choosing one of the macros in TestShared\rpc_test_shared.h:

  • SCENARIO_SINGLE_BOX_LOCAL_SEC is the most simple scenario, requires all test applications to run in a single box using Windows local security for authentication (NTLM) of calls using both local and TCP protocol sequences
  • SCENARIO_SINGLE_BOX_AD_SEC also has all test applications running in a single box, but there is availability of Microsoft Active Directory Services for authentication (Kerberos) of calls using both local and TCP protocol sequences
  • SCENARIO_REMOTE_WITH_AD_SEC uses only TCP protocol sequence, hence allowing the client and test applications to run in separate boxes and authenticate using Microsoft Active Directory Services (Kerberos)

You are also required to specify you RPC server location and its SPN (registered Server Principal Name, which unless set otherwise is the FQDN of the user account running IntegrationTests.exe) in the beginning of TestRpcClient\tests_rpc.cpp.

For the SCHANNEL test case you must set up client and server side certificates. For Windows 8.1 and later you can do it with a Powershell prompt elevated with admin privileges:

(in server machine)

New-SelfSignedCertificate -Type SSLServerAuthentication -Subject "CN=MySelfSignedCert4DevTestsClient" -Provider "Microsoft Strong Cryptographic Provider" -TextExtension @("2.5.29.37={text}1.3.6.1.5.5.7.3.2","2.5.29.17={text}upn=[email protected]") -KeyUsage DataEncipherment -KeyAlgorithm RSA -KeyExportPolicy ExportableEncrypted -KeyLength 2048 -CertStoreLocation "Cert:\LocalMachine\My" -NotAfter $(get-date -year 2026)

(in client machine)

New-SelfSignedCertificate -Type SSLServerAuthentication -Subject "CN=MySelfSignedCert4DevTestsServer" -Provider "Microsoft Strong Cryptographic Provider" -TextExtension @("2.5.29.37={text}1.3.6.1.5.5.7.3.1","2.5.29.17={text}upn=[email protected]") -KeyUsage DataEncipherment -KeyAlgorithm RSA -KeyExportPolicy ExportableEncrypted -KeyLength 2048 -CertStoreLocation "Cert:\LocalMachine\My" -NotAfter $(get-date -year 2026)

In case you are running the tests in an older Windows version, you will need to find a Win8.1+ box to generate a single certificate with the command

New-SelfSignedCertificate -Type SSLServerAuthentication -Subject "CN=MySelfSignedCert4DevTests" -Provider "Microsoft Strong Cryptographic Provider" -TextExtension @("2.5.29.37={text}1.3.6.1.5.5.7.3.2,1.3.6.1.5.5.7.3.1","2.5.29.17={text}upn=[email protected]") -KeyUsage DataEncipherment -KeyAlgorithm RSA -KeyExportPolicy ExportableEncrypted -KeyLength 2048 -CertStoreLocation "Cert:\LocalMachine\My" -NotAfter $(get-date -year 2026)

than you can export it to a pfx file (including the private key) and import in your test machine(s).

Notice that in parameter Subject you must have the same name used in source files IntegrationTests\tests_rpc.cpp and TestRpcClient\tests_rpc.cpp. Finally, having all the applications built with those changes and certificates in place, remember to allow all TCP connections to IntegrationTests.exe in your firewall.

Windows Web Services API

Wrappers for WWS API set up the web server using the provided WSDL. You must specify valid addresses for the endpoints in IntegrationTests\calculator.wsdl at //wsdl:definitions/wsdl:service/. Then you use the same addresses in TestWwsClient\tests_wws.cpp.

For transport security, again you need to set up some certificates. In the Visual Studio developer prompt, run

makecert -pe -n "CN=Certification Authority for Development" -cy authority -ss root -sr LocalMachine -r

This creates a certification authority and install it under "Trusted Root Certification Authorities\Certificates" (in management console). Then, use the new CA to create a new certificate for your local machine. Notice you must use your computer name in the commands below:

(n server machine)

makecert -pe -n "CN=MyComputerName" -ss My -sr LocalMachine -in "Certification Authority for Development" -is root -ir LocalMachine -eku 1.3.6.1.5.5.7.3.1

(in client machine)

makecert -pe -n "CN=MyComputerName" -ss My -sr LocalMachine -a sha256 -in "Certification Authority for Development" -is root -ir LocalMachine -eku 1.3.6.1.5.5.7.3.2

(In a development environment confined within a single box, you can choose to have only one certificate for both client and server authentication, hence eliminating the "-eku" option.)

Because both client and server must rely in the same certificate CA, when you create the CA in one box, you must export the certificate to a pfx file (including the private key) and import it into the other box. This way, the certificates in both boxes will have been issued by a common trusted CA.

In the server machine, go to the Windows management console (mmc.exe) and use the snap-in for certificate management to look for the certificate you have just created. The thumbprint is the hash you will use in the command below. These commands will assign the new certificate to the endpoints localhost:8888 and localhost:8989:

netsh http add sslcert ipport=0.0.0.0:8888 certhash=8f43288ad272f3103b6fb1428485ea3014c0bcfe appid={04EBD759-F3FF-4992-8D02-C1BFDB027A58} clientcertnegotiation=enable

netsh http add sslcert ipport=0.0.0.0:8989 certhash=8f43288ad272f3103b6fb1428485ea3014c0bcfe appid={0670F436-50EA-4B8A-B62A-02EF8B39F9C9} clientcertnegotiation=enable

(The application ID can be any GUID.)

In TestWwsClient\tests_wws.cpp the same hash must be set to variable clientCertificateThumbprint. Finally, having all the applications built with those changes and certificates in place, remember to allow all TCP connections to IntegrationTests.exe in your firewall.

Broker

The helpers for Microsoft SQL Server Service Broker rely on an ODBC connection to the SQL Server instance. For that to work, you need to make sure that

  • The SQL Server instance allows connections via TCP (take note of the port, which is 1433 by default)
  • The firewall allows incoming connections to such TCP port (by default, Windows Firewall does not)
  • You have created the database for tests, which can be done by running the script CreateMsSqlSvcBrokerDatabase.sql You have created the login "tester" with password "tester", and granted db_owner privilege to such user in SvcBrokerDatabase
  • You have installed the Microsoft ODBC driver for SQL Server
  • The hardcoded connection string is right in IntegrationTests\tests_broker.cpp, which means there you need to have the version of the driver you installed and the network address of you SQL Server instance (including the TCP port)

Setting up your build environment to use 3FD

First you need to compile the framework. Remember POCO (libraries Foundation, XML, Util and Data/ODBC) and Boost are dependencies that must be present and built by your development environment. The Visual Studio solution uses the macro variables BOOST_HOME and POCO_ROOT (set in the property sheets) to find the location of these libraries. The procedure for the POSIX port is here.

Once the framework is compiled, you are going to set up your solution to link to the framework just like you do for any other static library. Then there are just a couple of things to set:

  • Define the macro ENABLE_3FD_CST in your environment so as to enable the stack tracing feature in your release (recommended);

  • Provide a XML configuration file for the framework instance used by you application. This file must have the same name of the application executable (including the extension) plus ".3fd.config". A template for this file is provided in the source.

Now you are ready to go!

A quick start guided by examples

For a quick start, please refer to the source code where you can find examples of utilization:

SOURCE CODE >> Files >> IntegrationTests | TestRpcClient | TestWwsClient | VideoTranscoder | ImageTranscoder

This is rather the integration tests used in the framework development. Because it covers all important features of 3FD, newcomers can use it as a pretty comprehensive sample code. (The googletest framework syntax for "assert's" and "expect's" do obfuscate it a litte bit, but not that much.)

Also, every application that makes use of 3FD needs to provide a XML file that configures the framework instance in use by the application. This file is of utter importance. It controls the behavior of all the components of the framework instance, from GC tunning to error reporting settings. For an example, please refer to:

SOURCE CODE >> Files >> IntegrationTests >> application.config

Clone this wiki locally