mirror of
https://github.com/badaix/snapcast.git
synced 2025-05-19 12:06:15 +02:00
serialize stream to json
This commit is contained in:
parent
7bec5b8744
commit
95e4a98fef
11 changed files with 172 additions and 113 deletions
|
@ -9,6 +9,14 @@ else
|
|||
TARGET_DIR ?= /usr
|
||||
endif
|
||||
|
||||
# MIPS
|
||||
# export STAGING_DIR=/home/johannes/Develop/openwrt/staging_dir
|
||||
# TOOLCHAIN_DIR=$(STAGING_DIR)/toolchain-mips_34kc_gcc-4.8-linaro_uClibc-0.9.33.2
|
||||
# LDCFLAGS=$(TOOLCHAIN_DIR)/usr/lib
|
||||
# LD_LIBRARY_PATH=$(TOOLCHAIN_DIR)/usr/lib
|
||||
# PATH=$(TOOLCHAIN_DIR)/bin:$PATH
|
||||
# CXX = $(TOOLCHAIN_DIR)/bin/mips-openwrt-linux-g++
|
||||
|
||||
ifdef ANDROID
|
||||
# Android NDK setup: (http://developer.android.com/ndk/guides/standalone_toolchain.html)
|
||||
# 1. Download NDK: http://dl.google.com/android/ndk/android-ndk-r10e-linux-x86_64.bin
|
||||
|
|
|
@ -19,6 +19,7 @@
|
|||
#ifndef MESSAGE_H
|
||||
#define MESSAGE_H
|
||||
|
||||
#include <cstdlib>
|
||||
#include <cstring>
|
||||
#include <iostream>
|
||||
#include <streambuf>
|
||||
|
|
|
@ -13,7 +13,7 @@ CXX = /usr/bin/g++
|
|||
CFLAGS = -std=c++0x -Wall -Wno-unused-function -O3 -pthread -DASIO_STANDALONE -DVERSION=\"$(VERSION)\" -I.. -I../externals/asio/asio/include -I../externals/popl/include
|
||||
LDFLAGS = -lrt -lvorbis -lvorbisenc -logg -lFLAC -lavahi-client -lavahi-common
|
||||
|
||||
OBJ = snapServer.o config.o controlServer.o controlSession.o streamServer.o streamSession.o json/jsonrpc.o pcmreader/streamManager.o pcmreader/pcmReader.o pcmreader/pipeReader.o encoder/encoderFactory.o encoder/flacEncoder.o encoder/pcmEncoder.o encoder/oggEncoder.o publishAvahi.o ../common/log.o ../message/pcmChunk.o ../message/sampleFormat.o
|
||||
OBJ = snapServer.o config.o controlServer.o controlSession.o streamServer.o streamSession.o json/jsonrpc.o pcmreader/readerUri.o pcmreader/streamManager.o pcmreader/pcmReader.o pcmreader/pipeReader.o encoder/encoderFactory.o encoder/flacEncoder.o encoder/pcmEncoder.o encoder/oggEncoder.o publishAvahi.o ../common/log.o ../message/pcmChunk.o ../message/sampleFormat.o
|
||||
BIN = snapserver
|
||||
|
||||
all: $(TARGET)
|
||||
|
|
|
@ -66,34 +66,6 @@ struct Volume
|
|||
};
|
||||
|
||||
|
||||
struct Stream
|
||||
{
|
||||
//TODO
|
||||
Stream() : id("TODO_ID"), name("TODO_NAME"), uri("file:///tmp/snapfifo")
|
||||
{
|
||||
}
|
||||
|
||||
void fromJson(const json& j)
|
||||
{
|
||||
id = jGet<std::string>(j, "id", "TODO_ID");
|
||||
name = jGet<std::string>(j, "name", "TODO_NAME");
|
||||
uri = jGet<std::string>(j, "uri", "file:///tmp/snapfifo");
|
||||
}
|
||||
|
||||
json toJson()
|
||||
{
|
||||
json j;
|
||||
j["id"] = id;
|
||||
j["name"] = name;
|
||||
j["uri"] = uri;
|
||||
return j;
|
||||
}
|
||||
|
||||
std::string id;
|
||||
std::string name;
|
||||
std::string uri;
|
||||
};
|
||||
|
||||
|
||||
struct ClientInfo
|
||||
{
|
||||
|
|
|
@ -21,75 +21,16 @@
|
|||
#include <fcntl.h>
|
||||
#include <unistd.h>
|
||||
|
||||
#include "pcmReader.h"
|
||||
#include "../encoder/encoderFactory.h"
|
||||
#include "common/utils.h"
|
||||
#include "common/compat.h"
|
||||
#include "common/snapException.h"
|
||||
#include "common/compat.h"
|
||||
#include "pcmReader.h"
|
||||
#include "common/log.h"
|
||||
|
||||
|
||||
using namespace std;
|
||||
|
||||
|
||||
ReaderUri::ReaderUri(const std::string& uri)
|
||||
{
|
||||
// https://en.wikipedia.org/wiki/Uniform_Resource_Identifier
|
||||
// scheme:[//[user:password@]host[:port]][/]path[?query][#fragment]
|
||||
// would be more elegant with regex. Not yet supported on my dev machine's gcc 4.8 :(
|
||||
size_t pos;
|
||||
this->uri = uri;
|
||||
string tmp(uri);
|
||||
|
||||
pos = tmp.find(':');
|
||||
if (pos == string::npos)
|
||||
throw invalid_argument("missing ':'");
|
||||
scheme = tmp.substr(0, pos);
|
||||
tmp = tmp.substr(pos + 1);
|
||||
logE << "scheme: '" << scheme << "' tmp: '" << tmp << "'\n";
|
||||
|
||||
if (tmp.find("//") != 0)
|
||||
throw invalid_argument("missing host separator: '//'");
|
||||
tmp = tmp.substr(2);
|
||||
|
||||
pos = tmp.find('/');
|
||||
if (pos == string::npos)
|
||||
throw invalid_argument("missing path separator: '/'");
|
||||
host = tmp.substr(0, pos);
|
||||
tmp = tmp.substr(pos);
|
||||
path = tmp;
|
||||
logE << "host: '" << host << "' tmp: '" << tmp << "' path: '" << path << "'\n";
|
||||
|
||||
pos = tmp.find('?');
|
||||
if (pos == string::npos)
|
||||
return;
|
||||
|
||||
path = tmp.substr(0, pos);
|
||||
tmp = tmp.substr(pos + 1);
|
||||
string queryStr = tmp;
|
||||
logE << "path: '" << path << "' tmp: '" << tmp << "' query: '" << queryStr << "'\n";
|
||||
|
||||
pos = tmp.find('#');
|
||||
if (pos != string::npos)
|
||||
{
|
||||
queryStr = tmp.substr(0, pos);
|
||||
tmp = tmp.substr(pos + 1);
|
||||
fragment = tmp;
|
||||
logE << "query: '" << queryStr << "' fragment: '" << fragment << "' tmp: '" << tmp << "'\n";
|
||||
}
|
||||
|
||||
vector<string> keyValueList = split(queryStr, '&');
|
||||
for (auto& kv: keyValueList)
|
||||
{
|
||||
pos = kv.find('=');
|
||||
if (pos != string::npos)
|
||||
query[kv.substr(0, pos)] = kv.substr(pos+1);
|
||||
}
|
||||
|
||||
// for (auto& kv: query)
|
||||
// logD << "key: '" << kv.first << "' value: '" << kv.second << "'\n";
|
||||
}
|
||||
|
||||
|
||||
PcmReader::PcmReader(PcmListener* pcmListener, const ReaderUri& uri) : pcmListener_(pcmListener), uri_(uri), pcmReadMs_(20)
|
||||
{
|
||||
|
|
|
@ -23,6 +23,7 @@
|
|||
#include <atomic>
|
||||
#include <string>
|
||||
#include <map>
|
||||
#include "readerUri.h"
|
||||
#include "../encoder/encoder.h"
|
||||
#include "message/sampleFormat.h"
|
||||
#include "message/header.h"
|
||||
|
@ -43,27 +44,6 @@ public:
|
|||
};
|
||||
|
||||
|
||||
struct ReaderUri
|
||||
{
|
||||
ReaderUri(const std::string& uri);
|
||||
std::string uri;
|
||||
std::string scheme;
|
||||
/* struct Authority
|
||||
{
|
||||
std::string username;
|
||||
std::string password;
|
||||
std::string host;
|
||||
size_t port;
|
||||
};
|
||||
Authority authority;
|
||||
*/
|
||||
std::string host;
|
||||
std::string path;
|
||||
std::map<std::string, std::string> query;
|
||||
std::string fragment;
|
||||
};
|
||||
|
||||
|
||||
/// Reads and decodes PCM data
|
||||
/**
|
||||
* Reads PCM and passes the data to an encoder.
|
||||
|
|
96
server/pcmreader/readerUri.cpp
Normal file
96
server/pcmreader/readerUri.cpp
Normal file
|
@ -0,0 +1,96 @@
|
|||
/***
|
||||
This file is part of snapcast
|
||||
Copyright (C) 2014-2016 Johannes Pohl
|
||||
|
||||
This program is free software: you can redistribute it and/or modify
|
||||
it under the terms of the GNU General Public License as published by
|
||||
the Free Software Foundation, either version 3 of the License, or
|
||||
(at your option) any later version.
|
||||
|
||||
This program is distributed in the hope that it will be useful,
|
||||
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
GNU General Public License for more details.
|
||||
|
||||
You should have received a copy of the GNU General Public License
|
||||
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||
***/
|
||||
|
||||
|
||||
#include <common/utils.h>
|
||||
#include <common/compat.h>
|
||||
#include <common/log.h>
|
||||
#include "readerUri.h"
|
||||
|
||||
|
||||
using namespace std;
|
||||
|
||||
|
||||
ReaderUri::ReaderUri(const std::string& uri)
|
||||
{
|
||||
// https://en.wikipedia.org/wiki/Uniform_Resource_Identifier
|
||||
// scheme:[//[user:password@]host[:port]][/]path[?query][#fragment]
|
||||
// would be more elegant with regex. Not yet supported on my dev machine's gcc 4.8 :(
|
||||
size_t pos;
|
||||
this->uri = uri;
|
||||
string tmp(uri);
|
||||
|
||||
pos = tmp.find(':');
|
||||
if (pos == string::npos)
|
||||
throw invalid_argument("missing ':'");
|
||||
scheme = tmp.substr(0, pos);
|
||||
tmp = tmp.substr(pos + 1);
|
||||
logE << "scheme: '" << scheme << "' tmp: '" << tmp << "'\n";
|
||||
|
||||
if (tmp.find("//") != 0)
|
||||
throw invalid_argument("missing host separator: '//'");
|
||||
tmp = tmp.substr(2);
|
||||
|
||||
pos = tmp.find('/');
|
||||
if (pos == string::npos)
|
||||
throw invalid_argument("missing path separator: '/'");
|
||||
host = tmp.substr(0, pos);
|
||||
tmp = tmp.substr(pos);
|
||||
path = tmp;
|
||||
logE << "host: '" << host << "' tmp: '" << tmp << "' path: '" << path << "'\n";
|
||||
|
||||
pos = tmp.find('?');
|
||||
if (pos == string::npos)
|
||||
return;
|
||||
|
||||
path = tmp.substr(0, pos);
|
||||
tmp = tmp.substr(pos + 1);
|
||||
string queryStr = tmp;
|
||||
logE << "path: '" << path << "' tmp: '" << tmp << "' query: '" << queryStr << "'\n";
|
||||
|
||||
pos = tmp.find('#');
|
||||
if (pos != string::npos)
|
||||
{
|
||||
queryStr = tmp.substr(0, pos);
|
||||
tmp = tmp.substr(pos + 1);
|
||||
fragment = tmp;
|
||||
logE << "query: '" << queryStr << "' fragment: '" << fragment << "' tmp: '" << tmp << "'\n";
|
||||
}
|
||||
|
||||
vector<string> keyValueList = split(queryStr, '&');
|
||||
for (auto& kv: keyValueList)
|
||||
{
|
||||
pos = kv.find('=');
|
||||
if (pos != string::npos)
|
||||
query[kv.substr(0, pos)] = kv.substr(pos+1);
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
json ReaderUri::toJson() const
|
||||
{
|
||||
json j;
|
||||
j["uri"] = uri;
|
||||
j["scheme"] = scheme;
|
||||
j["host"] = host;
|
||||
j["path"] = path;
|
||||
j["fragment"] = fragment;
|
||||
j["query"] = json(query);
|
||||
return j;
|
||||
}
|
||||
|
52
server/pcmreader/readerUri.h
Normal file
52
server/pcmreader/readerUri.h
Normal file
|
@ -0,0 +1,52 @@
|
|||
/***
|
||||
This file is part of snapcast
|
||||
Copyright (C) 2014-2016 Johannes Pohl
|
||||
|
||||
This program is free software: you can redistribute it and/or modify
|
||||
it under the terms of the GNU General Public License as published by
|
||||
the Free Software Foundation, either version 3 of the License, or
|
||||
(at your option) any later version.
|
||||
|
||||
This program is distributed in the hope that it will be useful,
|
||||
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
GNU General Public License for more details.
|
||||
|
||||
You should have received a copy of the GNU General Public License
|
||||
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||
***/
|
||||
|
||||
#ifndef READER_URI_H
|
||||
#define READER_URI_H
|
||||
|
||||
#include <string>
|
||||
#include <map>
|
||||
#include "../json/json.hpp"
|
||||
|
||||
using json = nlohmann::json;
|
||||
|
||||
|
||||
struct ReaderUri
|
||||
{
|
||||
ReaderUri(const std::string& uri);
|
||||
std::string uri;
|
||||
std::string scheme;
|
||||
/* struct Authority
|
||||
{
|
||||
std::string username;
|
||||
std::string password;
|
||||
std::string host;
|
||||
size_t port;
|
||||
};
|
||||
Authority authority;
|
||||
*/
|
||||
std::string host;
|
||||
std::string path;
|
||||
std::map<std::string, std::string> query;
|
||||
std::string fragment;
|
||||
|
||||
json toJson() const;
|
||||
};
|
||||
|
||||
|
||||
#endif
|
|
@ -88,4 +88,12 @@ void StreamManager::stop()
|
|||
}
|
||||
|
||||
|
||||
json StreamManager::toJson() const
|
||||
{
|
||||
json result = json::array();
|
||||
for (auto stream: streams_)
|
||||
result.push_back(stream->getUri().toJson());
|
||||
return result;
|
||||
}
|
||||
|
||||
|
||||
|
|
|
@ -18,6 +18,7 @@ public:
|
|||
void stop();
|
||||
const std::vector<PcmReaderPtr>& getStreams();
|
||||
const PcmReaderPtr getDefaultStream();
|
||||
json toJson() const;
|
||||
|
||||
private:
|
||||
std::vector<PcmReaderPtr> streams_;
|
||||
|
|
|
@ -164,7 +164,7 @@ void StreamServer::onMessageReceived(ControlSession* controlSession, const std::
|
|||
{"version", VERSION}
|
||||
}},
|
||||
{"clients", jClient},
|
||||
{"streams", "TODO"}
|
||||
{"streams", streamManager_->toJson()}
|
||||
};
|
||||
}
|
||||
else if (request.method == "Server.DeleteClient")
|
||||
|
@ -342,7 +342,7 @@ void StreamServer::start()
|
|||
|
||||
streamManager_.reset(new StreamManager(this, settings_.sampleFormat, settings_.codec, settings_.streamReadMs));
|
||||
for (auto& streamUri: settings_.pcmStreams)
|
||||
streamManager_->addStream(streamUri);
|
||||
logE << "Stream: " << streamManager_->addStream(streamUri)->getUri().toJson() << "\n";
|
||||
|
||||
streamManager_->start();
|
||||
|
||||
|
|
Loading…
Add table
Add a link
Reference in a new issue