aboutsummaryrefslogtreecommitdiffstats
path: root/library/cpp/tvmauth/client/misc/threaded_updater.cpp
blob: 5d21ce67a75038e2236f45d8529382cce53497ad (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
#include "threaded_updater.h"

#include <library/cpp/tvmauth/client/exception.h>

#include <util/string/builder.h>
#include <util/system/spin_wait.h>
#include <util/system/thread.h>

namespace NTvmAuth {
    TThreadedUpdaterBase::TThreadedUpdaterBase(TDuration workerAwakingPeriod,
                                               TLoggerPtr logger,
                                               const TString& url,
                                               ui16 port,
                                               TDuration socketTimeout,
                                               TDuration connectTimeout)
        : WorkerAwakingPeriod_(workerAwakingPeriod)
        , Logger_(std::move(logger))
        , TvmUrl_(url)
        , TvmPort_(port)
        , TvmSocketTimeout_(socketTimeout)
        , TvmConnectTimeout_(connectTimeout)
        , IsStopped_(true)
    {
        Y_ENSURE_EX(Logger_, TNonRetriableException() << "Logger is required");

        ServiceTicketsDurations_.RefreshPeriod = TDuration::Hours(1);
        ServiceTicketsDurations_.Expiring = TDuration::Hours(2);
        ServiceTicketsDurations_.Invalid = TDuration::Hours(11);

        PublicKeysDurations_.RefreshPeriod = TDuration::Days(1);
        PublicKeysDurations_.Expiring = TDuration::Days(2);
        PublicKeysDurations_.Invalid = TDuration::Days(6);
    }

    TThreadedUpdaterBase::~TThreadedUpdaterBase() {
        StopWorker();
    }

    void TThreadedUpdaterBase::StartWorker() {
        if (HttpClient_) {
            HttpClient_->ResetConnection();
        }
        Thread_ = MakeHolder<TThread>(WorkerWrap, this);
        Thread_->Start();
        Started_.Wait();
        IsStopped_ = false;
    }

    void TThreadedUpdaterBase::StopWorker() {
        Event_.Signal();
        if (Thread_) {
            Thread_.Reset();
        }
    }

    TKeepAliveHttpClient& TThreadedUpdaterBase::GetClient() const {
        if (!HttpClient_) {
            HttpClient_ = MakeHolder<TKeepAliveHttpClient>(TvmUrl_, TvmPort_, TvmSocketTimeout_, TvmConnectTimeout_);
        }

        return *HttpClient_;
    }

    void TThreadedUpdaterBase::LogDebug(const TString& msg) const {
        if (Logger_) {
            Logger_->Debug(msg);
        }
    }

    void TThreadedUpdaterBase::LogInfo(const TString& msg) const {
        if (Logger_) {
            Logger_->Info(msg);
        }
    }

    void TThreadedUpdaterBase::LogWarning(const TString& msg) const {
        if (Logger_) {
            Logger_->Warning(msg);
        }
    }

    void TThreadedUpdaterBase::LogError(const TString& msg) const {
        if (Logger_) {
            Logger_->Error(msg);
        }
    }

    void* TThreadedUpdaterBase::WorkerWrap(void* arg) {
        TThread::SetCurrentThreadName("TicketParserUpd");
        TThreadedUpdaterBase& this_ = *reinterpret_cast<TThreadedUpdaterBase*>(arg);
        this_.Started_.Signal();
        this_.LogDebug("Thread-worker started");

        while (true) {
            if (this_.Event_.WaitT(this_.WorkerAwakingPeriod_)) {
                break;
            }

            try {
                this_.Worker();
                this_.GetClient().ResetConnection();
            } catch (const std::exception& e) { // impossible now
                this_.LogError(TStringBuilder() << "Failed to generate new cache: " << e.what());
            }
        }

        this_.LogDebug("Thread-worker stopped");
        this_.IsStopped_ = true;
        return nullptr;
    }
}