/* 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 . */ #include "main.h" #include #include #include #include #include #include "libtgvoip/VoIPServerConfig.h" #include "audio/AudioInputPHP.h" #include "audio/AudioOutputPHP.h" using namespace tgvoip; using namespace tgvoip::audio; void VoIP::__construct(Php::Parameters ¶ms) { madeline = params[1]; current_call = params[2]; inst = new VoIPController(); inst->implData = static_cast(this); inst->SetStateCallback([](tgvoip::VoIPController *controller, int state) { static_cast(controller->implData)->updateConnectionState(controller, state); }); } void VoIP::start() { inst->Start(); } void VoIP::connect() { inst->Connect(); } void VoIP::setEncryptionKey(Php::Parameters ¶ms) { char *key = (char *)malloc(256); memcpy(key, params[0], 256); inst->SetEncryptionKey(key, (bool)params[1]); free(key); } void VoIP::setRemoteEndpoints(Php::Parameters ¶ms) { size_t len = (size_t)params[0].size(); //voip_endpoint_t* eps=(voip_endpoint_t *) malloc(sizeof(voip_endpoint_t)*len); std::vector eps; uint i; for (i = 0; i < len; i++) { std::string ip = params[0][i]["ip"]; std::string ipv6 = params[0][i]["ipv6"]; std::string peer_tag = params[0][i]["peer_tag"]; tgvoip::IPv4Address v4addr(ip); tgvoip::IPv6Address v6addr("::0"); unsigned char pTag[16]; if (ipv6 != "") { v6addr = IPv6Address(ipv6); } if (peer_tag != "") { memcpy(pTag, peer_tag.c_str(), 16); } eps.push_back(Endpoint(params[0][i]["id"], (int32_t)params[0][i]["port"], v4addr, v6addr, EP_TYPE_UDP_RELAY, pTag)); } inst->SetRemoteEndpoints(eps, params[1]); } void VoIP::release() { delete inst; } Php::Value VoIP::writeFrames(Php::Parameters ¶ms) { AudioInputPHP *in = (AudioInputPHP *)(intptr_t)inst; return in->writeFrames(params[0]); } Php::Value VoIP::readFrames() { AudioOutputPHP *out = (AudioOutputPHP *)(intptr_t)inst; return out->readFrames(); } Php::Value VoIP::getDebugString() { char buf[10240]; inst->GetDebugString(buf, 10240); return buf; } void VoIP::setNetworkType(Php::Parameters ¶ms) { inst->SetNetworkType(params[0]); } void VoIP::setMicMute(Php::Parameters ¶ms) { inst->SetMicMute(params[0]); } // jdouble recvTimeout, jdouble initTimeout, jint dataSavingMode, jboolean enableAEC, jboolean enableNS, jboolean enableAGC, jstring logFilePath void VoIP::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]; if (params.size() == 7) { strncpy(cfg.logFilePath, params[6], sizeof(cfg.logFilePath)); cfg.logFilePath[sizeof(cfg.logFilePath) - 1] = 0; } else { memset(cfg.logFilePath, 0, sizeof(cfg.logFilePath)); } 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)); } inst->SetConfig(&cfg); } void VoIP::debugCtl(Php::Parameters ¶ms) { inst->DebugCtl(params[0], params[1]); } Php::Value VoIP::getVersion() { return VoIPController::GetVersion(); } Php::Value VoIP::getPreferredRelayID() { return inst->GetPreferredRelayID(); } Php::Value VoIP::getLastError() { return inst->GetLastError(); } Php::Value VoIP::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 VoIP::setSharedConfig(Php::Parameters ¶ms) { ServerConfig::GetSharedInstance()->Update(params[0]); } Php::Value VoIP::getDebugLog() { return inst->GetDebugLog(); } void VoIP::updateConnectionState(VoIPController *cntrlr, int state) { ((Php::Object)this).call("setState", state); //setStateMethod(state); } void VoIP::startInput() { ((Php::Object)this).call("startInput"); } void VoIP::startOutput() { ((Php::Object)this).call("startOutput"); } void VoIP::stopInput() { ((Php::Object)this).call("stopInput"); } void VoIP::stopOutput() { ((Php::Object)this).call("startInput"); } void VoIP::configureAudioInput(uint32_t sampleRate, uint32_t bitsPerSample, uint32_t channels) { ((Php::Object)this).call("configureAudioInput"); } void VoIP::configureAudioOutput(uint32_t sampleRate, uint32_t bitsPerSample, uint32_t channels) { ((Php::Object)this).call("configureAudioOutput"); } float VoIP::getOutputLevel() { return (double)((Php::Object)this).call("getOutputLevel"); } 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.method("setState", { Php::ByVal("state", Php::Type::Numeric), }); voip.method("startOutput"); voip.method("stopOutput"); voip.method("stopInput"); voip.method("configureAudioOutput", { Php::ByVal("sampleRate", Php::Type::Numeric), Php::ByVal("bitsPerSample", Php::Type::Numeric), Php::ByVal("channels", Php::Type::Numeric), }); voip.method("configureAudioInput", { Php::ByVal("sampleRate", Php::Type::Numeric), Php::ByVal("bitsPerSample", Php::Type::Numeric), Php::ByVal("channels", Php::Type::Numeric), }); voip.method("getOutputLevel"); voip.method<&VoIP::__construct>("__construct", Php::Public | Php::Final, { Php::ByRef("madelineProto", Php::Type::Object), Php::ByVal("currentCall", Php::Type::Numeric), }); voip.method<&VoIP::setEncryptionKey>("setEncryptionKey", Php::Public | Php::Final, { Php::ByVal("key", Php::Type::String), Php::ByVal("isOutgoing", Php::Type::Bool), }); voip.method<&VoIP::setNetworkType>("setNetworkType", Php::Public | Php::Final, { Php::ByVal("type", Php::Type::Numeric), }); voip.method<&VoIP::setMicMute>("setMicMute", Php::Public | Php::Final, { Php::ByVal("type", Php::Type::Bool), }); voip.method<&VoIP::debugCtl>("debugCtl", Php::Public | Php::Final, { Php::ByVal("request", Php::Type::Numeric), Php::ByVal("param", Php::Type::Numeric), }); voip.method<&VoIP::setConfig>("setConfig", Php::Public | Php::Final, { // 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), Php::ByVal("dataSavingMode", Php::Type::Bool), Php::ByVal("enableAEC", Php::Type::Bool), Php::ByVal("enableNS", Php::Type::Bool), Php::ByVal("enableAGC", Php::Type::Bool), Php::ByVal("logFilePath", Php::Type::String, false), Php::ByVal("statsDumpFilePath", Php::Type::String, false), }); voip.method<&VoIP::setSharedConfig>("setSharedConfig", Php::Public | Php::Final, {Php::ByVal("config", Php::Type::Array)}); voip.method<&VoIP::setRemoteEndpoints>("setRemoteEndpoints", Php::Public | Php::Final, {Php::ByVal("endpoints", Php::Type::Array), Php::ByVal("allowP2P", Php::Type::Bool)}); voip.method<&VoIP::getDebugLog>("getDebugLog", Php::Public | Php::Final); voip.method<&VoIP::getLastError>("getLastError", Php::Public | Php::Final); voip.method<&VoIP::getPreferredRelayID>("getPreferredRelayID", Php::Public | Php::Final); voip.method<&VoIP::getVersion>("getVersion", Php::Public | Php::Final); voip.method<&VoIP::getDebugString>("getDebugString", Php::Public | Php::Final); voip.method<&VoIP::getStats>("getStats", Php::Public | Php::Final); voip.method<&VoIP::release>("release", Php::Public | Php::Final); voip.method<&VoIP::start>("start", Php::Public | Php::Final); voip.method<&VoIP::connect>("connect", Php::Public | Php::Final); voip.method<&VoIP::readFrames>("readFrames", Php::Public | Php::Final); voip.method<&VoIP::writeFrames>("writeFrames", Php::Public | Php::Final, {Php::ByVal("frames", Php::Type::String)}); 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); voip.constant("DATA_SAVING_NEVER", 0); voip.constant("DATA_SAVING_MOBILE", 1); voip.constant("DATA_SAVING_ALWAYS", 2); 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; } }