folly/folly/coro/test/ScopeExitTest.cpp

/*
 * Copyright (c) Meta Platforms, Inc. and affiliates.
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *     http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

#include <folly/experimental/coro/AsyncGenerator.h>
#include <folly/experimental/coro/AsyncScope.h>
#include <folly/experimental/coro/AutoCleanup.h>
#include <folly/experimental/coro/Baton.h>
#include <folly/experimental/coro/BlockingWait.h>
#include <folly/experimental/coro/Task.h>
#include <folly/portability/GTest.h>

#include <stdexcept>
#include <type_traits>

#if FOLLY_HAS_COROUTINES

using namespace folly::coro;

namespace {
class ScopeExitTest : public testing::Test {
 protected:
  int count = 0;
};

class AsyncGeneratorScopeExitTest : public ScopeExitTest {};

class AutoCleanupScopeExitTest : public ScopeExitTest {};
} // namespace

TEST_F(ScopeExitTest, OneExitAction) {
  folly::coro::blockingWait([this]() -> Task<> {
    ++count;
    co_await co_scope_exit([this]() -> Task<> {
      count *= 2;
      co_return;
    });
    ++count;
  }());
  EXPECT_EQ(count, 4);
}

TEST_F(ScopeExitTest, TwoExitActions) {
  folly::coro::blockingWait([this]() -> Task<> {
    ++count;
    co_await co_scope_exit([this]() -> Task<> {
      count *= count;
      co_return;
    });
    co_await co_scope_exit([this]() -> Task<> {
      count *= 2;
      co_return;
    });
    ++count;
  }());
  EXPECT_EQ(count, 16);
}

TEST_F(ScopeExitTest, OneExitActionWithException) {
  EXPECT_THROW(
      folly::coro::blockingWait([this]() -> Task<> {
        ++count;
        co_await co_scope_exit([this]() -> Task<> {
          count *= 2;
          co_return;
        });
        throw std::runtime_error("Something bad happened!");
      }()),
      std::runtime_error);
  EXPECT_EQ(count, 2);
}

TEST_F(ScopeExitTest, ExceptionInExitActionCausesTermination) {
  ASSERT_DEATH(
      folly::coro::blockingWait([]() -> Task<> {
        co_await co_scope_exit([]() -> Task<> {
          throw std::runtime_error("Something bad happened!");
        });
      }()),
      "");
}

TEST_F(ScopeExitTest, ExceptionInExitActionDuringExceptionCausesTermination) {
  ASSERT_DEATH(
      folly::coro::blockingWait([]() -> Task<> {
        co_await co_scope_exit([]() -> Task<> {
          throw std::runtime_error("Something bad happened!");
        });
        throw std::runtime_error("Throwing from parent");
      }()),
      "Something bad happened!");
}

TEST_F(ScopeExitTest, StatefulExitAction) {
  folly::coro::blockingWait([this]() -> Task<> {
    auto&& [i] = co_await co_scope_exit(
        [this](int&& ii) -> Task<void> {
          count += ii;
          co_return;
        },
        3);
    ++count;
    i *= i;
  }());
  EXPECT_EQ(count, 10);
}

TEST_F(ScopeExitTest, NonMoveableState) {
  folly::coro::blockingWait([this]() -> Task<> {
    auto&& [asyncScope] = co_await co_scope_exit(
        [this](auto&& scope) -> Task<void> {
          co_await scope->joinAsync();
          count *= 2;
        },
        std::make_unique<AsyncScope>());

    auto ex = co_await co_current_executor;
    asyncScope->add(co_invoke([this]() -> Task<> {
                      ++count;
                      co_return;
                    }).scheduleOn(ex));
  }());
  EXPECT_EQ(count, 2);
}

TEST_F(ScopeExitTest, OneExitActionThroughNoThrow) {
  EXPECT_THROW(
      folly::coro::blockingWait([this]() -> Task<> {
        co_await co_scope_exit([this]() -> Task<> {
          ++count;
          co_return;
        });
        co_await co_nothrow([]() -> Task<> {
          throw std::runtime_error("Something bad happened!");
          co_return;
        }());
      }()),
      std::runtime_error);
  EXPECT_EQ(count, 1);
}

TEST_F(ScopeExitTest, ExitActionInsideNoThrow) {
  EXPECT_THROW(
      folly::coro::blockingWait([this]() -> Task<> {
        try {
          co_await co_nothrow([this]() -> Task<> {
            co_await co_scope_exit([this]() -> Task<> {
              ++count;
              co_return;
            });
            throw std::runtime_error("Something bad happened!");
            co_return;
          }());
        } catch (...) {
          ADD_FAILURE();
        }
      }()),
      std::runtime_error);
  EXPECT_EQ(count, 1);
}

TEST_F(AsyncGeneratorScopeExitTest, PartiallyConsumed) {
  auto makeGenerator = [&]() -> folly::coro::CleanableAsyncGenerator<int> {
    co_await co_scope_exit([&]() -> folly::coro::Task<> {
      ++count;
      co_return;
    });
    co_yield 1;
    co_yield 2;
  };

  auto gen = makeGenerator();

  folly::coro::blockingWait([&]() -> folly::coro::Task<> {
    auto result = co_await gen.next();
    EXPECT_TRUE(result);
    EXPECT_EQ(count, 0);
    co_await std::move(gen).cleanup();
  }());

  EXPECT_EQ(count, 1);
}

TEST_F(AsyncGeneratorScopeExitTest, FullyConsumed) {
  auto makeGenerator = [&]() -> folly::coro::CleanableAsyncGenerator<int> {
    co_await co_scope_exit([&]() -> folly::coro::Task<> {
      ++count;
      co_return;
    });
    co_yield 1;
  };

  auto gen = makeGenerator();

  folly::coro::blockingWait([&]() -> folly::coro::Task<> {
    auto result = co_await gen.next();
    EXPECT_TRUE(result);
    result = co_await gen.next();
    EXPECT_FALSE(result);
    EXPECT_EQ(count, 0);
    co_await std::move(gen).cleanup();
  }());

  EXPECT_EQ(count, 1);
}

TEST_F(AsyncGeneratorScopeExitTest, TwoExitActions) {
  auto makeGenerator = [&]() -> folly::coro::CleanableAsyncGenerator<int> {
    co_await co_scope_exit([&]() -> folly::coro::Task<> {
      EXPECT_EQ(count, 1);
      ++count;
      co_return;
    });
    co_await co_scope_exit([&]() -> folly::coro::Task<> {
      EXPECT_EQ(count, 0);
      ++count;
      co_return;
    });
    co_return;
  };

  auto gen = makeGenerator();

  folly::coro::blockingWait([&]() -> folly::coro::Task<> {
    auto result = co_await gen.next();
    EXPECT_FALSE(result);
    EXPECT_EQ(count, 0);
    co_await std::move(gen).cleanup();
  }());

  EXPECT_EQ(count, 2);
}

TEST_F(AsyncGeneratorScopeExitTest, StatefulExitAction) {
  auto makeGenerator = [&]() -> folly::coro::CleanableAsyncGenerator<int> {
    auto&& [i] = co_await co_scope_exit(
        [&](int&& ii) -> Task<void> {
          count += ii;
          co_return;
        },
        3);
    ++count;
    i *= i;
    co_return;
  };

  auto gen = makeGenerator();

  folly::coro::blockingWait([&]() -> folly::coro::Task<> {
    EXPECT_EQ(count, 0);
    auto result = co_await gen.next();
    EXPECT_FALSE(result);
    EXPECT_EQ(count, 1);
    co_await std::move(gen).cleanup();
  }());

  EXPECT_EQ(count, 10);
}

TEST_F(AsyncGeneratorScopeExitTest, OneExitActionThroughNoThrow) {
  struct Error : std::exception {};

  auto makeGenerator = [&]() -> folly::coro::CleanableAsyncGenerator<int> {
    co_await co_scope_exit([&]() -> folly::coro::Task<> {
      ++count;
      co_return;
    });
    co_await co_nothrow([]() -> Task<> {
      throw Error{};
      co_return;
    }());
  };

  auto gen = makeGenerator();

  folly::coro::blockingWait([&]() -> folly::coro::Task<> {
    EXPECT_THROW(co_await gen.next(), Error);
    co_await std::move(gen).cleanup();
  }());

  EXPECT_EQ(count, 1);
}

TEST_F(AsyncGeneratorScopeExitTest, NextAfterCancel) {
  folly::coro::blockingWait([&]() -> folly::coro::Task<void> {
    folly::CancellationSource cancelSrc;
    auto gen = folly::coro::co_invoke(
        [&]() -> folly::coro::CleanableAsyncGenerator<int> {
          co_await co_scope_exit([&]() -> folly::coro::Task<> {
            ++count;
            co_return;
          });
          co_yield 1;
          co_yield 2;
          co_await folly::coro::co_safe_point;
          co_await co_scope_exit([&]() -> folly::coro::Task<> {
            ++count;
            co_return;
          });
          co_yield 3;
        });

    auto result = co_await folly::coro::co_awaitTry(
        folly::coro::co_withCancellation(cancelSrc.getToken(), gen.next()));
    EXPECT_TRUE(result.hasValue());
    EXPECT_EQ(1, *result.value());

    cancelSrc.requestCancellation();
    result = co_await folly::coro::co_awaitTry(
        folly::coro::co_withCancellation(cancelSrc.getToken(), gen.next()));
    EXPECT_TRUE(result.hasValue());
    EXPECT_EQ(2, *result.value());

    result = co_await folly::coro::co_awaitTry(
        folly::coro::co_withCancellation(cancelSrc.getToken(), gen.next()));
    EXPECT_TRUE(result.hasException());
    EXPECT_THROW(result.value(), folly::OperationCancelled);

    EXPECT_EQ(count, 0);
    co_await std::move(gen).cleanup();
    EXPECT_EQ(count, 1);
  }());
}

TEST_F(AutoCleanupScopeExitTest, AsyncGeneratorAutoCleanup) {
  blockingWait([&]() -> Task<> {
    auto gen = co_invoke([&]() -> CleanableAsyncGenerator<int> {
      co_await co_scope_exit([&]() -> Task<> {
        ++count;
        co_return;
      });
      co_yield 1;
    });
    co_await gen.next();
    gen = co_invoke(
        [](auto&&, auto) -> CleanableAsyncGenerator<int> { co_return; },
        AutoCleanup{std::move(gen)},
        42);
    co_await std::move(gen).cleanup();
    EXPECT_EQ(count, 1);
  }());
}

TEST_F(AutoCleanupScopeExitTest, AsyncGeneratorAutoCleanupMove) {
  blockingWait([&]() -> Task<> {
    auto gen = co_invoke([&]() -> CleanableAsyncGenerator<int> {
      co_await co_scope_exit([&]() -> Task<> {
        ++count;
        co_return;
      });
      co_yield 1;
    });
    gen = co_invoke(
        [](auto autoCleanupGen, auto) -> CleanableAsyncGenerator<int> {
          co_await autoCleanupGen->next();
        },
        AutoCleanup{std::move(gen)},
        42);
    co_await gen.next();
    co_await std::move(gen).cleanup();
    EXPECT_EQ(count, 1);
  }());
}

TEST_F(AsyncGeneratorScopeExitTest, AsyncGeneratorAutoCleanupFn) {
  std::function cleanupFn{[&](int&&) -> folly::coro::Task<> {
    ++count;
    co_return;
  }};
  blockingWait([&]() -> Task<> {
    auto gen = co_invoke(
        [](auto) -> CleanableAsyncGenerator<int> { co_return; },
        AutoCleanup{1, cleanupFn});
    co_await std::move(gen).cleanup();
    EXPECT_EQ(count, 1);
  }());
}

TEST_F(AsyncGeneratorScopeExitTest, AsyncGeneratorAutoCleanupFnMove) {
  std::function cleanupFn{[&](int&&) -> folly::coro::Task<> {
    ++count;
    co_return;
  }};
  blockingWait([&]() -> Task<> {
    auto gen = co_invoke(
        [](auto) -> CleanableAsyncGenerator<int> { co_return; },
        AutoCleanup{1, cleanupFn});
    co_await gen.next();
    co_await std::move(gen).cleanup();
    EXPECT_EQ(count, 2);
  }());
}

#endif // FOLLY_HAS_COROUTINES