diff --git a/CMakeLists.txt b/CMakeLists.txt
index 19de2b55eeedd216bc2415ab693b7e49eb877a45..060e39e7cd3313512519ffad474a5040b51966c3 100644
--- a/CMakeLists.txt
+++ b/CMakeLists.txt
@@ -117,8 +117,8 @@ else()
     set(THREADS_PREFER_PTHREAD_FLAG ON)
     find_package(Threads REQUIRED)
 
-    target_link_libraries(Blackjack-client ${CMAKE_SOURCE_DIR}/sockpp/cmake-build-debug/libsockpp.so Threads::Threads)
-    target_link_libraries(Blackjack-server ${CMAKE_SOURCE_DIR}/sockpp/cmake-build-debug/libsockpp.so Threads::Threads)
+    target_link_libraries(Blackjack-client ${CMAKE_SOURCE_DIR}/sockpp/cmake-build-debug/libsockpp.dylib Threads::Threads)
+    target_link_libraries(Blackjack-server ${CMAKE_SOURCE_DIR}/sockpp/cmake-build-debug/libsockpp.dylib Threads::Threads)
 endif()
 
 # copy assets (images) to binary directory
@@ -130,3 +130,6 @@ set(CMAKE_CXX_FLAGS "--coverage")
 add_library(Blackjack-lib ${SERVER_SOURCE_FILES})
 # set compile directives for server-library
 target_compile_definitions(Blackjack-lib PRIVATE BLACKJACK_SERVER=1 RAPIDJSON_HAS_STDSTRING=1)
+
+add_subdirectory(googletest)
+add_subdirectory(unit-tests)
\ No newline at end of file
diff --git a/source/general/game_state/card.cpp b/source/general/game_state/card.cpp
index 68c2ec90f1edb93c662ca8fe26868d1dfd37c51c..0015d98f163d8511cd872a70d2f748794be4bf1a 100644
--- a/source/general/game_state/card.cpp
+++ b/source/general/game_state/card.cpp
@@ -1,11 +1,13 @@
 #include "card.hpp"
 
+#include <utility>
+
 #include "../exceptions/BlackjackException.hpp"
 
-card::card(std::string id) : unique_serializable(id) { }
+card::card(std::string id) : unique_serializable(std::move(id)) { }
 
 card::card(std::string id, serializable_value<int> *val, serializable_value<int>* suit)
-        : unique_serializable(id), _value(val), _suit(suit)
+        : unique_serializable(std::move(id)), _value(val), _suit(suit)
 { }
 
 card::card(int val, int suit) :
diff --git a/source/general/game_state/game_state.cpp b/source/general/game_state/game_state.cpp
index abd2e521bab2024fa26499f6cdec0e3dba18584b..91d26ff6d764926f43f59d6a7207b20f8b7dc7c8 100644
--- a/source/general/game_state/game_state.cpp
+++ b/source/general/game_state/game_state.cpp
@@ -4,6 +4,8 @@
 
 #include "game_state.hpp"
 
+#include <utility>
+
 #include "../exceptions/BlackjackException.hpp"
 #include "../serialization/vector_utils.h"
 
@@ -19,7 +21,7 @@ game_state::game_state() : unique_serializable() {
     this->_starting_player_idx = new serializable_value<int>(0);
 }
 
-game_state::game_state(std::string id) : unique_serializable(id) {
+game_state::game_state(std::string id) : unique_serializable(std::move(id)) {
     this->_players = std::vector<player*>();
     this->_shoe = new shoe();
     this->_dealers_hand = new hand();
@@ -34,7 +36,7 @@ game_state::game_state(std::string id, std::vector<player*>& players, shoe* shoe
                        hand* dealers_hand, serializable_value<bool>* is_started,
                        serializable_value<bool>* is_finished, serializable_value<int>* round_number,
                        serializable_value<int>* current_player_idx, serializable_value<int>* starting_player_idx)
-        : unique_serializable(id),
+        : unique_serializable(std::move(id)),
           _players(players),
           _shoe(shoe),
           _dealers_hand(dealers_hand),
@@ -109,7 +111,7 @@ shoe* game_state::get_shoe() const {
 }
 
 player* game_state::get_current_player() const {
-    if(_current_player_idx == nullptr || _players.size() == 0) {
+    if(_current_player_idx == nullptr || _players.empty()) {
         return nullptr;
     }
     return _players[_current_player_idx->get_value()];
diff --git a/source/general/game_state/hand.hpp b/source/general/game_state/hand.hpp
index 7c1cc530bad80ef7e8b39968bb16e59b330429d4..793f3ecf96fca1012c594ac864dc1258a05b787f 100644
--- a/source/general/game_state/hand.hpp
+++ b/source/general/game_state/hand.hpp
@@ -23,15 +23,15 @@ public:
     virtual void write_into_json(rapidjson::Value& json, rapidjson::Document::AllocatorType& allocator) const override;
 
 // accessors
-    int get_nof_cards() const;
-    const std::vector<card*> get_cards() const;
+    int get_nof_cards() const; //checked
+    const std::vector<card*> get_cards() const; //checked
 
 // state update functions
-    void setup_round(std::string& err);
-    bool add_card(card* card, std::string& err);
+    void setup_round(std::string& err); //TODO
+    bool add_card(card* card, std::string& err); //checked
 
-    int get_points(std::string &err);
-    bool is_over_21(std::string &err);
+    int get_points(std::string &err); //checked
+    bool is_over_21(std::string &err); //TODO
     std::vector<card*>::iterator get_card_iterator();
 };
 
diff --git a/unit-tests/CMakeLists.txt b/unit-tests/CMakeLists.txt
new file mode 100644
index 0000000000000000000000000000000000000000..dee96892d5549094e85220f229626b299e933bff
--- /dev/null
+++ b/unit-tests/CMakeLists.txt
@@ -0,0 +1,13 @@
+project(Blackjack-unit-tests)
+
+set(TEST_SOURCE_FILES
+        hand.cpp
+        card.cpp
+        player.cpp
+        shoe.cpp)
+
+add_executable(Blackjack-tests ${TEST_SOURCE_FILES})
+
+target_compile_definitions(Blackjack-tests PRIVATE BLACKJACK_SERVER=1 RAPIDJSON_HAS_STDSTRING=1)
+
+target_link_libraries(Blackjack-tests gtest gtest_main Blackjack-lib)
\ No newline at end of file
diff --git a/unit-tests/card.cpp b/unit-tests/card.cpp
new file mode 100644
index 0000000000000000000000000000000000000000..6dfad54aa7c2d5eaa9f3a527cb54617d12c8bcd0
--- /dev/null
+++ b/unit-tests/card.cpp
@@ -0,0 +1,4 @@
+//
+// Created by Flavia Taras on 24.05.22.
+//
+
diff --git a/unit-tests/hand.cpp b/unit-tests/hand.cpp
new file mode 100644
index 0000000000000000000000000000000000000000..14a4caa0b92ac4edfb0957dc67472ba807faa647
--- /dev/null
+++ b/unit-tests/hand.cpp
@@ -0,0 +1,234 @@
+//
+// Created by Flavia Taras on 20.05.22.
+//
+
+//TODO: check the get_points function in many different ways with different amounts of aces and stuff
+//TODO try to get extra cards when you already have over 21 points
+//TODO: adding the same card more than the number of decks in a shoe should not work
+
+#include "gtest/gtest.h"
+#include "../source/general/game_state/hand.hpp"
+
+
+/* A test fixture allows to reuse the same configuration of objects for all
+ * tests in a test suite. The name of the fixture must match the test suite.
+ *
+ * For each test defined with TEST_F(), googletest will create a fresh test
+ * fixture at runtime, immediately initialize it via SetUp(), run the test,
+ * clean up by calling TearDown(), and then delete the test fixture.
+ * Note that different tests in the same test suite have different test fixture
+ * objects, and googletest always deletes a test fixture before it creates the
+ * next one. googletest does not reuse the same test fixture for multiple
+ * tests. Any changes one test makes to the fixture do not affect other tests.
+ */
+class HandTest : public ::testing::Test {
+
+protected:
+    virtual void SetUp() {
+        cards.resize(8); //TODO: do we need this... rather just declare a shoe...
+        for (int i = 1; i < 14; ++i) {
+            for (int j = 0; j < 5; ++j) {
+                cards[i].push_back(new card(i, j));
+            }
+        }
+    }
+
+    /* Any object and subroutine declared here can be accessed in the tests */
+
+    // cards[i][j] holds a pointer to the j-th copy of a card of value i
+    std::vector<std::vector<card*>> cards;
+    hand player_hand;
+    std::string err;
+};
+
+// Adding one card to an empty hand must succeed
+TEST_F(HandTest, AddOneCard) {
+    EXPECT_TRUE(player_hand.add_card(cards[1][0], err));
+    std::vector<card*> expected_hand = {cards[1][0]};
+    EXPECT_EQ(expected_hand, player_hand.get_cards());
+}
+
+// The initial state must be an empty hand
+TEST_F(HandTest, AddNoCards) {
+    std::vector<card*> expected_hand = {};
+    EXPECT_EQ(expected_hand, player_hand.get_cards());
+}
+
+// Adding several cards with different values to an empty hand must succeed
+TEST_F(HandTest, AddManyCards) {
+    EXPECT_TRUE(player_hand.add_card(cards[1][0], err));
+    EXPECT_TRUE(player_hand.add_card(cards[3][0], err));
+    EXPECT_TRUE(player_hand.add_card(cards[13][0], err));
+    std::vector<card*> expected_hand = {cards[1][0], cards[3][0], cards[13][0]};
+    EXPECT_EQ(expected_hand, player_hand.get_cards());
+}
+
+// Adding several cards with duplicate values to an empty hand must succeed
+TEST_F(HandTest, AddManyCardsWithDuplicates) {
+    EXPECT_TRUE(player_hand.add_card(cards[1][0], err));
+    EXPECT_TRUE(player_hand.add_card(cards[1][1], err));
+    EXPECT_TRUE(player_hand.add_card(cards[1][2], err));
+    EXPECT_TRUE(player_hand.add_card(cards[3][0], err));
+    EXPECT_TRUE(player_hand.add_card(cards[3][1], err));
+    EXPECT_TRUE(player_hand.add_card(cards[13][0], err));
+    std::vector<card*> expected_hand = {cards[1][0], cards[1][1], cards[1][2],
+                                        cards[3][0], cards[3][1], cards[13][0]};
+    EXPECT_EQ(expected_hand, player_hand.get_cards());
+}
+
+// The score of a hand with a single ace must be 11
+TEST_F(HandTest, ScoreOneAce) {
+    player_hand.add_card(cards[1][0], err);
+    EXPECT_EQ(11, player_hand.get_points(err));
+}
+
+// The score of a hand with a single card must be equal to the card's value
+TEST_F(HandTest, ScoreOneCard) {
+    player_hand.add_card(cards[8][0], err);
+    EXPECT_EQ(8, player_hand.get_points(err));
+}
+
+//The score of a hand with a single face card must be 10
+TEST_F(HandTest, ScoreOneFaceCard) {
+    player_hand.add_card(cards[12][0], err);
+    EXPECT_EQ(10, player_hand.get_points(err));
+}
+
+//TODO: do we even test for this?
+// The score of an empty hand must be zero
+TEST_F(HandTest, ScoreNoCards) {
+    EXPECT_EQ(0, player_hand.get_points(err));
+}
+
+//The score of a hand with 2 aces must be 12
+TEST_F(HandTest, ScoreTwoAces) {
+    player_hand.add_card(cards[1][0], err);
+    EXPECT_EQ(11, player_hand.get_points(err));
+    player_hand.add_card(cards[1][1], err);
+    EXPECT_EQ(12, player_hand.get_points(err));
+}
+
+// Each addition of a card must increase the score by that card's value
+TEST_F(HandTest, ScoreManyCards) {
+    player_hand.add_card(cards[2][0], err);
+    ASSERT_EQ(2, player_hand.get_points(err));
+    player_hand.add_card(cards[3][0], err);
+    ASSERT_EQ(5, player_hand.get_points(err));
+    player_hand.add_card(cards[9][0], err);
+    EXPECT_EQ(14, player_hand.get_points(err));
+}
+
+// Each addition of a card must increase the score by that card's value
+//also if one of the cards is a face card
+TEST_F(HandTest, ScoreManyCardsWithFaceCard) {
+    player_hand.add_card(cards[1][0], err);
+    ASSERT_EQ(11, player_hand.get_points(err));
+    player_hand.add_card(cards[3][0], err);
+    ASSERT_EQ(14, player_hand.get_points(err));
+    player_hand.add_card(cards[13][0], err);
+    EXPECT_EQ(14, player_hand.get_points(err));
+}
+
+// Each addition of a card must increase the score by that card's value
+//also in the special case of having an ace (drawn first)
+TEST_F(HandTest, ScoreManyCardsWithAceFirst) {
+    player_hand.add_card(cards[1][0], err);
+    ASSERT_EQ(11, player_hand.get_points(err));
+    player_hand.add_card(cards[3][0], err);
+    ASSERT_EQ(14, player_hand.get_points(err));
+    player_hand.add_card(cards[7][0], err);
+    EXPECT_EQ(21, player_hand.get_points(err));
+}
+
+// Each addition of a card must increase the score by that card's value
+//also in the special case of having an ace (drawn last)
+TEST_F(HandTest, ScoreManyCardsWithAceLast) {
+    player_hand.add_card(cards[7][0], err);
+    ASSERT_EQ(7, player_hand.get_points(err));
+    player_hand.add_card(cards[3][0], err);
+    ASSERT_EQ(10, player_hand.get_points(err));
+    player_hand.add_card(cards[1][0], err);
+    EXPECT_EQ(21, player_hand.get_points(err));
+}
+
+// Each addition of a card must increase the score by that card's value even
+// if several cards have the same value
+TEST_F(HandTest, ScoreManyCardsWithDuplicates) {
+    player_hand.add_card(cards[2][0], err);
+    ASSERT_EQ(2, player_hand.get_points(err));
+    player_hand.add_card(cards[2][1], err);
+    ASSERT_EQ(4, player_hand.get_points(err));
+    player_hand.add_card(cards[2][2], err);
+    ASSERT_EQ(6, player_hand.get_points(err));
+    player_hand.add_card(cards[7][0], err);
+    ASSERT_EQ(13, player_hand.get_points(err));
+    player_hand.add_card(cards[7][1], err);
+    ASSERT_EQ(20, player_hand.get_points(err));
+    player_hand.add_card(cards[9][0], err);
+    EXPECT_EQ(29, player_hand.get_points(err));
+}
+
+// Each addition of a card must increase the score by that card's value even
+// if several cards have the same value (with aces)
+TEST_F(HandTest, ScoreManyCardsWithDuplicatesAces) {
+    player_hand.add_card(cards[2][0], err);
+    ASSERT_EQ(2, player_hand.get_points(err));
+    player_hand.add_card(cards[2][1], err);
+    ASSERT_EQ(4, player_hand.get_points(err));
+    player_hand.add_card(cards[2][2], err);
+    ASSERT_EQ(6, player_hand.get_points(err));
+    player_hand.add_card(cards[1][0], err);
+    ASSERT_EQ(17, player_hand.get_points(err));
+    player_hand.add_card(cards[1][1], err);
+    ASSERT_EQ(18, player_hand.get_points(err));
+    player_hand.add_card(cards[9][0], err);
+    EXPECT_EQ(17, player_hand.get_points(err));
+}
+
+// A hand of one card must have count 1
+TEST_F(HandTest, CountOneCard) {
+    player_hand.add_card(cards[1][0], err);
+    EXPECT_EQ(1, player_hand.get_nof_cards());
+}
+
+// An empty hand must have count 0
+TEST_F(HandTest, CountNoCards) {
+    EXPECT_EQ(0, player_hand.get_nof_cards());
+}
+
+// Each addition of a card must increase the count by 1
+TEST_F(HandTest, CountManyCards) {
+    player_hand.add_card(cards[1][0], err);
+    ASSERT_EQ(1, player_hand.get_nof_cards());
+    player_hand.add_card(cards[3][0], err);
+    ASSERT_EQ(2, player_hand.get_nof_cards());
+    player_hand.add_card(cards[7][0], err);
+    EXPECT_EQ(3, player_hand.get_nof_cards());
+}
+
+// Each addition of a card must increase the count by 1 even if several cards
+// have the same value
+TEST_F(HandTest, CountManyCardsWithDuplicates) {
+    player_hand.add_card(cards[1][0], err);
+    ASSERT_EQ(1, player_hand.get_nof_cards());
+    player_hand.add_card(cards[1][1], err);
+    ASSERT_EQ(2, player_hand.get_nof_cards());
+    player_hand.add_card(cards[1][2], err);
+    ASSERT_EQ(3, player_hand.get_nof_cards());
+    player_hand.add_card(cards[3][0], err);
+    ASSERT_EQ(4, player_hand.get_nof_cards());
+    player_hand.add_card(cards[3][1], err);
+    ASSERT_EQ(5, player_hand.get_nof_cards());
+    player_hand.add_card(cards[7][0], err);
+    EXPECT_EQ(6, player_hand.get_nof_cards());
+}
+
+// The setup function has to remove all cards in a hand
+TEST_F(HandTest, SetupRound) {
+    player_hand.add_card(cards[1][0], err);
+    player_hand.add_card(cards[7][0], err);
+    player_hand.add_card(cards[9][0], err);
+    player_hand.setup_round(err);
+    std::vector<card*> expected_hand = {};
+    EXPECT_EQ(expected_hand, player_hand);
+}
\ No newline at end of file
diff --git a/unit-tests/player.cpp b/unit-tests/player.cpp
new file mode 100644
index 0000000000000000000000000000000000000000..6dfad54aa7c2d5eaa9f3a527cb54617d12c8bcd0
--- /dev/null
+++ b/unit-tests/player.cpp
@@ -0,0 +1,4 @@
+//
+// Created by Flavia Taras on 24.05.22.
+//
+
diff --git a/unit-tests/shoe.cpp b/unit-tests/shoe.cpp
new file mode 100644
index 0000000000000000000000000000000000000000..6dfad54aa7c2d5eaa9f3a527cb54617d12c8bcd0
--- /dev/null
+++ b/unit-tests/shoe.cpp
@@ -0,0 +1,4 @@
+//
+// Created by Flavia Taras on 24.05.22.
+//
+