2017-04-05 03:46:16 +02:00
|
|
|
/*
|
|
|
|
Copyright 2016-2017 Daniil Gentili
|
|
|
|
(https://daniil.it)
|
|
|
|
This file is part of php-libtgvoip.
|
|
|
|
php-libtgvoip is free software: you can redistribute it and/or modify it under the terms of the GNU Affero General Public License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any later version.
|
|
|
|
The PWRTelegram API 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 Affero General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License along with php-libtgvoip.
|
|
|
|
If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
|
2017-06-22 14:48:52 +02:00
|
|
|
|
2017-04-05 03:46:16 +02:00
|
|
|
#include <phpcpp.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <wchar.h>
|
|
|
|
#include <map>
|
|
|
|
#include <string>
|
2017-06-11 18:13:25 +02:00
|
|
|
#include <vector>
|
2017-04-05 03:46:16 +02:00
|
|
|
#include "libtgvoip/VoIPServerConfig.h"
|
|
|
|
#include "libtgvoip/VoIPController.h"
|
2017-06-17 17:24:34 +02:00
|
|
|
#include "libtgvoip/NetworkSocket.h"
|
2017-06-17 17:59:56 +02:00
|
|
|
#include "libtgvoip/logging.cpp"
|
|
|
|
#include "libtgvoip/VoIPController.cpp"
|
|
|
|
#include "libtgvoip/BufferInputStream.cpp"
|
|
|
|
#include "libtgvoip/BufferOutputStream.cpp"
|
|
|
|
#include "libtgvoip/BlockingQueue.cpp"
|
|
|
|
#include "libtgvoip/MediaStreamItf.cpp"
|
|
|
|
#include "libtgvoip/OpusEncoder.cpp"
|
|
|
|
#include "libtgvoip/JitterBuffer.cpp"
|
|
|
|
#include "libtgvoip/OpusDecoder.cpp"
|
|
|
|
#include "libtgvoip/BufferPool.cpp"
|
|
|
|
#include "libtgvoip/EchoCanceller.cpp"
|
|
|
|
#include "libtgvoip/CongestionControl.cpp"
|
|
|
|
#include "libtgvoip/VoIPServerConfig.cpp"
|
|
|
|
#include "libtgvoip/NetworkSocket.cpp"
|
2017-06-18 19:32:20 +02:00
|
|
|
#include "libtgvoip/os/posix/NetworkSocketPosix.cpp"
|
2017-06-17 17:24:34 +02:00
|
|
|
|
2017-06-22 14:48:52 +02:00
|
|
|
#include "libtgvoip/audio/AudioInput.cpp"
|
|
|
|
#include "libtgvoip/audio/AudioOutput.cpp"
|
2017-04-05 03:46:16 +02:00
|
|
|
|
2017-06-22 20:38:20 +02:00
|
|
|
#include "audio/AudioInputPHP.h"
|
|
|
|
#include "audio/AudioOutputPHP.h"
|
|
|
|
|
|
|
|
|
2017-06-11 18:13:25 +02:00
|
|
|
using namespace tgvoip;
|
2017-06-22 14:48:52 +02:00
|
|
|
using namespace tgvoip::audio;
|
2017-04-05 03:46:16 +02:00
|
|
|
|
|
|
|
class VoIP : public Php::Base {
|
|
|
|
public:
|
|
|
|
|
2017-06-17 17:24:34 +02:00
|
|
|
void __construct(Php::Parameters ¶ms) {
|
2017-06-18 19:32:20 +02:00
|
|
|
setStateMethod = params[0];
|
2017-06-23 22:41:23 +02:00
|
|
|
madeline = params[3];
|
|
|
|
current_call = params[4];
|
|
|
|
params[1]["call"] = static_cast<void*>(this);
|
|
|
|
params[2]["call"] = static_cast<void*>(this);
|
2017-06-19 15:53:50 +02:00
|
|
|
inst=new VoIPController(params[1], params[2]);
|
2017-06-17 17:24:34 +02:00
|
|
|
inst->implData = static_cast<void*>(this);
|
|
|
|
inst->SetStateCallback([](tgvoip::VoIPController *controller, int state) {
|
|
|
|
static_cast<VoIP*>(controller->implData)->updateConnectionState(controller, state);
|
|
|
|
});
|
|
|
|
|
2017-04-05 03:46:16 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void start() {
|
|
|
|
inst->Start();
|
|
|
|
}
|
|
|
|
|
|
|
|
void connect() {
|
|
|
|
inst->Connect();
|
|
|
|
}
|
|
|
|
void setEncryptionKey(Php::Parameters ¶ms) {
|
2017-06-25 20:36:40 +02:00
|
|
|
|
|
|
|
char* key = strdup(params[0]);
|
2017-06-25 13:12:36 +02:00
|
|
|
inst->SetEncryptionKey(key, (bool) params[1]);
|
2017-04-05 03:46:16 +02:00
|
|
|
free(key);
|
|
|
|
}
|
|
|
|
|
|
|
|
void setRemoteEndpoints(Php::Parameters ¶ms) {
|
|
|
|
|
|
|
|
size_t len=(size_t) params[0].size();
|
2017-06-11 18:13:25 +02:00
|
|
|
//voip_endpoint_t* eps=(voip_endpoint_t *) malloc(sizeof(voip_endpoint_t)*len);
|
|
|
|
std::vector<Endpoint> eps;
|
2017-04-05 03:46:16 +02:00
|
|
|
uint i;
|
|
|
|
|
|
|
|
for(i=0; i<len; i++) {
|
2017-06-11 18:13:25 +02:00
|
|
|
std::string ip = params[0][i]["ip"];
|
2017-04-05 03:46:16 +02:00
|
|
|
std::string ipv6 = params[0][i]["ipv6"];
|
|
|
|
std::string peer_tag = params[0][i]["peer_tag"];
|
2017-06-11 18:13:25 +02:00
|
|
|
|
|
|
|
|
2017-06-17 17:24:34 +02:00
|
|
|
tgvoip::IPv4Address v4addr(ip);
|
|
|
|
tgvoip::IPv6Address v6addr("::0");
|
2017-06-11 18:13:25 +02:00
|
|
|
unsigned char pTag[16];
|
|
|
|
|
|
|
|
if(ipv6 != ""){
|
|
|
|
v6addr=IPv6Address(ipv6);
|
|
|
|
}
|
2017-06-25 20:36:40 +02:00
|
|
|
|
2017-04-05 03:46:16 +02:00
|
|
|
if(peer_tag != "") {
|
2017-06-26 10:37:07 +02:00
|
|
|
memcpy(pTag, peer_tag.c_str(), 16);
|
2017-04-05 03:46:16 +02:00
|
|
|
}
|
|
|
|
|
2017-06-11 18:13:25 +02:00
|
|
|
eps.push_back(Endpoint(params[0][i]["id"], (int32_t)params[0][i]["port"], v4addr, v6addr, EP_TYPE_UDP_RELAY, pTag));
|
|
|
|
|
2017-04-05 03:46:16 +02:00
|
|
|
}
|
2017-06-11 18:13:25 +02:00
|
|
|
inst->SetRemoteEndpoints(eps, params[1]);
|
2017-04-05 03:46:16 +02:00
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void release() {
|
|
|
|
delete inst;
|
|
|
|
}
|
|
|
|
|
2017-06-19 15:53:50 +02:00
|
|
|
|
|
|
|
void writeFrames(Php::Parameters ¶ms) {
|
2017-06-19 23:38:03 +02:00
|
|
|
AudioInputPHP* in=(AudioInputPHP*)(intptr_t)inst;
|
2017-06-19 16:07:24 +02:00
|
|
|
in->writeFrames(params);
|
2017-06-19 15:53:50 +02:00
|
|
|
}
|
2017-04-05 03:46:16 +02:00
|
|
|
|
2017-06-19 15:53:50 +02:00
|
|
|
Php::Value readFrames() {
|
2017-06-19 23:38:03 +02:00
|
|
|
AudioOutputPHP* out=(AudioOutputPHP*)(intptr_t)inst;
|
2017-06-19 15:53:50 +02:00
|
|
|
return out->readFrames();
|
|
|
|
}
|
2017-06-19 16:07:24 +02:00
|
|
|
|
2017-04-05 03:46:16 +02:00
|
|
|
|
|
|
|
Php::Value getDebugString() {
|
|
|
|
char buf[10240];
|
|
|
|
inst->GetDebugString(buf, 10240);
|
|
|
|
return buf;
|
|
|
|
}
|
|
|
|
|
|
|
|
void setNetworkType(Php::Parameters ¶ms) {
|
|
|
|
inst->SetNetworkType(params[0]);
|
|
|
|
}
|
|
|
|
|
|
|
|
void setMicMute(Php::Parameters ¶ms) {
|
|
|
|
inst->SetMicMute(params[0]);
|
|
|
|
}
|
|
|
|
// jdouble recvTimeout, jdouble initTimeout, jint dataSavingMode, jboolean enableAEC, jboolean enableNS, jboolean enableAGC, jstring logFilePath
|
|
|
|
void setConfig(Php::Parameters ¶ms) {
|
|
|
|
voip_config_t cfg;
|
|
|
|
cfg.recv_timeout=params[0];
|
|
|
|
cfg.init_timeout=params[1];
|
|
|
|
cfg.data_saving=params[2];
|
|
|
|
cfg.enableAEC=params[3];
|
|
|
|
cfg.enableNS=params[4];
|
|
|
|
cfg.enableAGC=params[5];
|
2017-06-11 18:13:25 +02:00
|
|
|
if (params.size() == 7) {
|
2017-04-05 03:46:16 +02:00
|
|
|
strncpy(cfg.logFilePath, params[6], sizeof(cfg.logFilePath));
|
|
|
|
cfg.logFilePath[sizeof(cfg.logFilePath) - 1] = 0;
|
|
|
|
} else {
|
|
|
|
memset(cfg.logFilePath, 0, sizeof(cfg.logFilePath));
|
|
|
|
}
|
2017-06-11 18:13:25 +02:00
|
|
|
|
|
|
|
if (params.size() == 8) {
|
|
|
|
strncpy(cfg.statsDumpFilePath, params[7], sizeof(cfg.statsDumpFilePath));
|
|
|
|
cfg.statsDumpFilePath[sizeof(cfg.statsDumpFilePath) - 1] = 0;
|
|
|
|
} else {
|
|
|
|
memset(cfg.statsDumpFilePath, 0, sizeof(cfg.statsDumpFilePath));
|
|
|
|
}
|
2017-04-05 03:46:16 +02:00
|
|
|
inst->SetConfig(&cfg);
|
|
|
|
}
|
|
|
|
|
|
|
|
void debugCtl(Php::Parameters ¶ms) {
|
|
|
|
inst->DebugCtl(params[0], params[1]);
|
|
|
|
}
|
|
|
|
|
|
|
|
Php::Value getVersion() {
|
2017-06-11 18:13:25 +02:00
|
|
|
return VoIPController::GetVersion();
|
2017-04-05 03:46:16 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
Php::Value getPreferredRelayID() {
|
|
|
|
return inst->GetPreferredRelayID();
|
|
|
|
}
|
|
|
|
|
|
|
|
Php::Value getLastError() {
|
|
|
|
return inst->GetLastError();
|
|
|
|
}
|
|
|
|
|
|
|
|
Php::Value getStats() {
|
|
|
|
voip_stats_t _stats;
|
|
|
|
inst->GetStats(&_stats);
|
|
|
|
Php::Array stats;
|
|
|
|
stats["bytesSentWifi"] = (int64_t)_stats.bytesSentWifi;
|
|
|
|
stats["bytesSentMobile"] = (int64_t)_stats.bytesSentMobile;
|
|
|
|
stats["bytesRecvdWifi"] = (int64_t)_stats.bytesRecvdWifi;
|
|
|
|
stats["bytesRecvdMobile"] = (int64_t)_stats.bytesRecvdMobile;
|
|
|
|
return stats;
|
|
|
|
}
|
|
|
|
|
|
|
|
void setSharedConfig(Php::Parameters ¶ms) {
|
2017-06-11 18:13:25 +02:00
|
|
|
ServerConfig::GetSharedInstance()->Update(params[0]);
|
2017-04-05 03:46:16 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
Php::Value getDebugLog() {
|
|
|
|
return inst->GetDebugLog();
|
|
|
|
}
|
|
|
|
|
2017-06-17 17:24:34 +02:00
|
|
|
void updateConnectionState(VoIPController* cntrlr, int state) {
|
2017-06-23 22:41:23 +02:00
|
|
|
setStateMethod(this, state);
|
2017-04-05 03:46:16 +02:00
|
|
|
}
|
2017-06-17 17:24:34 +02:00
|
|
|
private:
|
|
|
|
|
|
|
|
VoIPController* inst;
|
|
|
|
Php::Value setStateMethod;
|
2017-06-23 22:41:23 +02:00
|
|
|
Php::Value madeline;
|
|
|
|
Php::Value current_call;
|
2017-04-05 03:46:16 +02:00
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
extern "C" {
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Function that is called by PHP right after the PHP process
|
|
|
|
* has started, and that returns an address of an internal PHP
|
|
|
|
* strucure with all the details and features of your extension
|
|
|
|
*
|
|
|
|
* @return void* a pointer to an address that is understood by PHP
|
|
|
|
*/
|
|
|
|
PHPCPP_EXPORT void *get_module()
|
|
|
|
{
|
|
|
|
// static(!) Php::Extension object that should stay in memory
|
|
|
|
// for the entire duration of the process (that's why it's static)
|
|
|
|
static Php::Extension extension("php-libtgvoip", "1.0");
|
|
|
|
|
|
|
|
|
|
|
|
// description of the class so that PHP knows which methods are accessible
|
|
|
|
Php::Class<VoIP> voip("VoIP");
|
2017-06-17 17:24:34 +02:00
|
|
|
voip.method<&VoIP::__construct> ("__construct", {
|
2017-06-23 00:29:39 +02:00
|
|
|
Php::ByVal("setStateCallable", Php::Type::Callable),
|
|
|
|
Php::ByVal("inputCallables", Php::Type::Array),
|
|
|
|
Php::ByVal("outputCallables", Php::Type::Array),
|
2017-06-23 22:41:23 +02:00
|
|
|
Php::ByRef("madelineProto", Php::Type::Object),
|
|
|
|
Php::ByRef("currentCall", Php::Type::Array),
|
2017-06-17 17:24:34 +02:00
|
|
|
});
|
2017-04-05 03:46:16 +02:00
|
|
|
voip.method<&VoIP::setEncryptionKey> ("setEncryptionKey", {
|
|
|
|
Php::ByVal("key", Php::Type::String),
|
|
|
|
Php::ByVal("isOutgoing", Php::Type::Bool),
|
|
|
|
});
|
|
|
|
voip.method<&VoIP::setNetworkType> ("setNetworkType", {
|
|
|
|
Php::ByVal("type", Php::Type::Numeric),
|
|
|
|
});
|
|
|
|
voip.method<&VoIP::setMicMute> ("setMicMute", {
|
|
|
|
Php::ByVal("type", Php::Type::Bool),
|
|
|
|
});
|
|
|
|
voip.method<&VoIP::debugCtl> ("debugCtl", {
|
|
|
|
Php::ByVal("request", Php::Type::Numeric),
|
|
|
|
Php::ByVal("param", Php::Type::Numeric),
|
|
|
|
});
|
|
|
|
voip.method<&VoIP::setConfig> ("setConfig", { // jdouble recvTimeout, jdouble initTimeout, jint dataSavingMode, jboolean enableAEC, jboolean enableNS, jboolean enableAGC, jstring logFilePath
|
|
|
|
Php::ByVal("recvTimeout", Php::Type::Float),
|
|
|
|
Php::ByVal("initTimeout", Php::Type::Float),
|
2017-06-23 22:41:23 +02:00
|
|
|
Php::ByVal("dataSavingMode", Php::Type::Bool),
|
2017-04-05 03:46:16 +02:00
|
|
|
Php::ByVal("enableAEC", Php::Type::Bool),
|
2017-06-23 22:41:23 +02:00
|
|
|
Php::ByVal("enableNS", Php::Type::Bool),
|
2017-04-05 03:46:16 +02:00
|
|
|
Php::ByVal("enableAGC", Php::Type::Bool),
|
|
|
|
Php::ByVal("logFilePath", Php::Type::String, false),
|
2017-06-11 18:13:25 +02:00
|
|
|
Php::ByVal("statsDumpFilePath", Php::Type::String, false),
|
2017-04-05 03:46:16 +02:00
|
|
|
});
|
|
|
|
voip.method<&VoIP::setSharedConfig> ("setSharedConfig", {
|
|
|
|
Php::ByVal("config", Php::Type::Array)
|
|
|
|
});
|
|
|
|
voip.method<&VoIP::setRemoteEndpoints> ("setRemoteEndpoints", {
|
|
|
|
Php::ByVal("endpoints", Php::Type::Array),
|
|
|
|
Php::ByVal("allowP2P", Php::Type::Bool)
|
|
|
|
});
|
|
|
|
voip.method<&VoIP::getDebugLog> ("getDebugLog");
|
|
|
|
voip.method<&VoIP::getLastError> ("getLastError");
|
|
|
|
voip.method<&VoIP::getPreferredRelayID> ("getPreferredRelayID");
|
|
|
|
voip.method<&VoIP::getVersion> ("getVersion");
|
|
|
|
voip.method<&VoIP::getDebugString> ("getDebugString");
|
|
|
|
voip.method<&VoIP::getStats> ("getStats");
|
|
|
|
voip.method<&VoIP::release> ("release");
|
|
|
|
voip.method<&VoIP::start> ("start");
|
|
|
|
voip.method<&VoIP::connect> ("connect");
|
2017-06-19 15:53:50 +02:00
|
|
|
|
|
|
|
voip.method<&VoIP::readFrames> ("readFrames");
|
|
|
|
voip.method<&VoIP::writeFrames> ("writeFrames", {
|
|
|
|
Php::ByVal("frames", Php::Type::String)
|
|
|
|
});
|
2017-04-05 03:46:16 +02:00
|
|
|
|
2017-06-23 22:41:23 +02:00
|
|
|
voip.constant("STATE_READY_TO_INIT", 0);
|
2017-06-23 00:49:47 +02:00
|
|
|
voip.constant("STATE_WAIT_INIT", 1);
|
|
|
|
voip.constant("STATE_WAIT_INIT_ACK", 2);
|
|
|
|
voip.constant("STATE_ESTABLISHED", 3);
|
|
|
|
voip.constant("STATE_FAILED", 4);
|
|
|
|
|
|
|
|
voip.constant("TGVOIP_ERROR_UNKNOWN", 0);
|
|
|
|
voip.constant("TGVOIP_ERROR_INCOMPATIBLE", 1);
|
|
|
|
voip.constant("TGVOIP_ERROR_TIMEOUT", 2);
|
|
|
|
voip.constant("TGVOIP_ERROR_AUDIO_IO", 3);
|
|
|
|
|
|
|
|
voip.constant("NET_TYPE_UNKNOWN", 0);
|
|
|
|
voip.constant("NET_TYPE_GPRS", 1);
|
|
|
|
voip.constant("NET_TYPE_EDGE", 2);
|
|
|
|
voip.constant("NET_TYPE_3G", 3);
|
|
|
|
voip.constant("NET_TYPE_HSPA", 4);
|
|
|
|
voip.constant("NET_TYPE_LTE", 5);
|
|
|
|
voip.constant("NET_TYPE_WIFI", 6);
|
|
|
|
voip.constant("NET_TYPE_ETHERNET", 7);
|
|
|
|
voip.constant("NET_TYPE_OTHER_HIGH_SPEED", 8);
|
|
|
|
voip.constant("NET_TYPE_OTHER_LOW_SPEED", 9);
|
|
|
|
voip.constant("NET_TYPE_DIALUP", 10);
|
|
|
|
voip.constant("NET_TYPE_OTHER_MOBILE", 11);
|
|
|
|
|
2017-04-05 03:46:16 +02:00
|
|
|
Php::Namespace danog("danog");
|
|
|
|
Php::Namespace MadelineProto("MadelineProto");
|
|
|
|
|
|
|
|
MadelineProto.add(std::move(voip));
|
|
|
|
danog.add(std::move(MadelineProto));
|
|
|
|
extension.add(std::move(danog));
|
|
|
|
|
|
|
|
return extension;
|
|
|
|
}
|
|
|
|
}
|