SpircHandler.cpp 8.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304
  1. #include "SpircHandler.h"
  2. #include <cstdint> // for uint8_t
  3. #include <memory> // for shared_ptr, make_unique, unique_ptr
  4. #include <type_traits> // for remove_extent_t
  5. #include <utility> // for move
  6. #include "BellLogger.h" // for AbstractLogger
  7. #include "CSpotContext.h" // for Context::ConfigState, Context (ptr only)
  8. #include "Logger.h" // for CSPOT_LOG
  9. #include "MercurySession.h" // for MercurySession, MercurySession::Response
  10. #include "NanoPBHelper.h" // for pbDecode
  11. #include "Packet.h" // for cspot
  12. #include "PlaybackState.h" // for PlaybackState, PlaybackState::State
  13. #include "TrackPlayer.h" // for TrackPlayer
  14. #include "TrackQueue.h"
  15. #include "TrackReference.h" // for TrackReference
  16. #include "Utils.h" // for stringHexToBytes
  17. #include "pb_decode.h" // for pb_release
  18. #include "protobuf/spirc.pb.h" // for Frame, State, Frame_fields, MessageTy...
  19. using namespace cspot;
  20. SpircHandler::SpircHandler(std::shared_ptr<cspot::Context> ctx) {
  21. this->playbackState = std::make_shared<PlaybackState>(ctx);
  22. this->trackQueue = std::make_shared<cspot::TrackQueue>(ctx, playbackState);
  23. auto EOFCallback = [this]() {
  24. if (trackQueue->isFinished()) {
  25. sendEvent(EventType::DEPLETED);
  26. }
  27. };
  28. auto trackLoadedCallback = [this](std::shared_ptr<QueuedTrack> track) {
  29. playbackState->setPlaybackState(PlaybackState::State::Playing);
  30. playbackState->updatePositionMs(track->requestedPosition);
  31. this->notify();
  32. // Send playback start event, unpause
  33. sendEvent(EventType::PLAYBACK_START, (int)track->requestedPosition);
  34. sendEvent(EventType::PLAY_PAUSE, false);
  35. };
  36. this->ctx = ctx;
  37. this->trackPlayer = std::make_shared<TrackPlayer>(
  38. ctx, trackQueue, EOFCallback, trackLoadedCallback);
  39. // Subscribe to mercury on session ready
  40. ctx->session->setConnectedHandler([this]() { this->subscribeToMercury(); });
  41. }
  42. void SpircHandler::subscribeToMercury() {
  43. auto responseLambda = [this](MercurySession::Response& res) {
  44. if (res.fail)
  45. return;
  46. sendCmd(MessageType_kMessageTypeHello);
  47. CSPOT_LOG(debug, "Sent kMessageTypeHello!");
  48. // Assign country code
  49. this->ctx->config.countryCode = this->ctx->session->getCountryCode();
  50. };
  51. auto subscriptionLambda = [this](MercurySession::Response& res) {
  52. if (res.fail)
  53. return;
  54. CSPOT_LOG(debug, "Received subscription response");
  55. this->handleFrame(res.parts[0]);
  56. };
  57. ctx->session->executeSubscription(
  58. MercurySession::RequestType::SUB,
  59. "hm://remote/user/" + ctx->config.username + "/", responseLambda,
  60. subscriptionLambda);
  61. }
  62. void SpircHandler::loadTrackFromURI(const std::string& uri) {}
  63. void SpircHandler::notifyAudioReachedPlayback() {
  64. int offset = 0;
  65. // get HEAD track
  66. auto currentTrack = trackQueue->consumeTrack(nullptr, offset);
  67. // Do not execute when meta is already updated
  68. if (trackQueue->notifyPending) {
  69. trackQueue->notifyPending = false;
  70. playbackState->updatePositionMs(currentTrack->requestedPosition);
  71. // Reset position in queued track
  72. currentTrack->requestedPosition = 0;
  73. } else {
  74. trackQueue->skipTrack(TrackQueue::SkipDirection::NEXT, false);
  75. playbackState->updatePositionMs(0);
  76. // we moved to next track, re-acquire currentTrack again
  77. currentTrack = trackQueue->consumeTrack(nullptr, offset);
  78. }
  79. this->notify();
  80. sendEvent(EventType::TRACK_INFO, currentTrack->trackInfo);
  81. }
  82. void SpircHandler::updatePositionMs(uint32_t position) {
  83. playbackState->updatePositionMs(position);
  84. notify();
  85. }
  86. void SpircHandler::disconnect() {
  87. this->trackQueue->stopTask();
  88. this->trackPlayer->stop();
  89. this->ctx->session->disconnect();
  90. }
  91. void SpircHandler::handleFrame(std::vector<uint8_t>& data) {
  92. // Decode received spirc frame
  93. playbackState->decodeRemoteFrame(data);
  94. switch (playbackState->remoteFrame.typ) {
  95. case MessageType_kMessageTypeNotify: {
  96. CSPOT_LOG(debug, "Notify frame");
  97. // Pause the playback if another player took control
  98. if (playbackState->isActive() &&
  99. playbackState->remoteFrame.device_state.is_active) {
  100. CSPOT_LOG(debug, "Another player took control, pausing playback");
  101. playbackState->setActive(false);
  102. this->trackPlayer->stop();
  103. sendEvent(EventType::DISC);
  104. }
  105. break;
  106. }
  107. case MessageType_kMessageTypeSeek: {
  108. this->trackPlayer->seekMs(playbackState->remoteFrame.position);
  109. playbackState->updatePositionMs(playbackState->remoteFrame.position);
  110. notify();
  111. sendEvent(EventType::SEEK, (int)playbackState->remoteFrame.position);
  112. //sendEvent(EventType::FLUSH);
  113. break;
  114. }
  115. case MessageType_kMessageTypeVolume:
  116. playbackState->setVolume(playbackState->remoteFrame.volume);
  117. this->notify();
  118. sendEvent(EventType::VOLUME, (int)playbackState->remoteFrame.volume);
  119. break;
  120. case MessageType_kMessageTypePause:
  121. setPause(true);
  122. break;
  123. case MessageType_kMessageTypePlay:
  124. setPause(false);
  125. break;
  126. case MessageType_kMessageTypeNext:
  127. nextSong();
  128. sendEvent(EventType::NEXT);
  129. break;
  130. case MessageType_kMessageTypePrev:
  131. previousSong();
  132. sendEvent(EventType::PREV);
  133. break;
  134. case MessageType_kMessageTypeLoad: {
  135. this->trackPlayer->start();
  136. CSPOT_LOG(debug, "Load frame %d!", playbackState->remoteTracks.size());
  137. if (playbackState->remoteTracks.size() == 0) {
  138. CSPOT_LOG(info, "No tracks in frame, stopping playback");
  139. break;
  140. }
  141. playbackState->setActive(true);
  142. playbackState->updatePositionMs(playbackState->remoteFrame.position);
  143. playbackState->setPlaybackState(PlaybackState::State::Playing);
  144. playbackState->syncWithRemote();
  145. // Update track list in case we have a new one
  146. trackQueue->updateTracks(playbackState->remoteFrame.state.position_ms,
  147. true);
  148. this->notify();
  149. // Stop the current track, if any
  150. trackPlayer->resetState();
  151. break;
  152. }
  153. case MessageType_kMessageTypeReplace: {
  154. CSPOT_LOG(debug, "Got replace frame");
  155. playbackState->syncWithRemote();
  156. trackQueue->updateTracks(playbackState->remoteFrame.state.position_ms,
  157. false);
  158. this->notify();
  159. trackPlayer->resetState();
  160. sendEvent(EventType::FLUSH);
  161. break;
  162. }
  163. case MessageType_kMessageTypeShuffle: {
  164. CSPOT_LOG(debug, "Got shuffle frame");
  165. this->notify();
  166. break;
  167. }
  168. case MessageType_kMessageTypeRepeat: {
  169. CSPOT_LOG(debug, "Got repeat frame");
  170. this->notify();
  171. break;
  172. }
  173. default:
  174. break;
  175. }
  176. }
  177. void SpircHandler::setRemoteVolume(int volume) {
  178. playbackState->setVolume(volume);
  179. notify();
  180. }
  181. void SpircHandler::notify() {
  182. this->sendCmd(MessageType_kMessageTypeNotify);
  183. }
  184. void SpircHandler::skipSong(TrackQueue::SkipDirection dir) {
  185. if (trackQueue->skipTrack(dir)) {
  186. playbackState->setPlaybackState(PlaybackState::State::Playing);
  187. notify();
  188. // Reset track state
  189. trackPlayer->resetState();
  190. sendEvent(EventType::PLAY_PAUSE, false);
  191. } else {
  192. playbackState->setPlaybackState(PlaybackState::State::Paused);
  193. playbackState->updatePositionMs(0);
  194. notify();
  195. sendEvent(EventType::PLAY_PAUSE, true);
  196. }
  197. notify();
  198. sendEvent(EventType::FLUSH);
  199. }
  200. void SpircHandler::nextSong() {
  201. skipSong(TrackQueue::SkipDirection::NEXT);
  202. }
  203. void SpircHandler::previousSong() {
  204. skipSong(TrackQueue::SkipDirection::PREV);
  205. }
  206. std::shared_ptr<TrackPlayer> SpircHandler::getTrackPlayer() {
  207. return this->trackPlayer;
  208. }
  209. void SpircHandler::sendCmd(MessageType typ) {
  210. // Serialize current player state
  211. auto encodedFrame = playbackState->encodeCurrentFrame(typ);
  212. auto responseLambda = [=](MercurySession::Response& res) {
  213. };
  214. auto parts = MercurySession::DataParts({encodedFrame});
  215. ctx->session->execute(MercurySession::RequestType::SEND,
  216. "hm://remote/user/" + ctx->config.username + "/",
  217. responseLambda, parts);
  218. }
  219. void SpircHandler::setEventHandler(EventHandler handler) {
  220. this->eventHandler = handler;
  221. }
  222. void SpircHandler::setPause(bool isPaused) {
  223. if (isPaused) {
  224. CSPOT_LOG(debug, "External pause command");
  225. playbackState->setPlaybackState(PlaybackState::State::Paused);
  226. } else {
  227. CSPOT_LOG(debug, "External play command");
  228. playbackState->setPlaybackState(PlaybackState::State::Playing);
  229. }
  230. notify();
  231. sendEvent(EventType::PLAY_PAUSE, isPaused);
  232. }
  233. void SpircHandler::sendEvent(EventType type) {
  234. auto event = std::make_unique<Event>();
  235. event->eventType = type;
  236. event->data = {};
  237. eventHandler(std::move(event));
  238. }
  239. void SpircHandler::sendEvent(EventType type, EventData data) {
  240. auto event = std::make_unique<Event>();
  241. event->eventType = type;
  242. event->data = data;
  243. eventHandler(std::move(event));
  244. }