mirror of
https://github.com/mollyim/webrtc.git
synced 2025-05-12 13:20:44 +01:00

Bug: webrtc:13579 Change-Id: Ia9f90e6c3b008fc614d378cae4c407becfc597c9 Reviewed-on: https://webrtc-review.googlesource.com/c/src/+/298447 Reviewed-by: Tomas Gunnarsson <tommi@webrtc.org> Commit-Queue: Danil Chapovalov <danilchap@webrtc.org> Cr-Commit-Position: refs/heads/main@{#39610}
235 lines
7.1 KiB
C++
235 lines
7.1 KiB
C++
/*
|
|
* Copyright 2017 The WebRTC Project Authors. All rights reserved.
|
|
*
|
|
* Use of this source code is governed by a BSD-style license
|
|
* that can be found in the LICENSE file in the root of the source
|
|
* tree. An additional intellectual property rights grant can be found
|
|
* in the file PATENTS. All contributing project authors may
|
|
* be found in the AUTHORS file in the root of the source tree.
|
|
*/
|
|
|
|
#include "api/rtc_error.h"
|
|
|
|
#include <utility>
|
|
|
|
#include "test/gtest.h"
|
|
|
|
namespace webrtc {
|
|
namespace {
|
|
|
|
constexpr int kDefaultMoveOnlyIntValue = 0xbadf00d;
|
|
|
|
// Class that has no copy constructor, ensuring that RTCErrorOr can
|
|
struct MoveOnlyInt {
|
|
MoveOnlyInt() {}
|
|
explicit MoveOnlyInt(int value) : value(value) {}
|
|
MoveOnlyInt(const MoveOnlyInt& other) = delete;
|
|
MoveOnlyInt& operator=(const MoveOnlyInt& other) = delete;
|
|
MoveOnlyInt(MoveOnlyInt&& other) : value(other.value) {}
|
|
MoveOnlyInt& operator=(MoveOnlyInt&& other) {
|
|
value = other.value;
|
|
return *this;
|
|
}
|
|
|
|
int value = kDefaultMoveOnlyIntValue;
|
|
};
|
|
|
|
// Same as above. Used to test conversion from RTCErrorOr<A> to RTCErrorOr<B>
|
|
// when A can be converted to B.
|
|
struct MoveOnlyInt2 {
|
|
MoveOnlyInt2() {}
|
|
explicit MoveOnlyInt2(int value) : value(value) {}
|
|
MoveOnlyInt2(const MoveOnlyInt2& other) = delete;
|
|
MoveOnlyInt2& operator=(const MoveOnlyInt2& other) = delete;
|
|
MoveOnlyInt2(MoveOnlyInt2&& other) : value(other.value) {}
|
|
MoveOnlyInt2& operator=(MoveOnlyInt2&& other) {
|
|
value = other.value;
|
|
return *this;
|
|
}
|
|
|
|
explicit MoveOnlyInt2(MoveOnlyInt&& other) : value(other.value) {}
|
|
MoveOnlyInt2& operator=(MoveOnlyInt&& other) {
|
|
value = other.value;
|
|
return *this;
|
|
}
|
|
|
|
int value = kDefaultMoveOnlyIntValue;
|
|
};
|
|
|
|
// Test that the default constructor creates a "no error" error.
|
|
TEST(RTCErrorTest, DefaultConstructor) {
|
|
RTCError e;
|
|
EXPECT_EQ(e.type(), RTCErrorType::NONE);
|
|
EXPECT_STREQ(e.message(), "");
|
|
EXPECT_TRUE(e.ok());
|
|
}
|
|
|
|
TEST(RTCErrorTest, NormalConstructors) {
|
|
RTCError a(RTCErrorType::INVALID_PARAMETER);
|
|
EXPECT_EQ(a.type(), RTCErrorType::INVALID_PARAMETER);
|
|
EXPECT_STREQ(a.message(), "");
|
|
|
|
// Constructor that takes const char* message.
|
|
RTCError b(RTCErrorType::UNSUPPORTED_PARAMETER, "foobar");
|
|
EXPECT_EQ(b.type(), RTCErrorType::UNSUPPORTED_PARAMETER);
|
|
EXPECT_STREQ(b.message(), "foobar");
|
|
|
|
// Constructor that takes absl::string_view message.
|
|
RTCError c(RTCErrorType::SYNTAX_ERROR, absl::string_view("baz"));
|
|
EXPECT_EQ(c.type(), RTCErrorType::SYNTAX_ERROR);
|
|
EXPECT_STREQ(c.message(), "baz");
|
|
|
|
// Constructor that takes std::string message.
|
|
RTCError d(RTCErrorType::INVALID_RANGE, std::string("new"));
|
|
EXPECT_EQ(d.type(), RTCErrorType::INVALID_RANGE);
|
|
EXPECT_STREQ(d.message(), "new");
|
|
}
|
|
|
|
TEST(RTCErrorTest, MoveConstructor) {
|
|
// Static string.
|
|
RTCError a(RTCErrorType::INVALID_PARAMETER, "foo");
|
|
RTCError b(std::move(a));
|
|
EXPECT_EQ(b.type(), RTCErrorType::INVALID_PARAMETER);
|
|
EXPECT_STREQ(b.message(), "foo");
|
|
|
|
// Non-static string.
|
|
RTCError c(RTCErrorType::UNSUPPORTED_PARAMETER, std::string("bar"));
|
|
RTCError d(std::move(c));
|
|
EXPECT_EQ(d.type(), RTCErrorType::UNSUPPORTED_PARAMETER);
|
|
EXPECT_STREQ(d.message(), "bar");
|
|
}
|
|
|
|
TEST(RTCErrorTest, MoveAssignment) {
|
|
// Try all combinations of "is static string"/"is non-static string" moves.
|
|
RTCError e(RTCErrorType::INVALID_PARAMETER, "foo");
|
|
|
|
e = RTCError(RTCErrorType::UNSUPPORTED_PARAMETER, "bar");
|
|
EXPECT_EQ(e.type(), RTCErrorType::UNSUPPORTED_PARAMETER);
|
|
EXPECT_STREQ(e.message(), "bar");
|
|
|
|
e = RTCError(RTCErrorType::SYNTAX_ERROR, absl::string_view("baz"));
|
|
EXPECT_STREQ(e.message(), "baz");
|
|
|
|
e = RTCError(RTCErrorType::SYNTAX_ERROR, std::string("another"));
|
|
EXPECT_STREQ(e.message(), "another");
|
|
}
|
|
|
|
// Test that the error returned by RTCError::OK() is a "no error" error.
|
|
TEST(RTCErrorTest, OKConstant) {
|
|
RTCError ok = RTCError::OK();
|
|
EXPECT_EQ(ok.type(), RTCErrorType::NONE);
|
|
EXPECT_STREQ(ok.message(), "");
|
|
EXPECT_TRUE(ok.ok());
|
|
}
|
|
|
|
// Test that "error.ok()" behaves as expected.
|
|
TEST(RTCErrorTest, OkMethod) {
|
|
RTCError success;
|
|
RTCError failure(RTCErrorType::INTERNAL_ERROR);
|
|
EXPECT_TRUE(success.ok());
|
|
EXPECT_FALSE(failure.ok());
|
|
}
|
|
|
|
// Test that a message can be set using either static const strings or
|
|
// std::strings.
|
|
TEST(RTCErrorTest, SetMessage) {
|
|
RTCError e;
|
|
e.set_message("foo");
|
|
EXPECT_STREQ(e.message(), "foo");
|
|
|
|
e.set_message(absl::string_view("bar"));
|
|
EXPECT_STREQ(e.message(), "bar");
|
|
|
|
e.set_message(std::string("string"));
|
|
EXPECT_STREQ(e.message(), "string");
|
|
}
|
|
|
|
// Test that the default constructor creates an "INTERNAL_ERROR".
|
|
TEST(RTCErrorOrTest, DefaultConstructor) {
|
|
RTCErrorOr<MoveOnlyInt> e;
|
|
EXPECT_EQ(e.error().type(), RTCErrorType::INTERNAL_ERROR);
|
|
}
|
|
|
|
// Test that an RTCErrorOr can be implicitly constructed from a value.
|
|
TEST(RTCErrorOrTest, ImplicitValueConstructor) {
|
|
RTCErrorOr<MoveOnlyInt> e = [] { return MoveOnlyInt(100); }();
|
|
EXPECT_EQ(e.value().value, 100);
|
|
}
|
|
|
|
// Test that an RTCErrorOr can be implicitly constructed from an RTCError.
|
|
TEST(RTCErrorOrTest, ImplicitErrorConstructor) {
|
|
RTCErrorOr<MoveOnlyInt> e = [] {
|
|
return RTCError(RTCErrorType::SYNTAX_ERROR);
|
|
}();
|
|
EXPECT_EQ(e.error().type(), RTCErrorType::SYNTAX_ERROR);
|
|
}
|
|
|
|
TEST(RTCErrorOrTest, MoveConstructor) {
|
|
RTCErrorOr<MoveOnlyInt> a(MoveOnlyInt(5));
|
|
RTCErrorOr<MoveOnlyInt> b(std::move(a));
|
|
EXPECT_EQ(b.value().value, 5);
|
|
}
|
|
|
|
TEST(RTCErrorOrTest, MoveAssignment) {
|
|
RTCErrorOr<MoveOnlyInt> a(MoveOnlyInt(5));
|
|
RTCErrorOr<MoveOnlyInt> b(MoveOnlyInt(10));
|
|
a = std::move(b);
|
|
EXPECT_EQ(a.value().value, 10);
|
|
}
|
|
|
|
TEST(RTCErrorOrTest, ConversionConstructor) {
|
|
RTCErrorOr<MoveOnlyInt> a(MoveOnlyInt(1));
|
|
RTCErrorOr<MoveOnlyInt2> b(std::move(a));
|
|
}
|
|
|
|
TEST(RTCErrorOrTest, ConversionAssignment) {
|
|
RTCErrorOr<MoveOnlyInt> a(MoveOnlyInt(5));
|
|
RTCErrorOr<MoveOnlyInt2> b(MoveOnlyInt2(10));
|
|
b = std::move(a);
|
|
EXPECT_EQ(b.value().value, 5);
|
|
}
|
|
|
|
TEST(RTCErrorOrTest, OkMethod) {
|
|
RTCErrorOr<int> success(1337);
|
|
RTCErrorOr<int> error = RTCError(RTCErrorType::INTERNAL_ERROR);
|
|
EXPECT_TRUE(success.ok());
|
|
EXPECT_FALSE(error.ok());
|
|
}
|
|
|
|
TEST(RTCErrorOrTest, MoveError) {
|
|
RTCErrorOr<int> e({RTCErrorType::SYNTAX_ERROR, "message"});
|
|
RTCError err = e.MoveError();
|
|
EXPECT_EQ(err.type(), RTCErrorType::SYNTAX_ERROR);
|
|
EXPECT_STREQ(err.message(), "message");
|
|
}
|
|
|
|
TEST(RTCErrorOrTest, MoveValue) {
|
|
RTCErrorOr<MoveOnlyInt> e(MoveOnlyInt(88));
|
|
MoveOnlyInt value = e.MoveValue();
|
|
EXPECT_EQ(value.value, 88);
|
|
}
|
|
|
|
// Death tests.
|
|
// Disabled on Android because death tests misbehave on Android, see
|
|
// base/test/gtest_util.h.
|
|
#if RTC_DCHECK_IS_ON && GTEST_HAS_DEATH_TEST && !defined(WEBRTC_ANDROID)
|
|
|
|
TEST(RTCErrorOrDeathTest, ConstructWithOkError) {
|
|
RTCErrorOr<int> err;
|
|
EXPECT_DEATH(err = RTCError::OK(), "");
|
|
}
|
|
|
|
TEST(RTCErrorOrDeathTest, DereferenceErrorValue) {
|
|
RTCErrorOr<int> error = RTCError(RTCErrorType::INTERNAL_ERROR);
|
|
EXPECT_DEATH(error.value(), "");
|
|
}
|
|
|
|
TEST(RTCErrorOrDeathTest, MoveErrorValue) {
|
|
RTCErrorOr<int> error = RTCError(RTCErrorType::INTERNAL_ERROR);
|
|
EXPECT_DEATH(error.MoveValue(), "");
|
|
}
|
|
|
|
#endif // RTC_DCHECK_IS_ON && GTEST_HAS_DEATH_TEST && !defined(WEBRTC_ANDROID)
|
|
|
|
} // namespace
|
|
} // namespace webrtc
|