From 7410eddf0d0020aeaab621fe793c6d0c8ce421ae Mon Sep 17 00:00:00 2001 From: Lev Brouk Date: Mon, 22 Apr 2024 12:17:41 -0700 Subject: [PATCH] wip: all combined --- .github/workflows/build-test.yml | 182 ++++++++++++++++++++++++++ .github/workflows/on-pr-debug.yml | 58 ++++++++ .github/workflows/on-push-quick.yml | 28 ++++ .github/workflows/on-push-release.yml | 65 +++++++++ .gitignore | 1 + CMakeLists.txt | 2 +- codecov.yml | 10 ++ test/test.c | 26 ++++ 8 files changed, 371 insertions(+), 1 deletion(-) create mode 100644 .github/workflows/build-test.yml create mode 100644 .github/workflows/on-pr-debug.yml create mode 100644 .github/workflows/on-push-quick.yml create mode 100644 .github/workflows/on-push-release.yml create mode 100644 codecov.yml diff --git a/.github/workflows/build-test.yml b/.github/workflows/build-test.yml new file mode 100644 index 000000000..788dda26f --- /dev/null +++ b/.github/workflows/build-test.yml @@ -0,0 +1,182 @@ +on: + workflow_call: + inputs: + arch: + type: string + default: "64" + compiler: + type: string + default: "gcc" + coverage: + type: string + default: "OFF" + lib_msg_delivery: + type: string + default: "OFF" + lib_write_deadline: + type: string + default: "OFF" + quick_test: + type: string + default: "OFF" + sanitize: + type: string + description: "sanitize option to use, 'address' or 'thread'" + server_version: + type: string + description: "nats-server version to test with" + default: "latest" + streaming: + type: string + default: "ON" + tls: + type: string + default: "ON" + type: + type: string + description: "Debug or Release." + default: "Release" + ubuntu_version: + type: string + description: "Ubuntu version to use, e.g. '20.04'" + default: "latest" + secrets: + COVERALLS_REPO_TOKEN: + description: "Coveralls repo token" + CODECOV_TOKEN: + description: "Codecov repo token" + +permissions: + contents: write # to comment on coverage. + +defaults: + run: + shell: bash --noprofile --norc -x -eo pipefail {0} + +jobs: + do: + runs-on: ubuntu-${{ inputs.ubuntu_version }} + name: "ubuntu:${{ inputs.ubuntu_version }} ${{ inputs.compiler }} nats:${{ inputs.server_version }}" + steps: + - name: Checkout nats.c + uses: actions/checkout@v4 + + - name: "Checkout dependencies (nats.c.deps)" + uses: actions/checkout@v4 + with: + repository: nats-io/nats.c.deps + path: deps + + # configure the cmake flags and NATS_... environment variables + - id: cmake-flags + name: Configure cmake flags + env: + flags: -DNATS_BUILD_ARCH=${{ inputs.arch }} + -DCMAKE_BUILD_TYPE=${{ inputs.type }} + -DNATS_BUILD_STREAMING=${{ inputs.streaming }} + -DNATS_BUILD_WITH_TLS=${{ inputs.tls }} + -DNATS_PROTOBUF_DIR=${{ github.workspace}}/deps/pbuf + -DNATS_BUILD_USE_SODIUM=ON + -DNATS_SODIUM_DIR=${{ github.workspace}}/deps/sodium + run: | + if [[ -n "${{ inputs.sanitize }}" ]]; then + flags="$flags -DNATS_SANITIZE=ON -DCMAKE_C_FLAGS=-fsanitize=${{ inputs.sanitize }}" + fi + if [[ "${{ inputs.coverage }}" == "ON" ]]; then + flags="$flags -DNATS_COVERAGE=ON" + fi + echo "flags=$flags" >> $GITHUB_OUTPUT + + - id: nats-vars + name: Configure NATS_ environment variables + run: | + if [[ -n "${{ inputs.sanitize }}" ]]; then + echo "NATS_TEST_VALGRIND=yes" >> $GITHUB_ENV + fi + if [[ "${{ inputs.lib_msg_delivery }}" == "ON" ]]; then + echo "NATS_DEFAULT_TO_LIB_MSG_DELIVERY=yes" >> $GITHUB_ENV + fi + if [[ "${{ inputs.lib_write_deadline }}" == "ON" ]]; then + echo "NATS_DEFAULT_LIB_WRITE_DEADLINE=2000" >> $GITHUB_ENV + fi + echo "CC=${{ inputs.compiler }}" >> $GITHUB_ENV + + # install build dependencies + - name: Install ${{ inputs.compiler }} if needed + if: startsWith(inputs.compiler, 'clang-') || startsWith(inputs.compiler, 'gcc-') + run: | + sudo apt-get -q update + sudo apt-get -y install ${{ inputs.compiler }} + + # otherwise, configure cmake, build, archive and upload + - name: CMake + run: | + mkdir -p build + cd build + cmake .. ${{ steps.cmake-flags.outputs.flags }} + make rebuild_cache && make + + - name: "Rebuild the list of tests to match the compile flags" + working-directory: ./build + run: | + ./test/testsuite + if [[ $(diff list.txt ../test/list.txt; echo $?) != 0 ]]; then + mv list.txt ../test/list.txt + make rebuild_cache + fi + + # testing + + - name: "Download nats-server version ${{ inputs.server_version }}" + if: inputs.quick_test != 'ON' && inputs.server_version + working-directory: ./build + run: | + rel=${{ inputs.server_version }} + if [ "$rel" = "latest" ]; then + rel=$(curl -s https://api.github.com/repos/nats-io/nats-server/releases/latest | jq -r '.tag_name') + fi + if [ "$rel" != "${rel#v}" ] && wget https://github.com/nats-io/nats-server/releases/download/$rel/nats-server-$rel-linux-amd64.tar.gz; then + tar -xzf nats-server-$rel-linux-amd64.tar.gz + ls -lsra nats-server-$rel-linux-amd64 + cp nats-server-$rel-linux-amd64/nats-server ../deps/nats-server/nats-server + else + for c in 1 2 3 4 5 + do + echo "Attempt $c to download binary for main" + rm -f ./nats-server + curl -sf "https://binaries.nats.dev/nats-io/nats-server/v2@$rel" | PREFIX=. sh + # We are sometimes getting nats-server of size 0. Make sure we have a + # working nats-server by making sure we get a version number. + v="$(./nats-server -v)" + if [ "$v" != "" ]; then + break + fi + done + mv ./nats-server ../deps/nats-server/nats-server + fi + echo "------------------------------------------" + ../deps/nats-server/nats-server -v # <>/<> + + - name: "Test" + working-directory: ./build + run: | + export PATH=../deps/nats-server:../deps/nats-streaming-server:$PATH + export NATS_TEST_SERVER_VERSION="$(nats-server -v)" + if [[ "${{ inputs.quick_test }}" == "ON" ]]; then + ctest --timeout 60 --output-on-failure -I 1,1 + else + ctest --timeout 60 --output-on-failure --repeat-until-fail 3 2>&1 | tee /tmp/out.txt + if [[ $(grep -q 'WARNING: ThreadSanitizer: ' /tmp/out.txt; echo $?) == 0 ]]; then + echo "!!! ThreadSanitizer detected WARNING(s) !!!" + exit 1 + fi + fi + + - name: Upload coverage reports to Codecov + if: inputs.quick_test != 'ON' && inputs.coverage == 'ON' + env: + CODECOV_TOKEN: ${{ secrets.CODECOV_TOKEN }} + run: | + # curl -Os https://uploader.codecov.io/latest/linux/codecov + # chmod +x codecov + # ./codecov -Q 'issue1011' -C ${{ github.sha }} -Z -g --ga '-lp' --gi 'examples/**' --gi 'src/win/**' --gi 'src/unix/**' --gi 'src/stan/protocol.pb-c.c' diff --git a/.github/workflows/on-pr-debug.yml b/.github/workflows/on-pr-debug.yml new file mode 100644 index 000000000..c89161428 --- /dev/null +++ b/.github/workflows/on-pr-debug.yml @@ -0,0 +1,58 @@ +name: "Debug" +on: + pull_request: + +permissions: + contents: write # so it can comment + +defaults: + run: + shell: bash --noprofile --norc -x -eo pipefail {0} + +jobs: + gcc-cov: + name: "Default (coverage)" + uses: ./.github/workflows/build-test.yml + with: + coverage: ON + server_version: main + type: Debug + + clang: + name: "Default" + uses: ./.github/workflows/build-test.yml + with: + compiler: clang + server_version: main + type: Debug + + sanitize-addr: + name: "Sanitize address" + uses: ./.github/workflows/build-test.yml + with: + sanitize: address + server_version: main + type: Debug + + sanitize-addr-lib-msg-delivery: + name: "Sanitize address (lib_msg_delivery)" + uses: ./.github/workflows/build-test.yml + with: + sanitize: address + server_version: main + lib_msg_delivery: ON + + san-addr: + name: "Sanitize address (lib_write_deadline)" + uses: ./.github/workflows/build-test.yml + with: + sanitize: address + server_version: main + lib_write_deadline: ON + + san-thread: + name: "Sanitize thread" + uses: ./.github/workflows/build-test.yml + with: + sanitize: thread + server_version: main diff --git a/.github/workflows/on-push-quick.yml b/.github/workflows/on-push-quick.yml new file mode 100644 index 000000000..3133463b1 --- /dev/null +++ b/.github/workflows/on-push-quick.yml @@ -0,0 +1,28 @@ +name: "Release(quick)" +on: + push: + branches-ignore: + - main + - release_* + +permissions: + contents: write # required by build-test to comment on coverage but not used here. + +defaults: + run: + shell: bash --noprofile --norc -x -eo pipefail {0} + +jobs: + quick: + strategy: + fail-fast: false + matrix: + compiler: [gcc, clang] + ubuntu_version: [latest, 20.04] + uses: ./.github/workflows/build-test.yml + name: "Default" + with: + server_version: main + ubuntu_version: ${{ matrix.ubuntu_version }} + compiler: ${{ matrix.compiler }} + diff --git a/.github/workflows/on-push-release.yml b/.github/workflows/on-push-release.yml new file mode 100644 index 000000000..794bd82cc --- /dev/null +++ b/.github/workflows/on-push-release.yml @@ -0,0 +1,65 @@ +name: "Release" +on: + push: + # branches: + # - main + # - release_* + + +permissions: + contents: write # required by build-test to comment on coverage but not used here. + +defaults: + run: + shell: bash --noprofile --norc -x -eo pipefail {0} + +jobs: + default: + strategy: + fail-fast: false + matrix: + compiler: [gcc, clang] + ubuntu_version: [latest, 20.04] + uses: ./.github/workflows/build-test.yml + name: "Default" + with: + ubuntu_version: ${{ matrix.ubuntu_version }} + compiler: ${{ matrix.compiler }} + + more-server-versions: + strategy: + fail-fast: false + matrix: + server_version: [main, v2.9.11] + uses: ./.github/workflows/build-test.yml + name: "Other servers" + with: + server_version: ${{ matrix.server_version }} + + TLS-OFF: + strategy: + fail-fast: false + matrix: + compiler: [gcc, clang] + uses: ./.github/workflows/build-test.yml + name: "No TLS" + with: + tls: OFF + + older-cc: + strategy: + fail-fast: false + matrix: + compiler: [gcc-7, gcc-8, clang-8] + uses: ./.github/workflows/build-test.yml + name: "Older compilers" + with: + ubuntu_version: 20.04 + compiler: ${{ matrix.compiler }} + quick_test: ON + + no-streaming: + uses: ./.github/workflows/build-test.yml + name: "No Streaming" + with: + streaming: OFF diff --git a/.gitignore b/.gitignore index fdedaffa8..d68a2cc42 100644 --- a/.gitignore +++ b/.gitignore @@ -3,6 +3,7 @@ Compiled Object files, Static and Dynamic libs (Shared Objects) *.a *.so *.dylib +*.gcov # Folders build*/ diff --git a/CMakeLists.txt b/CMakeLists.txt index 1c2f60df7..8595370e5 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -5,7 +5,7 @@ include(CTest) include(FindPackageHandleStandardArgs) # Uncomment to have the build process verbose -#set(CMAKE_VERBOSE_MAKEFILE TRUE) +set(CMAKE_VERBOSE_MAKEFILE TRUE) # Uncomment to have the executable moved to 'build' instead of their respective 'build/xxx' directories #set(CMAKE_RUNTIME_OUTPUT_DIRECTORY ${PROJECT_BINARY_DIR}) diff --git a/codecov.yml b/codecov.yml new file mode 100644 index 000000000..2f3aa590f --- /dev/null +++ b/codecov.yml @@ -0,0 +1,10 @@ +coverage: + status: + project: + default: + target: 80% + threshold: 0.5% +github_checks: + annotations: true +comment: + require_changes: false \ No newline at end of file diff --git a/test/test.c b/test/test.c index d0a85759d..e0a68f481 100644 --- a/test/test.c +++ b/test/test.c @@ -20,6 +20,7 @@ #else #include #include +#include #endif #include "buf.h" @@ -32391,6 +32392,15 @@ test_KeyValueMirrorCrossDomains(void) test("Get key: "); s = kvStore_Get(&e, mkv, "name"); + // levb: we seem to get the old value at times here, so try one more time if + // there's a mismatch + for (i = 0; i < 3; i++) + if ((s == NATS_OK) && (e != NULL) && (strcmp(kvEntry_ValueString(e), "rip") != 0)) + { + kvEntry_Destroy(e); + e = NULL; + s = kvStore_Get(&e, mkv, "name"); + } if ((s == NATS_OK) && (e != NULL)) s = (strcmp(kvEntry_ValueString(e), "rip") == 0 ? NATS_OK : NATS_ERR); testCond(s == NATS_OK); @@ -36335,6 +36345,18 @@ generateList(void) fclose(list); } +#ifndef _WIN32 +static void _sigsegv_handler(int sig) { + void *array[20]; + size_t size = backtrace(array, 20); + + // print out all the frames to stderr + fprintf(stderr, "Error: signal %d:\n", sig); + backtrace_symbols_fd(array, size, STDERR_FILENO); + exit(1); +} +#endif // _WIN32 + int main(int argc, char **argv) { const char *envStr; @@ -36363,6 +36385,10 @@ int main(int argc, char **argv) return 1; } +#ifndef _WIN32 + signal(SIGSEGV, _sigsegv_handler); +#endif // _WIN32 + envStr = getenv("NATS_TEST_TRAVIS"); if ((envStr != NULL) && (envStr[0] != '\0')) runOnTravis = true;