Add locking to FlipperState
Summary: FlipperState may be updated from multiple threads, so it needs synchronization. Note the comments in the diff about why we drop the lock before calling out to the update listener -- this can be changed if necessary in the future. Reviewed By: jknoxville Differential Revision: D18095471 fbshipit-source-id: 95d558394ae1a9b7583e5a61969e1eeda6448cff
This commit is contained in:
committed by
Facebook Github Bot
parent
f00f0ad4b9
commit
8dbf60e82e
@@ -6,9 +6,9 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
#include "FlipperState.h"
|
#include "FlipperState.h"
|
||||||
|
#include <vector>
|
||||||
#include "FlipperStateUpdateListener.h"
|
#include "FlipperStateUpdateListener.h"
|
||||||
#include "FlipperStep.h"
|
#include "FlipperStep.h"
|
||||||
#include <vector>
|
|
||||||
|
|
||||||
#if FLIPPER_DEBUG_LOG
|
#if FLIPPER_DEBUG_LOG
|
||||||
#include "Log.h"
|
#include "Log.h"
|
||||||
@@ -22,10 +22,14 @@ using namespace facebook::flipper;
|
|||||||
FlipperState::FlipperState() : logs("") {}
|
FlipperState::FlipperState() : logs("") {}
|
||||||
void FlipperState::setUpdateListener(
|
void FlipperState::setUpdateListener(
|
||||||
std::shared_ptr<FlipperStateUpdateListener> listener) {
|
std::shared_ptr<FlipperStateUpdateListener> listener) {
|
||||||
|
std::lock_guard<std::mutex> lock(mutex);
|
||||||
mListener = listener;
|
mListener = listener;
|
||||||
}
|
}
|
||||||
|
|
||||||
void FlipperState::started(std::string step) {
|
void FlipperState::started(std::string step) {
|
||||||
|
std::shared_ptr<FlipperStateUpdateListener> localListener;
|
||||||
|
{
|
||||||
|
std::lock_guard<std::mutex> lock(mutex);
|
||||||
#if FLIPPER_DEBUG_LOG
|
#if FLIPPER_DEBUG_LOG
|
||||||
log("[started] " + step);
|
log("[started] " + step);
|
||||||
#endif
|
#endif
|
||||||
@@ -33,31 +37,49 @@ void FlipperState::started(std::string step) {
|
|||||||
insertOrder.push_back(step);
|
insertOrder.push_back(step);
|
||||||
}
|
}
|
||||||
stateMap[step] = State::in_progress;
|
stateMap[step] = State::in_progress;
|
||||||
if (mListener) {
|
localListener = mListener;
|
||||||
mListener->onUpdate();
|
}
|
||||||
|
// Need to drop the lock before issuing callback because the caller
|
||||||
|
// might call us back (and is responsible for their own locking).
|
||||||
|
if (localListener) {
|
||||||
|
localListener->onUpdate();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void FlipperState::success(std::string step) {
|
void FlipperState::success(std::string step) {
|
||||||
|
std::shared_ptr<FlipperStateUpdateListener> localListener;
|
||||||
|
{
|
||||||
|
std::lock_guard<std::mutex> lock(mutex);
|
||||||
#if FLIPPER_DEBUG_LOG
|
#if FLIPPER_DEBUG_LOG
|
||||||
log("[finished] " + step);
|
log("[finished] " + step);
|
||||||
#endif
|
#endif
|
||||||
logs = logs + "[Success] " + step + "\n";
|
logs = logs + "[Success] " + step + "\n";
|
||||||
stateMap[step] = State::success;
|
stateMap[step] = State::success;
|
||||||
if (mListener) {
|
localListener = mListener;
|
||||||
mListener->onUpdate();
|
}
|
||||||
|
// Need to drop the lock before issuing callback because the caller
|
||||||
|
// might call us back (and is responsible for their own locking).
|
||||||
|
if (localListener) {
|
||||||
|
localListener->onUpdate();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void FlipperState::failed(std::string step, std::string errorMessage) {
|
void FlipperState::failed(std::string step, std::string errorMessage) {
|
||||||
|
std::shared_ptr<FlipperStateUpdateListener> localListener;
|
||||||
|
{
|
||||||
|
std::lock_guard<std::mutex> lock(mutex);
|
||||||
std::string message = "[Failed] " + step + ": " + errorMessage;
|
std::string message = "[Failed] " + step + ": " + errorMessage;
|
||||||
#if FLIPPER_DEBUG_LOG
|
#if FLIPPER_DEBUG_LOG
|
||||||
log(message);
|
log(message);
|
||||||
#endif
|
#endif
|
||||||
logs = logs + message + "\n";
|
logs = logs + message + "\n";
|
||||||
stateMap[step] = State::failed;
|
stateMap[step] = State::failed;
|
||||||
if (mListener) {
|
localListener = mListener;
|
||||||
mListener->onUpdate();
|
}
|
||||||
|
// Need to drop the lock before issuing callback because the caller
|
||||||
|
// might call us back (and is responsible for their own locking).
|
||||||
|
if (localListener) {
|
||||||
|
localListener->onUpdate();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -65,10 +87,12 @@ void FlipperState::failed(std::string step, std::string errorMessage) {
|
|||||||
// representation of the current state so the UI can show it in a more intuitive
|
// representation of the current state so the UI can show it in a more intuitive
|
||||||
// way
|
// way
|
||||||
std::string FlipperState::getState() {
|
std::string FlipperState::getState() {
|
||||||
|
std::lock_guard<std::mutex> lock(mutex);
|
||||||
return logs;
|
return logs;
|
||||||
}
|
}
|
||||||
|
|
||||||
std::vector<StateElement> FlipperState::getStateElements() {
|
std::vector<StateElement> FlipperState::getStateElements() {
|
||||||
|
std::lock_guard<std::mutex> lock(mutex);
|
||||||
std::vector<StateElement> v;
|
std::vector<StateElement> v;
|
||||||
for (auto stepName : insertOrder) {
|
for (auto stepName : insertOrder) {
|
||||||
v.push_back(StateElement(stepName, stateMap[stepName]));
|
v.push_back(StateElement(stepName, stateMap[stepName]));
|
||||||
@@ -77,6 +101,8 @@ std::vector<StateElement> FlipperState::getStateElements() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
std::shared_ptr<FlipperStep> FlipperState::start(std::string step_name) {
|
std::shared_ptr<FlipperStep> FlipperState::start(std::string step_name) {
|
||||||
|
// started() acquires the lock and we don't access any of our members below,
|
||||||
|
// so we needn't take the lock.
|
||||||
started(step_name);
|
started(step_name);
|
||||||
return std::make_shared<FlipperStep>(step_name, this);
|
return std::make_shared<FlipperStep>(step_name, this);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -7,10 +7,11 @@
|
|||||||
|
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
|
#include <map>
|
||||||
#include <memory>
|
#include <memory>
|
||||||
|
#include <mutex>
|
||||||
#include <string>
|
#include <string>
|
||||||
#include <vector>
|
#include <vector>
|
||||||
#include <map>
|
|
||||||
|
|
||||||
class FlipperStep;
|
class FlipperStep;
|
||||||
class FlipperStateUpdateListener;
|
class FlipperStateUpdateListener;
|
||||||
@@ -21,20 +22,23 @@ namespace flipper {
|
|||||||
enum State { success, in_progress, failed };
|
enum State { success, in_progress, failed };
|
||||||
|
|
||||||
class StateElement {
|
class StateElement {
|
||||||
public:
|
public:
|
||||||
StateElement(std::string name, State state): name_(name), state_(state) {};
|
StateElement(std::string name, State state) : name_(name), state_(state){};
|
||||||
std::string name_;
|
std::string name_;
|
||||||
State state_;
|
State state_;
|
||||||
};
|
};
|
||||||
|
|
||||||
}
|
} // namespace flipper
|
||||||
}
|
} // namespace facebook
|
||||||
|
|
||||||
class FlipperState {
|
class FlipperState {
|
||||||
friend FlipperStep;
|
friend FlipperStep;
|
||||||
|
|
||||||
public:
|
public:
|
||||||
FlipperState();
|
FlipperState();
|
||||||
|
// Update listeners are responsible for their own
|
||||||
|
// synchronization. There is no guarantee about which thread they
|
||||||
|
// may be called on.
|
||||||
void setUpdateListener(std::shared_ptr<FlipperStateUpdateListener>);
|
void setUpdateListener(std::shared_ptr<FlipperStateUpdateListener>);
|
||||||
std::string getState();
|
std::string getState();
|
||||||
std::vector<facebook::flipper::StateElement> getStateElements();
|
std::vector<facebook::flipper::StateElement> getStateElements();
|
||||||
@@ -50,6 +54,7 @@ class FlipperState {
|
|||||||
void failed(std::string, std::string);
|
void failed(std::string, std::string);
|
||||||
void started(std::string);
|
void started(std::string);
|
||||||
|
|
||||||
|
std::mutex mutex; // Protects all our member variables.
|
||||||
std::shared_ptr<FlipperStateUpdateListener> mListener = nullptr;
|
std::shared_ptr<FlipperStateUpdateListener> mListener = nullptr;
|
||||||
std::string logs;
|
std::string logs;
|
||||||
std::vector<std::string> insertOrder;
|
std::vector<std::string> insertOrder;
|
||||||
|
|||||||
Reference in New Issue
Block a user