oscam-2.26.01-11942-802-wit.../module-webif.c

10378 lines
320 KiB
C
Raw Normal View History

#define MODULE_LOG_PREFIX "webif"
#include "globals.h"
#ifdef WEBIF
//
// OSCam HTTP server module
//
#include <locale.h>
#include "cscrypt/md5.h"
#include "module-anticasc.h"
#include "module-cacheex.h"
#include "module-cccam.h"
#include "module-cccam-data.h"
#include "module-dvbapi.h"
#include "module-newcamd.h"
#include "module-stat.h"
#include "module-webif.h"
#include "module-webif-lib.h"
#include "module-webif-tpl.h"
#include "oscam-conf-mk.h"
#include "oscam-config.h"
#include "oscam-files.h"
#include "oscam-garbage.h"
#include "oscam-cache.h"
#include "oscam-client.h"
#include "oscam-lock.h"
#include "oscam-net.h"
#include "oscam-reader.h"
#include "oscam-string.h"
#include "oscam-time.h"
#include "oscam-work.h"
#include "oscam-ecm.h"
#define WEBIF_MAX_NODES 512
static uint64_t webif_last_nodeid[WEBIF_MAX_NODES];
#ifdef MODULE_GBOX
#include "module-gbox-sms.h"
#include "module-gbox.h"
#include "module-gbox-cards.h"
#endif
#ifdef WEBIF_WIKI
#include "webif/pages_wiki.h"
#endif
extern const struct s_cardreader *cardreaders[];
extern char cs_confdir[];
extern uint32_t ecmcwcache_size;
extern uint32_t cfg_sidtab_generation;
extern int32_t exit_oscam;
extern uint8_t cacheex_peer_id[8];
extern char *entitlement_type[];
extern char *RDR_CD_TXT[];
int8_t isactive;
int32_t ssl_active = 0;
char noncekey[33];
pthread_key_t getkeepalive;
static pthread_key_t getip;
pthread_key_t getssl;
static CS_MUTEX_LOCK http_lock;
CS_MUTEX_LOCK *lock_cs;
static uint8_t useLocal = 1;
#define PRINTF_LOCAL_D useLocal ? "%'d" : "%d"
#define PRINTF_LOCAL_F useLocal ? "%'.0f" : "%.0f"
#define PRINTF_LOCAL_MB useLocal ? "%'.2f MB" : "%.2f MB"
static int8_t httpthread_running = 0;
static pthread_t httpthread;
static int32_t sock;
enum refreshtypes { REFR_ACCOUNTS, REFR_READERS, REFR_CLIENTS, REFR_SERVER, REFR_ANTICASC, REFR_SERVICES };
//initialize structs for calculating cpu-usage depending on time between refresh of status_page
static struct pstat p_stat_cur;
static struct pstat p_stat_old;
static bool use_srvid2 = false;
/* constants for menuactivating */
#define MNU_STATUS 0
#define MNU_LIVELOG 1
#define MNU_CONFIG 2
#define MNU_READERS 3
#define MNU_USERS 4
#define MNU_SERVICES 5
#define MNU_FILES 6
#define MNU_FAILBAN 7
#define MNU_CACHEEX 8
#define MNU_SCRIPT 9
#define MNU_SHUTDOWN 10
#define MNU_TOTAL_ITEMS 11 // sum of items above
/* constants for config.html submenuactivating */
#define MNU_CFG_GLOBAL 0
#define MNU_CFG_ANTICASC 1
#define MNU_CFG_CACHE 2
#define MNU_CFG_LOADBAL 3
#define MNU_CFG_CAMD33 4
#define MNU_CFG_CAMD35 5
#define MNU_CFG_CAMD35TCP 6
#define MNU_CFG_CCCAM 7
#define MNU_CFG_NEWCAMD 8
#define MNU_CFG_GBOX 9
#define MNU_CFG_RADEGAST 10
#define MNU_CFG_SCAM 11
#define MNU_CFG_SERIAL 12
#define MNU_CFG_DVBAPI 13
#define MNU_CFG_LCD 14
#define MNU_CFG_MONITOR 15
#define MNU_CFG_WEBIF 16
#define MNU_CFG_STREAMRELAY 17
/* constants for files.html submenuactivating */
#define MNU_CFG_FVERSION 0
#define MNU_CFG_FCONF 1
#define MNU_CFG_FUSER 2
#define MNU_CFG_FSERVER 3
#define MNU_CFG_FSRVID 4
#define MNU_CFG_FDVBAPI 5
#define MNU_CFG_FACLOG 6
#define MNU_CFG_FLOGFILE 7
#define MNU_CFG_FUSERFILE 8
#define MNU_CFG_FSERVICES 9
#define MNU_CFG_FPROVID 10
#define MNU_CFG_FTIERS 11
#define MNU_CFG_FRATELIMIT 12
#define MNU_CFG_FWHITELIST 13
#define MNU_CFG_FSRVID2 14
#define MNU_CFG_FFAKECWS 15
#define MNU_CFG_FCSS 16
#define MNU_CFG_FTWIN 17
#define MNU_CFG_FKEYCW 18
/* constants for files.html for GBOX submenuactivating */
#define MNU_GBX_FSCINF 19
#define MNU_GBX_FSHRINF 20
#define MNU_GBX_FSHRONL 21
#define MNU_GBX_FVERS 22
#define MNU_GBX_FATTACK 23
#define MNU_GBX_FSMSLOG 24
#define MNU_GBX_FSMSACK 25
#define MNU_GBX_FSMSNACK 26
#define MNU_GBX_FSTAINF 27
#define MNU_GBX_FEXPINF 28
#define MNU_GBX_INFOLOG 29
#define MNU_CFG_FSOFTCAMKEY 30
#define MNU_CFG_TOTAL_ITEMS 31 // sum of items above. Use it for "All inactive" in function calls too.
static void set_status_info_var(struct templatevars *vars, char *varname, int no_data, char *fmt, double value)
{
if (no_data)
tpl_addVar(vars, TPLADD, varname, "N/A");
else
tpl_printf(vars, TPLADD, varname, fmt, value);
}
/*
* Creates vars Memory/CPU/OSCAM info in for status_page
* if check_available == 0 N/A will be displayed
* Bit mapping
* mem 0 total, 1 used & free, 2 buff, cached & free incl. buff, 3 share
* swap 4 total, 5 used & free,
* proc 6 count
* cpu 7 load
* oscam 8 vsize & rssize, 9 cpu user, 10 cpu sys, 11 cpu sum, 12 cpu refreshtime
* unused 13 - 15
*/
static void set_status_info(struct templatevars *vars, struct pstat stats){
set_status_info_var(vars, "MEM_CUR_TOTAL", stats.check_available & (1 << 0), PRINTF_LOCAL_MB , (double)stats.mem_total/(1024.0*1024.0));
set_status_info_var(vars, "MEM_CUR_FREE", stats.check_available & (1 << 1), PRINTF_LOCAL_MB , (double)stats.mem_free/(1024.0*1024.0));
set_status_info_var(vars, "MEM_CUR_USED", stats.check_available & (1 << 1), PRINTF_LOCAL_MB , (double)stats.mem_used/(1024.0*1024.0));
set_status_info_var(vars, "MEM_CUR_BUFF", stats.check_available & (1 << 2), PRINTF_LOCAL_MB , (double)stats.mem_buff/(1024.0*1024.0));
set_status_info_var(vars, "MEM_CUR_CACHED", stats.check_available & (1 << 2), PRINTF_LOCAL_MB , (double)stats.mem_cached/(1024.0*1024.0));
set_status_info_var(vars, "MEM_CUR_FREEM", stats.check_available & (1 << 2), PRINTF_LOCAL_MB , (double)stats.mem_freem/(1024.0*1024.0));
set_status_info_var(vars, "MEM_CUR_SHARE", stats.check_available & (1 << 3), PRINTF_LOCAL_MB , (double)stats.mem_share/(1024.0*1024.0));
set_status_info_var(vars, "MEM_CUR_TOTSW", stats.check_available & (1 << 4), PRINTF_LOCAL_MB , (double)stats.mem_total_swap/(1024.0*1024.0));
set_status_info_var(vars, "MEM_CUR_FRESW", stats.check_available & (1 << 5), PRINTF_LOCAL_MB , (double)stats.mem_free_swap/(1024.0*1024.0));
set_status_info_var(vars, "MEM_CUR_USESW", stats.check_available & (1 << 5), PRINTF_LOCAL_MB , (double)stats.mem_used_swap/(1024.0*1024.0));
set_status_info_var(vars, "SERVER_PROCS", stats.check_available & (1 << 6), PRINTF_LOCAL_F , stats.info_procs);
set_status_info_var(vars, "CPU_LOAD_0", stats.check_available & (1 << 7), "%.2f" , stats.cpu_avg[0]);
set_status_info_var(vars, "CPU_LOAD_1", stats.check_available & (1 << 7), "%.2f" , stats.cpu_avg[1]);
set_status_info_var(vars, "CPU_LOAD_2", stats.check_available & (1 << 7), "%.2f" , stats.cpu_avg[2]);
set_status_info_var(vars, "OSCAM_VMSIZE", stats.check_available & (1 << 8), PRINTF_LOCAL_MB , (double)stats.vsize/(1024.0*1024.0));
set_status_info_var(vars, "OSCAM_RSSSIZE", stats.check_available & (1 << 8), PRINTF_LOCAL_MB , (double)stats.rss/(1024.0*1024.0));
set_status_info_var(vars, "OSCAM_CPU_USER", stats.check_available & (1 << 9), "%.2f %%" , stats.cpu_usage_user);
set_status_info_var(vars, "OSCAM_CPU_SYS", stats.check_available & (1 << 10), "%.2f %%" , stats.cpu_usage_sys);
double sum_cpu = stats.cpu_usage_sys + stats.cpu_usage_user;
set_status_info_var(vars, "OSCAM_CPU_SUM", stats.check_available & (1 << 11), "%.2f %%" , sum_cpu);
if (stats.check_available & (1 << 12))
{
tpl_addVar(vars, TPLADD, "OSCAM_REFRESH" , "N/A");
}
else
{
tpl_printf(vars, TPLADD, "OSCAM_REFRESH" , "%02"PRId64":%02"PRId64":%02"PRId64"h",
stats.gone_refresh / 3600,
(stats.gone_refresh / 60) % 60,
stats.gone_refresh % 60);
}
}
static void clear_account_stats(struct s_auth *account)
{
account->cwfound = 0;
account->cwcache = 0;
account->cwnot = 0;
account->cwtun = 0;
account->cwignored = 0;
account->cwtout = 0;
account->emmok = 0;
account->emmnok = 0;
#ifdef CW_CYCLE_CHECK
account->cwcycledchecked = 0;
account->cwcycledok = 0;
account->cwcyclednok = 0;
account->cwcycledign = 0;
#endif
cacheex_clear_account_stats(account);
}
static void clear_all_account_stats(void)
{
struct s_auth *account = cfg.account;
while(account)
{
clear_account_stats(account);
account = account->next;
}
}
#ifdef CS_CACHEEX
static void cacheex_clear_all_stats(void)
{
struct s_auth *account = cfg.account;
while(account)
{
cacheex_clear_account_stats(account);
account = account->next;
}
struct s_client *cl;
for(cl = first_client->next; cl ; cl = cl->next)
{
cacheex_clear_client_stats(cl);
ll_clear_data(cl->ll_cacheex_stats);
}
cacheex_clear_client_stats(first_client);
}
#endif
static void clear_info_clients_stats(void)
{
first_client->cwfound = 0;
first_client->cwcache = 0;
first_client->cwnot = 0;
first_client->cwtun = 0;
first_client->cwignored = 0;
first_client->cwtout = 0;
first_client->emmok = 0;
first_client->emmnok = 0;
cacheex_clear_client_stats(first_client);
}
static void clear_info_readers_stats(void)
{
int8_t i;
cs_writelock(__func__, &readerlist_lock);
LL_ITER itr = ll_iter_create(configured_readers);
struct s_reader *rdr;
while((rdr = ll_iter_next(&itr)))
{
rdr->webif_ecmsok = 0;
rdr->webif_ecmsnok = 0;
rdr->webif_ecmstout = 0;
rdr->webif_ecmsfilteredhead = 0;
rdr->webif_ecmsfilteredlen = 0;
for(i = 0; i < 4; i++)
{
rdr->webif_emmerror[i] = 0;
rdr->webif_emmwritten[i] = 0;
rdr->webif_emmskipped[i] = 0;
rdr->webif_emmblocked[i] = 0;
}
}
cs_writeunlock(__func__, &readerlist_lock);
}
static void set_ecm_info(struct templatevars * vars)
{
//if one of the stats overloaded, reset all stats!
if(first_client->cwfound<0
|| first_client->cwnot<0
|| first_client->cwignored<0
|| first_client->cwtout<0
|| first_client->cwcache<0
|| first_client->cwtun<0
|| first_client->emmok<0
|| first_client->emmnok<0
#ifdef CS_CACHEEX
|| first_client->cwcacheexgot<0
|| first_client->cwcacheexpush<0
|| first_client->cwcacheexhit<0
#ifdef CS_CACHEEX_AIO
|| first_client->cwcacheexgotlg<0
|| first_client->cwcacheexpushlg<0
#endif
#endif
){
clear_info_clients_stats();
}
//end reset stats
int ecm = 0, emm = 0;
double ecmsum = first_client->cwfound + first_client->cwcache + first_client->cwnot + first_client->cwtout; //dont count TUN its included
if(ecmsum < 1) {ecmsum = 1; ecm = 1;}
double ecmpos = first_client->cwfound + first_client->cwcache; // dont count TUN its included
if(ecmpos < 1) {ecmpos = 1;}
double ecmneg = first_client->cwnot + first_client->cwtout; //dont count IGN its not part of neagtiv
if(ecmneg < 1) {ecmneg = 1;}
double emmsum = first_client->emmok + first_client->emmnok;
if(emmsum < 1) {emmsum = 1; emm = 1;}
tpl_printf(vars, TPLADD, "TOTAL_ECM_MIN", "%d", first_client->n_request[0]);
tpl_printf(vars, TPLADD, "TOTAL_CW", PRINTF_LOCAL_F, !ecm ? ecmsum : 0);
tpl_printf(vars, TPLADD, "TOTAL_CWOK", PRINTF_LOCAL_F, (double)first_client->cwfound);
tpl_printf(vars, TPLADD, "TOTAL_CWNOK", PRINTF_LOCAL_F, (double)first_client->cwnot);
tpl_printf(vars, TPLADD, "TOTAL_CWIGN", PRINTF_LOCAL_F, (double)first_client->cwignored);
tpl_printf(vars, TPLADD, "TOTAL_CWTOUT", PRINTF_LOCAL_F, (double)first_client->cwtout);
tpl_printf(vars, TPLADD, "TOTAL_CWCACHE", PRINTF_LOCAL_F, (double)first_client->cwcache);
tpl_printf(vars, TPLADD, "TOTAL_CWTUN", PRINTF_LOCAL_F, (double)first_client->cwtun);
tpl_printf(vars, TPLADD, "TOTAL_CWPOS", PRINTF_LOCAL_F, (double)first_client->cwfound + (double)first_client->cwcache);
tpl_printf(vars, TPLADD, "TOTAL_CWNEG", PRINTF_LOCAL_F, (double)first_client->cwnot + (double)first_client->cwtout);
tpl_printf(vars, TPLADD, "TOTAL_EM", PRINTF_LOCAL_F, !emm ? emmsum : 0);
tpl_printf(vars, TPLADD, "TOTAL_EMOK", PRINTF_LOCAL_F, (double)first_client->emmok);
tpl_printf(vars, TPLADD, "TOTAL_EMNOK", PRINTF_LOCAL_F, (double)first_client->emmnok);
tpl_printf(vars, TPLADD, "REL_CWOK", "%.2f", (double)first_client->cwfound * 100 / ecmsum);
tpl_printf(vars, TPLADD, "REL_CWNOK", "%.2f", (double)first_client->cwnot * 100 / ecmsum);
//tpl_printf(vars, TPLADD, "REL_CWIGN", "%.2f", (double)first_client->cwignored * 100 / ecmsum);
tpl_printf(vars, TPLADD, "REL_CWTOUT", "%.2f", (double)first_client->cwtout * 100 / ecmsum);
tpl_printf(vars, TPLADD, "REL_CWCACHE", "%.2f", (double)first_client->cwcache * 100 / ecmsum);
tpl_printf(vars, TPLADD, "REL_CWTUN", "%.2f", (double)first_client->cwtun * 100 / ecmsum);
tpl_printf(vars, TPLADD, "REL_CWPOS", "%.2f", (double)(first_client->cwfound + first_client->cwcache) * 100 / ecmsum);
tpl_printf(vars, TPLADD, "REL_CWNEG", "%.2f", (double)(first_client->cwnot + first_client->cwtout) * 100 / ecmsum);
tpl_printf(vars, TPLADD, "REL_EMOK", "%.2f", (double)first_client->emmok * 100 / emmsum);
tpl_printf(vars, TPLADD, "REL_EMNOK", "%.2f", (double)first_client->emmnok * 100 / emmsum);
tpl_printf(vars, TPLADD, "REL_CWPOSOK", "%.2f", (double)first_client->cwfound * 100 / ecmpos);
tpl_printf(vars, TPLADD, "REL_CWPOSCACHE", "%.2f", (double)first_client->cwcache * 100 / ecmpos);
tpl_printf(vars, TPLADD, "REL_CWNEGNOK", "%.2f", (double)first_client->cwnot * 100 / ecmneg);
//tpl_printf(vars, TPLADD, "REL_CWNEGIGN", "%.2f", (double)first_client->cwignored * 100 / ecmneg);
tpl_printf(vars, TPLADD, "REL_CWNEGTOUT", "%.2f", (double)first_client->cwtout * 100 / ecmneg);
double totalrdrneg = 0, totalrdrpos = 0;
double totalrdrok = 0, totalrdrnok = 0, totalrdrtout = 0;
double flen = 0, fhead = 0;
double teruk = 0, terg = 0, ters = 0, teruq = 0;
double twruk = 0, twrg = 0, twrs = 0, twruq = 0;
double tskuk = 0, tskg = 0, tsks = 0, tskuq = 0;
double tbluk = 0, tblg = 0, tbls = 0, tbluq = 0;
cs_readlock(__func__, &readerlist_lock);
LL_ITER itr = ll_iter_create(configured_readers);
struct s_reader *rdr;
while((rdr = ll_iter_next(&itr)))
{
if (rdr->webif_ecmsok) { totalrdrok += rdr->webif_ecmsok; }
if (rdr->webif_ecmsnok) { totalrdrnok += rdr->webif_ecmsnok; }
if (rdr->webif_ecmstout) { totalrdrtout += rdr->webif_ecmstout; }
if (rdr->webif_ecmsfilteredlen) { flen += rdr->webif_ecmsfilteredlen; }
if (rdr->webif_ecmsfilteredhead) { fhead += rdr->webif_ecmsfilteredhead; }
if (rdr->webif_emmerror[0]) { teruk += rdr->webif_emmerror[0]; }
if (rdr->webif_emmerror[1]) { teruq += rdr->webif_emmerror[1]; }
if (rdr->webif_emmerror[2]) { ters += rdr->webif_emmerror[2]; }
if (rdr->webif_emmerror[3]) { terg += rdr->webif_emmerror[3]; }
if (rdr->webif_emmwritten[0]) { twruk += rdr->webif_emmwritten[0]; }
if (rdr->webif_emmwritten[1]) { twruq += rdr->webif_emmwritten[1]; }
if (rdr->webif_emmwritten[2]) { twrs += rdr->webif_emmwritten[2]; }
if (rdr->webif_emmwritten[3]) { twrg += rdr->webif_emmwritten[3]; }
if (rdr->webif_emmskipped[0]) { tskuk += rdr->webif_emmskipped[0]; }
if (rdr->webif_emmskipped[1]) { tskuq += rdr->webif_emmskipped[1]; }
if (rdr->webif_emmskipped[2]) { tsks += rdr->webif_emmskipped[2]; }
if (rdr->webif_emmskipped[3]) { tskg += rdr->webif_emmskipped[3]; }
if (rdr->webif_emmblocked[0]) { tbluk += rdr->webif_emmblocked[0]; }
if (rdr->webif_emmblocked[1]) { tbluq += rdr->webif_emmblocked[1]; }
if (rdr->webif_emmblocked[2]) { tbls += rdr->webif_emmblocked[2]; }
if (rdr->webif_emmblocked[3]) { tblg += rdr->webif_emmblocked[3]; }
}
cs_readunlock(__func__, &readerlist_lock);
totalrdrneg = totalrdrnok + totalrdrtout;
totalrdrpos = totalrdrok;
ecmsum = totalrdrok + totalrdrnok + totalrdrtout;
tpl_printf(vars, TPLADD, "TOTAL_CWOK_READERS", PRINTF_LOCAL_F, totalrdrok);
tpl_printf(vars, TPLADD, "TOTAL_CWNOK_READERS", PRINTF_LOCAL_F, totalrdrnok);
tpl_printf(vars, TPLADD, "TOTAL_CWTOUT_READERS", PRINTF_LOCAL_F, totalrdrtout);
tpl_printf(vars, TPLADD, "REL_CWOK_READERS", "%.2f", ecmsum ? totalrdrok * 100 / ecmsum : 0);
tpl_printf(vars, TPLADD, "REL_CWNOK_READERS", "%.2f", ecmsum ? totalrdrnok * 100 / ecmsum : 0);
tpl_printf(vars, TPLADD, "REL_CWTOUT_READERS", "%.2f", ecmsum ? totalrdrtout * 100 / ecmsum : 0);
tpl_printf(vars, TPLADD, "TOTAL_CWPOS_READERS", PRINTF_LOCAL_F, totalrdrpos);
tpl_printf(vars, TPLADD, "TOTAL_CWNEG_READERS", PRINTF_LOCAL_F, totalrdrneg);
tpl_printf(vars, TPLADD, "REL_CWPOS_READERS", "%.2f", ecmsum ? totalrdrpos * 100 / ecmsum : 0);
tpl_printf(vars, TPLADD, "REL_CWNEG_READERS", "%.2f", ecmsum ? totalrdrneg * 100 / ecmsum : 0);
tpl_printf(vars, TPLADD, "TOTAL_ELENR", PRINTF_LOCAL_F, flen);
tpl_printf(vars, TPLADD, "TOTAL_EHEADR", PRINTF_LOCAL_F, fhead);
tpl_printf(vars, TPLADD, "TOTAL_SUM_ALL_READERS_ECM", PRINTF_LOCAL_F, ecmsum);
tpl_printf(vars, TPLADD, "TOTAL_EMMERRORUK_READERS", PRINTF_LOCAL_F, teruk);
tpl_printf(vars, TPLADD, "TOTAL_EMMERRORG_READERS", PRINTF_LOCAL_F, terg);
tpl_printf(vars, TPLADD, "TOTAL_EMMERRORS_READERS", PRINTF_LOCAL_F, ters);
tpl_printf(vars, TPLADD, "TOTAL_EMMERRORUQ_READERS", PRINTF_LOCAL_F, teruq);
tpl_printf(vars, TPLADD, "TOTAL_EMMWRITTENUK_READERS", PRINTF_LOCAL_F, twruk);
tpl_printf(vars, TPLADD, "TOTAL_EMMWRITTENG_READERS", PRINTF_LOCAL_F, twrg);
tpl_printf(vars, TPLADD, "TOTAL_EMMWRITTENS_READERS", PRINTF_LOCAL_F, twrs);
tpl_printf(vars, TPLADD, "TOTAL_EMMWRITTENUQ_READERS", PRINTF_LOCAL_F, twruq);
tpl_printf(vars, TPLADD, "TOTAL_EMMSKIPPEDUK_READERS", PRINTF_LOCAL_F, tskuk);
tpl_printf(vars, TPLADD, "TOTAL_EMMSKIPPEDG_READERS", PRINTF_LOCAL_F, tskg);
tpl_printf(vars, TPLADD, "TOTAL_EMMSKIPPEDS_READERS", PRINTF_LOCAL_F, tsks);
tpl_printf(vars, TPLADD, "TOTAL_EMMSKIPPEDUQ_READERS", PRINTF_LOCAL_F, tskuq);
tpl_printf(vars, TPLADD, "TOTAL_EMMBLOCKEDUK_READERS", PRINTF_LOCAL_F, tbluk);
tpl_printf(vars, TPLADD, "TOTAL_EMMBLOCKEDG_READERS", PRINTF_LOCAL_F, tblg);
tpl_printf(vars, TPLADD, "TOTAL_EMMBLOCKEDS_READERS", PRINTF_LOCAL_F, tbls);
tpl_printf(vars, TPLADD, "TOTAL_EMMBLOCKEDUQ_READERS", PRINTF_LOCAL_F, tbluq);
emmsum = teruk + terg + ters + teruq + twruk + twrg + twrs + twruq + tskuk + tskg + tsks + tskuq + tbluk + tblg + tbls + tbluq;
tpl_printf(vars, TPLADD, "TOTAL_SUM_ALL_READERS_EMM", PRINTF_LOCAL_F, emmsum);
}
static void refresh_oscam(enum refreshtypes refreshtype)
{
switch(refreshtype)
{
case REFR_ACCOUNTS:
cs_log("Refresh Accounts requested by WebIF from %s", cs_inet_ntoa(GET_IP()));
cs_accounts_chk();
break;
case REFR_READERS:
cs_log("Refresh Readers requested by WebIF from %s", cs_inet_ntoa(GET_IP()));
reload_readerdb();
break;
case REFR_CLIENTS:
cs_log("Refresh Clients requested by WebIF from %s", cs_inet_ntoa(GET_IP()));
cs_reinit_clients(cfg.account);
break;
case REFR_SERVER:
cs_log("Refresh Server requested by WebIF from %s", cs_inet_ntoa(GET_IP()));
reload_global_config(); // Wczytaj ponownie globalną konfigurację
break;
case REFR_SERVICES:
cs_log("Refresh Services requested by WebIF from %s", cs_inet_ntoa(GET_IP()));
//init_sidtab();
cs_accounts_chk();
break;
#ifdef CS_ANTICASC
case REFR_ANTICASC:
cs_log("Refresh Anticascading requested by WebIF from %s", cs_inet_ntoa(GET_IP()));
ac_init_stat();
struct s_client *cl;
struct s_auth *account;
for(cl = first_client->next; cl ; cl = cl->next)
{
if(cl->typ == 'c' && (account = cl->account))
{
cl->ac_limit = (account->ac_users * 100 + 80) * cfg.ac_stime;
}
}
break;
#endif
default:
break;
}
}
/*
* load historical values from ringbuffer and return it in the right order
* as string. Value should be freed with free_mk_t()
*/
static char *get_ecm_historystring(struct s_client *cl)
{
if(cl)
{
int32_t k, i, pos = 0, needed = 1, v;
char *value, *dot = "";
int32_t ptr = cl->cwlastresptimes_last;
needed = CS_ECM_RINGBUFFER_MAX * 6; //5 digits + delimiter
if(!cs_malloc(&value, needed)) { return ""; }
k = ptr + 1;
for(i = 0; i < CS_ECM_RINGBUFFER_MAX; i++)
{
if(k >= CS_ECM_RINGBUFFER_MAX)
{ k = 0; }
v = cl->cwlastresptimes[k].duration;
if(v > 0 && v < (int32_t)cfg.ctimeout * 5)
{
pos += snprintf(value + pos, needed - pos, "%s%d", dot, v);
dot = ",";
}
k++;
}
if(cs_strlen(value) == 0)
{
NULLFREE(value);
return "";
}
else { return value; }
}
else
{
return "";
}
}
static char *get_ecm_fullhistorystring(struct s_client *cl)
{
if(cl)
{
int32_t k, i, pos = 0, needed = 1, v;
char *value, *dot = "";
int32_t ptr = cl->cwlastresptimes_last;
needed = CS_ECM_RINGBUFFER_MAX * 20; //5 digits + : + returncode(2) + : + time(10) + delimiter
if(!cs_malloc(&value, needed)) { return ""; }
k = ptr + 1;
for(i = 0; i < CS_ECM_RINGBUFFER_MAX; i++)
{
if(k >= CS_ECM_RINGBUFFER_MAX)
{ k = 0; }
v = cl->cwlastresptimes[k].duration;
if(v > 0 && v < (int32_t)cfg.ctimeout * 5)
{
pos += snprintf(value + pos, needed - pos, "%s%d:%d:%" PRId64, dot, cl->cwlastresptimes[k].duration, cl->cwlastresptimes[k].rc, (int64_t)cl->cwlastresptimes[k].timestamp);
dot = ",";
}
k++;
}
return (value);
}
else
{
return "";
}
}
/*
* Set the active menu to a different CSS class
*/
static void setActiveMenu(struct templatevars *vars, int8_t active)
{
int8_t i;
for(i = 0; i < MNU_TOTAL_ITEMS; i++)
{
tpl_printf(vars, TPLADD, "TMP", "MENUACTIVE%d", i);
if(i == active)
{ tpl_addVar(vars, TPLADD, tpl_getVar(vars, "TMP"), "menu_selected"); }
else
{ tpl_addVar(vars, TPLADD, tpl_getVar(vars, "TMP"), "menu"); }
}
#ifdef WEBIF_LIVELOG
tpl_addVar(vars, TPLADD, "LOGPAGEMENU", tpl_getTpl(vars, "LOGMENU"));
#endif
}
/*
* Set the active submenu to a different CSS class
*/
static void setActiveSubMenu(struct templatevars *vars, int8_t active)
{
int8_t i;
for(i = 0; i < MNU_CFG_TOTAL_ITEMS; i++)
{
tpl_printf(vars, TPLADD, "TMP", "CMENUACTIVE%d", i);
if(i == active)
{ tpl_addVar(vars, TPLADD, tpl_getVar(vars, "TMP"), "configmenu_selected"); }
else
{ tpl_addVar(vars, TPLADD, tpl_getVar(vars, "TMP"), "configmenu"); }
}
}
static void webif_save_config(char *section, struct templatevars *vars, struct uriparams *params)
{
if(!streq(getParam(params, "action"), "execute"))
{ return; }
if(cfg.http_readonly)
{
tpl_addMsg(vars, "WebIf is in readonly mode. No changes are possible!");
return;
}
int i;
int cnt = (*params).paramcount;
// First pass: check for checkbox values (value=1) to identify which checkboxes are checked
// This is needed because hidden fields (value=0) should not override checkbox values
bool *checkbox_checked = NULL;
if(cnt > 0 && !cs_malloc(&checkbox_checked, cnt * sizeof(bool)))
{
return; // Allocation failed, proceed with original behavior
}
for(i = 0; i < cnt; i++)
{
char *token = (*params).params[i];
char *value = (*params).values[i];
if(!streq(token, "part") && !streq(token, "action"))
{
// Check if this is a checkbox with value=1
if(strcmp(value, "1") == 0)
{
checkbox_checked[i] = true;
}
}
}
// Second pass: apply settings, skip value=0 if checkbox is checked (hidden field)
for(i = 0; i < cnt; i++)
{
char *token = (*params).params[i];
char *value = (*params).values[i];
if(!streq(token, "part") && !streq(token, "action"))
{
// Skip hidden field value=0 if checkbox is checked (value=1 exists)
if(strcmp(value, "0") == 0)
{
bool checkbox_will_be_checked = false;
int j;
for(j = i + 1; j < cnt; j++)
{
if(strcmp((*params).params[j], token) == 0 && strcmp((*params).values[j], "1") == 0)
{
checkbox_will_be_checked = true;
break;
}
}
if(checkbox_will_be_checked)
{
continue; // Skip this hidden field, checkbox will set it to 1
}
}
config_set(section, token, value);
}
}
free(checkbox_checked);
if(write_config() == 0)
{
tpl_addMsg(vars, "Configuration was saved.");
enum refreshtypes ref_type = REFR_SERVER;
if(streq(getParam(params, "part"), "anticasc"))
{ ref_type = REFR_ANTICASC; }
refresh_oscam(ref_type);
}
else
{
tpl_addMsg(vars, "ERROR: Failed to write config file!!!");
}
}
static char *send_oscam_config_global(struct templatevars *vars, struct uriparams *params)
{
setActiveSubMenu(vars, MNU_CFG_GLOBAL);
webif_save_config("global", vars, params);
if(IP_ISSET(cfg.srvip))
{ tpl_addVar(vars, TPLADD, "SERVERIP", cs_inet_ntoa(cfg.srvip)); }
tpl_printf(vars, TPLADD, "NICE", "%d", cfg.nice);
tpl_printf(vars, TPLADD, "BINDWAIT", "%d", cfg.bindwait);
tpl_printf(vars, TPLADD, "TMP", "NETPRIO%d", cfg.netprio);
tpl_addVar(vars, TPLADD, tpl_getVar(vars, "TMP"), "selected");
tpl_printf(vars, TPLADD, "PIDFILE", "%s", ESTR(cfg.pidfile));
if(cfg.usrfile != NULL) { tpl_addVar(vars, TPLADD, "USERFILE", cfg.usrfile); }
if(cfg.disableuserfile == 0) { tpl_addVar(vars, TPLADD, "DISABLEUSERFILECHECKED", "checked"); }
if(cfg.usrfileflag == 1) { tpl_addVar(vars, TPLADD, "USERFILEFLAGCHECKED", "selected"); }
if(cfg.mailfile != NULL) { tpl_addVar(vars, TPLADD, "MAILFILE", cfg.mailfile); }
if(cfg.disablemail == 0) { tpl_addVar(vars, TPLADD, "DISABLEMAILCHECKED", "checked"); }
char *value = mk_t_logfile();
tpl_addVar(vars, TPLADD, "LOGFILE", value);
free_mk_t(value);
if(cfg.disablelog == 0) { tpl_addVar(vars, TPLADD, "DISABLELOGCHECKED", "checked"); }
tpl_printf(vars, TPLADD, "MAXLOGSIZE", "%d", cfg.max_log_size);
tpl_addVar(vars, TPLADD, "LOGDUPSCHECKED", (cfg.logduplicatelines == 1) ? "checked" : "");
tpl_printf(vars, TPLADD, "INITIALDEBUGLEVEL", "%u", cfg.initial_debuglevel);
if(cfg.cwlogdir != NULL) { tpl_addVar(vars, TPLADD, "CWLOGDIR", cfg.cwlogdir); }
if(cfg.emmlogdir != NULL) { tpl_addVar(vars, TPLADD, "EMMLOGDIR", cfg.emmlogdir); }
tpl_addVar(vars, TPLADD, "ECMFMT", cfg.ecmfmt);
tpl_printf(vars, TPLADD, "LOGHISTORYLINES", "%u", cfg.loghistorylines);
if(cfg.sysloghost != NULL) { tpl_addVar(vars, TPLADD, "SYSLOGHOST", cfg.sysloghost); }
tpl_printf(vars, TPLADD, "SYSLOGPORT", "%u", cfg.syslogport);
tpl_printf(vars, TPLADD, "CLIENTTIMEOUT", "%u", cfg.ctimeout);
value = mk_t_caidvaluetab(&cfg.ctimeouttab);
tpl_addVar(vars, TPLADD, "CLIENTTIMEOUT_PERCAID", value);
free_mk_t(value);
tpl_printf(vars, TPLADD, "FALLBACKTIMEOUT", "%u", cfg.ftimeout);
tpl_printf(vars, TPLADD, "CLIENTMAXIDLE", "%u", cfg.cmaxidle);
value = mk_t_caidvaluetab(&cfg.ftimeouttab);
tpl_addVar(vars, TPLADD, "FALLBACKTIMEOUT_PERCAID", value);
free_mk_t(value);
tpl_printf(vars, TPLADD, "SLEEP", "%d", cfg.tosleep);
tpl_addVar(vars, TPLADD, "UNLOCKPARENTALCHECKED", (cfg.ulparent == 1) ? "checked" : "");
if(cfg.reload_useraccounts) { tpl_addVar(vars, TPLADD, "RELOADUSERACCOUNTSCHECKED", "checked"); }
if(cfg.reload_readers) { tpl_addVar(vars, TPLADD, "RELOADREADERSCHECKED", "checked"); }
if(cfg.reload_provid) { tpl_addVar(vars, TPLADD, "RELOADPROVIDCHECKED", "checked"); }
if(cfg.reload_services_ids) { tpl_addVar(vars, TPLADD, "RELOADSERVICESIDSCHECKED", "checked"); }
if(cfg.reload_tier_ids) { tpl_addVar(vars, TPLADD, "RELOADTIERUDSCHECKED", "checked"); }
if(cfg.reload_fakecws) { tpl_addVar(vars, TPLADD, "RELOADFAKECWSCHECKED", "checked"); }
if(cfg.reload_ac_stat) { tpl_addVar(vars, TPLADD, "RELOADACSTATCHECKED", "checked"); }
if(cfg.reload_log) { tpl_addVar(vars, TPLADD, "RELOADLOGCHECKED", "checked"); }
if(cfg.block_same_ip) { tpl_addVar(vars, TPLADD, "BLOCKSAMEIPCHECKED", "checked"); }
if(cfg.block_same_name) { tpl_addVar(vars, TPLADD, "BLOCKSAMENAMECHECKED", "checked"); }
if(cfg.waitforcards == 1) { tpl_addVar(vars, TPLADD, "WAITFORCARDSCHECKED", "checked"); }
tpl_printf(vars, TPLADD, "EXTRADELAY", "%d", cfg.waitforcards_extra_delay);
if(cfg.preferlocalcards == 1)
{
tpl_addVar(vars, TPLADD, "PREFERCACHEEX", "selected");
}
else if(cfg.preferlocalcards == 2)
{
tpl_addVar(vars, TPLADD, "PREFERLOCALCARDS", "selected");
}
if(cfg.c35_suppresscmd08)
{ tpl_addVar(vars, TPLADD, "SUPPRESSCMD08", "checked"); }
if(cfg.getblockemmauprovid > 0)
{
tpl_addVar(vars, TPLADD, "GETBLOCKEMMAUPROVID", "checked");
}
if(cfg.reader_restart_seconds)
{ tpl_printf(vars, TPLADD, "READERRESTARTSECONDS", "%d", cfg.reader_restart_seconds); }
tpl_addVar(vars, TPLADD, "DROPDUPSCHECKED", (cfg.dropdups == 1) ? "checked" : "");
if(cfg.resolve_gethostbyname == 1)
{ tpl_addVar(vars, TPLADD, "RESOLVER1", "selected"); }
else
{ tpl_addVar(vars, TPLADD, "RESOLVER0", "selected"); }
tpl_printf(vars, TPLADD, "FAILBANTIME", "%d", cfg.failbantime);
tpl_printf(vars, TPLADD, "FAILBANCOUNT", "%d", cfg.failbancount);
tpl_addVar(vars, TPLADD, "DCHECKCSELECTED", (cfg.double_check == 1) ? "checked" : "");
value = mk_t_ftab(&cfg.double_check_caid);
tpl_addVar(vars, TPLADD, "DOUBLECHECKCAID", value);
free_mk_t(value);
tpl_addVar(vars, TPLADD, "DISABLECRCCWSCHECKEDGLOBAL", (cfg.disablecrccws == 1) ? "checked" : "");
value = mk_t_ftab(&cfg.disablecrccws_only_for);
tpl_addVar(vars, TPLADD, "IGNCHKSUMONLYFORGLOBAL", value);
free_mk_t(value);
tpl_addVar(vars, TPLADD, "CWVOTEENABLEDCHECKED", (cfg.cwvote_enabled == 1) ? "checked" : "");
tpl_addVar(vars, TPLADD, "CWVOTELOGENABLEDCHECKED", (cfg.cwvote_log_enabled == 1) ? "checked" : "");
tpl_printf(vars, TPLADD, "CWVOTETIMEOUT", "%d", cfg.cwvote_timeout);
tpl_printf(vars, TPLADD, "CWVOTEMINVOTES", "%d", cfg.cwvote_min_votes);
tpl_printf(vars, TPLADD, "CWVOTELOCALWEIGHT", "%.1f", cfg.cwvote_local_weight);
tpl_printf(vars, TPLADD, "CWVOTEMAXCANDIDATES", "%d", cfg.cwvote_max_candidates);
tpl_addVar(vars, TPLADD, "CWVOTECOMPARE8", (cfg.cwvote_compare_len == 8) ? "selected" : "");
tpl_addVar(vars, TPLADD, "CWVOTECOMPARE16", (cfg.cwvote_compare_len == 16) ? "selected" : "");
tpl_addVar(vars, TPLADD, "CWVOTEFALLBACK0", (cfg.cwvote_fallback == 0) ? "selected" : "");
tpl_addVar(vars, TPLADD, "CWVOTEFALLBACK1", (cfg.cwvote_fallback == 1) ? "selected" : "");
tpl_addVar(vars, TPLADD, "CWVOTEFALLBACK2", (cfg.cwvote_fallback == 2) ? "selected" : "");
value = mk_t_cwvote_caidtab(&cfg.cwvote_caids);
tpl_addVar(vars, TPLADD, "CWVOTECAIDS", value);
free_mk_t(value);
#ifdef CS_CACHEEX_AIO
tpl_addVar(vars, TPLADD, "CACHEEXSRCNAME", (cfg.cacheex_srcname_webif == 1) ? "checked" : "");
#endif
#ifdef LEDSUPPORT
if(cfg.enableled == 1)
{ tpl_addVar(vars, TPLADD, "ENABLELEDSELECTED1", "selected"); }
else if(cfg.enableled == 2)
{ tpl_addVar(vars, TPLADD, "ENABLELEDSELECTED2", "selected"); }
#endif
return tpl_getTpl(vars, "CONFIGGLOBAL");
}
#ifdef WITH_LB
static char *send_oscam_config_loadbalancer(struct templatevars *vars, struct uriparams *params)
{
setActiveSubMenu(vars, MNU_CFG_LOADBAL);
if(cs_strlen(getParam(params, "button")) > 0)
{
if(cfg.http_readonly)
{
tpl_addMsg(vars, "WebIf is in readonly mode. No changes are possible!");
}
else
{
if(strcmp(getParam(params, "button"), "Load Stats") == 0)
{
clear_all_stat();
load_stat_from_file();
tpl_addMsg(vars, "Stats loaded from file");
}
if(strcmp(getParam(params, "button"), "Save Stats") == 0)
{
save_stat_to_file(1);
tpl_addMsg(vars, "Stats saved to file");
}
if(strcmp(getParam(params, "button"), "Clear Stats") == 0)
{
clear_all_stat();
tpl_addMsg(vars, "Stats cleared completly");
}
if(strcmp(getParam(params, "button"), "Clear Timeouts") == 0)
{
clean_all_stats_by_rc(E_TIMEOUT, 0);
tpl_addMsg(vars, "Timeout cleared from Stats");
}
if(strcmp(getParam(params, "button"), "Clear Not Found") == 0)
{
clean_all_stats_by_rc(E_NOTFOUND, 0);
tpl_addMsg(vars, "Not Found cleared from Stats");
}
if(strcmp(getParam(params, "button"), "Clear Invalid") == 0)
{
clean_all_stats_by_rc(E_INVALID, 0);
tpl_addMsg(vars, "Invalid cleared from Stats");
}
}
}
webif_save_config("global", vars, params);
tpl_printf(vars, TPLADD, "TMP", "LBMODE%d", cfg.lb_mode);
tpl_addVar(vars, TPLADD, tpl_getVar(vars, "TMP"), "selected");
tpl_printf(vars, TPLADD, "LBSAVE", "%d", cfg.lb_save);
if(cfg.lb_savepath) { tpl_addVar(vars, TPLADD, "LBSAVEPATH", cfg.lb_savepath); }
tpl_printf(vars, TPLADD, "LBNBESTREADERS", "%d", cfg.lb_nbest_readers);
char *value = mk_t_caidvaluetab(&cfg.lb_nbest_readers_tab);
tpl_addVar(vars, TPLADD, "LBNBESTPERCAID", value);
free_mk_t(value);
tpl_printf(vars, TPLADD, "LBNFBREADERS", "%d", cfg.lb_nfb_readers);
tpl_printf(vars, TPLADD, "LBMAXREADERS", "%d", cfg.lb_max_readers);
tpl_printf(vars, TPLADD, "LBMINECMCOUNT", "%d", cfg.lb_min_ecmcount);
tpl_printf(vars, TPLADD, "LBMAXECEMCOUNT", "%d", cfg.lb_max_ecmcount);
tpl_printf(vars, TPLADD, "LBRETRYLIMIT", "%d", cfg.lb_retrylimit);
value = mk_t_caidvaluetab(&cfg.lb_retrylimittab);
tpl_addVar(vars, TPLADD, "LBRETRYLIMITS", value);
free_mk_t(value);
tpl_printf(vars, TPLADD, "LBREOPENSECONDS", "%d", cfg.lb_reopen_seconds);
tpl_printf(vars, TPLADD, "LBCLEANUP", "%d", cfg.lb_stat_cleanup);
tpl_addVar(vars, TPLADD, "LBREOPENINVALID", (cfg.lb_reopen_invalid == 1) ? "checked" : "");
tpl_addVar(vars, TPLADD, "LBFORCEALWAYS", (cfg.lb_force_reopen_always == 1) ? "checked" : "");
value = mk_t_caidtab(&cfg.lb_noproviderforcaid);
tpl_addVar(vars, TPLADD, "LBNOPROVIDERFORCAID", value);
free_mk_t(value);
tpl_addVar(vars, TPLADD, "LBAUTOBETATUNNEL", (cfg.lb_auto_betatunnel == 1) ? "checked" : "");
if(cfg.lb_auto_betatunnel_mode)
{
tpl_printf(vars, TPLADD, "TMP", "LBAUTOBETATUNNELMODE%d", cfg.lb_auto_betatunnel_mode);
tpl_addVar(vars, TPLADD, tpl_getVar(vars, "TMP"), "selected");
}
tpl_printf(vars, TPLADD, "LBPREFERBETA", "%d", cfg.lb_auto_betatunnel_prefer_beta);
tpl_addVar(vars, TPLADD, "LBAUTOTIMEOUT", (cfg.lb_auto_timeout == 1) ? "checked" : "");
tpl_printf(vars, TPLADD, "LBAUTOTIMEOUTP", "%d", cfg.lb_auto_timeout_p);
tpl_printf(vars, TPLADD, "LBAUTOTIMEOUTT", "%d", cfg.lb_auto_timeout_t);
tpl_addVar(vars, TPLADDONCE, "CONFIG_CONTROL", tpl_getTpl(vars, "CONFIGLOADBALANCERCTRL"));
return tpl_getTpl(vars, "CONFIGLOADBALANCER");
}
#endif
#ifdef MODULE_CAMD33
static char *send_oscam_config_camd33(struct templatevars *vars, struct uriparams *params)
{
int32_t i;
setActiveSubMenu(vars, MNU_CFG_CAMD33);
webif_save_config("camd33", vars, params);
if(cfg.c33_port)
{
tpl_printf(vars, TPLADD, "PORT", "%d", cfg.c33_port);
if(IP_ISSET(cfg.c33_srvip)) { tpl_addVar(vars, TPLADD, "SERVERIP", cs_inet_ntoa(cfg.c33_srvip)); }
tpl_addVar(vars, TPLADD, "PASSIVECHECKED", (cfg.c33_passive == 1) ? "checked" : "");
for(i = 0; i < (int) sizeof(cfg.c33_key); ++i) { tpl_printf(vars, TPLAPPEND, "KEY", "%02X", cfg.c33_key[i]); }
char *value = mk_t_iprange(cfg.c33_plain);
tpl_addVar(vars, TPLADD, "NOCRYPT", value);
free_mk_t(value);
}
return tpl_getTpl(vars, "CONFIGCAMD33");
}
#endif
#ifdef MODULE_CAMD35
static char *send_oscam_config_camd35(struct templatevars *vars, struct uriparams *params)
{
setActiveSubMenu(vars, MNU_CFG_CAMD35);
webif_save_config("cs357x", vars, params);
if(cfg.c35_port)
{
tpl_printf(vars, TPLADD, "PORT", "%d", cfg.c35_port);
if(IP_ISSET(cfg.c35_srvip))
{ tpl_addVar(vars, TPLAPPEND, "SERVERIP", cs_inet_ntoa(cfg.c35_srvip)); }
if(cfg.c35_udp_suppresscmd08)
{ tpl_addVar(vars, TPLADD, "SUPPRESSCMD08UDP", "checked"); }
}
return tpl_getTpl(vars, "CONFIGCAMD35");
}
#endif
#ifdef MODULE_CAMD35_TCP
static char *send_oscam_config_camd35tcp(struct templatevars *vars, struct uriparams *params)
{
setActiveSubMenu(vars, MNU_CFG_CAMD35TCP);
webif_save_config("cs378x", vars, params);
if((cfg.c35_tcp_ptab.nports > 0) && (cfg.c35_tcp_ptab.ports[0].s_port > 0))
{
char *value = mk_t_camd35tcp_port();
tpl_addVar(vars, TPLADD, "PORT", value);
free_mk_t(value);
if(IP_ISSET(cfg.c35_tcp_srvip))
{ tpl_addVar(vars, TPLAPPEND, "SERVERIP", cs_inet_ntoa(cfg.c35_tcp_srvip)); }
if(cfg.c35_tcp_suppresscmd08)
{ tpl_addVar(vars, TPLADD, "SUPPRESSCMD08TCP", "checked"); }
}
return tpl_getTpl(vars, "CONFIGCAMD35TCP");
}
#endif
static char *send_oscam_config_cache(struct templatevars *vars, struct uriparams *params)
{
setActiveSubMenu(vars, MNU_CFG_CACHE);
webif_save_config("cache", vars, params);
tpl_printf(vars, TPLADD, "CACHEDELAY", "%u", cfg.delay);
tpl_printf(vars, TPLADD, "MAXCACHETIME", "%d", cfg.max_cache_time);
#ifdef CS_CACHEEX
char *value = NULL;
#ifdef CS_CACHEEX_AIO
value = mk_t_cacheex_cwcheck_valuetab(&cfg.cw_cache_settings);
tpl_addVar(vars, TPLADD, "CWCACHESETTINGS", value);
free_mk_t(value);
tpl_printf(vars, TPLADD, "CWCACHESIZE", "%d", cfg.cw_cache_size);
tpl_printf(vars, TPLADD, "CWCACHEMEMORY", "%d", cfg.cw_cache_memory);
tpl_printf(vars, TPLADD, "ECMCACHESIZE", "%d", cfg.ecm_cache_size);
tpl_printf(vars, TPLADD, "ECMCACHEMEMORY", "%d", cfg.ecm_cache_memory);
tpl_printf(vars, TPLADD, "ECMDROPTIME", "%d", cfg.ecm_cache_droptime);
#endif
value = mk_t_cacheex_valuetab(&cfg.cacheex_wait_timetab);
tpl_addVar(vars, TPLADD, "WAIT_TIME", value);
free_mk_t(value);
#ifdef CS_CACHEEX_AIO
tpl_printf(vars, TPLADD, "WAITTIME_BLOCK_START", "%d", cfg.waittime_block_start);
tpl_printf(vars, TPLADD, "WAITTIME_BLOCK_TIME", "%d", cfg.waittime_block_time);
#endif
value = mk_t_caidvaluetab(&cfg.cacheex_mode1_delay_tab);
tpl_addVar(vars, TPLADD, "CACHEEXMODE1DELAY", value);
free_mk_t(value);
#ifdef CS_CACHEEX_AIO
value = mk_t_caidvaluetab(&cfg.cacheex_nopushafter_tab);
tpl_addVar(vars, TPLADD, "CACHEEXNOPUSHAFTER", value);
free_mk_t(value);
#endif
tpl_printf(vars, TPLADD, "MAX_HIT_TIME", "%d", cfg.max_hitcache_time);
2026-02-17 09:41:05 +00:00
tpl_addVar(vars, TPLADD, "CACHEEXSTATSSELECTED", (cfg.cacheex_enable_stats == 1) ? "checked" : "");
/* GLOBAL CACHEEX MAXHOP */
tpl_printf(vars, TPLADD, "CACHEEX_GLOBAL_MAXHOP",cfg.cacheex_global_maxhop ? "checked" : "");
tpl_printf(vars, TPLADD, "CACHEEX_MAXHOP_GLOBAL", "%d", cfg.cacheex_maxhop);
tpl_printf(vars, TPLADD, "CACHEEX_MAXHOP_LG_GLOBAL", "%d", cfg.cacheex_maxhop_lg);
tpl_printf(vars, TPLADD, "CACHEEX_MAXHOP_PERCAID", "%s", mk_t_caidvaluetab(&cfg.cacheex_maxhop_percaid));
tpl_printf(vars, TPLADD, "CACHEEX_MAXHOP_LG_PERCAID", "%s", mk_t_caidvaluetab(&cfg.cacheex_maxhop_lg_percaid));
tpl_addVar(vars, TPLADD, "WTTCHECKED", (cfg.wait_until_ctimeout == 1) ? "checked" : "");
#ifdef CS_CACHEEX_AIO
tpl_addVar(vars, TPLADD, "CACHEEXDROPDIFFS", (cfg.cacheex_dropdiffs == 1) ? "checked" : "");
value = mk_t_group(cfg.cacheex_push_lg_groups);
tpl_addVar(vars, TPLADD, "CACHEEXPUSHLGGRPS", value);
free_mk_t(value);
tpl_addVar(vars, TPLADD, "LGONLYREMOTESETTINGSCHECKED", (cfg.cacheex_lg_only_remote_settings == 1) ? "checked" : "");
tpl_addVar(vars, TPLADD, "LOCALGENERATEDONLYCHECKED", (cfg.cacheex_localgenerated_only == 1) ? "checked" : "");
value = mk_t_ftab(&cfg.cacheex_lg_only_tab);
tpl_addVar(vars, TPLADD, "LGONLYTAB", value);
free_mk_t(value);
tpl_addVar(vars, TPLADD, "LOCALGENERATEDONLYINCHECKED", (cfg.cacheex_localgenerated_only_in == 1) ? "checked" : "");
tpl_addVar(vars, TPLADD, "LGONLYINAIOONLYCHECKED", (cfg.cacheex_lg_only_in_aio_only == 1) ? "checked" : "");
value = mk_t_ftab(&cfg.cacheex_lg_only_in_tab);
tpl_addVar(vars, TPLADD, "LGONLYINTAB", value);
free_mk_t(value);
value = mk_t_cacheex_hitvaluetab(&cfg.cacheex_filter_caidtab);
tpl_addVar(vars, TPLADD, "CACHEEXECMFILTER", value);
free_mk_t(value);
value = mk_t_cacheex_hitvaluetab(&cfg.cacheex_filter_caidtab_aio);
tpl_addVar(vars, TPLADD, "CACHEEXECMFILTERAIO", value);
free_mk_t(value);
#endif
if(cfg.csp_port)
{ tpl_printf(vars, TPLADD, "PORT", "%d", cfg.csp_port); }
if(IP_ISSET(cfg.csp_srvip))
{ tpl_addVar(vars, TPLAPPEND, "SERVERIP", cs_inet_ntoa(cfg.csp_srvip)); }
value = mk_t_cacheex_hitvaluetab(&cfg.csp.filter_caidtab);
tpl_addVar(vars, TPLADD, "CSP_ECM_FILTER", value);
free_mk_t(value);
value = mk_t_cacheex_cwcheck_valuetab(&cfg.cacheex_cwcheck_tab);
tpl_addVar(vars, TPLADD, "CACHEEXCWCHECK", value);
free_mk_t(value);
tpl_addVar(vars, TPLADD, "ARCHECKED", (cfg.csp.allow_request == 1) ? "checked" : "");
tpl_addVar(vars, TPLADD, "ARFCHECKED", (cfg.csp.allow_reforward == 1) ? "checked" : "");
tpl_addVar(vars, TPLADD, "BLOCKFAKECWSCHECKED", (cfg.csp.block_fakecws == 1) ? "checked" : "");
#endif
#ifdef CW_CYCLE_CHECK
#ifndef CS_CACHEEX
char *value = NULL;
#endif
tpl_addVar(vars, TPLADD, "CWCYCLECHECK", (cfg.cwcycle_check_enable == 1) ? "checked" : "");
value = mk_t_caidtab(&cfg.cwcycle_check_caidtab);
tpl_addVar(vars, TPLADD, "CWCYCLECHECKCAID", value);
free_mk_t(value);
tpl_printf(vars, TPLADD, "MAXCYCLELIST", "%d", cfg.maxcyclelist);
tpl_printf(vars, TPLADD, "KEEPCYCLETIME", "%d", cfg.keepcycletime);
if(cfg.onbadcycle)
{
tpl_printf(vars, TPLADD, "TMP", "ONBADCYCLE%d", cfg.onbadcycle);
tpl_addVar(vars, TPLADD, tpl_getVar(vars, "TMP"), "selected");
}
tpl_addVar(vars, TPLADD, "DROPOLD", (cfg.cwcycle_dropold == 1) ? "checked" : "");
if(cfg.cwcycle_sensitive)
{
tpl_printf(vars, TPLADD, "TMP", "CWCSEN%d", cfg.cwcycle_sensitive);
tpl_addVar(vars, TPLADD, tpl_getVar(vars, "TMP"), "selected");
}
tpl_addVar(vars, TPLADD, "ALLOWBADFROMFFB", (cfg.cwcycle_allowbadfromffb == 1) ? "checked" : "");
tpl_addVar(vars, TPLADD, "USECWCFROMCE", (cfg.cwcycle_usecwcfromce == 1) ? "checked" : "");
#endif
#ifdef CS_CACHEEX_AIO
return tpl_getTpl(vars, "CONFIGCACHEAIO");
#else
return tpl_getTpl(vars, "CONFIGCACHE");
#endif
}
#ifdef MODULE_NEWCAMD
static char *send_oscam_config_newcamd(struct templatevars *vars, struct uriparams *params)
{
int32_t i;
setActiveSubMenu(vars, MNU_CFG_NEWCAMD);
webif_save_config("newcamd", vars, params);
if((cfg.ncd_ptab.nports > 0) && (cfg.ncd_ptab.ports[0].s_port > 0))
{
char *value = mk_t_newcamd_port();
tpl_addVar(vars, TPLADD, "PORT", value);
free_mk_t(value);
if(IP_ISSET(cfg.ncd_srvip))
{ tpl_addVar(vars, TPLADD, "SERVERIP", cs_inet_ntoa(cfg.ncd_srvip)); }
for(i = 0; i < (int32_t)sizeof(cfg.ncd_key); i++)
{ tpl_printf(vars, TPLAPPEND, "KEY", "%02X", cfg.ncd_key[i]); }
value = mk_t_iprange(cfg.ncd_allowed);
tpl_addVar(vars, TPLADD, "ALLOWED", value);
free_mk_t(value);
if(cfg.ncd_keepalive)
{ tpl_addVar(vars, TPLADD, "KEEPALIVE", "checked"); }
if(cfg.ncd_mgclient)
{ tpl_addVar(vars, TPLADD, "MGCLIENTCHK", "checked"); }
}
return tpl_getTpl(vars, "CONFIGNEWCAMD");
}
#endif
#ifdef MODULE_GBOX
static char *send_oscam_config_gbox(struct templatevars *vars, struct uriparams *params)
{
uint8_t i=0;
char local_gbox_save_gsms[2],local_gbox_msg_type[3], local_gbox_dest_peers[GBOX_MAX_DEST_PEERS*5], tmp_gbox_dest_peers[GBOX_MAX_DEST_PEERS*5] ;
int n=0, len_gbox_save_gsms=0, len_gbox_msg_type=0, len_gbox_dest_peers=0, len_gbox_msg_txt=0;
char *ptr1, *saveptr1, *isbroadcast = NULL;
const char *s;
uint16_t gbox_dest_peers_tmp;
setActiveSubMenu(vars, MNU_CFG_GBOX);
webif_save_config("gbox", vars, params);
/*
* Action when GetOnlinePeers is pressed
*/
if(streq(getParam(params, "action"), "Online peers"))
{
gbox_get_online_peers();
// init var
len_gbox_save_gsms=cs_strlen(getParam(params, "gbox_msg_type"));
len_gbox_msg_type=cs_strlen(getParam(params, "gbox_msg_type"));
len_gbox_msg_txt=cs_strlen(getParam(params, "gbox_msg_txt"));
if(len_gbox_msg_txt>GBOX_MAX_MSG_TXT) { len_gbox_msg_txt=GBOX_MAX_MSG_TXT; }
// retrieve value from Webif
cs_strncpy(local_gbox_save_gsms, getParam(params, "gbox_save_gsms"), len_gbox_save_gsms+1);
cfg.gbox_save_gsms=atoi(local_gbox_save_gsms);
cs_strncpy(local_gbox_msg_type, getParam(params, "gbox_msg_type"), len_gbox_msg_type+1);
cfg.gbox_msg_type=atoi(local_gbox_msg_type);
cs_strncpy(cfg.gbox_msg_txt,getParam(params, "gbox_msg_txt"), len_gbox_msg_txt+1);
}
/*
* Action when ResetGSMS button is pressed
*/
if(streq(getParam(params, "action"), "resetallgsms"))
{
cfg.gbox_save_gsms = 0;
cfg.gbox_msg_type = 0;
for(i = 0; i < GBOX_MAX_DEST_PEERS; i++)
{
cfg.gbox_dest_peers[i]='\0';
}
cfg.gbox_dest_peers_num=0;
for(i = 0; i < GBOX_MAX_MSG_TXT; i++)
{
cfg.gbox_msg_txt[i]='\0';
}
tpl_addMsg(vars, "GBOX: Reset GSMS datas done!");
}
/*
* Action when Send GSMS is pressed
*/
if(streq(getParam(params, "action"), "Send GSMS"))
{
// init var
len_gbox_msg_type=cs_strlen(getParam(params, "gbox_msg_type"));
len_gbox_dest_peers=cs_strlen(trim(getParam(params, "gbox_dest_peers")));
len_gbox_msg_txt=cs_strlen(getParam(params, "gbox_msg_txt"));
if(len_gbox_msg_txt>GBOX_MAX_MSG_TXT) { len_gbox_msg_txt=GBOX_MAX_MSG_TXT; }
// retrieve value from Webif
cs_strncpy(local_gbox_msg_type, getParam(params, "gbox_msg_type"), len_gbox_msg_type+1);
cfg.gbox_msg_type=atoi(local_gbox_msg_type);
cs_strncpy(local_gbox_dest_peers, strtoupper(trim(getParam(params, "gbox_dest_peers"))), len_gbox_dest_peers+1);
cs_strncpy(tmp_gbox_dest_peers, strtoupper(trim(getParam(params, "gbox_dest_peers"))), len_gbox_dest_peers+1);
cs_strncpy(cfg.gbox_msg_txt,getParam(params, "gbox_msg_txt"), len_gbox_msg_txt+1);
n=0;
for (ptr1 = strtok_r(tmp_gbox_dest_peers, ",", &saveptr1); (ptr1); ptr1 = strtok_r(NULL, ",", &saveptr1))
{
s=trim(ptr1);
if ((n < GBOX_MAX_DEST_PEERS) && (s[strspn(s, "0123456789abcdefABCDEF")] == 0))
{ cfg.gbox_dest_peers[n++] = a2i(trim(ptr1), cs_strlen(trim(ptr1))); }
}
cfg.gbox_dest_peers_num = n;
/*
Start sending GBox SMS
*/
if((cs_strlen(cfg.gbox_msg_txt) > 5))
{
isbroadcast=strstr(local_gbox_dest_peers, "FFFF");
if(isbroadcast == NULL)
{
n =0;
for (i = 0, ptr1 = strtok_r(local_gbox_dest_peers, ",", &saveptr1); (i < 4) && (ptr1); ptr1 = strtok_r(NULL, ",", &saveptr1))
{
s=ptr1;
if ((n < GBOX_MAX_DEST_PEERS) && (s[strspn(s, "0123456789abcdefABCDEF")] == 0))
{
gbox_dest_peers_tmp = a2i(ptr1, 4);
if(gbox_direct_send_gsms(gbox_dest_peers_tmp, cfg.gbox_msg_type, cfg.gbox_msg_txt)) { cs_log("GBOX message sent to[%04X] type[%d] text[%s] ", gbox_dest_peers_tmp, cfg.gbox_msg_type, cfg.gbox_msg_txt);}
n++;
}
}
tpl_addMsg(vars, "GBOX Send SMS: individual messages started.");
}
else
{
if(gbox_direct_send_gsms(0xFFFF, cfg.gbox_msg_type, cfg.gbox_msg_txt)) { cs_log("GBOX broadcast message sent type[%d] text[%s] ", cfg.gbox_msg_type, cfg.gbox_msg_txt);}
tpl_addMsg(vars, "GBOX Send SMS: broadcast started.");
}
}
else
{
cs_log("GBox SMS: destination peers or message text not specified or too short");
tpl_addMsg(vars, "GBOX: Send SMS failed - error in input fields: dest peers or text message.");
}
}
tpl_addVar(vars, TPLADD, "HOSTNAME", xml_encode(vars, cfg.gbox_hostname));
char *value0 = mk_t_gbox_port();
tpl_addVar(vars, TPLAPPEND, "PORT", value0);
free_mk_t(value0);
tpl_printf(vars, TPLADD, "MYGBOXPASSWORD", "%08X", cfg.gbox_password);
tpl_printf(vars, TPLADD, "MYGBOXID", "%04X", gbox_get_local_gbox_id());
tpl_printf(vars, TPLADD, "GBOXRECONNECT", "%d", cfg.gbox_reconnect);
tpl_printf(vars, TPLADD, "GBOXMYVERS", "%02X", cfg.gbox_my_vers);
tpl_printf(vars, TPLAPPEND, "GBOXMYCPUAPI", "%02X", cfg.gbox_my_cpu_api);
#ifdef MODULE_CCCAM
if(cfg.cc_gbx_reshare_en == 1) { tpl_addVar(vars, TPLADD, "GBOXCCCRESHARE", "checked"); }
tpl_addVar(vars, TPLAPPEND, "CCCDEPENDINGCONFIG", tpl_getTpl(vars, "CCCAMRESHAREBIT"));
#endif
if(cfg.log_hello == 1) { tpl_addVar(vars, TPLADD, "GBOXLOGHELLO", "checked"); }
if(cfg.gsms_dis == 1) { tpl_addVar(vars, TPLADD, "GBOXGSMSDISABLE", "checked"); }
if(cfg.dis_attack_txt == 1) { tpl_addVar(vars, TPLADD, "GBOXDISATTACKTXT", "checked"); }
if(cfg.gbox_tmp_dir != NULL) { tpl_addVar(vars, TPLADD, "GBOXTMPDIR", cfg.gbox_tmp_dir); }
char *value1 = mk_t_gbox_proxy_card();
tpl_addVar(vars, TPLAPPEND, "GBOXPROXYCARD", value1);
free_mk_t(value1);
char *value2 = mk_t_gbox_ignored_peer();
tpl_addVar(vars, TPLAPPEND, "GBOXIGNOREDPEER", value2);
free_mk_t(value2);
char *value3 = mk_t_gbox_block_ecm();
tpl_addVar(vars, TPLAPPEND, "GBOXBLOCKECM", value3);
free_mk_t(value3);
char *value4 = mk_t_accept_remm_peer();
tpl_addVar(vars, TPLAPPEND, "GBOXACCEPTREMM", value4);
free_mk_t(value4);
/*
* GBOX SMS
*/
tpl_addVar(vars, TPLADD, "GBOXSAVEGSMS", (cfg.gbox_save_gsms == 1) ? "checked" : "");
if(cfg.gbox_msg_type == 0)
{
tpl_addVar(vars, TPLADD, "GBOXMSGTYPENORMAL", "selected");
}
else if(cfg.gbox_msg_type == 1)
{
tpl_addVar(vars, TPLADD, "GBOXMSGTYPEOSD", "selected");
}
char *gmsg_dest_peers = mk_t_gbox_dest_peers();
tpl_addVar(vars, TPLADD, "GBOXMSGDESTPEERS", gmsg_dest_peers);
free_mk_t(gmsg_dest_peers);
tpl_addVar(vars, TPLADD, "GBOXMSGTXT", cfg.gbox_msg_txt);
return tpl_getTpl(vars, "CONFIGGBOX");
}
#endif
#ifdef MODULE_RADEGAST
static char *send_oscam_config_radegast(struct templatevars *vars, struct uriparams *params)
{
setActiveSubMenu(vars, MNU_CFG_RADEGAST);
webif_save_config("radegast", vars, params);
tpl_printf(vars, TPLADD, "PORT", "%d", cfg.rad_port);
if(IP_ISSET(cfg.rad_srvip))
{ tpl_addVar(vars, TPLADD, "SERVERIP", cs_inet_ntoa(cfg.rad_srvip)); }
tpl_addVar(vars, TPLADD, "USERNAME", xml_encode(vars, cfg.rad_usr));
char *value = mk_t_iprange(cfg.rad_allowed);
tpl_addVar(vars, TPLADD, "ALLOWED", value);
free_mk_t(value);
return tpl_getTpl(vars, "CONFIGRADEGAST");
}
#endif
#ifdef MODULE_SCAM
static char *send_oscam_config_scam(struct templatevars *vars, struct uriparams *params)
{
setActiveSubMenu(vars, MNU_CFG_SCAM);
webif_save_config("scam", vars, params);
tpl_printf(vars, TPLADD, "PORT", "%d", cfg.scam_port);
if(IP_ISSET(cfg.scam_srvip))
{ tpl_addVar(vars, TPLADD, "SERVERIP", cs_inet_ntoa(cfg.scam_srvip)); }
char *value = mk_t_iprange(cfg.scam_allowed);
tpl_addVar(vars, TPLADD, "ALLOWED", value);
free_mk_t(value);
return tpl_getTpl(vars, "CONFIGSCAM");
}
#endif
#ifdef MODULE_STREAMRELAY
static char *send_oscam_config_streamrelay(struct templatevars *vars, struct uriparams *params)
{
char *value;
setActiveSubMenu(vars, MNU_CFG_STREAMRELAY);
webif_save_config("streamrelay", vars, params);
tpl_printf(vars, TPLADD, "TMP", "STREAMRELAYENABLEDSELECTED%d", cfg.stream_relay_enabled);
tpl_addVar(vars, TPLADD, tpl_getVar(vars, "TMP"), "selected");
tpl_printf(vars, TPLADD, "STREAM_RELAY_PORT", "%d", cfg.stream_relay_port);
if(cfg.stream_relay_user)
{ tpl_printf(vars, TPLADD, "STREAM_RELAY_USER", "%s", cfg.stream_relay_user); }
value = mk_t_caidtab(&cfg.stream_relay_ctab);
tpl_addVar(vars, TPLADD, "STREAM_RELAY_CTAB", value);
free_mk_t(value);
tpl_printf(vars, TPLADD, "STREAM_SOURCE_HOST", "%s", cfg.stream_source_host);
tpl_addVar(vars, TPLADD, "STREAM_CLIENT_SOURCE_HOST", (cfg.stream_client_source_host == 1) ? "checked" : "");
tpl_printf(vars, TPLADD, "STREAM_SOURCE_PORT", "%d", cfg.stream_source_port);
if(cfg.stream_source_auth_user)
{ tpl_printf(vars, TPLADD, "STREAM_SOURCE_AUTH_USER", "%s", cfg.stream_source_auth_user); }
if(cfg.stream_source_auth_password)
{ tpl_printf(vars, TPLADD, "STREAM_SOURCE_AUTH_PASSWORD", "%s", cfg.stream_source_auth_password); }
tpl_printf(vars, TPLADD, "STREAM_RELAY_BUFFER_TIME", "%d", cfg.stream_relay_buffer_time);
tpl_printf(vars, TPLADD, "STREAM_RELAY_RECONNECT_COUNT", "%d", cfg.stream_relay_reconnect_count);
#ifdef WITH_EMU
tpl_printf(vars, TPLADD, "TMP", "STREAMEMMENABLEDSELECTED%d", cfg.emu_stream_emm_enabled);
tpl_addVar(vars, TPLADD, tpl_getVar(vars, "TMP"), "selected");
tpl_printf(vars, TPLADD, "STREAM_ECM_DELAY", "%d", cfg.emu_stream_ecm_delay);
#endif
tpl_printf(vars, TPLADD, "TMP", "STREAMCONFIGCLIENTSELECTED%d", cfg.stream_display_client);
tpl_addVar(vars, TPLADD, tpl_getVar(vars, "TMP"), "selected");
tpl_addVar(vars, TPLADD, "STREAM_REUSE_CLIENT", (cfg.stream_reuse_client == 1) ? "checked" : "");
#ifdef WEBIF
tpl_addVar(vars, TPLADD, "STREAM_HIDE_CLIENT", (cfg.stream_hide_client == 1) ? "checked" : "");
#endif
return tpl_getTpl(vars, "CONFIGSTREAMRELAY");
}
#endif
#ifdef MODULE_CCCAM
static char *send_oscam_config_cccam(struct templatevars *vars, struct uriparams *params)
{
setActiveSubMenu(vars, MNU_CFG_CCCAM);
if(strcmp(getParam(params, "button"), "Refresh list") == 0)
{
cs_log_dbg(D_TRACE, "Entitlements: Refresh Shares start");
#ifdef MODULE_CCCSHARE
refresh_shares();
#endif
cs_log_dbg(D_TRACE, "Entitlements: Refresh Shares finished");
tpl_addMsg(vars, "Refresh Shares started");
}
webif_save_config("cccam", vars, params);
if(streq(getParam(params, "action"), "execute") && !cfg.http_readonly)
{ cc_update_nodeid(); }
char *value = mk_t_cccam_port();
tpl_addVar(vars, TPLAPPEND, "PORT", value);
free_mk_t(value);
if(IP_ISSET(cfg.cc_srvip))
{ tpl_addVar(vars, TPLADD, "SERVERIP", cs_inet_ntoa(cfg.cc_srvip)); }
tpl_printf(vars, TPLADD, "RESHARE", "%d", cfg.cc_reshare);
if(!strcmp((char *)cfg.cc_version, "2.0.11"))
{
tpl_addVar(vars, TPLADD, "VERSIONSELECTED0", "selected");
}
else if(!strcmp((char *)cfg.cc_version, "2.1.1"))
{
tpl_addVar(vars, TPLADD, "VERSIONSELECTED1", "selected");
}
else if(!strcmp((char *)cfg.cc_version, "2.1.2"))
{
tpl_addVar(vars, TPLADD, "VERSIONSELECTED2", "selected");
}
else if(!strcmp((char *)cfg.cc_version, "2.1.3"))
{
tpl_addVar(vars, TPLADD, "VERSIONSELECTED3", "selected");
}
else if(!strcmp((char *)cfg.cc_version, "2.1.4"))
{
tpl_addVar(vars, TPLADD, "VERSIONSELECTED4", "selected");
}
else if(!strcmp((char *)cfg.cc_version, "2.2.0"))
{
tpl_addVar(vars, TPLADD, "VERSIONSELECTED5", "selected");
}
else if(!strcmp((char *)cfg.cc_version, "2.2.1"))
{
tpl_addVar(vars, TPLADD, "VERSIONSELECTED6", "selected");
}
else if(!strcmp((char *)cfg.cc_version, "2.3.0"))
{
tpl_addVar(vars, TPLADD, "VERSIONSELECTED7", "selected");
}
else if(!strcmp((char *)cfg.cc_version, "2.3.1"))
{
tpl_addVar(vars, TPLADD, "VERSIONSELECTED8", "selected");
}
else if(!strcmp((char *)cfg.cc_version, "2.3.2"))
{
tpl_addVar(vars, TPLADD, "VERSIONSELECTED9", "selected");
}
tpl_printf(vars, TPLADD, "UPDATEINTERVAL", "%d", cfg.cc_update_interval);
tpl_printf(vars, TPLADD, "RECV_TIMEOUT", "%u", cfg.cc_recv_timeout);
tpl_addVar(vars, TPLADD, "STEALTH", (cfg.cc_stealth == 1) ? "checked" : "");
tpl_printf(vars, TPLADD, "NODEID", "%02X%02X%02X%02X%02X%02X%02X%02X",
cfg.cc_fixed_nodeid[0], cfg.cc_fixed_nodeid[1], cfg.cc_fixed_nodeid[2], cfg.cc_fixed_nodeid[3],
cfg.cc_fixed_nodeid[4], cfg.cc_fixed_nodeid[5], cfg.cc_fixed_nodeid[6], cfg.cc_fixed_nodeid[7]);
tpl_printf(vars, TPLADD, "TMP", "MINIMIZECARDSELECTED%d", cfg.cc_minimize_cards);
tpl_addVar(vars, TPLADD, tpl_getVar(vars, "TMP"), "selected");
tpl_printf(vars, TPLADD, "TMP", "RESHAREMODE%d", cfg.cc_reshare_services);
tpl_addVar(vars, TPLADD, tpl_getVar(vars, "TMP"), "selected");
tpl_printf(vars, TPLADD, "TMP", "IGNRSHRSELECTED%d", cfg.cc_ignore_reshare);
tpl_addVar(vars, TPLADD, tpl_getVar(vars, "TMP"), "selected");
tpl_addVar(vars, TPLADD, "FORWARDORIGINCARD", (cfg.cc_forward_origin_card == 1) ? "checked" : "");
tpl_addVar(vars, TPLADD, "KEEPCONNECTED", (cfg.cc_keep_connected == 1) ? "checked" : "");
tpl_addVar(vars, TPLADDONCE, "CONFIG_CONTROL", tpl_getTpl(vars, "CONFIGCCCAMCTRL"));
return tpl_getTpl(vars, "CONFIGCCCAM");
}
#endif
static bool is_ext(const char *path, const char *ext)
{
size_t lenpath = cs_strlen(path);
size_t lenext = cs_strlen(ext);
if(lenext > lenpath)
{ return 0; }
return memcmp(path + lenpath - lenext, ext, lenext) == 0;
}
static char *send_oscam_config_webif(struct templatevars *vars, struct uriparams *params)
{
int32_t i;
setActiveSubMenu(vars, MNU_CFG_WEBIF);
webif_save_config("webif", vars, params);
tpl_printf(vars, TPLADD, "HTTPPORT", "%s%d", cfg.http_use_ssl ? "+" : "", cfg.http_port);
if(IP_ISSET(cfg.http_srvip))
{ tpl_addVar(vars, TPLAPPEND, "SERVERIP", cs_inet_ntoa(cfg.http_srvip)); }
tpl_addVar(vars, TPLADD, "HTTPUSER", cfg.http_user);
tpl_addVar(vars, TPLADD, "HTTPPASSWORD", cfg.http_pwd);
tpl_addVar(vars, TPLADD, "HTTPOSCAMLABEL", cfg.http_oscam_label);
// css style selector
tpl_printf(vars, TPLADD, "CSSOPTIONS", "\t\t\t\t\t\t<option value=\"\"%s>embedded</option>\n",
!cfg.http_css ? " selected" : "");
if(cfg.http_tpl)
{
char path[255];
tpl_getFilePathInSubdir(cfg.http_tpl, "", "style", ".css", path, 255);
if(file_exists(path))
tpl_printf(vars, TPLAPPEND, "CSSOPTIONS", "\t\t\t\t\t\t<option value=\"%s\"%s>%s (template)</option>\n",
path,
cfg.http_css && strstr(cfg.http_css, path) ? " selected" : "",
path);
}
struct dirent **namelist;
int count = scandir(cs_confdir, &namelist, 0, NULL);
if( count >= 0 )
{
for( i = 0 ; i < count; i++ )
{
if(is_ext(namelist[i]->d_name, ".css"))
{
tpl_printf(vars, TPLAPPEND, "CSSOPTIONS", "\t\t\t\t\t\t<option value=\"%s%s\"%s>%s%s</option>\n",
cs_confdir,
namelist[i]->d_name,
cfg.http_css && strstr(cfg.http_css, namelist[i]->d_name) ? " selected" : "",
cs_confdir, namelist[i]->d_name);
}
free( namelist[i] );
}
free(namelist);
}
if(cfg.http_prepend_embedded_css)
{ tpl_addVar(vars, TPLADD, "HTTPPREPENDEMBEDDEDCSS", "checked"); }
tpl_addVar(vars, TPLADD, "HTTPHELPLANG", cfg.http_help_lang);
tpl_addVar(vars, TPLADD, "HTTPLOCALE", cfg.http_locale);
tpl_printf(vars, TPLADD, "HTTPEMMUCLEAN", "%d", cfg.http_emmu_clean);
tpl_printf(vars, TPLADD, "HTTPEMMSCLEAN", "%d", cfg.http_emms_clean);
tpl_printf(vars, TPLADD, "HTTPEMMGCLEAN", "%d", cfg.http_emmg_clean);
tpl_printf(vars, TPLADD, "HTTPREFRESH", "%d", cfg.http_refresh);
tpl_printf(vars, TPLADD, "HTTPPOLLREFRESH", "%d", cfg.poll_refresh);
tpl_addVar(vars, TPLADD, "HTTPTPL", cfg.http_tpl);
tpl_addVar(vars, TPLADD, "HTTPPICONPATH", cfg.http_piconpath);
tpl_addVar(vars, TPLADD, "HTTPSCRIPT", cfg.http_script);
tpl_addVar(vars, TPLADD, "HTTPJSCRIPT", cfg.http_jscript);
#ifndef WEBIF_JQUERY
tpl_addVar(vars, TPLADD, "HTTPEXTERNJQUERY", cfg.http_extern_jquery);
#endif
tpl_printf(vars, TPLADD, "HTTPPICONSIZE", "%d", cfg.http_picon_size);
if(cfg.http_hide_idle_clients > 0) { tpl_addVar(vars, TPLADD, "CHECKED", "checked"); }
tpl_addVar(vars, TPLADD, "HTTPHIDETYPE", cfg.http_hide_type);
if(cfg.http_status_log > 0) { tpl_addVar(vars, TPLADD, "SHOWLOGCHECKED", "checked"); }
if(cfg.http_showpicons > 0) { tpl_addVar(vars, TPLADD, "SHOWPICONSCHECKED", "checked"); }
if(cfg.http_showmeminfo > 0) { tpl_addVar(vars, TPLADD, "SHOWMEMINFOCHECKED", "checked"); }
if(cfg.http_showuserinfo > 0) { tpl_addVar(vars, TPLADD, "SHOWUSERINFOCHECKED", "checked"); }
if(cfg.http_showreaderinfo > 0) { tpl_addVar(vars, TPLADD, "SHOWREADERINFOCHECKED", "checked"); }
if(cfg.http_showcacheexinfo > 0) { tpl_addVar(vars, TPLADD, "SHOWCACHEEXINFOCHECKED", "checked"); }
if(cfg.http_showloadinfo > 0) { tpl_addVar(vars, TPLADD, "SHOWLOADINFOCHECKED", "checked"); }
if(cfg.http_showecminfo > 0) { tpl_addVar(vars, TPLADD, "SHOWECMINFOCHECKED", "checked"); }
char *value = mk_t_iprange(cfg.http_allowed);
tpl_addVar(vars, TPLADD, "HTTPALLOW", value);
free_mk_t(value);
for(i = 0; i < MAX_HTTP_DYNDNS; i++)
{
if(cfg.http_dyndns[i][0])
{
tpl_addVar(vars, TPLAPPEND, "HTTPDYNDNS", i > 0 ? "," : "");
tpl_addVar(vars, TPLAPPEND, "HTTPDYNDNS", (char *)cfg.http_dyndns[i]);
}
}
tpl_addVar(vars, TPLADD, "HTTPSAVEFULLSELECT", (cfg.http_full_cfg == 1) ? "checked" : "");
tpl_addVar(vars, TPLADD, "HTTPOVERWRITEBAKFILE", (cfg.http_overwrite_bak_file == 1) ? "checked" : "");
tpl_addVar(vars, TPLADD, "HTTPREADONLY", (cfg.http_readonly == 1) ? "checked" : "");
#ifdef WITH_SSL
if(cfg.http_cert != NULL) { tpl_addVar(vars, TPLADD, "HTTPCERT", cfg.http_cert); }
tpl_addVar(vars, TPLADD, "HTTPFORCESECUREMODESELECT", (cfg.https_force_secure_mode == 1) ? "checked" : "");
tpl_addVar(vars, TPLADD, "HTTPAUTOCREATECERTSELECT", (cfg.https_auto_create_cert == 1) ? "checked" : "");
#endif
#ifndef WEBIF_JQUERY
tpl_addVar(vars, TPLADDONCE, "CONFIGWEBIFJQUERY", tpl_getTpl(vars, "CONFIGWEBIFJQUERYBIT"));
#endif
tpl_printf(vars, TPLADD, "AULOW", "%d", cfg.aulow);
tpl_printf(vars, TPLADD, "HIDECLIENTTO", "%d", cfg.hideclient_to);
return tpl_getTpl(vars, "CONFIGWEBIF");
}
#ifdef LCDSUPPORT
static char *send_oscam_config_lcd(struct templatevars *vars, struct uriparams *params)
{
setActiveSubMenu(vars, MNU_CFG_LCD);
webif_save_config("lcd", vars, params);
tpl_addVar(vars, TPLADD, "ENABLELCDSELECTED", (cfg.enablelcd == 1) ? "checked" : "");
if(cfg.lcd_output_path != NULL)
{ tpl_addVar(vars, TPLADD, "LCDOUTPUTPATH", cfg.lcd_output_path); }
tpl_addVar(vars, TPLADD, "LCDHIDEIDLE", (cfg.lcd_hide_idle == 1) ? "checked" : "");
tpl_printf(vars, TPLADD, "LCDREFRESHINTERVAL", "%d", cfg.lcd_write_intervall);
return tpl_getTpl(vars, "CONFIGLCD");
}
#endif
#ifdef MODULE_MONITOR
static char *send_oscam_config_monitor(struct templatevars *vars, struct uriparams *params)
{
setActiveSubMenu(vars, MNU_CFG_MONITOR);
webif_save_config("monitor", vars, params);
tpl_printf(vars, TPLADD, "MONPORT", "%d", cfg.mon_port);
if(IP_ISSET(cfg.mon_srvip))
{ tpl_addVar(vars, TPLADD, "SERVERIP", cs_inet_ntoa(cfg.mon_srvip)); }
tpl_printf(vars, TPLADD, "AULOW", "%d", cfg.aulow);
tpl_printf(vars, TPLADD, "HIDECLIENTTO", "%d", cfg.hideclient_to);
char *value = mk_t_iprange(cfg.mon_allowed);
tpl_addVar(vars, TPLADD, "NOCRYPT", value);
free_mk_t(value);
//Monlevel selector
tpl_printf(vars, TPLADD, "TMP", "MONSELECTED%d", cfg.mon_level);
tpl_addVar(vars, TPLADD, tpl_getVar(vars, "TMP"), "selected");
return tpl_getTpl(vars, "CONFIGMONITOR");
}
#endif
#ifdef MODULE_SERIAL
static char *send_oscam_config_serial(struct templatevars *vars, struct uriparams *params)
{
setActiveSubMenu(vars, MNU_CFG_SERIAL);
webif_save_config("serial", vars, params);
if(cfg.ser_device)
{
char sdevice[cs_strlen(cfg.ser_device)];
cs_strncpy(sdevice, cfg.ser_device, sizeof(sdevice));
char *ptr, *saveptr1 = NULL;
char delimiter[2];
delimiter[0] = 1;
delimiter[1] = '\0';
for(ptr = strtok_r(sdevice, delimiter, &saveptr1); ptr; ptr = strtok_r(NULL, delimiter, &saveptr1))
{
tpl_addVar(vars, TPLADD, "SERIALDEVICE", xml_encode(vars, ptr));
tpl_addVar(vars, TPLAPPEND, "DEVICES", tpl_getTpl(vars, "CONFIGSERIALDEVICEBIT"));
}
}
tpl_addVar(vars, TPLADD, "SERIALDEVICE", "");
tpl_addVar(vars, TPLAPPEND, "DEVICES", tpl_getTpl(vars, "CONFIGSERIALDEVICEBIT"));
return tpl_getTpl(vars, "CONFIGSERIAL");
}
#endif
#ifdef HAVE_DVBAPI
extern const char *boxdesc[];
static char *send_oscam_config_dvbapi(struct templatevars *vars, struct uriparams *params)
{
int32_t i;
setActiveSubMenu(vars, MNU_CFG_DVBAPI);
webif_save_config("dvbapi", vars, params);
if(cfg.dvbapi_enabled > 0)
{ tpl_addVar(vars, TPLADD, "ENABLEDCHECKED", "checked"); }
if(cfg.dvbapi_au > 0)
{ tpl_addVar(vars, TPLADD, "AUCHECKED", "checked"); }
if(cfg.dvbapi_delayer > 0)
{ tpl_printf(vars, TPLADD, "DELAYER", "%d", cfg.dvbapi_delayer); }
tpl_printf(vars, TPLADD, "BOXTYPE", "<option value=\"\"%s>None</option>\n", cfg.dvbapi_boxtype == 0 ? " selected" : "");
for(i = 1; i <= BOXTYPES; i++)
{
tpl_printf(vars, TPLAPPEND, "BOXTYPE", "<option%s>%s</option>\n", cfg.dvbapi_boxtype == i ? " selected" : "", boxdesc[i]);
}
tpl_addVar(vars, TPLADD, "USERNAME", xml_encode(vars, cfg.dvbapi_usr));
//PMT Mode
tpl_printf(vars, TPLADD, "TMP", "PMTMODESELECTED%d", cfg.dvbapi_pmtmode);
tpl_addVar(vars, TPLADD, tpl_getVar(vars, "TMP"), "selected");
//Request Mode
tpl_printf(vars, TPLADD, "TMP", "REQMODESELECTED%d", cfg.dvbapi_requestmode);
tpl_addVar(vars, TPLADD, tpl_getVar(vars, "TMP"), "selected");
//ecminfo_file
if(cfg.dvbapi_ecminfo_file > 0)
{ tpl_addVar(vars, TPLADD, "ECMINFOFILECHECKED", "checked"); }
//ecminfo_type
tpl_printf(vars, TPLADD, "TMP", "ECMINFOTYPESELECTED%d", cfg.dvbapi_ecminfo_type);
tpl_addVar(vars, TPLADD, tpl_getVar(vars, "TMP"), "selected");
//read_sdt
tpl_printf(vars, TPLADD, "TMP", "READSDTSELECTED%d", cfg.dvbapi_read_sdt);
tpl_addVar(vars, TPLADD, tpl_getVar(vars, "TMP"), "selected");
#ifdef WITH_EXTENDED_CW
//extended_cw_api
tpl_printf(vars, TPLADD, "TMP", "EXTENDEDCWAPISELECTED%d", cfg.dvbapi_extended_cw_api);
tpl_addVar(vars, TPLADD, tpl_getVar(vars, "TMP"), "selected");
#endif
//write_sdt_prov
if(cfg.dvbapi_write_sdt_prov > 0)
{ tpl_addVar(vars, TPLADD, "WRITESDTPROVCHECKED", "checked"); }
#ifdef MODULE_STREAMRELAY
//demuxer_fix
if(cfg.dvbapi_demuxer_fix > 0)
{ tpl_addVar(vars, TPLADD, "DEMUXERFIXCHECKED", "checked"); }
#endif
//TCP listen port
if(cfg.dvbapi_listenport > 0)
{ tpl_printf(vars, TPLADD, "LISTENPORT", "%d", cfg.dvbapi_listenport); }
if(IP_ISSET(cfg.dvbapi_srvip))
{ tpl_addVar(vars, TPLADD, "SERVERIP", cs_inet_ntoa(cfg.dvbapi_srvip)); }
return tpl_getTpl(vars, "CONFIGDVBAPI");
}
#endif
#ifdef CS_ANTICASC
static char *send_oscam_config_anticasc(struct templatevars *vars, struct uriparams *params)
{
setActiveSubMenu(vars, MNU_CFG_ANTICASC);
webif_save_config("anticasc", vars, params);
if(cfg.ac_enabled > 0) { tpl_addVar(vars, TPLADD, "CHECKED", "checked"); }
tpl_printf(vars, TPLADD, "NUMUSERS", "%d", cfg.ac_users);
tpl_printf(vars, TPLADD, "SAMPLETIME", "%d", cfg.ac_stime);
tpl_printf(vars, TPLADD, "SAMPLES", "%d", cfg.ac_samples);
tpl_printf(vars, TPLADD, "TMP", "PENALTY%d", cfg.ac_penalty);
tpl_addVar(vars, TPLADD, tpl_getVar(vars, "TMP"), "selected");
if(cfg.ac_logfile)
{ tpl_addVar(vars, TPLADD, "ACLOGFILE", cfg.ac_logfile); }
tpl_printf(vars, TPLADD, "FAKEDELAY", "%d", cfg.ac_fakedelay);
tpl_printf(vars, TPLADD, "DENYSAMPLES", "%d", cfg.ac_denysamples);
if(cfg.acosc_enabled == 1)
{ tpl_addVar(vars, TPLADD, "ACOSC_CHECKED", "checked"); }
tpl_printf(vars, TPLADD, "ACOSC_MAX_ECMS_PER_MINUTE", "%d", cfg.acosc_max_ecms_per_minute);
tpl_printf(vars, TPLADD, "ACOSC_MAX_ACTIVE_SIDS", "%d", cfg.acosc_max_active_sids);
tpl_printf(vars, TPLADD, "ACOSC_ZAP_LIMIT", "%d", cfg.acosc_zap_limit);
tpl_printf(vars, TPLADD, "TMP", "ACOSC_PENALTY%d", cfg.acosc_penalty);
tpl_addVar(vars, TPLADD, tpl_getVar(vars, "TMP"), "selected");
tpl_printf(vars, TPLADD, "ACOSC_PENALTY_DURATION", "%d", cfg.acosc_penalty_duration);
tpl_printf(vars, TPLADD, "ACOSC_DELAY", "%d", cfg.acosc_delay);
return tpl_getTpl(vars, "CONFIGANTICASC");
}
#endif
static char *send_oscam_config(struct templatevars *vars, struct uriparams *params)
{
setActiveMenu(vars, MNU_CONFIG);
char *part = getParam(params, "part");
if(!strcmp(part, "webif")) { return send_oscam_config_webif(vars, params); }
#ifdef MODULE_MONITOR
else if(!strcmp(part, "monitor")) { return send_oscam_config_monitor(vars, params); }
#endif
#ifdef LCDSUPPORT
else if(!strcmp(part, "lcd")) { return send_oscam_config_lcd(vars, params); }
#endif
#ifdef MODULE_CAMD33
else if(!strcmp(part, "camd33")) { return send_oscam_config_camd33(vars, params); }
#endif
#ifdef MODULE_CAMD35
else if(!strcmp(part, "camd35")) { return send_oscam_config_camd35(vars, params); }
#endif
#ifdef MODULE_CAMD35_TCP
else if(!strcmp(part, "camd35tcp")) { return send_oscam_config_camd35tcp(vars, params); }
#endif
else if(!strcmp(part, "cache")) { return send_oscam_config_cache(vars, params); }
#ifdef MODULE_NEWCAMD
else if(!strcmp(part, "newcamd")) { return send_oscam_config_newcamd(vars, params); }
#endif
#ifdef MODULE_RADEGAST
else if(!strcmp(part, "radegast")) { return send_oscam_config_radegast(vars, params); }
#endif
#ifdef MODULE_SCAM
else if(!strcmp(part, "scam")) { return send_oscam_config_scam(vars, params); }
#endif
#ifdef MODULE_STREAMRELAY
else if(!strcmp(part, "streamrelay")) { return send_oscam_config_streamrelay(vars, params); }
#endif
#ifdef MODULE_CCCAM
else if(!strcmp(part, "cccam")) { return send_oscam_config_cccam(vars, params); }
#endif
#ifdef MODULE_GBOX
else if(!strcmp(part, "gbox")) { return send_oscam_config_gbox(vars, params); }
#endif
#ifdef HAVE_DVBAPI
else if(!strcmp(part, "dvbapi")) { return send_oscam_config_dvbapi(vars, params); }
#endif
#ifdef CS_ANTICASC
else if(!strcmp(part, "anticasc")) { return send_oscam_config_anticasc(vars, params); }
#endif
#ifdef MODULE_SERIAL
else if(!strcmp(part, "serial")) { return send_oscam_config_serial(vars, params); }
#endif
#ifdef WITH_LB
else if(!strcmp(part, "loadbalancer")) { return send_oscam_config_loadbalancer(vars, params); }
#endif
else { return send_oscam_config_global(vars, params); }
}
static void inactivate_reader(struct s_reader *rdr)
{
struct s_client *cl = rdr->client;
if(cl)
{ kill_thread(cl); }
}
static bool picon_exists(char *name)
{
char picon_name[255], path[255];
char *tpl_path;
tpl_path = cfg.http_piconpath ? cfg.http_piconpath : cfg.http_tpl;
if(!tpl_path)
{ return false; }
snprintf(picon_name, sizeof(picon_name) - 1, "IC_%s", name);
return cs_strlen(tpl_getTplPath(picon_name, tpl_path, path, sizeof(path) - 1)) && file_exists(path);
}
static void clear_rdr_stats(struct s_reader *rdr)
{
int i;
for(i = 0; i < 4; i++)
{
rdr->emmerror[i] = 0;
rdr->emmwritten[i] = 0;
rdr->emmskipped[i] = 0;
rdr->emmblocked[i] = 0;
}
rdr->ecmsok = 0;
#ifdef CS_CACHEEX_AIO
rdr->ecmsoklg = 0;
#endif
rdr->ecmsnok = 0;
rdr->ecmstout = 0;
rdr->ecmshealthok = 0;
#ifdef CS_CACHEEX_AIO
rdr->ecmshealthoklg = 0;
#endif
rdr->ecmshealthnok = 0;
rdr->ecmshealthtout = 0;
rdr->ecmsfilteredhead = 0;
rdr->ecmsfilteredlen = 0;
}
static void clear_all_rdr_stats(void)
{
struct s_reader *rdr;
LL_ITER itr = ll_iter_create(configured_readers);
while((rdr = ll_iter_next(&itr)))
{
clear_rdr_stats(rdr);
}
}
static char *send_oscam_reader(struct templatevars *vars, struct uriparams *params, int32_t apicall)
{
struct s_reader *rdr;
int32_t i;
uint8_t md5tmp[MD5_DIGEST_LENGTH];
char *status;
if(!apicall) { setActiveMenu(vars, MNU_READERS); }
if(!apicall)
{
if(strcmp(getParam(params, "action"), "resetallrdrstats") == 0)
{
clear_all_rdr_stats();
}
}
tpl_addVar(vars, TPLADD, "READERACTIONCOLS", config_enabled(WITH_LB) ? "6" : "5");
if(strcmp(getParam(params, "action"), "resetuserstats") == 0)
{
clear_info_clients_stats();
}
if(strcmp(getParam(params, "action"), "resetreaderstats") == 0)
{
clear_info_readers_stats();
}
if(strcmp(getParam(params, "action"), "reloadreaders") == 0)
{
if(!cfg.http_readonly)
{
refresh_oscam(REFR_READERS);
refresh_oscam(REFR_ACCOUNTS);
}
}
if((strcmp(getParam(params, "action"), "disable") == 0) || (strcmp(getParam(params, "action"), "enable") == 0))
{
if(cfg.http_readonly)
{
tpl_addMsg(vars, "WebIf is in readonly mode. Enabling or disabling readers is not possible!");
}
else
{
rdr = get_reader_by_label(getParam(params, "label"));
if(rdr)
{
if(strcmp(getParam(params, "action"), "enable") == 0)
{
if(!rdr->enable)
{
rdr->enable = 1;
}
}
else
{
if(rdr->enable)
{
rdr->enable = 0;
}
}
if(rdr->typ != R_GBOX)
{
restart_cardreader(rdr, 1);
}
#ifdef MODULE_GBOX
else
{
restart_gbox_peer(rdr->label, 0, 0);
cs_log("gbox -> you must restart oscam so that setting becomes effective");
}
#endif
cs_log("reader %s %s by WebIf", rdr->label, rdr->enable == 1 ? "enabled":"disabled");
if(write_server() != 0) { tpl_addMsg(vars, "Write Config failed!"); }
#ifdef MODULE_GBOX
if(!is_network_reader(rdr) && !rdr->enable)
{
gbx_local_card_stat(LOCALCARDDISABLED, 0);
}
#endif
}
}
}
if(strcmp(getParam(params, "action"), "delete") == 0)
{
if(cfg.http_readonly)
{
tpl_addMsg(vars, "WebIf is in readonly mode. No deletion will be made!");
}
else
{
rdr = get_reader_by_label(getParam(params, "label"));
if(rdr)
{
inactivate_reader(rdr);
ll_remove(configured_readers, rdr);
free_reader(rdr);
if(write_server() != 0) { tpl_addMsg(vars, "Write Config failed!"); }
}
}
}
if(strcmp(getParam(params, "action"), "reread") == 0)
{
rdr = get_reader_by_label(getParam(params, "label"));
if(rdr)
{
struct s_client *cl = rdr->client;
//reset the counters
for(i = 0; i < 4; i++)
{
rdr->emmerror[i] = 0;
rdr->emmwritten[i] = 0;
rdr->emmskipped[i] = 0;
rdr->emmblocked[i] = 0;
}
if(rdr->enable == 1 && cl && cl->typ == 'r')
{
add_job(cl, ACTION_READER_CARDINFO, NULL, 0);
}
}
}
LL_ITER itr = ll_iter_create(configured_readers);
if(!apicall)
{
for(i = 0, rdr = ll_iter_next(&itr); rdr && rdr->label[0]; rdr = ll_iter_next(&itr), i++) { ; }
tpl_printf(vars, TPLADD, "NEXTREADER", "Reader-%d", i); //Next Readername
}
int jsondelimiter = 0;
int existing_insert = 0;
int32_t total_readers = 0;
int32_t disabled_readers = 0;
int32_t active_readers = 0;
int32_t connected_readers = 0;
ll_iter_reset(&itr); //going to iterate all configured readers
while((rdr = ll_iter_next(&itr)))
{
#ifdef CS_CACHEEX_AIO
const char *proto = reader_get_type_desc(rdr, 0);
#endif
struct s_client *cl = rdr->client;
if(rdr->label[0] && rdr->typ)
{
#ifdef CS_CACHEEX_AIO
char *new_proto;
#if defined(MODULE_CAMD35) || defined (MODULE_CAMD35_TCP)
if(rdr->cacheex.feature_bitfield || (cl && cl->c35_extmode > 1))
#else
if(rdr->cacheex.feature_bitfield)
#endif
{
const char *aio_suffix = " (cx-aio)";
if(cs_malloc(&new_proto, cs_strlen(proto)+cs_strlen(aio_suffix)+1)) {
if (!cs_strncat(new_proto, (char *)proto, cs_strlen(proto)+cs_strlen(aio_suffix)+1)) {
cs_log("FIXME!");
}
if (!cs_strncat(new_proto, (char *)aio_suffix, cs_strlen(proto)+cs_strlen(aio_suffix)+1)) {
cs_log("FIXME!");
}
}
}
#endif
total_readers += 1;
// used for API and WebIf
tpl_addVar(vars, TPLADD, "READERNAME", xml_encode(vars, rdr->label));
MD5((uint8_t *)rdr->label, cs_strlen(rdr->label), md5tmp);
int z;
tpl_addVar(vars, TPLADD, "LABELMD5","id_");
for (z = 0; z < MD5_DIGEST_LENGTH; z++)
{
tpl_printf(vars, TPLAPPEND, "LABELMD5", "%02x", md5tmp[z]);
}
#ifdef MODULE_GBOX
if(apicall)
{
tpl_addVar(vars, TPLADD, "LASTGSMS", "");
tpl_addVar(vars, TPLADD, "LASTGSMS", rdr->last_gsms);
}
#endif
if(apicall)
{
tpl_printf(vars, TPLADD, "PICONENABLED", "%d", cfg.http_showpicons?1:0);
}
tpl_addVar(vars, TPLADD, "READERNAMEENC", urlencode(vars, rdr->label));
if(!existing_insert)
{
tpl_printf(vars, TPLADD, "EXISTING_INS", "'%s'", urlencode(vars, rdr->label));
existing_insert++;
}else
{
tpl_printf(vars, TPLAPPEND, "EXISTING_INS", ",'%s'", urlencode(vars, rdr->label));
}
tpl_addVar(vars, TPLADD, "READERCLASS", rdr->enable ? "enabledreader" : "disabledreader");
if(rdr->enable) { active_readers += 1; }
else { disabled_readers += 1; }
if(rdr->tcp_connected)
{
connected_readers += 1;
#ifdef CS_CACHEEX_AIO
if(rdr->cacheex.feature_bitfield)
{
tpl_addVar(vars, TPLADD, "CLIENTPROTOSORT", (const char*)new_proto);
tpl_addVar(vars, TPLADD, "CLIENTPROTO", (const char*)new_proto);
if(cfg.http_showpicons)
{
char picon_name[32];
snprintf(picon_name, sizeof(picon_name) / sizeof(char) - 1, "%s", new_proto);
if(picon_exists(picon_name))
{
tpl_printf(vars, TPLADDONCE, "PROTOICON", "%s",(char*)new_proto);
}
}
if(rdr->cacheex.feature_bitfield & 32)
tpl_addVar(vars, TPLADD, "CLIENTPROTOTITLE", rdr->cacheex.aio_version);
else if(cl->reader->cacheex.feature_bitfield)
tpl_addVar(vars, TPLADD, "CLIENTPROTOTITLE", "[cx-aio < 9.2.3]");
}
else
{
tpl_addVar(vars, TPLADD, "CLIENTPROTOSORT", proto);
tpl_addVar(vars, TPLADD, "CLIENTPROTO", proto);
if(cfg.http_showpicons)
{
char picon_name[32];
snprintf(picon_name, sizeof(picon_name) / sizeof(char) - 1, "%s", proto);
if(picon_exists(picon_name))
{
tpl_printf(vars, TPLADDONCE, "PROTOICON", "%s",(char *)proto);
}
}
}
#else
tpl_addVar(vars, TPLADD, "CLIENTPROTO", reader_get_type_desc(rdr, 0));
tpl_addVar(vars, TPLADD, "CLIENTPROTOSORT", reader_get_type_desc(rdr, 0));
if(cfg.http_showpicons)
{
char picon_name[32];
snprintf(picon_name, sizeof(picon_name) / sizeof(char) - 1, "%s", reader_get_type_desc(rdr, 0));
if(picon_exists(picon_name))
{
tpl_printf(vars, TPLADDONCE, "PROTOICON", "%s", reader_get_type_desc(rdr, 0));
}
}
#endif
switch(rdr->card_status)
{
case CARD_INSERTED:
status = "<B>online</B>";
tpl_addVar(vars, TPLADD, "RSTATUS", status);
tpl_addVar(vars, TPLADD, "READERCLASS", "r_connected");
break;
case NO_CARD:
case UNKNOWN:
case READER_DEVICE_ERROR:
case CARD_NEED_INIT:
case CARD_FAILURE:
default:
status = "<B>connected</B>";
tpl_addVar(vars, TPLADD, "RSTATUS", status);
tpl_addVar(vars, TPLADD, "READERCLASS", "r_undefined");
break;
}
tpl_addVar(vars, TPLADD, "READERIP", cs_inet_ntoa(rdr->client->ip));
}
else
{
/* default initial values */
tpl_addVar(vars, TPLADDONCE, "RSTATUS", "offline");
tpl_addVar(vars, TPLADDONCE, "READERIP", "");
tpl_addVar(vars, TPLADDONCE, "CLIENTPROTO", "");
tpl_addVar(vars, TPLADDONCE, "CLIENTPROTOSORT", "");
tpl_addVar(vars, TPLADDONCE, "CLIENTPROTOTITLE", "");
tpl_addVar(vars, TPLADDONCE, "PROTOICON", "");
if(!is_network_reader(rdr) && rdr->enable)
{
switch(rdr->card_status)
{
case CARD_INSERTED:
status = "<B>active</B>";
tpl_addVar(vars, TPLADD, "RSTATUS", status);
tpl_addVar(vars, TPLADD, "READERCLASS", "r_connected");
break;
case NO_CARD:
case UNKNOWN:
case READER_DEVICE_ERROR:
case CARD_NEED_INIT:
case CARD_FAILURE:
default:
status = "<B>connected</B>";
tpl_addVar(vars, TPLADD, "RSTATUS", status);
tpl_addVar(vars, TPLADD, "READERCLASS", "r_undefined");
break;
}
tpl_addVar(vars, TPLADD, "CLIENTPROTO", reader_get_type_desc(rdr, 0));
tpl_addVar(vars, TPLADD, "CLIENTPROTOSORT", reader_get_type_desc(rdr, 0));
if(cfg.http_showpicons)
{
char picon_name[32];
snprintf(picon_name, sizeof(picon_name) / sizeof(char) - 1, "%s", reader_get_type_desc(rdr, 0));
if(picon_exists(picon_name))
{
tpl_printf(vars, TPLADDONCE, "PROTOICON", "%s", reader_get_type_desc(rdr, 0));
}
}
}
}
if(rdr->description)
tpl_printf(vars, TPLADD, "DESCRIPTION","%s(%s)",!apicall?"&#13;":"",xml_encode(vars, rdr->description));
else
tpl_addVar(vars, TPLADD, "DESCRIPTION", "");
if(cfg.http_showpicons && !apicall)
{
tpl_addVar(vars, TPLADD, "READERBIT", tpl_getTpl(vars, picon_exists(xml_encode(vars, rdr->label)) ? "READERNAMEBIT" : "READERNOICON"));
#ifdef CS_CACHEEX_AIO
if(rdr->cacheex.feature_bitfield)
{
tpl_addVar(vars, TPLADD, "CLIENTPROTO", picon_exists(xml_encode(vars, (const char*)new_proto)) ? tpl_getTpl(vars, "READERCTYPBIT") : tpl_getTpl(vars, "READERCTYPNOICON"));
}
else
{
#endif
tpl_addVar(vars, TPLADD, "CLIENTPROTO", picon_exists(xml_encode(vars, reader_get_type_desc(rdr, 0))) ? tpl_getTpl(vars, "READERCTYPBIT") : tpl_getTpl(vars, "READERCTYPNOICON"));
#ifdef CS_CACHEEX_AIO
}
#endif
}
else
tpl_addVar(vars, TPLADD, "READERBIT", tpl_getTpl(vars, "READERLABEL"));
char *value = mk_t_group(rdr->grp);
tpl_addVar(vars, TPLADD, "GROUPS", value);
free_mk_t(value);
tpl_printf(vars, TPLADD, "EMMERRORUK", PRINTF_LOCAL_D, rdr->emmerror[UNKNOWN]);
tpl_printf(vars, TPLADD, "EMMERRORG", PRINTF_LOCAL_D, rdr->emmerror[GLOBAL]);
tpl_printf(vars, TPLADD, "EMMERRORS", PRINTF_LOCAL_D, rdr->emmerror[SHARED]);
tpl_printf(vars, TPLADD, "EMMERRORUQ", PRINTF_LOCAL_D, rdr->emmerror[UNIQUE]);
tpl_printf(vars, TPLADD, "EMMWRITTENUK", PRINTF_LOCAL_D, rdr->emmwritten[UNKNOWN]);
tpl_printf(vars, TPLADD, "EMMWRITTENG", PRINTF_LOCAL_D, rdr->emmwritten[GLOBAL]);
tpl_printf(vars, TPLADD, "EMMWRITTENS", PRINTF_LOCAL_D, rdr->emmwritten[SHARED]);
tpl_printf(vars, TPLADD, "EMMWRITTENUQ", PRINTF_LOCAL_D, rdr->emmwritten[UNIQUE]);
tpl_printf(vars, TPLADD, "EMMSKIPPEDUK", PRINTF_LOCAL_D, rdr->emmskipped[UNKNOWN]);
tpl_printf(vars, TPLADD, "EMMSKIPPEDG", PRINTF_LOCAL_D, rdr->emmskipped[GLOBAL]);
tpl_printf(vars, TPLADD, "EMMSKIPPEDS", PRINTF_LOCAL_D, rdr->emmskipped[SHARED]);
tpl_printf(vars, TPLADD, "EMMSKIPPEDUQ", PRINTF_LOCAL_D, rdr->emmskipped[UNIQUE]);
tpl_printf(vars, TPLADD, "EMMBLOCKEDUK", PRINTF_LOCAL_D, rdr->emmblocked[UNKNOWN]);
tpl_printf(vars, TPLADD, "EMMBLOCKEDG", PRINTF_LOCAL_D, rdr->emmblocked[GLOBAL]);
tpl_printf(vars, TPLADD, "EMMBLOCKEDS", PRINTF_LOCAL_D, rdr->emmblocked[SHARED]);
tpl_printf(vars, TPLADD, "EMMBLOCKEDUQ", PRINTF_LOCAL_D, rdr->emmblocked[UNIQUE]);
tpl_printf(vars, TPLADD, "ECMSOK", PRINTF_LOCAL_D, rdr->ecmsok);
tpl_printf(vars, TPLADD, "ECMSOKREL", " (%.2f %%)", rdr->ecmshealthok);
#ifdef CS_CACHEEX_AIO
tpl_printf(vars, TPLADD, "ECMSOKLG", PRINTF_LOCAL_D, rdr->ecmsoklg);
tpl_printf(vars, TPLADD, "ECMSOKLGREL", " (%.2f %%)", rdr->ecmshealthoklg);
#endif
tpl_printf(vars, TPLADD, "ECMSNOK", PRINTF_LOCAL_D, rdr->ecmsnok);
tpl_printf(vars, TPLADD, "ECMSNOKREL", " (%.2f %%)",rdr->ecmshealthnok);
tpl_printf(vars, TPLADD, "ECMSTOUT", PRINTF_LOCAL_D, rdr->ecmstout);
tpl_printf(vars, TPLADD, "ECMSTOUTREL", " (%.2f %%)",rdr->ecmshealthtout);
tpl_printf(vars, TPLADD, "ECMSFILTEREDHEAD", PRINTF_LOCAL_D, rdr->ecmsfilteredhead);
tpl_printf(vars, TPLADD, "ECMSFILTEREDLEN", PRINTF_LOCAL_D, rdr->ecmsfilteredlen);
#ifdef WITH_LB
tpl_printf(vars, TPLADD, "LBWEIGHT", "%d", rdr->lb_weight);
#endif
if(!is_network_reader(rdr)) //reader is physical
{
tpl_addVar(vars, TPLADD, "REFRICO", "image?i=ICREF");
tpl_addVar(vars, TPLADD, "READERREFRESH", tpl_getTpl(vars, "READERREFRESHBIT"));
tpl_addVar(vars, TPLADD, "ENTICO", "image?i=ICENT");
tpl_addVar(vars, TPLADD, "ENTITLEMENT", tpl_getTpl(vars, "READERENTITLEBIT"));
}
else
{
tpl_addVar(vars, TPLADD, "READERREFRESH", "");
if(rdr->typ == R_CCCAM)
{
tpl_addVar(vars, TPLADD, "ENTICO", "image?i=ICENT");
tpl_addVar(vars, TPLADD, "ENTITLEMENT", tpl_getTpl(vars, "READERENTITLEBIT"));
}
else
{
tpl_addVar(vars, TPLADD, "ENTITLEMENT", "");
}
}
if(rdr->enable == 0)
{
tpl_addVar(vars, TPLADD, "SWITCHICO", "image?i=ICENA");
tpl_addVar(vars, TPLADD, "SWITCHTITLE", "Enable");
tpl_addVar(vars, TPLADD, "SWITCH", "enable");
tpl_addVar(vars, TPLADD, "WRITEEMM", "");
}
else
{
tpl_addVar(vars, TPLADD, "SWITCHICO", "image?i=ICDIS");
tpl_addVar(vars, TPLADD, "SWITCHTITLE", "Disable");
tpl_addVar(vars, TPLADD, "SWITCH", "disable");
tpl_addVar(vars, TPLADD, "EMMICO", "image?i=ICEMM");
tpl_addVar(vars, TPLADD, "WRITEEMM", tpl_getTpl(vars, "READERWRITEEMMBIT"));
}
if(!apicall)
{
// Add to WebIf Template
#ifdef CS_CACHEEX_AIO
tpl_addVar(vars, TPLAPPEND, "READERLIST", tpl_getTpl(vars, "READERSBITAIO"));
#else
tpl_addVar(vars, TPLAPPEND, "READERLIST", tpl_getTpl(vars, "READERSBIT"));
#endif
}
else
{
// used only for API
tpl_addVar(vars, TPLADD, "APIREADERENABLED", !rdr->enable ? "0" : "1");
if(cl)
{
tpl_printf(vars, TPLADD, "APIREADERTYPE", "%c", cl->typ ? cl->typ : 'x');
}
if(apicall==1)
{
// Add to API Template
tpl_addVar(vars, TPLAPPEND, "APIREADERLIST", tpl_getTpl(vars, "APIREADERSBIT"));
}
if(apicall==2)
{
tpl_printf(vars, TPLAPPEND, "APIREADERLIST","%s%s",jsondelimiter?",":"", tpl_getTpl(vars, "JSONREADERBIT"));
jsondelimiter++;
}
}
#ifdef CS_CACHEEX_AIO
if(rdr->cacheex.feature_bitfield)
{
free(new_proto);
}
#endif
}
}
tpl_printf(vars, TPLADD, "TOTAL_READERS", "%d", total_readers);
tpl_printf(vars, TPLADD, "TOTAL_DISABLED_READERS", "%d", disabled_readers);
tpl_printf(vars, TPLADD, "TOTAL_ACTIVE_READERS", "%d", active_readers);
tpl_printf(vars, TPLADD, "TOTAL_CONNECTED_READERS", "%d", connected_readers);
//CM info
tpl_addVar(vars, TPLADD, "DISPLAYUSERINFO", "hidden"); // no userinfo in readers
set_ecm_info(vars);
if(!apicall)
{
#ifdef MODULE_CAMD33
tpl_addVar(vars, TPLAPPEND, "ADDPROTOCOL", "<option>camd33</option>\n");
#endif
#ifdef MODULE_CAMD35
tpl_addVar(vars, TPLAPPEND, "ADDPROTOCOL", "<option>cs357x</option>\n");
#endif
#ifdef MODULE_CAMD35_TCP
tpl_addVar(vars, TPLAPPEND, "ADDPROTOCOL", "<option>cs378x</option>\n");
#endif
#ifdef MODULE_NEWCAMD
tpl_addVar(vars, TPLAPPEND, "ADDPROTOCOL", "<option>newcamd</option>\n");
tpl_addVar(vars, TPLAPPEND, "ADDPROTOCOL", "<option>newcamd524</option>\n");
#endif
#ifdef MODULE_CCCAM
tpl_addVar(vars, TPLAPPEND, "ADDPROTOCOL", "<option>cccam</option>\n");
#endif
#ifdef MODULE_GBOX
tpl_addVar(vars, TPLAPPEND, "ADDPROTOCOL", "<option>gbox</option>\n");
#endif
#ifdef MODULE_RADEGAST
tpl_addVar(vars, TPLAPPEND, "ADDPROTOCOL", "<option>radegast</option>\n");
#endif
#ifdef MODULE_SERIAL
tpl_addVar(vars, TPLAPPEND, "ADDPROTOCOL", "<option>serial</option>\n");
#endif
#ifdef MODULE_CONSTCW
tpl_addVar(vars, TPLAPPEND, "ADDPROTOCOL", "<option>constcw</option>\n");
#endif
#ifdef MODULE_SCAM
tpl_addVar(vars, TPLAPPEND, "ADDPROTOCOL", "<option>scam</option>\n");
#endif
for(i = 0; cardreaders[i]; i++)
{
tpl_printf(vars, TPLAPPEND, "ADDPROTOCOL", "<option>%s</option>\n", xml_encode(vars, cardreaders[i]->desc));
}
#ifdef CS_CACHEEX_AIO
return tpl_getTpl(vars, "READERSAIO");
#else
return tpl_getTpl(vars, "READERS");
#endif
}
else
{
if(apicall == 1)
{
return tpl_getTpl(vars, "APIREADERS");
}
else
{
return tpl_getTpl(vars, "JSONREADER");
}
}
}
static char *send_oscam_reader_config(struct templatevars *vars, struct uriparams *params)
{
int32_t i;
int32_t apicall = 0;
char *reader_ = getParam(params, "label");
char *value;
struct s_reader *rdr;
if(!apicall) { setActiveMenu(vars, MNU_READERS); }
if(strcmp(getParam(params, "action"), "Add") == 0)
{
// Add new reader
struct s_reader *newrdr;
if(!cs_malloc(&newrdr, sizeof(struct s_reader))) { return "0"; }
for(i = 0; i < (*params).paramcount; ++i)
{
if(strcmp((*params).params[i], "action"))
{ chk_reader((*params).params[i], (*params).values[i], newrdr); }
}
module_reader_set(newrdr);
reader_ = newrdr->label;
reader_set_defaults(newrdr);
newrdr->enable = 0; // do not start the reader because must configured before
ll_append(configured_readers, newrdr);
tpl_addMsg(vars, "New Reader has been added with default settings");
}
else if(strcmp(getParam(params, "action"), "Save") == 0)
{
rdr = get_reader_by_label(getParam(params, "label"));
if(!rdr)
{ return NULL; }
//if (is_network_reader(rdr))
// inactivate_reader(rdr); //Stop reader before reinitialization
char servicelabels[1024] = "";
char servicelabelslb[1024] = "";
for(i = 0; i < (*params).paramcount; ++i)
{
if((strcmp((*params).params[i], "reader")) && (strcmp((*params).params[i], "action")))
{
if(!strcmp((*params).params[i], "services"))
{ snprintf(servicelabels + cs_strlen(servicelabels), sizeof(servicelabels) - cs_strlen(servicelabels), "%s,", (*params).values[i]); }
else if(!strcmp((*params).params[i], "lb_whitelist_services"))
{ snprintf(servicelabelslb + cs_strlen(servicelabelslb), sizeof(servicelabelslb) - cs_strlen(servicelabelslb), "%s,", (*params).values[i]); }
else
/*if(cs_strlen((*params).values[i]) > 0)*/
{ chk_reader((*params).params[i], (*params).values[i], rdr); }
}
//printf("param %s value %s\n",(*params).params[i], (*params).values[i]);
}
chk_reader("services", servicelabels, rdr);
chk_reader("lb_whitelist_services", servicelabelslb, rdr);
if(is_network_reader(rdr) || rdr->typ == R_EMU) //physical readers make trouble if re-started
{
if(rdr)
{
if(rdr->typ != R_GBOX)
{
restart_cardreader(rdr, 1);
}
#ifdef MODULE_GBOX
else
{
//cs_log("SAVE - single gbox reader %s restarted by WebIf", rdr->label);
restart_gbox_peer(rdr->label, 0, 0);
}
#endif
}
}
if(write_server() != 0) { tpl_addMsg(vars, "Write Config failed!"); } else { tpl_addMsg(vars, "Reader config updated and saved"); }
}
rdr = get_reader_by_label(reader_);
if(!rdr)
{ return NULL; }
// Label, Description
tpl_addVar(vars, TPLADD, "READERNAME", xml_encode(vars, rdr->label));
tpl_addVar(vars, TPLADD, "DESCRIPTION", xml_encode(vars, rdr->description));
// enabled
if(!apicall)
{
tpl_addVar(vars, TPLADD, "ENABLED", (rdr->enable == 1) ? "checked" : "");
}
else
{
tpl_addVar(vars, TPLADD, "ENABLEDVALUE", (rdr->enable == 1) ? "1" : "0");
}
tpl_addVar(vars, TPLADD, "PASSWORD", xml_encode(vars, rdr->r_pwd));
tpl_addVar(vars, TPLADD, "USERNAME", xml_encode(vars, rdr->r_usr));
tpl_addVar(vars, TPLADD, "PASS", xml_encode(vars, rdr->r_pwd));
// Key Newcamd
for(i = 0; i < (int32_t)sizeof(rdr->ncd_key); i++)
{ tpl_printf(vars, TPLAPPEND, "NCD_KEY", "%02X", rdr->ncd_key[i]); }
// Pincode
tpl_addVar(vars, TPLADD, "PINCODE", rdr->pincode);
// Emmfile Path
if(rdr->emmfile) { tpl_addVar(vars, TPLADD, "EMMFILE", (char *)rdr->emmfile); }
// Inactivity timeout
tpl_printf(vars, TPLADD, "INACTIVITYTIMEOUT", "%d", rdr->tcp_ito);
// Receive timeout
tpl_printf(vars, TPLADD, "RECEIVETIMEOUT", "%d", rdr->tcp_rto);
// keepalive
tpl_addVar(vars, TPLADD, "RDRKEEPALIVE", (rdr->keepalive == 1) ? "checked" : "");
// Connect on init (newcamd)
if(!apicall)
{
tpl_addVar(vars, TPLADD, "CONNECTONINITCHECKED", (rdr->ncd_connect_on_init == 1) ? "checked" : "");
}
else
{
tpl_addVar(vars, TPLADD, "CONNECTONINITCHECKED", (rdr->ncd_connect_on_init == 1) ? "1" : "0");
}
// Reset Cycle
tpl_printf(vars, TPLADD, "RESETCYCLE", "%d", rdr->resetcycle);
// Disable Serverfilter
if(!apicall)
{
tpl_addVar(vars, TPLADD, "DISABLESERVERFILTERCHECKED", (rdr->ncd_disable_server_filt == 1) ? "checked" : "");
}
else
{
tpl_addVar(vars, TPLADD, "DISABLESERVERFILTERVALUE", (rdr->ncd_disable_server_filt == 1) ? "1" : "0");
}
#ifdef MODULE_GHTTP
// Use SSL
if(!apicall)
{
tpl_addVar(vars, TPLADD, "USESSLCHECKED", (rdr->ghttp_use_ssl == 1) ? "checked" : "");
}
else
{
tpl_addVar(vars, TPLADD, "USESSLVALUE", (rdr->ghttp_use_ssl == 1) ? "1" : "0");
}
#endif
// IPv4 force
if (!apicall)
{
tpl_addVar(vars, TPLADD, "IPV4FORCE", (rdr->ipv4force == 1) ? "checked" : "");
}
else
{
tpl_addVar(vars, TPLADD, "IPV4FORCE", (rdr->ipv4force == 1) ? "1" : "0");
}
// Fallback
if(!apicall)
{
tpl_addVar(vars, TPLADD, "FALLBACKCHECKED", (rdr->fallback == 1) ? "checked" : "");
}
else
{
tpl_addVar(vars, TPLADD, "FALLBACKVALUE", (rdr->fallback == 1) ? "1" : "0");
}
// Fallback per caid
value = mk_t_ftab(&rdr->fallback_percaid);
tpl_addVar(vars, TPLADD, "FALLBACK_PERCAID", value);
free_mk_t(value);
// disable checksum test only for selected caid/provid
value = mk_t_ftab(&rdr->disablecrccws_only_for);
tpl_addVar(vars, TPLADD, "IGN_CHKSUM_ONLYFOR", value);
free_mk_t(value);
#ifdef WITH_LB
tpl_addVar(vars, TPLADD, "LBFORCEFALLBACK", (rdr->lb_force_fallback == 1) ? "checked" : "");
#endif
#ifdef CS_CACHEEX
// Cacheex
if(!apicall)
{
tpl_printf(vars, TPLADD, "TMP", "CACHEEXSELECTED%d", rdr->cacheex.mode);
tpl_addVar(vars, TPLADD, tpl_getVar(vars, "TMP"), "selected");
}
else
{
tpl_printf(vars, TPLADD, "CACHEEX", "%d", rdr->cacheex.mode);
}
tpl_printf(vars, TPLADD, "CACHEEX_MAXHOP", "%d", rdr->cacheex.maxhop);
#ifdef CS_CACHEEX_AIO
tpl_printf(vars, TPLADD, "CACHEEX_MAXHOP_LG", "%d", rdr->cacheex.maxhop_lg);
#endif
value = mk_t_cacheex_hitvaluetab(&rdr->cacheex.filter_caidtab);
//if (cs_strlen(value) > 0)
tpl_printf(vars, TPLADD, "CACHEEX_ECM_FILTER", "%s", value);
free_mk_t(value);
tpl_addVar(vars, TPLADD, "DCCHECKED", (rdr->cacheex.drop_csp == 1) ? "checked" : "");
tpl_addVar(vars, TPLADD, "ARCHECKED", (rdr->cacheex.allow_request == 1) ? "checked" : "");
tpl_addVar(vars, TPLADD, "AFCHECKED", (rdr->cacheex.allow_filter == 1) ? "checked" : "");
#ifdef CS_CACHEEX_AIO
tpl_addVar(vars, TPLADD, "AMCHECKED", (rdr->cacheex.allow_maxhop == 1) ? "checked" : "");
#endif
tpl_addVar(vars, TPLADD, "BLOCKFAKECWSCHECKED", (rdr->cacheex.block_fakecws == 1) ? "checked" : "");
#ifdef CS_CACHEEX_AIO
tpl_addVar(vars, TPLADD, "USECWCHECKFORPUSHCHECKED", (rdr->cacheex.cw_check_for_push == 1) ? "checked" : "");
tpl_addVar(vars, TPLADD, "LGONLYREMOTESETTINGSCHECKED", (rdr->cacheex.lg_only_remote_settings == 1) ? "checked" : "");
tpl_addVar(vars, TPLADD, "LOCALGENERATEDONLYCHECKED", (rdr->cacheex.localgenerated_only == 1) ? "checked" : "");
value = mk_t_ftab(&rdr->cacheex.lg_only_tab);
tpl_addVar(vars, TPLADD, "LGONLYTAB", value);
free_mk_t(value);
tpl_addVar(vars, TPLADD, "LOCALGENERATEDONLYINCHECKED", (rdr->cacheex.localgenerated_only_in == 1) ? "checked" : "");
tpl_addVar(vars, TPLADD, "LGONLYINAIOONLYCHECKED", (rdr->cacheex.lg_only_in_aio_only == 1) ? "checked" : "");
value = mk_t_ftab(&rdr->cacheex.lg_only_in_tab);
tpl_addVar(vars, TPLADD, "LGONLYINTAB", value);
free_mk_t(value);
value = mk_t_caidvaluetab(&rdr->cacheex.cacheex_nopushafter_tab);
tpl_addVar(vars, TPLADD, "CACHEEXNOPUSHAFTER", value);
free_mk_t(value);
#endif
#endif
// BoxID
if(rdr->boxid)
{ tpl_printf(vars, TPLADD, "BOXID", "%08X", rdr->boxid); }
#ifdef READER_VIDEOGUARD
// Filt 07
if(!apicall)
{
tpl_addVar(vars, TPLADD, "FIX07CHECKED", (rdr->fix_07 == 1) ? "checked" : "");
}
else
{
tpl_addVar(vars, TPLADD, "FIX07VALUE", (rdr->fix_07 == 1) ? "1" : "0");
}
// Fix 9993
if(!apicall)
{
tpl_addVar(vars, TPLADD, "FIX9993CHECKED", (rdr->fix_9993 == 1) ? "checked" : "");
}
else
{
tpl_addVar(vars, TPLADD, "FIX9993VALUE", (rdr->fix_9993 == 1) ? "1" : "0");
}
#endif
// Drop CWs with wrong checksum:
if(!apicall)
{
tpl_addVar(vars, TPLADD, "DROPBADCWSCHECKED", (rdr->dropbadcws == 1) ? "checked" : "");
}
else
{
tpl_addVar(vars, TPLADD, "DROPBADCWSVALUE", (rdr->dropbadcws == 1) ? "1" : "0");
}
// Disable CWs checksum test:
if(!apicall)
{
tpl_addVar(vars, TPLADD, "DISABLECRCCWSCHECKED", (rdr->disablecrccws == 1) ? "checked" : "");
}
else
{
tpl_addVar(vars, TPLADD, "DISABLECRCCWSVALUE", (rdr->disablecrccws == 1) ? "1" : "0");
}
#ifdef WITH_CARDREADER
// Set reader to use GPIO
if(!apicall)
{
tpl_addVar(vars, TPLADD, "USE_GPIOCHECKED", rdr->use_gpio ? "checked" : "");
}
else
{
tpl_addVar(vars, TPLADD, "USE_GPIOVALUE", rdr->use_gpio ? "1" : "0");
}
#endif
// AUdisabled
if(!apicall)
{
tpl_addVar(vars, TPLADD, "AUDISABLED", (rdr->audisabled == 1) ? "checked" : "");
}
else
{
tpl_addVar(vars, TPLADD, "AUDISABLEDVALUE", (rdr->audisabled == 1) ? "1" : "0");
}
// AUprovid
if(rdr->auprovid)
{ tpl_printf(vars, TPLADD, "AUPROVID", "%06X", rdr->auprovid); }
if(rdr->ecmnotfoundlimit)
{ tpl_printf(vars, TPLADD, "ECMNOTFOUNDLIMIT", "%u", rdr->ecmnotfoundlimit); }
#ifdef READER_IRDETO
// Force Irdeto
if(!apicall)
{
tpl_addVar(vars, TPLADD, "FORCEIRDETOCHECKED", (rdr->force_irdeto == 1) ? "checked" : "");
}
else
{
tpl_addVar(vars, TPLADD, "FORCEIRDETOVALUE", (rdr->force_irdeto == 1) ? "1" : "0");
}
#endif
#ifdef READER_CRYPTOWORKS
// needsglobalfirst
if(!apicall)
{
tpl_addVar(vars, TPLADD, "NEEDSGLOBALFIRST", (rdr->needsglobalfirst == 1) ? "checked" : "");
}
else
{
tpl_addVar(vars, TPLADD, "NEEDSGLOBALFIRST", (rdr->needsglobalfirst == 1) ? "1" : "0");
}
#endif
#ifdef WITH_CARDREADER
// RSA Key
for(i = 0; i < rdr->rsa_mod_length; i++)
{ tpl_printf(vars, TPLAPPEND, "RSAKEY", "%02X", rdr->rsa_mod[i]); }
// 3DES Key
for(i = 0; i < rdr->des_key_length; i++)
{ tpl_printf(vars, TPLAPPEND, "DESKEY", "%02X", rdr->des_key[i]); }
// BoxKey
for(i = 0; i < rdr->boxkey_length ; i++)
{ tpl_printf(vars, TPLAPPEND, "BOXKEY", "%02X", rdr->boxkey[i]); }
#endif
#ifdef READER_CONAX
for(i = 0; i < rdr->cwpk_mod_length; i++)
{ tpl_printf(vars, TPLAPPEND, "CWPKKEY", "%02X", rdr->cwpk_mod[i]); }
#endif
#ifdef READER_NAGRA
// nuid (CAK6.3)
for(i = 0; i < rdr->cak63nuid_length; i++)
{ tpl_printf(vars, TPLAPPEND, "CAK63NUID", "%02X", rdr->cak63nuid[i]); }
// cwekey (CAK6.3)
for(i = 0; i < rdr->cak63cwekey_length; i++)
{ tpl_printf(vars, TPLAPPEND, "CAK63CWEKEY", "%02X", rdr->cak63cwekey[i]); }
#endif
#ifdef READER_NAGRA_MERLIN
int32_t j;
// idird (CAK7)
for(i = 0; i < rdr->idird_length; i++)
{ tpl_printf(vars, TPLAPPEND, "IDIRD", "%02X", rdr->idird[i]); }
// cmd0e_provider (CAK7)
for(i = 0; i < rdr->cmd0eprov_length; i++)
{ tpl_printf(vars, TPLAPPEND, "CMD0EPROV", "%02X", rdr->cmd0eprov[i]); }
// mod1 (CAK7)
for(i = 0; i < rdr->mod1_length ; i++)
{ tpl_printf(vars, TPLAPPEND, "MOD1", "%02X", rdr->mod1[i]); }
// mod2 (CAK7)
for(i = 0; i < rdr->mod2_length ; i++)
{ tpl_printf(vars, TPLAPPEND, "MOD2", "%02X", rdr->mod2[i]); }
// key3588 (CAK7)
for(i = 0; i < rdr->key3588_length; i++)
{ tpl_printf(vars, TPLAPPEND, "KEY3588", "%02X", rdr->key3588[i]); }
// key3310 (CAK7)
for(i = 0; i < rdr->key3310_length; i++)
{ tpl_printf(vars, TPLAPPEND, "KEY3310", "%02X", rdr->key3310[i]); }
// key3460 (CAK7)
for(i = 0; i < rdr->key3460_length; i++)
{ tpl_printf(vars, TPLAPPEND, "KEY3460", "%02X", rdr->key3460[i]); }
// data50 (CAK7)
for(i = 0; i < rdr->data50_length; i++)
{ tpl_printf(vars, TPLAPPEND, "DATA50", "%02X", rdr->data50[i]); }
// mod50 (CAK7)
for(i = 0; i < rdr->mod50_length; i++)
{ tpl_printf(vars, TPLAPPEND, "MOD50", "%02X", rdr->mod50[i]); }
// nuid (CAK7)
for(i = 0; i < rdr->nuid_length; i++)
{ tpl_printf(vars, TPLAPPEND, "NUID", "%02X", rdr->nuid[i]); }
// OTP CSC (CAK7)
for(i = 0; i < rdr->otpcsc_length; i++)
{ tpl_printf(vars, TPLAPPEND, "OTPCSC", "%02X", rdr->otpcsc[i]); }
// OTA CSC (CAK7)
for(i = 0; i < rdr->otacsc_length; i++)
{ tpl_printf(vars, TPLAPPEND, "OTACSC", "%02X", rdr->otacsc[i]); }
// Force Pairing Type (CAK7)
for(i = 0; i < rdr->forcepair_length; i++)
{ tpl_printf(vars, TPLAPPEND, "FORCEPAIR", "%02X", rdr->forcepair[i]); }
// cwekeys (CAK7)
for(j = 0; j < 17; j++)
{
char key[9] = "CWEKEY";
if(j > 9)
{
key[6] = '1';
key[7] = '0' + (j - 10);
}
else
{
key[6] = '0' + j;
}
for(i = 0; i < rdr->cwekey_length[j]; i++)
{ tpl_printf(vars, TPLAPPEND, key, "%02X", rdr->cwekey[j][i]); }
}
// force_cw_swap
if(rdr->forcecwswap)
{ tpl_addVar(vars, TPLADD, "FORCECWSWAPCHECKED", "checked"); }
// only_even_SA
if(rdr->evensa)
{ tpl_addVar(vars, TPLADD, "EVENSACHECKED", "checked"); }
// force_EMM_82
if(rdr->forceemmg)
{ tpl_addVar(vars, TPLADD, "FORCEEMMGCHECKED", "checked"); }
// OTA_CWPKs
if(rdr->cwpkota)
{ tpl_addVar(vars, TPLADD, "CWPKOTACHECKED", "checked"); }
tpl_printf(vars, TPLADD, "TMP", "NAGRACAK7HEADERMODE%d", rdr->headermode);
tpl_addVar(vars, TPLADD, tpl_getVar(vars, "TMP"), "selected");
// CWPK CaID (CAK7)
for(i = 0; i < rdr->cwpkcaid_length ; i++)
{ tpl_printf(vars, TPLAPPEND, "CWPKCAID", "%02X", rdr->cwpkcaid[i]); }
// cak7_mode
if(rdr->cak7_mode)
{ tpl_addVar(vars, TPLADD, "NAGRACAK7MODECHECKED", "checked"); }
#endif
#ifdef READER_VIDEOGUARD
tpl_printf(vars, TPLADD, "TMP", "CARDSTARTDATEBASEMONTH%d", rdr->card_startdate_basemonth);
tpl_addVar(vars, TPLADD, tpl_getVar(vars, "TMP"), "selected");
tpl_printf(vars, TPLADD, "CARDSTARTDATEBASEYEAR", "%d", rdr->card_startdate_baseyear);
tpl_printf(vars, TPLADD, "TMP", "CARDEXPIREDATEBASEMONTH%d", rdr->card_expiredate_basemonth);
tpl_addVar(vars, TPLADD, tpl_getVar(vars, "TMP"), "selected");
tpl_printf(vars, TPLADD, "CARDEXPIREDATEBASEYEAR", "%d", rdr->card_expiredate_baseyear);
// ins7E
if(rdr->ins7E[0x1A])
{
for(i = 0; i < 26 ; i++) { tpl_printf(vars, TPLAPPEND, "INS7E", "%02X", rdr->ins7E[i]); }
}
// ins42
if(rdr->ins42[0x25])
{
for(i = 0; i < 37 ; i++) { tpl_printf(vars, TPLAPPEND, "INS42", "%02X", rdr->ins42[i]); }
}
// ins7E11
if(rdr->ins7E11[0x01])
{
tpl_printf(vars, TPLAPPEND, "INS7E11", "%02X", rdr->ins7E11[0]);
}
// ins2e06
if(rdr->ins2e06[0x04])
{
for(i = 0; i < 4 ; i++) { tpl_printf(vars, TPLAPPEND, "INS2E06", "%02X", rdr->ins2e06[i]); }
}
// k1 for generic pairing mode
if(rdr->k1_generic[0x10])
{
for(i = 0; i < rdr->k1_generic[0x10] ; i++) { tpl_printf(vars, TPLAPPEND, "K1_GENERIC", "%02X", rdr->k1_generic[i]); }
}
// k1 for unique pairing mode
if(rdr->k1_unique[0x10])
{
for(i = 0; i < rdr->k1_unique[0x10] ; i++) { tpl_printf(vars, TPLAPPEND, "K1_UNIQUE", "%02X", rdr->k1_unique[i]); }
}
#endif
#ifdef WITH_CARDREADER
// ATR
if(rdr->atr[0])
for(i = 0; i < rdr->atrlen / 2; i++)
{ tpl_printf(vars, TPLAPPEND, "ATR", "%02X", rdr->atr[i]); }
#endif
// ECM Whitelist
value = mk_t_ecm_whitelist(&rdr->ecm_whitelist);
tpl_addVar(vars, TPLADD, "ECMWHITELIST", value);
free_mk_t(value);
// ECM Header Whitelist
value = mk_t_ecm_hdr_whitelist(&rdr->ecm_hdr_whitelist);
tpl_addVar(vars, TPLADD, "ECMHEADERWHITELIST", value);
free_mk_t(value);
#ifdef WITH_CARDREADER
// Deprecated
if(!apicall)
{
tpl_addVar(vars, TPLADD, "DEPRECATEDCHECKED", (rdr->deprecated == 1) ? "checked" : "");
}
else
{
tpl_addVar(vars, TPLADD, "DEPRECATEDVALUE", (rdr->deprecated == 1) ? "1" : "0");
}
// Smargopatch
if(!apicall)
{
tpl_addVar(vars, TPLADD, "SMARGOPATCHCHECKED", (rdr->smargopatch == 1) ? "checked" : "");
}
else
{
tpl_addVar(vars, TPLADD, "SMARGOPATCHVALUE", (rdr->smargopatch == 1) ? "1" : "0");
}
// Autospeed
if(!apicall)
{
tpl_addVar(vars, TPLADD, "AUTOSPEEDCHECKED", (rdr->autospeed == 1) ? "checked" : "");
}
else
{
tpl_addVar(vars, TPLADD, "AUTOSPEEDVALUE", (rdr->autospeed == 1) ? "1" : "0");
}
// sc8in1 dtrrts patch
if(!apicall)
{
tpl_addVar(vars, TPLADD, "SC8IN1DTRRTSPATCHCHECKED", (rdr->sc8in1_dtrrts_patch == 1) ? "checked" : "");
}
else
{
tpl_addVar(vars, TPLADD, "SC8IN1DTRRTSPATCHVALUE", (rdr->sc8in1_dtrrts_patch == 1) ? "1" : "0");
}
#ifdef READER_VIACCESS
if(!apicall)
{
tpl_addVar(vars, TPLADD, "READOLDCLASSES", (rdr->read_old_classes == 1) ? "checked" : "");
}
else
{
tpl_addVar(vars, TPLADD, "READOLDCLASSES", (rdr->read_old_classes == 1) ? "1" : "0");
}
#endif
// Detect
if(rdr->detect & 0x80)
{ tpl_printf(vars, TPLADD, "DETECT", "!%s", RDR_CD_TXT[rdr->detect & 0x7f]); }
else
{ tpl_addVar(vars, TPLADD, "DETECT", RDR_CD_TXT[rdr->detect & 0x7f]); }
#endif
// Ratelimit
if(rdr->ratelimitecm)
{
tpl_printf(vars, TPLADD, "RATELIMITECM", "%d", rdr->ratelimitecm);
// ECMUNIQUE
if(!apicall)
{
tpl_addVar(vars, TPLADD, "ECMUNIQUECHECKED", (rdr->ecmunique == 1) ? "checked" : "");
}
else
{
tpl_addVar(vars, TPLADD, "ECMUNIQUE", (rdr->ecmunique == 1) ? "1" : "0");
}
tpl_printf(vars, TPLADD, "RATELIMITTIME", "%d", rdr->ratelimittime);
tpl_printf(vars, TPLADD, "SRVIDHOLDTIME", "%d", rdr->srvidholdtime);
}
// Cooldown
if(rdr->cooldown[0] && rdr->cooldown[1])
{
tpl_printf(vars, TPLADD, "COOLDOWNDELAY", "%d", rdr->cooldown[0]);
tpl_printf(vars, TPLADD, "COOLDOWNTIME", "%d", rdr->cooldown[1]);
}
// Max parallel services
tpl_printf(vars, TPLADD, "MAXPARALLEL", "%d", rdr->maxparallel);
// Parallelfactor: format as float with dot (comma would break URL parameters)
tpl_printf(vars, TPLADD, "PARALLELFACTOR", "%.1f", rdr->parallelfactor >= 0 ? rdr->parallelfactor : 2.0);
tpl_printf(vars, TPLADD, "PARALLELTIMEOUT", "%d", rdr->paralleltimeout);
#ifdef WITH_CARDREADER
// Frequencies
tpl_printf(vars, TPLADD, "MHZ", "%d", rdr->mhz);
tpl_printf(vars, TPLADD, "CARDMHZ", "%d", rdr->cardmhz);
#endif
// Device
if(!apicall)
{
tpl_addVar(vars, TPLADD, "DEVICE", xml_encode(vars, rdr->device));
}
else
{
tpl_addVar(vars, TPLADD, "DEVICE", rdr->device);
}
if(rdr->r_port)
{ tpl_printf(vars, TPLAPPEND, "DEVICE", ",%d", rdr->r_port); }
if(rdr->l_port)
{
if(rdr->r_port)
{ tpl_printf(vars, TPLAPPEND, "DEVICE", ",%d", rdr->l_port); }
else
{ tpl_printf(vars, TPLAPPEND, "DEVICE", ",,%d", rdr->l_port); }
}
// Group
value = mk_t_group(rdr->grp);
tpl_addVar(vars, TPLADD, "GRP", value);
free_mk_t(value);
#ifdef WITH_LB
if(rdr->lb_weight)
{ tpl_printf(vars, TPLADD, "LBWEIGHT", "%d", rdr->lb_weight); }
#endif
//services
if(!apicall)
{
struct s_sidtab *sidtab = cfg.sidtab;
//build matrix
i = 0;
while(sidtab != NULL)
{
tpl_addVar(vars, TPLADD, "SIDLABEL", xml_encode(vars, sidtab->label));
if(rdr->sidtabs.ok & ((SIDTABBITS)1 << i)) { tpl_addVar(vars, TPLADD, "CHECKED", "checked"); }
else { tpl_addVar(vars, TPLADD, "CHECKED", ""); }
tpl_addVar(vars, TPLAPPEND, "SIDS", tpl_getTpl(vars, "READERCONFIGSIDOKBIT"));
if(rdr->sidtabs.no & ((SIDTABBITS)1 << i)) { tpl_addVar(vars, TPLADD, "CHECKED", "checked"); }
else { tpl_addVar(vars, TPLADD, "CHECKED", ""); }
tpl_addVar(vars, TPLAPPEND, "SIDS", tpl_getTpl(vars, "READERCONFIGSIDNOBIT"));
if(rdr->lb_sidtabs.ok & ((SIDTABBITS)1 << i)) { tpl_addVar(vars, TPLADD, "CHECKED", "checked"); }
else { tpl_addVar(vars, TPLADD, "CHECKED", ""); }
tpl_addVar(vars, TPLAPPEND, "SIDS", tpl_getTpl(vars, "READERCONFIGSIDLBOKBIT"));
sidtab = sidtab->next;
i++;
}
if(i){
tpl_addVar(vars, TPLADD, "READERCONFIGSIDINS", tpl_getTpl(vars, "READERCONFIGSID"));
}
}
else
{
value = mk_t_service(&rdr->sidtabs);
if(cs_strlen(value) > 0)
{ tpl_addVar(vars, TPLADD, "SERVICES", value); }
free_mk_t(value);
}
// CAID
value = mk_t_caidtab(&rdr->ctab);
tpl_addVar(vars, TPLADD, "CAIDS", value);
free_mk_t(value);
#ifdef READER_VIACCESS
// AESkeys
value = mk_t_aeskeys(rdr);
tpl_addVar(vars, TPLADD, "AESKEYS", value);
free_mk_t(value);
#endif
//ident
value = mk_t_ftab(&rdr->ftab);
tpl_addVar(vars, TPLADD, "IDENTS", value);
free_mk_t(value);
//CHID
value = mk_t_ftab(&rdr->fchid);
tpl_addVar(vars, TPLADD, "CHIDS", value);
free_mk_t(value);
//Local cards
value = mk_t_ftab(&rdr->localcards);
tpl_addVar(vars, TPLADD, "LOCALCARDS", value);
free_mk_t(value);
//class
value = mk_t_cltab(&rdr->cltab);
tpl_addVar(vars, TPLADD, "CLASS", value);
free_mk_t(value);
if(rdr->cachemm || rdr->logemm)
{ tpl_printf(vars, TPLADD, "EMMCACHE", "%d,%d,%d,%d", rdr->cachemm, rdr->rewritemm, rdr->logemm, rdr->deviceemm); }
//savenano
value = mk_t_nano(rdr->s_nano);
tpl_addVar(vars, TPLADD, "SAVENANO", value);
free_mk_t(value);
//blocknano
value = mk_t_nano(rdr->b_nano);
tpl_addVar(vars, TPLADD, "BLOCKNANO", value);
free_mk_t(value);
// Blocke EMM
if(!apicall)
{
tpl_addVar(vars, TPLADD, "BLOCKEMMUNKNOWNCHK", (rdr->blockemm & EMM_UNKNOWN) ? "checked" : "");
tpl_addVar(vars, TPLADD, "BLOCKEMMUNIQCHK", (rdr->blockemm & EMM_UNIQUE) ? "checked" : "");
tpl_addVar(vars, TPLADD, "BLOCKEMMSHAREDCHK", (rdr->blockemm & EMM_SHARED) ? "checked" : "");
tpl_addVar(vars, TPLADD, "BLOCKEMMGLOBALCHK", (rdr->blockemm & EMM_GLOBAL) ? "checked" : "");
}
else
{
tpl_addVar(vars, TPLADD, "BLOCKEMMUNKNOWNVALUE", (rdr->blockemm & EMM_UNKNOWN) ? "1" : "0");
tpl_addVar(vars, TPLADD, "BLOCKEMMUNIQVALUE", (rdr->blockemm & EMM_UNIQUE) ? "1" : "0");
tpl_addVar(vars, TPLADD, "BLOCKEMMSHAREDVALUE", (rdr->blockemm & EMM_SHARED) ? "1" : "0");
tpl_addVar(vars, TPLADD, "BLOCKEMMGLOBALVALUE", (rdr->blockemm & EMM_GLOBAL) ? "1" : "0");
}
// Save EMM
if(!apicall)
{
tpl_addVar(vars, TPLADD, "SAVEEMMUNKNOWNCHK", (rdr->saveemm & EMM_UNKNOWN) ? "checked" : "");
tpl_addVar(vars, TPLADD, "SAVEEMMUNIQCHK", (rdr->saveemm & EMM_UNIQUE) ? "checked" : "");
tpl_addVar(vars, TPLADD, "SAVEEMMSHAREDCHK", (rdr->saveemm & EMM_SHARED) ? "checked" : "");
tpl_addVar(vars, TPLADD, "SAVEEMMGLOBALCHK", (rdr->saveemm & EMM_GLOBAL) ? "checked" : "");
}
else
{
tpl_addVar(vars, TPLADD, "SAVEEMMUNKNOWNVALUE", (rdr->saveemm & EMM_UNKNOWN) ? "1" : "0");
tpl_addVar(vars, TPLADD, "SAVEEMMUNIQVALUE", (rdr->saveemm & EMM_UNIQUE) ? "1" : "0");
tpl_addVar(vars, TPLADD, "SAVEEMMSHAREDVALUE", (rdr->saveemm & EMM_SHARED) ? "1" : "0");
tpl_addVar(vars, TPLADD, "SAVEEMMGLOBALVALUE", (rdr->saveemm & EMM_GLOBAL) ? "1" : "0");
}
value = mk_t_emmbylen(rdr);
if(cs_strlen(value) > 0)
{ tpl_addVar(vars, TPLADD, "BLOCKEMMBYLEN", value); }
free_mk_t(value);
#ifdef MODULE_CCCAM
if(!strcmp(rdr->cc_version, "2.0.11"))
{
tpl_addVar(vars, TPLADD, "CCCVERSIONSELECTED0", "selected");
}
else if(!strcmp(rdr->cc_version, "2.1.1"))
{
tpl_addVar(vars, TPLADD, "CCCVERSIONSELECTED1", "selected");
}
else if(!strcmp(rdr->cc_version, "2.1.2"))
{
tpl_addVar(vars, TPLADD, "CCCVERSIONSELECTED2", "selected");
}
else if(!strcmp(rdr->cc_version, "2.1.3"))
{
tpl_addVar(vars, TPLADD, "CCCVERSIONSELECTED3", "selected");
}
else if(!strcmp(rdr->cc_version, "2.1.4"))
{
tpl_addVar(vars, TPLADD, "CCCVERSIONSELECTED4", "selected");
}
else if(!strcmp(rdr->cc_version, "2.2.0"))
{
tpl_addVar(vars, TPLADD, "CCCVERSIONSELECTED5", "selected");
}
else if(!strcmp(rdr->cc_version, "2.2.1"))
{
tpl_addVar(vars, TPLADD, "CCCVERSIONSELECTED6", "selected");
}
else if(!strcmp(rdr->cc_version, "2.3.0"))
{
tpl_addVar(vars, TPLADD, "CCCVERSIONSELECTED7", "selected");
}
else if(!strcmp(rdr->cc_version, "2.3.1"))
{
tpl_addVar(vars, TPLADD, "CCCVERSIONSELECTED8", "selected");
}
else if(!strcmp(rdr->cc_version, "2.3.2"))
{
tpl_addVar(vars, TPLADD, "CCCVERSIONSELECTED9", "selected");
}
#endif
#ifdef READER_VIDEOGUARD
tpl_printf(vars, TPLADD, "TMP", "NDSVERSION%d", rdr->ndsversion);
tpl_addVar(vars, TPLADD, tpl_getVar(vars, "TMP"), "selected");
tpl_printf(vars, TPLADD, "TMP", "NDSREADTIERS%d", rdr->readtiers);
tpl_addVar(vars, TPLADD, tpl_getVar(vars, "TMP"), "selected");
#endif
#ifdef READER_NAGRA
tpl_printf(vars, TPLADD, "TMP", "NAGRAREAD%d", rdr->nagra_read);
tpl_addVar(vars, TPLADD, tpl_getVar(vars, "TMP"), "selected");
if(rdr->detect_seca_nagra_tunneled_card)
{ tpl_addVar(vars, TPLADD, "NAGRADETECTSECACARDCHECKED", "checked"); }
#endif
#ifdef READER_TONGFANG
if(rdr->tongfang3_calibsn)
{ tpl_printf(vars, TPLADD, "TONGFANGCALIBSN", "%08X", rdr->tongfang3_calibsn); }
if(rdr->tongfang_boxid)
{ tpl_printf(vars, TPLADD, "TONGFANGBOXID", "%08X", rdr->tongfang_boxid); }
if(rdr->tongfang3_deskey_length > 0)
{
for(i = 0; i < rdr->tongfang3_deskey_length ; i++)
{
tpl_printf(vars, TPLAPPEND, "TONGFANGDESKEY", "%02X", rdr->tongfang3_deskey[i]);
}
}
if(rdr->stbid_length > 0)
{
for(i = 0; i < rdr->stbid_length ; i++)
{
tpl_printf(vars, TPLAPPEND, "STBID", "%02X", rdr->stbid[i]);
}
}
#endif
#ifdef MODULE_CCCAM
tpl_printf(vars, TPLADD, "CCCMAXHOPS", "%d", rdr->cc_maxhops);
tpl_printf(vars, TPLADD, "CCCMINDOWN", "%d", rdr->cc_mindown);
tpl_printf(vars, TPLADD, "CCCRESHARE", "%d", rdr->cc_reshare);
tpl_printf(vars, TPLADD, "RESHARE", "%d", cfg.cc_reshare);
tpl_printf(vars, TPLADD, "CCCRECONNECT", "%d", rdr->cc_reconnect);
if(rdr->cc_want_emu)
{ tpl_addVar(vars, TPLADD, "CCCWANTEMUCHECKED", "checked"); }
if(rdr->cc_keepalive)
{ tpl_addVar(vars, TPLADD, "KEEPALIVECHECKED", "checked"); }
#endif
#ifdef MODULE_GBOX
tpl_printf(vars, TPLADD, "GBOXMAXDISTANCE", "%d", rdr->gbox_maxdist);
tpl_printf(vars, TPLADD, "GBOXMAXECMSEND", "%d", rdr->gbox_maxecmsend);
tpl_printf(vars, TPLADD, "GBOXRESHARE", "%d", rdr->gbox_reshare);
tpl_printf(vars, TPLADD, "PEERGBOXID", "%04X", gbox_convert_password_to_id((uint32_t)a2i(rdr->r_pwd, 4)));
tpl_addVar(vars, TPLADD, "PEERONLSTAT", (get_peer_onl_status(gbox_convert_password_to_id((uint32_t)a2i(rdr->r_pwd, 4)))) ? "checked" : "");
tpl_printf(vars, TPLADD, "FORCEREMM", "%d", rdr->gbox_force_remm);
tpl_printf(vars, TPLADD, "CMDHERE", rdr->send_offline_cmd ? "checked" : "");
if(rdr->blockemm & 0x80)
{
tpl_addVar(vars, TPLADD, "REMMCNLDCHK", "checked");
tpl_printf(vars, TPLADD, "GBOXREMMPEER", "%04X", rdr->gbox_remm_peer);
tpl_addVar(vars, TPLADD, "REMMUNIQCHK", (rdr->blockemm & EMM_UNIQUE) ? "" : "checked");
tpl_addVar(vars, TPLADD, "REMMSHAREDCHK", (rdr->blockemm & EMM_SHARED) ? "" : "checked");
tpl_addVar(vars, TPLADD, "REMMGLOBALCHK", (rdr->blockemm & EMM_GLOBAL) ? "" : "checked");
tpl_addVar(vars, TPLADD, "REMMEMMUNKNOWNCHK", (rdr->blockemm & EMM_UNKNOWN) ? "" : "checked");
}
if(rdr->gbox_gsms_peer)
{
tpl_printf(vars, TPLADD, "LASTGSMS", "%s", rdr->last_gsms);
tpl_printf(vars, TPLADD, "GBOXGSMSPEER", "%04X", rdr->gbox_gsms_peer);
}
#endif
#ifdef READER_DRECAS
tpl_addVar(vars, TPLADD, "STMKEYS", rdr->stmkeys);
#endif
#if defined(READER_DRE) || defined(READER_DRECAS)
tpl_addVar(vars, TPLADD, "USERSCRIPT", rdr->userscript);
#endif
#ifdef WITH_EMU
//emu_auproviders
value = mk_t_ftab(&rdr->emu_auproviders);
tpl_addVar(vars, TPLADD, "EMUAUPROVIDERS", value);
free_mk_t(value);
// Date-coded BISS keys
if(!apicall)
{
tpl_addVar(vars, TPLADD, "EMUDATECODEDENABLED", (rdr->emu_datecodedenabled == 1) ? "checked" : "");
}
else
{
tpl_addVar(vars, TPLADD, "EMUDATECODEDENABLED", (rdr->emu_datecodedenabled == 1) ? "1" : "0");
}
#endif
tpl_addVar(vars, TPLADD, "PROTOCOL", reader_get_type_desc(rdr, 0));
// Show only parameters which needed for the reader
switch(rdr->typ)
{
case R_CONSTCW:
case R_DB2COM1:
case R_DB2COM2:
case R_MOUSE :
case R_DRECAS :
case R_MP35:
case R_SC8in1 :
case R_SMART :
case R_INTERNAL:
case R_SERIAL :
case R_PCSC :
tpl_addVar(vars, TPLAPPEND, "READERDEPENDINGCONFIG", tpl_getTpl(vars, "READERCONFIGSTDHWREADERBIT"));
break;
case R_CAMD35 :
tpl_addVar(vars, TPLAPPEND, "READERDEPENDINGCONFIG", tpl_getTpl(vars, "READERCONFIGCAMD35BIT"));
break;
case R_EMU :
tpl_addVar(vars, TPLAPPEND, "READERDEPENDINGCONFIG", tpl_getTpl(vars, "READERCONFIGEMUBIT"));
break;
case R_CS378X :
tpl_addVar(vars, TPLAPPEND, "READERDEPENDINGCONFIG", tpl_getTpl(vars, "READERCONFIGCS378XBIT"));
break;
case R_RADEGAST:
tpl_addVar(vars, TPLAPPEND, "READERDEPENDINGCONFIG", tpl_getTpl(vars, "READERCONFIGRADEGASTBIT"));
break;
case R_SCAM:
tpl_addVar(vars, TPLAPPEND, "READERDEPENDINGCONFIG", tpl_getTpl(vars, "READERCONFIGSCAMBIT"));
break;
case R_GHTTP:
tpl_addVar(vars, TPLAPPEND, "READERDEPENDINGCONFIG", tpl_getTpl(vars, "READERCONFIGGHTTPBIT"));
break;
case R_GBOX:
tpl_addVar(vars, TPLAPPEND, "READERDEPENDINGCONFIG", tpl_getTpl(vars, "READERCONFIGGBOXBIT"));
#if defined (MODULE_CCCAM) && defined (MODULE_GBOX)
if(cfg.cc_gbx_reshare_en)
{
tpl_printf(vars, TPLADD, "GBOXCCCAMRESHARE", "%d", rdr->gbox_cccam_reshare);
value = mk_t_ftab(&rdr->ccc_gbx_reshare_ident);
tpl_addVar(vars, TPLADD, "CCCGBXRESHAREIDENT", value);
free_mk_t(value);
tpl_addVar(vars, TPLAPPEND, "READERDEPENDINGCONFIG", tpl_getTpl(vars, "GBOXCCCAMRESHAREBIT"));
}
#endif
tpl_addVar(vars, TPLAPPEND, "READERDEPENDINGCONFIG", tpl_getTpl(vars, "READERINFOGBOXREMM"));
break;
case R_NEWCAMD:
if(rdr->ncd_proto == NCD_525)
{
tpl_addVar(vars, TPLAPPEND, "READERDEPENDINGCONFIG", tpl_getTpl(vars, "READERCONFIGNCD525BIT"));
}
else if(rdr->ncd_proto == NCD_524)
{
tpl_addVar(vars, TPLAPPEND, "READERDEPENDINGCONFIG", tpl_getTpl(vars, "READERCONFIGNCD524BIT"));
}
break;
#ifdef MODULE_CCCAM
case R_CCCAM :
tpl_addVar(vars, TPLAPPEND, "READERDEPENDINGCONFIG", tpl_getTpl(vars, "READERCONFIGCCCAMBIT"));
break;
#endif
default :
tpl_addMsg(vars, "Error: protocol not resolvable");
break;
}
#ifdef MODULE_CCCAM
if(rdr->typ != R_CCCAM && rdr->typ != R_GBOX)
{
tpl_printf(vars, TPLADD, "CCCHOP", "%d", rdr->cc_hop);
tpl_addVar(vars, TPLAPPEND, "READERDEPENDINGCONFIG", tpl_getTpl(vars, "READERCONFIGHOPBIT"));
}
#endif
#ifdef CS_CACHEEX_AIO
return tpl_getTpl(vars, "READERCONFIGAIO");
#else
return tpl_getTpl(vars, "READERCONFIG");
#endif
}
static char *send_oscam_reader_stats(struct templatevars *vars, struct uriparams *params, int32_t apicall)
{
if(!apicall) { setActiveMenu(vars, MNU_READERS); }
int8_t error;
struct s_client *cl = NULL;
struct s_reader *rdr;
rdr = get_reader_by_label(getParam(params, "label"));
error = (rdr ? 0 : 1);
if(!error && rdr)
{
cl = rdr->client;
error = (cl ? 0 : 1);
}
if(error)
{
tpl_addVar(vars, TPLAPPEND, "READERSTATSROW", tpl_getTpl(vars, "READERSTATSROWBIT"));
if(!apicall)
{ return tpl_getTpl(vars, "READERSTATS"); }
else
{ return tpl_getTpl(vars, "APIREADERSTATS"); }
}
#ifdef WITH_LB
char *stxt[] = {"found", "cache1", "cache2", "cache3",
"not found", "timeout", "sleeping",
"fake", "invalid", "corrupt", "no card", "expdate",
"disabled", "stopped"
};
if(strcmp(getParam(params, "action"), "resetstat") == 0)
{
char *rcs = getParam(params, "rc");
int32_t retval = 0;
if(cs_strlen(rcs) > 0)
{
int8_t rc;
rc = atoi(rcs);
retval = clean_stat_by_rc(rdr, rc, 0);
cs_log("Reader %s stats %d %s entr%s deleted by WebIF from %s",
rdr->label, retval, stxt[rc],
retval == 1 ? "y" : "ies",
cs_inet_ntoa(GET_IP()));
}
else
{
clear_reader_stat(rdr);
cs_log("Reader %s stats resetted by WebIF from %s", rdr->label, cs_inet_ntoa(GET_IP()));
}
}
if(strcmp(getParam(params, "action"), "deleterecord") == 0)
{
char *record = getParam(params, "record");
if(cs_strlen(record) > 0)
{
int32_t retval = 0;
uint32_t caid, provid, sid, cid, len;
sscanf(record, "%4x@%6x:%4x:%4x:%4x", &caid, &provid, &sid, &cid, &len);
retval = clean_stat_by_id(rdr, caid, provid, sid, cid, len);
cs_log("Reader %s stats %d entr%s deleted by WebIF from %s",
rdr->label, retval,
retval == 1 ? "y" : "ies",
cs_inet_ntoa(GET_IP()));
}
}
if(strcmp(getParam(params, "action"), "updateecmlen") == 0)
{
update_ecmlen_from_stat(rdr);
write_server();
}
#endif
tpl_addVar(vars, TPLADD, "READERNAME", xml_encode(vars, rdr->label));
tpl_addVar(vars, TPLADD, "LABEL", xml_encode(vars, rdr->label));
tpl_addVar(vars, TPLADD, "ENCODEDLABEL", urlencode(vars, rdr->label));
if(apicall)
{
int32_t i, emmcount = 0;
char *ttxt[] = {"unknown", "unique", "shared", "global"};
for(i = 0; i < 4; i++)
{
tpl_addVar(vars, TPLADD, "EMMRESULT", "error");
tpl_addVar(vars, TPLADD, "EMMTYPE", ttxt[i]);
tpl_printf(vars, TPLADD, "EMMCOUNT", "%d", rdr->emmerror[i]);
tpl_addVar(vars, TPLAPPEND, "EMMSTATS", tpl_getTpl(vars, "APIREADERSTATSEMMBIT"));
emmcount += rdr->emmerror[i];
tpl_printf(vars, TPLADD, "TOTALERROR", "%d", emmcount);
}
emmcount = 0;
for(i = 0; i < 4; i++)
{
tpl_addVar(vars, TPLADD, "EMMRESULT", "written");
tpl_addVar(vars, TPLADD, "EMMTYPE", ttxt[i]);
tpl_printf(vars, TPLADD, "EMMCOUNT", "%d", rdr->emmwritten[i]);
tpl_addVar(vars, TPLAPPEND, "EMMSTATS", tpl_getTpl(vars, "APIREADERSTATSEMMBIT"));
emmcount += rdr->emmwritten[i];
tpl_printf(vars, TPLADD, "TOTALWRITTEN", "%d", emmcount);
}
emmcount = 0;
for(i = 0; i < 4; i++)
{
tpl_addVar(vars, TPLADD, "EMMRESULT", "skipped");
tpl_addVar(vars, TPLADD, "EMMTYPE", ttxt[i]);
tpl_printf(vars, TPLADD, "EMMCOUNT", "%d", rdr->emmskipped[i]);
tpl_addVar(vars, TPLAPPEND, "EMMSTATS", tpl_getTpl(vars, "APIREADERSTATSEMMBIT"));
emmcount += rdr->emmskipped[i];
tpl_printf(vars, TPLADD, "TOTALSKIPPED", "%d", emmcount);
}
emmcount = 0;
for(i = 0; i < 4; i++)
{
tpl_addVar(vars, TPLADD, "EMMRESULT", "blocked");
tpl_addVar(vars, TPLADD, "EMMTYPE", ttxt[i]);
tpl_printf(vars, TPLADD, "EMMCOUNT", "%d", rdr->emmblocked[i]);
tpl_addVar(vars, TPLAPPEND, "EMMSTATS", tpl_getTpl(vars, "APIREADERSTATSEMMBIT"));
emmcount += rdr->emmblocked[i];
tpl_printf(vars, TPLADD, "TOTALBLOCKED", "%d", emmcount);
}
}
if(apicall)
{
char *txt = "UNDEF";
switch(rdr->card_status)
{
case NO_CARD:
if(rdr->typ == R_GBOX)
{ txt = "ONL no crd"; }
else
{ txt = "OFF"; }
break;
case UNKNOWN:
txt = "UNKNOWN";
break;
case READER_DEVICE_ERROR:
txt = "READER DEVICE ERROR";
break;
case CARD_NEED_INIT:
if(rdr->typ == R_GBOX)
{ txt = "OFFLINE"; }
else
{ txt = "NEEDINIT"; }
break;
case CARD_INSERTED:
if(cl->typ == 'p')
{
if(rdr->typ == R_GBOX)
{ txt = "ONL w/crd"; }
else
{ txt = "CONNECTED"; }
}
else
{ txt = "CARDOK"; }
break;
case CARD_FAILURE:
txt = "ERROR";
break;
default:
txt = "UNDEF";
}
tpl_addVar(vars, TPLADD, "READERSTATUS", txt);
tpl_printf(vars, TPLADD, "READERCAID", "%04X", rdr->caid);
}
int32_t rowcount = 0;
uint64_t ecmcount = 0;
time_t lastaccess = 0;
#ifdef WITH_LB
int32_t rc2hide = (-1);
if(cs_strlen(getParam(params, "hide")) > 0)
{ rc2hide = atoi(getParam(params, "hide")); }
int32_t rc2show = (-1);
if(cs_strlen(getParam(params, "show")) > 0)
{ rc2show = atoi(getParam(params, "show")); }
if(rdr->lb_stat)
{
int32_t statsize;
// @todo alno: sort by click, 0=ascending, 1=descending (maybe two buttons or reverse on second click)
READER_STAT **statarray = get_sorted_stat_copy(rdr, 0, &statsize);
char channame[CS_SERVICENAME_SIZE];
for(; rowcount < statsize; ++rowcount)
{
READER_STAT *s = statarray[rowcount];
if(!(s->rc == rc2hide) && ((rc2show == -1) || (s->rc == rc2show)))
{
struct tm lt;
localtime_r(&s->last_received.time, &lt); // fixme we need walltime!
ecmcount += s->ecm_count;
if(!apicall)
{
tpl_printf(vars, TPLADD, "CHANNEL", "%04X@%06X:%04X:%04X", s->caid, s->prid, s->srvid, s->chid);
tpl_addVar(vars, TPLADD, "CHANNELNAME", xml_encode(vars, get_servicename(cur_client(), s->srvid, s->prid, s->caid, channame, sizeof(channame))));
tpl_printf(vars, TPLADD, "ECMLEN", "%04hX", s->ecmlen);
tpl_addVar(vars, TPLADD, "RC", stxt[s->rc]);
tpl_printf(vars, TPLADD, "TIME", PRINTF_LOCAL_D " ms", s->time_avg);
if(s->time_stat[s->time_idx])
{ tpl_printf(vars, TPLADD, "TIMELAST", PRINTF_LOCAL_D " ms", s->time_stat[s->time_idx]); }
else
{ tpl_addVar(vars, TPLADD, "TIMELAST", ""); }
tpl_printf(vars, TPLADD, "COUNT", PRINTF_LOCAL_D, s->ecm_count);
if(s->last_received.time)
{
tpl_printf(vars, TPLADD, "LAST", "%02d.%02d.%02d %02d:%02d:%02d", lt.tm_mday, lt.tm_mon + 1, lt.tm_year % 100, lt.tm_hour, lt.tm_min, lt.tm_sec);
}
else
{
tpl_addVar(vars, TPLADD, "LAST", "never");
}
}
else
{
tpl_printf(vars, TPLADD, "ECMCAID", "%04X", s->caid);
tpl_printf(vars, TPLADD, "ECMPROVID", "%06X", s->prid);
tpl_printf(vars, TPLADD, "ECMSRVID", "%04X", s->srvid);
tpl_printf(vars, TPLADD, "ECMLEN", "%04hX", s->ecmlen);
tpl_addVar(vars, TPLADD, "ECMCHANNELNAME", xml_encode(vars, get_servicename(cur_client(), s->srvid, s->prid, s->caid, channame, sizeof(channame))));
tpl_printf(vars, TPLADD, "ECMTIME", PRINTF_LOCAL_D, s->time_avg);
tpl_printf(vars, TPLADD, "ECMTIMELAST", PRINTF_LOCAL_D, s->time_stat[s->time_idx]);
tpl_printf(vars, TPLADD, "ECMRC", "%d", s->rc);
tpl_addVar(vars, TPLADD, "ECMRCS", stxt[s->rc]);
if(s->last_received.time)
{
char tbuffer [30];
strftime(tbuffer, 30, "%Y-%m-%dT%H:%M:%S%z", &lt);
tpl_addVar(vars, TPLADD, "ECMLAST", tbuffer);
}
else
{
tpl_addVar(vars, TPLADD, "ECMLAST", "");
}
tpl_printf(vars, TPLADD, "ECMCOUNT", PRINTF_LOCAL_D, s->ecm_count);
if(s->last_received.time > lastaccess)
{ lastaccess = s->last_received.time; }
}
if(!apicall)
{
if(s->rc == E_NOTFOUND)
{
tpl_addVar(vars, TPLAPPEND, "READERSTATSROWNOTFOUND", tpl_getTpl(vars, "READERSTATSBIT"));
tpl_addVar(vars, TPLADD, "RESETA", urlencode(vars, rdr->label));
tpl_addVar(vars, TPLADD, "READERSTATSNFHEADLINE", tpl_getTpl(vars, "READERSTATSROWNOTFOUNDBIT"));
}
else if(s->rc == E_TIMEOUT)
{
tpl_addVar(vars, TPLAPPEND, "READERSTATSROWTIMEOUT", tpl_getTpl(vars, "READERSTATSBIT"));
tpl_addVar(vars, TPLADD, "RESETB", urlencode(vars, rdr->label));
tpl_addVar(vars, TPLADD, "READERSTATSTOHEADLINE", tpl_getTpl(vars, "READERSTATSROWTIMEOUTBIT"));
}
else if(s->rc == E_INVALID)
{
tpl_addVar(vars, TPLAPPEND, "READERSTATSROWINVALID", tpl_getTpl(vars, "READERSTATSBIT"));
tpl_addVar(vars, TPLADD, "RESETC", urlencode(vars, rdr->label));
tpl_addVar(vars, TPLADD, "READERSTATSIVHEADLINE", tpl_getTpl(vars, "READERSTATSROWINVALIDBIT"));
}
else
{ tpl_addVar(vars, TPLAPPEND, "READERSTATSROWFOUND", tpl_getTpl(vars, "READERSTATSBIT")); }
}
else
{
tpl_addVar(vars, TPLAPPEND, "ECMSTATS", tpl_getTpl(vars, "APIREADERSTATSECMBIT"));
}
}
}
NULLFREE(statarray);
}
else
#endif
tpl_addVar(vars, TPLAPPEND, "READERSTATSROW", tpl_getTpl(vars, "READERSTATSNOSTATS"));
tpl_printf(vars, TPLADD, "ROWCOUNT", "%d", rowcount);
if(lastaccess > 0)
{
char tbuffer [30];
struct tm lt;
localtime_r(&lastaccess, &lt);
strftime(tbuffer, 30, "%Y-%m-%dT%H:%M:%S%z", &lt);
tpl_addVar(vars, TPLADD, "LASTACCESS", tbuffer);
}
else
{
tpl_addVar(vars, TPLADD, "LASTACCESS", "");
}
if(apicall)
{
if(cl)
{
char *value = get_ecm_historystring(cl);
tpl_addVar(vars, TPLADD, "ECMHISTORY", value);
free_mk_t(value);
}
}
tpl_printf(vars, TPLADD, "TOTALECM", "%'" PRIu64, ecmcount);
if(!apicall)
{ return tpl_getTpl(vars, "READERSTATS"); }
else
{ return tpl_getTpl(vars, "APIREADERSTATS"); }
}
static char *send_oscam_user_config_edit(struct templatevars *vars, struct uriparams *params, int32_t apicall)
{
struct s_auth *account, *ptr, *chk;
char user[sizeof(first_client->account->usr)];
int32_t i;
int existing_insert = 0;
if(!apicall) { setActiveMenu(vars, MNU_USERS); }
if(strcmp(getParam(params, "action"), "Save As") == 0) { cs_strncpy(user, getParam(params, "newuser"), sizeof(user)); }
else { cs_strncpy(user, getParam(params, "user"), sizeof(user)); }
account = NULL;
for(chk = cfg.account; chk != NULL; chk = chk->next)
{
if(strcmp(user, chk->usr) == 0)
{ account = chk; }
if(!existing_insert)
{
tpl_printf(vars, TPLADD, "EXISTING_INS", "'%s'", urlencode(vars, chk->usr));
existing_insert++;
}
else
{
tpl_printf(vars, TPLAPPEND, "EXISTING_INS", ",'%s'", urlencode(vars, chk->usr));
}
}
// Create a new user if it doesn't yet
if(account == NULL)
{
i = 1;
while(cs_strlen(user) < 1)
{
snprintf(user, sizeof(user) / sizeof(char) - 1, "NEWUSER%d", i);
for(account = cfg.account; account != NULL && strcmp(user, account->usr) != 0; account = account->next) { ; }
if(account != NULL) { user[0] = '\0'; }
++i;
}
if(!cs_malloc(&account, sizeof(struct s_auth))) { return "0"; }
if(cfg.account == NULL) { cfg.account = account; }
else
{
for(ptr = cfg.account; ptr != NULL && ptr->next != NULL; ptr = ptr->next) { ; }
ptr->next = account;
}
account_set_defaults(account);
account->disabled = 1;
cs_strncpy((char *)account->usr, user, sizeof(account->usr));
if(!account->grp)
{ account->grp = 1; }
if(write_userdb() != 0) { tpl_addMsg(vars, "Write Config failed!"); }
else if(strcmp(getParam(params, "action"), "Save As") == 0) { tpl_addMsg(vars, "New user has been added with cloned settings"); }
else { tpl_addMsg(vars, "New user has been added with default settings"); }
// no need to refresh anything here as the account is disabled by default and there's no client with this new account anyway!
}
if((strcmp(getParam(params, "action"), "Save") == 0) || (strcmp(getParam(params, "action"), "Save As") == 0))
{
char servicelabels[1024] = "";
for(i = 0; i < (*params).paramcount; i++)
{
if((strcmp((*params).params[i], "action")) &&
(strcmp((*params).params[i], "user")) &&
(strcmp((*params).params[i], "newuser")) &&
(strcmp((*params).params[i], "part")))
{
if(!strcmp((*params).params[i], "services"))
{ snprintf(servicelabels + cs_strlen(servicelabels), sizeof(servicelabels) - cs_strlen(servicelabels), "%s,", (*params).values[i]); }
else
{ chk_account((*params).params[i], (*params).values[i], account); }
}
}
chk_account("services", servicelabels, account);
refresh_oscam(REFR_CLIENTS);
if(write_userdb() != 0) { tpl_addMsg(vars, "Write Config failed!"); }
else if(strcmp(getParam(params, "action"), "Save As") != 0) { tpl_addMsg(vars, "User Account updated and saved"); }
}
tpl_addVar(vars, TPLADD, "USERNAME", xml_encode(vars, account->usr));
tpl_addVar(vars, TPLADD, "PASSWORD", xml_encode(vars, account->pwd));
tpl_addVar(vars, TPLADD, "DESCRIPTION", xml_encode(vars, account->description));
//Disabled
if(!apicall)
{
if(account->disabled)
{ tpl_addVar(vars, TPLADD, "DISABLEDCHECKED", "checked"); }
}
else
{
tpl_printf(vars, TPLADD, "DISABLEDVALUE", "%d", account->disabled);
}
//Expirationdate
struct tm timeinfo;
cs_gmtime_r(&account->expirationdate, &timeinfo);
char buf [80];
strftime(buf, 80, "%Y-%m-%d", &timeinfo);
if(strcmp(buf, "1970-01-01")) { tpl_addVar(vars, TPLADD, "EXPDATE", buf); }
//Allowed TimeFrame
char *allowedtf = mk_t_allowedtimeframe(account);
tpl_printf(vars, TPLADD, "ALLOWEDTIMEFRAME", "%s", allowedtf);
free_mk_t(allowedtf);
//Group
char *value = mk_t_group(account->grp);
tpl_addVar(vars, TPLADD, "GROUPS", value);
free_mk_t(value);
// allowed protocols
value = mk_t_allowedprotocols(account);
tpl_addVar(vars, TPLADD, "ALLOWEDPROTOCOLS", value);
free_mk_t(value);
//Hostname
tpl_addVar(vars, TPLADD, "DYNDNS", xml_encode(vars, account->dyndns));
//Uniq
if(!apicall)
{
tpl_printf(vars, TPLADD, "TMP", "UNIQSELECTED%d", account->uniq);
tpl_addVar(vars, TPLADD, tpl_getVar(vars, "TMP"), "selected");
}
else
{
tpl_printf(vars, TPLADD, "UNIQVALUE", "%d", account->uniq);
}
//Sleep
if(!account->tosleep) { tpl_addVar(vars, TPLADD, "SLEEP", "0"); }
else { tpl_printf(vars, TPLADD, "SLEEP", "%d", account->tosleep); }
//Monlevel selector
if(!apicall)
{
tpl_printf(vars, TPLADD, "TMP", "MONSELECTED%d", account->monlvl);
tpl_addVar(vars, TPLADD, tpl_getVar(vars, "TMP"), "selected");
}
else
{
tpl_printf(vars, TPLADD, "MONVALUE", "%d", account->monlvl);
}
//Au
if(account->autoau == 1)
{ tpl_addVar(vars, TPLADD, "AUREADER", "1"); }
else if(account->aureader_list)
{
value = mk_t_aureader(account);
tpl_addVar(vars, TPLADD, "AUREADER", value);
free_mk_t(value);
}
if(!apicall)
{
/* SERVICES */
struct s_sidtab *sidtab = cfg.sidtab;
//build matrix
i = 0;
while(sidtab != NULL)
{
tpl_addVar(vars, TPLADD, "SIDLABEL", xml_encode(vars, sidtab->label));
if(account->sidtabs.ok & ((SIDTABBITS)1 << i)) { tpl_addVar(vars, TPLADD, "CHECKED", "checked"); }
else { tpl_addVar(vars, TPLADD, "CHECKED", ""); }
tpl_addVar(vars, TPLAPPEND, "SIDS", tpl_getTpl(vars, "USEREDITSIDOKBIT"));
if(account->sidtabs.no & ((SIDTABBITS)1 << i)) { tpl_addVar(vars, TPLADD, "CHECKED", "checked"); }
else { tpl_addVar(vars, TPLADD, "CHECKED", ""); }
tpl_addVar(vars, TPLAPPEND, "SIDS", tpl_getTpl(vars, "USEREDITSIDNOBIT"));
sidtab = sidtab->next;
i++;
}
if(i){
tpl_addVar(vars, TPLADD, "USEREDITSIDINS", tpl_getTpl(vars, "USEREDITSID"));
}
}
else
{
value = mk_t_service(&account->sidtabs);
if(cs_strlen(value) > 0)
{ tpl_addVar(vars, TPLADD, "SERVICES", value); }
free_mk_t(value);
}
// CAID
value = mk_t_caidtab(&account->ctab);
tpl_addVar(vars, TPLADD, "CAIDS", value);
free_mk_t(value);
//ident
value = mk_t_ftab(&account->ftab);
tpl_addVar(vars, TPLADD, "IDENTS", value);
free_mk_t(value);
//CHID
value = mk_t_ftab(&account->fchid);
tpl_addVar(vars, TPLADD, "CHIDS", value);
free_mk_t(value);
//class
value = mk_t_cltab(&account->cltab);
tpl_addVar(vars, TPLADD, "CLASS", value);
free_mk_t(value);
//Betatunnel
value = mk_t_tuntab(&account->ttab);
tpl_addVar(vars, TPLADD, "BETATUNNELS", value);
free_mk_t(value);
//SUPPRESSCMD08
if(!apicall)
{
if(account->c35_suppresscmd08)
{ tpl_addVar(vars, TPLADD, "SUPPRESSCMD08", "selected"); }
}
else
{
tpl_printf(vars, TPLADD, "SUPPRESSCMD08VALUE", "%d", account->c35_suppresscmd08);
}
//Sleepsend
if(account->c35_sleepsend)
{
tpl_printf(vars, TPLADD, "SLEEPSEND", "selected");
}
//max_connections
tpl_printf(vars, TPLADD, "MAXCONNECTIONS", "%d", account->max_connections);
//User Max Idle
tpl_printf(vars, TPLADD, "UMAXIDLE", "%d", account->umaxidle);
//EMM Reassembly selector
if(!apicall)
{
tpl_printf(vars, TPLADD, "TMP", "EMMRSELECTED%d", account->emm_reassembly);
tpl_addVar(vars, TPLADD, tpl_getVar(vars, "TMP"), "selected");
}
else
{
tpl_printf(vars, TPLADD, "EMMRVALUE", "%d", account->emm_reassembly);
}
//Prefer local cards
if(!apicall)
{
tpl_printf(vars, TPLADD, "TMP", "PREFERLOCALCARDS%d", account->preferlocalcards);
tpl_addVar(vars, TPLADD, tpl_getVar(vars, "TMP"), "selected");
char *tmp = NULL;
switch(cfg.preferlocalcards)
{
case -1:
tmp = "-1 - Use Global prefer local cards";
break;
case 0:
tmp = "0 - local cards like proxied";
break;
case 1:
tmp = "1 - prefer cache-ex then local cards";
break;
case 2:
tmp = "2 - prefer local cards above cache-ex";
break;
}
tpl_addVar(vars, TPLADD, "CFGPREFERLOCALCARDS", tmp);
}
else
{
tpl_printf(vars, TPLADD, "PREFERLOCALCARDSVALUE", "%d", account->preferlocalcards);
}
#ifdef CS_CACHEEX
// Cacheex
if(!apicall)
{
tpl_printf(vars, TPLADD, "TMP", "CACHEEXSELECTED%d", account->cacheex.mode);
tpl_addVar(vars, TPLADD, tpl_getVar(vars, "TMP"), "selected");
}
else
{
tpl_printf(vars, TPLADD, "CACHEEX", "%d", account->cacheex.mode);
}
tpl_printf(vars, TPLADD, "CACHEEX_MAXHOP", "%d", account->cacheex.maxhop);
#ifdef CS_CACHEEX_AIO
tpl_printf(vars, TPLADD, "CACHEEX_MAXHOP_LG", "%d", account->cacheex.maxhop_lg);
#endif
value = mk_t_cacheex_hitvaluetab(&account->cacheex.filter_caidtab);
//if (cs_strlen(value) > 0)
tpl_printf(vars, TPLADD, "CACHEEX_ECM_FILTER", "%s", value);
free_mk_t(value);
tpl_addVar(vars, TPLADD, "DCCHECKED", (account->cacheex.drop_csp == 1) ? "checked" : "");
tpl_addVar(vars, TPLADD, "ARCHECKED", (account->cacheex.allow_request == 1) ? "checked" : "");
tpl_addVar(vars, TPLADD, "AFCHECKED", (account->cacheex.allow_filter == 1) ? "checked" : "");
#ifdef CS_CACHEEX_AIO
tpl_addVar(vars, TPLADD, "AMCHECKED", (account->cacheex.allow_maxhop == 1) ? "checked" : "");
#endif
tpl_addVar(vars, TPLADD, "BLOCKFAKECWSCHECKED", (account->cacheex.block_fakecws == 1) ? "checked" : "");
#ifdef CS_CACHEEX_AIO
tpl_addVar(vars, TPLADD, "USECWCHECKFORPUSHCHECKED", (account->cacheex.cw_check_for_push == 1) ? "checked" : "");
tpl_addVar(vars, TPLADD, "LGONLYREMOTESETTINGSCHECKED", (account->cacheex.lg_only_remote_settings == 1) ? "checked" : "");
tpl_addVar(vars, TPLADD, "LOCALGENERATEDONLYCHECKED", (account->cacheex.localgenerated_only == 1) ? "checked" : "");
value = mk_t_ftab(&account->cacheex.lg_only_tab);
tpl_addVar(vars, TPLADD, "LGONLYTAB", value);
free_mk_t(value);
tpl_addVar(vars, TPLADD, "LOCALGENERATEDONLYINCHECKED", (account->cacheex.localgenerated_only_in == 1) ? "checked" : "");
tpl_addVar(vars, TPLADD, "LGONLYINAIOONLYCHECKED", (account->cacheex.lg_only_in_aio_only == 1) ? "checked" : "");
value = mk_t_ftab(&account->cacheex.lg_only_in_tab);
tpl_addVar(vars, TPLADD, "LGONLYINTAB", value);
free_mk_t(value);
value = mk_t_caidvaluetab(&account->cacheex.cacheex_nopushafter_tab);
tpl_addVar(vars, TPLADD, "CACHEEXNOPUSHAFTER", value);
free_mk_t(value);
#endif
tpl_addVar(vars, TPLADD, "NWTCHECKED", (account->no_wait_time == 1) ? "checked" : "");
tpl_addVar(vars, TPLADD, "DISABLECRCCEX4USER", (account->disablecrccacheex == 1) ? "checked" : "");
value = mk_t_ftab(&account->disablecrccacheex_only_for);
tpl_addVar(vars, TPLADD, "IGNCRCCEX4USERONLYFOR", value);
free_mk_t(value);
#endif
#ifdef CW_CYCLE_CHECK
tpl_addVar(vars, TPLADD, "USERCWCYCLEDISABLE", (account->cwc_disable == 1) ? "checked" : "");
#endif
//Keepalive
if(!apicall)
{
if(account->ncd_keepalive)
{ tpl_addVar(vars, TPLADD, "KEEPALIVE", "checked"); }
}
else
{
tpl_printf(vars, TPLADD, "KEEPALIVEVALUE", "%d", account->ncd_keepalive);
}
#ifdef CS_ANTICASC
tpl_printf(vars, TPLADD, "AC_USERS", "%d", account->ac_users);
tpl_printf(vars, TPLADD, "CFGNUMUSERS", "%d", cfg.ac_users);
if(!apicall)
{
tpl_printf(vars, TPLADD, "TMP", "PENALTY%d", account->ac_penalty);
tpl_addVar(vars, TPLADD, tpl_getVar(vars, "TMP"), "selected");
char *tmp = NULL;
switch(cfg.ac_penalty)
{
case 0:
tmp = "(0) Only write to log";
break;
case 1:
tmp = "(1) NULL CW";
break;
case 2:
tmp = "(2) Ban";
break;
case 3:
tmp = "(3) Real CW delayed";
break;
}
tpl_addVar(vars, TPLADD, "CFGPENALTY", tmp);
}
else
{
tpl_printf(vars, TPLADD, "PENALTYVALUE", "%d", account->ac_penalty);
}
tpl_printf(vars, TPLADD, "ACOSC_MAX_ECMS_PER_MINUTE", "%d", account->acosc_max_ecms_per_minute);
tpl_printf(vars, TPLADD, "ACOSC_MAX_ACTIVE_SIDS", "%d", account->acosc_max_active_sids);
tpl_printf(vars, TPLADD, "CFG_ACOSC_MAX_ECMS_PER_MINUTE", "%d", cfg.acosc_max_ecms_per_minute);
tpl_printf(vars, TPLADD, "CFG_ACOSC_MAX_ACTIVE_SIDS", "%d", cfg.acosc_max_active_sids);
tpl_printf(vars, TPLADD, "ACOSC_ZAP_LIMIT", "%d", account->acosc_zap_limit);
tpl_printf(vars, TPLADD, "CFG_ACOSC_ZAP_LIMIT", "%d", cfg.acosc_zap_limit);
if(!apicall)
{
tpl_printf(vars, TPLADD, "TMP", "ACOSC_PENALTY%d", account->acosc_penalty);
tpl_addVar(vars, TPLADD, tpl_getVar(vars, "TMP"), "selected");
char *tmp = NULL;
switch(cfg.acosc_penalty)
{
case -1:
tmp = "(-1) Use Global Value";
break;
case 0:
tmp = "(0) Only write to log";
break;
case 1:
tmp = "(1) NULL CW";
break;
case 2:
tmp = "(2) Ban";
break;
case 3:
tmp = "(3) CW delayed";
break;
case 4:
tmp = "(4) Hidecards";
break;
}
tpl_addVar(vars, TPLADD, "CFG_ACOSC_PENALTY", tmp);
} else
{
tpl_printf(vars, TPLADD, "ACOSC_PENALTYVALUE", "%d", account->acosc_penalty);
}
tpl_printf(vars, TPLADD, "ACOSC_PENALTY_DURATION", "%d", account->acosc_penalty_duration);
tpl_printf(vars, TPLADD, "CFG_ACOSC_PENALTY_DURATION", "%d", cfg.acosc_penalty_duration);
tpl_printf(vars, TPLADD, "ACOSC_DELAY", "%d", account->acosc_delay);
tpl_printf(vars, TPLADD, "CFG_ACOSC_DELAY", "%d", cfg.acosc_delay);
#endif
#ifdef MODULE_CCCAM
tpl_printf(vars, TPLADD, "CCCMAXHOPS", "%d", account->cccmaxhops);
tpl_printf(vars, TPLADD, "CCCRESHARE", "%d", account->cccreshare);
tpl_printf(vars, TPLADD, "RESHARE", "%d", cfg.cc_reshare);
//CCcam Ignore Reshare
tpl_printf(vars, TPLADD, "TMP", "CCCIGNRSHRSELECTED%d", account->cccignorereshare);
tpl_addVar(vars, TPLADD, tpl_getVar(vars, "TMP"), "selected");
tpl_addVar(vars, TPLADD, "CFGIGNORERESHARE",
cfg.cc_ignore_reshare == 0 ?
"0 - use reshare level of Server" : "1 - use reshare level of Reader or User");
//CCcam Stealth Mode
tpl_printf(vars, TPLADD, "TMP", "CCCSTEALTHSELECTED%d", account->cccstealth);
tpl_addVar(vars, TPLADD, tpl_getVar(vars, "TMP"), "selected");
tpl_addVar(vars, TPLADD, "STEALTH", cfg.cc_stealth ? "enable" : "disable");
#endif
//Failban
tpl_printf(vars, TPLADD, "FAILBAN", "%d", account->failban);
if(!apicall)
{
#ifdef CS_CACHEEX_AIO
return tpl_getTpl(vars, "USEREDITAIO");
#else
return tpl_getTpl(vars, "USEREDIT");
#endif
}
else
{
return tpl_getTpl(vars, "APIUSEREDIT");
}
}
static void webif_add_client_proto(struct templatevars *vars, struct s_client *cl, const char *proto, int8_t apicall)
{
tpl_addVar(vars, TPLADDONCE, "CLIENTPROTO", "");
tpl_addVar(vars, TPLADDONCE, "CLIENTPROTOSORT", "");
tpl_addVar(vars, TPLADDONCE, "CLIENTPROTOTITLE", "");
tpl_addVar(vars, TPLADDONCE, "PROTOICON", "");
if(!cl) { return; }
#ifdef MODULE_NEWCAMD
if(streq(proto, "newcamd") && cl->typ == 'c')
{
tpl_printf(vars, TPLADD, "CLIENTPROTO", "%s (%s)", proto, newcamd_get_client_name(cl->ncd_client_id));
tpl_printf(vars, TPLADD, "CLIENTPROTOSORT", "%s (%s)", proto, newcamd_get_client_name(cl->ncd_client_id));
if(cfg.http_showpicons )
{
char picon_name[32];
snprintf(picon_name, sizeof(picon_name) / sizeof(char) - 1, "%s_%s", (char *)proto, newcamd_get_client_name(cl->ncd_client_id));
if(picon_exists(picon_name))
{
if (!apicall)
{
tpl_addVar(vars, TPLADD, "NCMDA", (char *)proto);
tpl_addVar(vars, TPLADD, "NCMDB", (char *)newcamd_get_client_name(cl->ncd_client_id));
tpl_addVar(vars, TPLADD, "CLIENTPROTO", tpl_getTpl(vars, "PROTONEWCAMDPIC"));
}
else
{
tpl_printf(vars, TPLADDONCE, "PROTOICON", "%s_%s",(char *)proto, (char *)newcamd_get_client_name(cl->ncd_client_id));
}
}
else
{
tpl_printf(vars, TPLADD, "CLIENTPROTOTITLE", "missing icon: IC_%s_%s.tpl", proto, newcamd_get_client_name(cl->ncd_client_id));
}
}
return;
}
#endif
#ifdef MODULE_CCCAM
if(strncmp(proto, "cccam", 5) == 0)
{
struct cc_data *cc = cl->cc;
if(cc && *cc->remote_version && *cc->remote_build)
{
tpl_printf(vars, TPLADD, "CLIENTPROTO", "%s (%s-%s)", proto, cc->remote_version, cc->remote_build);
tpl_printf(vars, TPLADD, "CLIENTPROTOSORT", "%s (%s-%s)", proto, cc->remote_version, cc->remote_build);
if(cccam_client_multics_mode(cl))
{
tpl_printf(vars, TPLADD, "CLIENTPROTOTITLE", "Multics, revision r%d", cc->multics_version[0] | (cc->multics_version[1] << 8));
}
else
{
#endif
#ifdef CS_CACHEEX_AIO
#if defined(MODULE_CCCAM) && defined(CS_CACHEEX)
if(cl->reader && cl->reader->cacheex.feature_bitfield)
{
if(cl->reader->cacheex.feature_bitfield & 32)
{
tpl_printf(vars, TPLADD, "CLIENTPROTOTITLE", "%s [cx-aio %s]", (cc->extended_mode ? cc->remote_oscam : ""), cl->reader->cacheex.aio_version);
}
else if(cl->reader->cacheex.feature_bitfield)
{
tpl_printf(vars, TPLADD, "CLIENTPROTOTITLE", "%s [cx-aio < 9.2.3]", (cc->extended_mode ? cc->remote_oscam : ""));
}
}
else if(cl->account && cl->account->cacheex.feature_bitfield)
{
if(cl->account->cacheex.feature_bitfield & 32)
{
tpl_printf(vars, TPLADD, "CLIENTPROTOTITLE", "%s [cx-aio %s]", (cc->extended_mode ? cc->remote_oscam : ""), cl->account->cacheex.aio_version);
}
else if(cl->account->cacheex.feature_bitfield)
{
tpl_printf(vars, TPLADD, "CLIENTPROTOTITLE", "%s [cx-aio < 9.2.3]", (cc->extended_mode ? cc->remote_oscam : ""));
}
}
else
{
#endif
#endif
#ifdef MODULE_CCCAM
tpl_addVar(vars, TPLADD, "CLIENTPROTOTITLE", cc->extended_mode ? cc->remote_oscam : "");
#endif
#ifdef CS_CACHEEX_AIO
#if defined(MODULE_CCCAM) && defined(CS_CACHEEX)
}
#endif
#endif
#ifdef MODULE_CCCAM
}
if(cfg.http_showpicons)
{
char picon_name[32];
int8_t is_other_proto = 0;
if(cccam_client_multics_mode(cl)) { is_other_proto = 1; }
switch(is_other_proto)
{
case 1:
snprintf(picon_name, sizeof(picon_name) / sizeof(char) - 1, "%s_r_%d", proto, cc->multics_version[0] | (cc->multics_version[1] << 8));
if(picon_exists(picon_name))
{
if (!apicall)
{
tpl_addVar(vars, TPLADD, "CCA", (char *)proto);
tpl_addVar(vars, TPLADD, "CCB", "r");
tpl_printf(vars, TPLADD, "CCC", "%d", cc->multics_version[0] | (cc->multics_version[1] << 8));
tpl_addVar(vars, TPLADD, "CCD", "");
tpl_addVar(vars, TPLADD, "CLIENTPROTO", tpl_getTpl(vars, "PROTOCCCAMPIC"));
}
else
{
tpl_printf(vars, TPLADDONCE, "PROTOICON", "%s_r_%d",(char *)proto, cc->multics_version[0] | (cc->multics_version[1] << 8));
}
}
else
{
tpl_printf(vars, TPLADD, "CLIENTPROTOTITLE", "Multics, revision r%d missing icon: IC_%s_r_%d.tpl",
cc->multics_version[0] | (cc->multics_version[1] << 8), proto, cc->multics_version[0] | (cc->multics_version[1] << 8));
}
break;
default:
snprintf(picon_name, sizeof(picon_name) / sizeof(char) - 1, "%s_%s_%s", proto, cc->remote_version, cc->remote_build);
if(picon_exists(picon_name))
{
if (!apicall)
{
tpl_addVar(vars, TPLADD, "CCA", (char *)proto);
tpl_addVar(vars, TPLADD, "CCB", cc->remote_version);
tpl_addVar(vars, TPLADD, "CCC", cc->remote_build);
#ifdef CS_CACHEEX_AIO
#if defined(MODULE_CCCAM) && defined(CS_CACHEEX)
if(cl->reader && cl->reader->cacheex.feature_bitfield)
{
if(cl->reader->cacheex.feature_bitfield & 32)
{
tpl_printf(vars, TPLADD, "CCD", "%s [cx-aio %s]", (cc->extended_mode ? cc->remote_oscam : ""), cl->reader->cacheex.aio_version);
}
else if(cl->reader->cacheex.feature_bitfield)
{
tpl_printf(vars, TPLADD, "CCD", "%s [cx-aio < 9.2.3]", (cc->extended_mode ? cc->remote_oscam : ""));
}
}
else if(cl->account && cl->account->cacheex.feature_bitfield)
{
if(cl->account->cacheex.feature_bitfield & 32)
{
tpl_printf(vars, TPLADD, "CCD", "%s [cx-aio %s]", (cc->extended_mode ? cc->remote_oscam : ""), cl->account->cacheex.aio_version);
}
else if(cl->account->cacheex.feature_bitfield)
{
tpl_printf(vars, TPLADD, "CCD", "%s [cx-aio < 9.2.3]", (cc->extended_mode ? cc->remote_oscam : ""));
}
}
else
{
#endif
#endif
tpl_addVar(vars, TPLADD, "CCD", cc->extended_mode ? cc->remote_oscam : "");
#ifdef CS_CACHEEX_AIO
#if defined(MODULE_CCCAM) && defined(CS_CACHEEX)
}
#endif
#endif
tpl_addVar(vars, TPLADD, "CLIENTPROTO", tpl_getTpl(vars, "PROTOCCCAMPIC"));
}
else
{
tpl_printf(vars, TPLADDONCE, "PROTOICON", "%s_%s_%s",(char *)proto, cc->remote_version, cc->remote_build);
}
}
else
{
#endif
#ifdef CS_CACHEEX_AIO
#if defined(MODULE_CCCAM) && defined(CS_CACHEEX)
if(cl->reader && cl->reader->cacheex.feature_bitfield)
{
if(cl->reader->cacheex.feature_bitfield & 32)
{
tpl_printf(vars, TPLADD, "CLIENTPROTOTITLE", "%s [cx-aio %s] missing icon: IC_%s_%s_%s.tpl", (cc->extended_mode ? cc->remote_oscam : ""), cl->reader->cacheex.aio_version, proto, cc->remote_version, cc->remote_build);
}
else if(cl->reader->cacheex.feature_bitfield)
{
tpl_printf(vars, TPLADD, "CLIENTPROTOTITLE", "%s [cx-aio < 9.2.3] missing icon: IC_%s_%s_%s.tpl", (cc->extended_mode ? cc->remote_oscam : ""), proto, cc->remote_version, cc->remote_build);
}
}
else if(cl->account && cl->account->cacheex.feature_bitfield)
{
if(cl->account->cacheex.feature_bitfield & 32)
{
tpl_printf(vars, TPLADD, "CLIENTPROTOTITLE", "%s [cx-aio %s] missing icon: IC_%s_%s_%s.tpl", (cc->extended_mode ? cc->remote_oscam : ""), cl->account->cacheex.aio_version, proto, cc->remote_version, cc->remote_build);
}
else if(cl->account->cacheex.feature_bitfield)
{
tpl_printf(vars, TPLADD, "CLIENTPROTOTITLE", "%s [cx-aio < 9.2.3] missing icon: IC_%s_%s_%s.tpl", (cc->extended_mode ? cc->remote_oscam : ""), proto, cc->remote_version, cc->remote_build);
}
}
else
{
#endif
#endif
#ifdef MODULE_CCCAM
tpl_printf(vars, TPLADD, "CLIENTPROTOTITLE", "%s missing icon: IC_%s_%s_%s.tpl",
cc->extended_mode ? cc->remote_oscam : "", proto, cc->remote_version, cc->remote_build);
#ifdef CS_CACHEEX_AIO
#if defined(MODULE_CCCAM) && defined(CS_CACHEEX)
}
#endif
#endif
}
break;
}
}
}
return;
}
#endif
#ifdef CS_CACHEEX_AIO
#if (defined(MODULE_CAMD35) || defined(MODULE_CAMD35_TCP)) && defined(CS_CACHEEX)
if(strncmp(proto, "cs3", 3) == 0)
{
tpl_addVar(vars, TPLADDONCE, "CLIENTPROTO", (char *)proto);
char aiover[32];
aiover[0] = '\0';
if(cl->account && cl->cacheex_aio_checked)
{
if(cl->account->cacheex.feature_bitfield & 32)
{
snprintf(aiover, sizeof(aiover) / sizeof(char) -1, "%s", cl->account->cacheex.aio_version);
tpl_addVar(vars, TPLADD, "CLIENTPROTOTITLE", aiover);
}
else if(cl->account->cacheex.feature_bitfield)
{
snprintf(aiover, sizeof(aiover) / sizeof(char) -1, "%s", "[cx-aio: < 9.2.3]");
tpl_addVar(vars, TPLADD, "CLIENTPROTOTITLE", aiover);
}
else
{
tpl_addVar(vars, TPLADD, "CLIENTPROTOTITLE", "");
}
}
if(cl->reader && cl->cacheex_aio_checked)
{
if(cl->reader->cacheex.feature_bitfield & 32)
{
snprintf(aiover, sizeof(aiover) / sizeof(char) -1, "%s", cl->reader->cacheex.aio_version);
tpl_addVar(vars, TPLADD, "CLIENTPROTOTITLE", aiover);
}
else if(cl->reader->cacheex.feature_bitfield)
{
snprintf(aiover, sizeof(aiover) / sizeof(char) -1, "%s", "[cx-aio: < 9.2.3]");
tpl_addVar(vars, TPLADD, "CLIENTPROTOTITLE", aiover);
}
else
{
tpl_addVar(vars, TPLADD, "CLIENTPROTOTITLE", "");
}
}
if(cfg.http_showpicons)
{
tpl_addVar(vars, TPLADDONCE, "CLIENTPROTO", (char *)proto);
char picon_name[32];
snprintf(picon_name, sizeof(picon_name) / sizeof(char) - 1, "%s", proto);
if(picon_exists(picon_name))
{
if (!apicall)
{
tpl_addVar(vars, TPLADD, "CAMD3A", (char *)proto);
if(aiover[0] == '\0')
tpl_addVar(vars, TPLADD, "AIOVER", "");
else
tpl_printf(vars, TPLADD, "AIOVER", "[cx-aio %s]", aiover);
tpl_addVar(vars, TPLADD, "CLIENTPROTO", tpl_getTpl(vars, "PROTOCAMD3AIOPIC"));
}
else
{
tpl_printf(vars, TPLADDONCE, "PROTOICON", "%s",(char *)proto);
}
}
else
{
if(cl->account && cl->cacheex_aio_checked)
{
if(cl->account->cacheex.feature_bitfield & 32)
tpl_printf(vars, TPLADD, "CLIENTPROTOTITLE", "missing icon: IC_%s.tpl [cx-aio %s]", proto, cl->account->cacheex.aio_version);
else if(cl->account->cacheex.feature_bitfield)
tpl_printf(vars, TPLADD, "CLIENTPROTOTITLE", "missing icon: IC_%s.tpl [cx-aio < 9.2.3]", proto);
else
tpl_printf(vars, TPLADD, "CLIENTPROTOTITLE", "missing icon: IC_%s.tpl", proto);
}
if(cl->reader && cl->cacheex_aio_checked)
{
if(cl->reader->cacheex.feature_bitfield & 32)
tpl_printf(vars, TPLADD, "CLIENTPROTOTITLE", "missing icon: IC_%s.tpl [cx-aio %s]", proto, cl->reader->cacheex.aio_version);
else if(cl->reader->cacheex.feature_bitfield)
tpl_printf(vars, TPLADD, "CLIENTPROTOTITLE", "missing icon: IC_%s.tpl [cx-aio < 9.2.3]", proto);
else
tpl_printf(vars, TPLADD, "CLIENTPROTOTITLE", "missing icon: IC_%s.tpl", proto);
}
}
}
return;
}
#endif
#endif
#ifdef HAVE_DVBAPI
if(streq(proto, "dvbapi") && cl->typ == 'c' && strcmp(dvbapi_get_client_name(), ""))
{
if (!apicall)
tpl_printf(vars, TPLADD, "CLIENTPROTO", "<A HREF=\"#\" CLASS=\"tooltip\">%s<SPAN>client: %s<BR>protocol version: %d</SPAN></A>", proto, dvbapi_get_client_name(), dvbapi_get_client_proto_version());
else
tpl_printf(vars, TPLADD, "CLIENTPROTO", "%s (client: %s, protocol version: %d)", proto, dvbapi_get_client_name(), dvbapi_get_client_proto_version());
tpl_printf(vars, TPLADD, "CLIENTPROTOSORT", "%s", proto);
return;
}
#endif
tpl_addVar(vars, TPLADDONCE, "CLIENTPROTO", (char *)proto);
tpl_addVar(vars, TPLADDONCE, "CLIENTPROTOSORT", (char *)proto);
if(cfg.http_showpicons)
{
char picon_name[32];
snprintf(picon_name, sizeof(picon_name) / sizeof(char) - 1, "%s", proto);
if(picon_exists(picon_name))
{
if (!apicall)
{
tpl_addVar(vars, TPLADD, "OTHER", (char *)proto);
tpl_addVar(vars, TPLADD, "CLIENTPROTO", tpl_getTpl(vars, "PROTOOTHERPIC"));
}
else
{
tpl_printf(vars, TPLADDONCE, "PROTOICON", "%s",(char *)proto);
}
}
else
{
tpl_printf(vars, TPLADD, "CLIENTPROTOTITLE", "missing icon: IC_%s.tpl", proto);
}
}
}
static void kill_account_thread(struct s_auth *account)
{
struct s_client *cl;
for(cl = first_client->next; cl ; cl = cl->next)
{
if(cl->account == account)
{
if(get_module(cl)->type & MOD_CONN_NET)
{
kill_thread(cl);
}
else
{
cl->account = first_client->account;
}
}
}
}
static char *send_oscam_user_config(struct templatevars *vars, struct uriparams *params, int32_t apicall)
{
struct s_auth *account;
struct s_client *cl;
char *user = getParam(params, "user");
int32_t found = 0;
uint8_t md5tmp[MD5_DIGEST_LENGTH];
if(!apicall)
{
setActiveMenu(vars, MNU_USERS);
}
if(strcmp(getParam(params, "action"), "reinit") == 0)
{
if(!cfg.http_readonly)
{ refresh_oscam(REFR_ACCOUNTS); }
}
if(strcmp(getParam(params, "action"), "delete") == 0)
{
if(cfg.http_readonly)
{
tpl_addMsg(vars, "WebIf is in readonly mode. No deletion will be made!");
}
else
{
struct s_auth *account_prev = NULL;
for(account = cfg.account; (account); account = account->next)
{
if(strcmp(account->usr, user) == 0)
{
if(account_prev == NULL)
{ cfg.account = account->next; }
else
{ account_prev->next = account->next; }
ll_clear(account->aureader_list);
kill_account_thread(account);
add_garbage(account);
found = 1;
break;
}
account_prev = account;
}
if(found > 0)
{
if(write_userdb() != 0) { tpl_addMsg(vars, "Write Config failed!"); }
}
else { tpl_addMsg(vars, "Sorry but the specified user doesn't exist. No deletion will be made!"); }
}
}
if((strcmp(getParam(params, "action"), "disable") == 0) || (strcmp(getParam(params, "action"), "enable") == 0))
{
account = get_account_by_name(getParam(params, "user"));
if(account)
{
if(strcmp(getParam(params, "action"), "disable") == 0)
{
account->disabled = 1;
kill_account_thread(account);
}
else
{ account->disabled = 0; }
if(write_userdb() != 0) { tpl_addMsg(vars, "Write Config failed!"); }
}
else
{
tpl_addMsg(vars, "Sorry but the specified user doesn't exist. No deletion will be made!");
}
}
if(strcmp(getParam(params, "action"), "resetstats") == 0)
{
account = get_account_by_name(getParam(params, "user"));
if(account) { clear_account_stats(account); }
}
if(strcmp(getParam(params, "action"), "resetuserstats") == 0)
{
clear_info_clients_stats();
}
if(strcmp(getParam(params, "action"), "resetreaderstats") == 0)
{
clear_info_readers_stats();
}
if(strcmp(getParam(params, "action"), "resetalluserstats") == 0)
{
clear_all_account_stats();
}
if((strcmp(getParam(params, "part"), "adduser") == 0) && (!cfg.http_readonly))
{
tpl_addVar(vars, TPLAPPEND, "NEWUSERFORM", tpl_getTpl(vars, "ADDNEWUSER"));
}
/* List accounts*/
char *status, *expired, *classname, *lastchan;
time_t now = time((time_t *)0);
int32_t isec = 0, chsec = 0;
char *filter = NULL;
int32_t clientcount = 0;
if(apicall)
{
filter = getParam(params, "label");
}
int8_t grp_set = 0;
int8_t expdate_set = 0;
int32_t total_users = 0;
int32_t disabled_users = 0;
int32_t expired_users = 0;
int32_t expired_or_disabled_users = 0;
int32_t connected_users = 0;
int32_t online_users = 0;
int32_t casc_users = 0;
int32_t casc_users2 = 0;
int32_t n_request = 0;
int existing_insert = 0;
for(account = cfg.account; (account); account = account->next)
{
if(account->expirationdate){
expdate_set = 1;
}
if(account->next){
if(account->grp != account->next->grp){
grp_set = 1;
}
}
if(expdate_set && grp_set)
break;
}
for(account = cfg.account; (account); account = account->next)
{
//clear for next client
total_users++;
isactive = 1;
status = "offline";
classname = "offline";
isec = 0;
chsec = 0;
//reset caid/srevid template variables
tpl_addVar(vars, TPLADD, "CLIENTCAID", "");
tpl_addVar(vars, TPLADD, "CLIENTPROVID", "");
tpl_addVar(vars, TPLADD, "CLIENTSRVID", "");
tpl_addVar(vars, TPLADD, "LASTCHANNEL", "");
tpl_addVar(vars, TPLADD, "LASTCHANNELSORT", "");
tpl_addVar(vars, TPLADD, "USERMD5", "");
tpl_addVar(vars, TPLADD, "CWLASTRESPONSET", "");
tpl_addVar(vars, TPLADD, "CWLASTRESPONSETMS", "");
tpl_addVar(vars, TPLADD, "CLIENTIP", "");
tpl_addVar(vars, TPLADD, "LASTCHANNELTITLE", "");
tpl_addVar(vars, TPLADD, "LASTCHANNELSORT", "");
tpl_addVar(vars, TPLADD, "CLIENTTIMEONCHANNELAPI", "");
tpl_addVar(vars, TPLADD, "CLIENTTIMEONCHANNEL", "");
tpl_addVar(vars, TPLADD, "CLIENTTIMETOSLEEP", "");
tpl_addVar(vars, TPLADD, "CLIENTTIMETOSLEEPAPI", "");
tpl_addVar(vars, TPLADD, "IDLESECS", "");
tpl_addVar(vars, TPLADD, "UNOTIFY", "");
if(account->expirationdate && account->expirationdate < now)
{
expired = " (expired)";
classname = "expired";
expired_users++;
isactive = 0;
}
else
{
expired = "";
}
if(account->disabled != 0)
{
expired = " (disabled)";
classname = "disabled";
tpl_addVar(vars, TPLADD, "SWITCHICO", "image?i=ICENA");
tpl_addVar(vars, TPLADD, "SWITCHTITLE", "Enable");
tpl_addVar(vars, TPLADD, "SWITCH", "enable");
disabled_users++;
isactive = 0;
}
else
{
tpl_addVar(vars, TPLADD, "SWITCHICO", "image?i=ICDIS");
tpl_addVar(vars, TPLADD, "SWITCHTITLE", "Disable");
tpl_addVar(vars, TPLADD, "SWITCH", "disable");
}
if((account->expirationdate && account->expirationdate < now)||account->disabled != 0)
{
expired_or_disabled_users++;
}
int32_t lastresponsetm = 0, latestactivity = 0;
const char *proto = "";
double cwrate = 0.0, cwrate2 = 0.0;
//search account in active clients
isactive = 0;
int16_t nrclients = 0;
struct s_client *latestclient = NULL;
for(cl = first_client->next; cl ; cl = cl->next)
{
if(cl->account && !strcmp(cl->account->usr, account->usr))
{
if(cl->lastecm > latestactivity || cl->login > latestactivity)
{
if(cl->lastecm > cl->login) { latestactivity = cl->lastecm; }
else { latestactivity = cl->login; }
latestclient = cl;
}
nrclients++;
}
}
if(account->cwfound + account->cwnot + account->cwcache > 0)
{
cwrate = now - account->firstlogin;
cwrate /= (account->cwfound + account->cwnot + account->cwcache);
}
casc_users = 0;
casc_users2 = 0;
int8_t conn = 0;
if(latestclient != NULL)
{
char channame[CS_SERVICENAME_SIZE];
status = (!apicall) ? "<B>connected</B>" : "connected";
if(account->expirationdate && account->expirationdate < now) { classname = "expired"; }
else { classname = "connected";conn = 1; }
proto = client_get_proto(latestclient);
int clientcaid = latestclient->last_caid;
int clientsrvid = latestclient->last_srvid;
int clientprovid = latestclient->last_provid;
tpl_printf(vars, TPLADD, "CLIENTCAID", "%04X", clientcaid);
tpl_printf(vars, TPLADD, "CLIENTPROVID", "%06X", clientprovid);
tpl_printf(vars, TPLADD, "CLIENTSRVID", "%04X", clientsrvid);
if(clientsrvid != NO_SRVID_VALUE || clientcaid != NO_CAID_VALUE)
{
lastchan = xml_encode(vars, get_servicename(latestclient, clientsrvid, clientprovid, clientcaid, channame, sizeof(channame)));
tpl_addVar(vars, TPLADD, "LASTCHANNELSORT", lastchan);
}
else
{
lastchan = "";
tpl_addVar(vars, TPLADD, "LASTCHANNELSORT", "~~~~~");
}
tpl_addVar(vars, TPLADDONCE, "LASTCHANNEL", lastchan);
tpl_addVar(vars, TPLADD, "LASTCHANNELTITLE", lastchan);
if(cfg.http_showpicons )
{
char picon_name[128];
char picon_channame[128];
int8_t picon_ok = 0;
get_picon_servicename_or_null(latestclient, clientsrvid, clientprovid, clientcaid, picon_channame, sizeof(picon_channame));
if(picon_channame[0])
{
snprintf(picon_name, sizeof(picon_name) / sizeof(char), "%s", picon_channame);
picon_ok = picon_exists(picon_name);
if(!picon_ok && picon_servicename_remve_hd(picon_channame, sizeof(picon_channame)))
{
snprintf(picon_name, sizeof(picon_name) / sizeof(char), "%s", picon_channame);
picon_ok = picon_exists(picon_name);
}
}
if(!picon_ok)
{
snprintf(picon_name, sizeof(picon_name) / sizeof(char), "%04X_%06X_%04X", clientcaid, clientprovid, clientsrvid);
picon_ok = picon_exists(picon_name);
}
if(!picon_ok)
{
snprintf(picon_name, sizeof(picon_name) / sizeof(char), "%04X_%04X", clientcaid, clientsrvid);
picon_ok = picon_exists(picon_name);
}
if(!picon_ok)
{
snprintf(picon_name, sizeof(picon_name) / sizeof(char), "0000_%04X", clientsrvid);
picon_ok = picon_exists(picon_name);
}
if(picon_ok)
{
tpl_addVar(vars, TPLADDONCE, "LCA", picon_name);
tpl_addVar(vars, TPLADDONCE, "LCB", lastchan);
if(!apicall)
{
tpl_addVar(vars, TPLADDONCE, "LASTCHANNEL", tpl_getTpl(vars, "USERCONFIGLASTCHANEL"));
}
}
else
{
tpl_printf(vars, TPLADD, "LASTCHANNELTITLE", "missing icon: IC_%s.tpl", picon_name);
}
}
lastresponsetm = latestclient->cwlastresptime;
if(IP_ISSET(latestclient->ip))
{ tpl_addVar(vars, TPLADD, "CLIENTIP", cs_inet_ntoa(latestclient->ip)); }
else if(latestclient->login > latestclient->logout)
{ tpl_addVar(vars, TPLADD, "CLIENTIP", "camd.socket"); }
else
{ tpl_addVar(vars, TPLADD, "CLIENTIP", ""); }
connected_users++;
casc_users = ll_count(latestclient->cascadeusers);
LL_ITER it = ll_iter_create(latestclient->cascadeusers);
struct s_cascadeuser *cu;
while((cu = ll_iter_next(&it)))
{
if(cu->cwrate > 0)
{ casc_users2++; }
}
if(latestactivity > 0)
{
isec = now - latestactivity;
chsec = latestclient->lastswitch ? now - latestclient->lastswitch : 0;
if(isec < cfg.hideclient_to)
{
isactive = 1;
status = (!apicall) ? "<B>online</B>" : "online";
if(account->expirationdate && account->expirationdate < now) { classname = "expired"; }
else { classname = "online"; }
if(latestclient->cwfound + latestclient->cwnot + latestclient->cwcache > 0)
{
cwrate2 = now - latestclient->login;
cwrate2 /= (latestclient->cwfound + latestclient->cwnot + latestclient->cwcache);
tpl_printf(vars, TPLADDONCE, "CWRATE2", " (%.2f)", cwrate2);
online_users++;
}
}
}
}
n_request = 0;
if(latestclient != NULL){
n_request = latestclient->n_request[0];
}
tpl_addVar(vars, TPLADD, "EXPIREVIEW", expdate_set ? "" : "exp");
tpl_addVar(vars, TPLADD, "GRPVIEW", grp_set ? "" : "grp");
#ifdef CS_ANTICASC
tpl_addVar(vars, TPLADD, "ANTICASCVIEW", cfg.ac_enabled ? "" : "acas");
#endif
tpl_printf(vars, TPLADD, "CWOK", PRINTF_LOCAL_D, account->cwfound);
tpl_printf(vars, TPLADD, "CWNOK", PRINTF_LOCAL_D, account->cwnot);
tpl_printf(vars, TPLADD, "CWIGN", PRINTF_LOCAL_D, account->cwignored);
tpl_printf(vars, TPLADD, "CWTOUT", PRINTF_LOCAL_D, account->cwtout);
#ifdef CW_CYCLE_CHECK
tpl_addVar(vars, TPLADD, "CWCCYCVIEW", cfg.cwcycle_check_enable ? "" : "cwc");
tpl_printf(vars, TPLADD, "CWCYCLECHECKED", "%d", account->cwcycledchecked);
tpl_printf(vars, TPLADD, "CWCYCLEOK", "%d", account->cwcycledok);
tpl_printf(vars, TPLADD, "CWCYCLENOK", "%d", account->cwcyclednok);
tpl_printf(vars, TPLADD, "CWCYCLEIGN", "%d", account->cwcycledign);
#endif
tpl_printf(vars, TPLADD, "CWCACHE", PRINTF_LOCAL_D, account->cwcache);
tpl_printf(vars, TPLADD, "CWTUN", PRINTF_LOCAL_D, account->cwtun);
tpl_printf(vars, TPLADD, "EMMOK", PRINTF_LOCAL_D, account->emmok);
tpl_printf(vars, TPLADD, "EMMNOK", PRINTF_LOCAL_D, account->emmnok);
tpl_printf(vars, TPLADD, "CWRATE", "%.2f", cwrate);
tpl_printf(vars, TPLADD, "CASCUSERS", "%d", casc_users);
tpl_printf(vars, TPLADD, "CASCUSERS2", "%d", casc_users2);
tpl_printf(vars, TPLADD, "CASCUSERSCOMB", "%d/%d", casc_users, casc_users2);
tpl_printf(vars, TPLADD, "N_REQUEST_MIN", "%d", n_request);
if(isactive > 0 || conn > 0)
{
if(casc_users+casc_users2>0)
{
tpl_printf(vars, TPLADD, "CWLASTRESPONSET", "%d", lastresponsetm);
tpl_printf(vars, TPLADD, "CWLASTRESPONSETMS", PRINTF_LOCAL_D " ms", lastresponsetm);
}
tpl_addVar(vars, TPLADD, "IDLESECS", sec2timeformat(vars, isec));
}
if(isactive > 0)
{
tpl_printf(vars, TPLADD, "CLIENTTIMEONCHANNELAPI", "%d", chsec);
tpl_addVar(vars, TPLADD, "CLIENTTIMEONCHANNEL", sec2timeformat(vars, chsec));
if(account->tosleep)
{
if(account->tosleep >0){
tpl_printf(vars, TPLADD, "CLIENTTIMETOSLEEP", "Sleeping in %d minutes", account->tosleep - (chsec / 60));
} else {
tpl_addVar(vars, TPLADD, "CLIENTTIMETOSLEEP", "Sleeping");
}
tpl_printf(vars, TPLADD, "CLIENTTIMETOSLEEPAPI", "%d", account->tosleep - (chsec / 60));
} else {
tpl_addVar(vars, TPLADD, "CLIENTTIMETOSLEEPAPI", "undefined");
}
}
#ifdef CS_CACHEEX_AIO
#if defined(MODULE_CAMD35) || defined (MODULE_CAMD35_TCP)
if(latestclient != NULL && (latestclient->account->cacheex.feature_bitfield || latestclient->c35_extmode > 1))
#else
if(latestclient != NULL && (latestclient->account->cacheex.feature_bitfield))
#endif
{
const char *aio_suffix = " (cx-aio)";
char *new_proto;
if(cs_malloc(&new_proto, cs_strlen(proto)+cs_strlen(aio_suffix)+1)) {
if (!cs_strncat(new_proto, (char *)proto, cs_strlen(proto)+cs_strlen(aio_suffix)+1)) {
cs_log("FIXME!");
}
if (cs_strncat(new_proto, (char *)aio_suffix, cs_strlen(proto)+cs_strlen(aio_suffix)+1)) {
webif_add_client_proto(vars, latestclient, (const char *)new_proto, apicall);
} else {
cs_log("FIXME!");
}
free(new_proto);
}
} else {
#endif
webif_add_client_proto(vars, latestclient, proto, apicall);
#ifdef CS_CACHEEX_AIO
}
#endif
tpl_addVar(vars, TPLADD, "CLASSNAME", classname);
MD5((uint8_t *)account->usr, cs_strlen(account->usr), md5tmp);
int z;
tpl_addVar(vars, TPLADD, "USERMD5","id_");
for (z = 0; z < MD5_DIGEST_LENGTH; z++)
{
tpl_printf(vars, TPLAPPEND, "USERMD5", "%02x", md5tmp[z]);
}
tpl_addVar(vars, TPLADD, "USERNAME", xml_encode(vars, account->usr));
tpl_addVar(vars, TPLADD, "USERNAMEENC", urlencode(vars, account->usr));
if(!existing_insert)
{
tpl_printf(vars, TPLADD, "EXISTING_INS", "'%s'", urlencode(vars, account->usr));
existing_insert++;
}else
{
tpl_printf(vars, TPLAPPEND, "EXISTING_INS", ",'%s'", urlencode(vars, account->usr));
}
if(account->description)
tpl_printf(vars, TPLADD, "DESCRIPTION","%s(%s)",!apicall?"&#13;":"",xml_encode(vars, account->description));
else
tpl_addVar(vars, TPLADD, "DESCRIPTION", "");
if(cfg.http_showpicons && !apicall)
tpl_addVar(vars, TPLADD, "USERBIT", tpl_getTpl(vars, picon_exists(xml_encode(vars, account->usr)) ? "USERICON" : "USERNOICON"));
else
tpl_addVar(vars, TPLADD, "USERBIT", tpl_getTpl(vars, "USERLABEL"));
char *value = mk_t_group(account->grp);
tpl_addVar(vars, TPLADD, "GROUPS", value);
free_mk_t(value);
tpl_addVar(vars, TPLADD, "STATUS", status);
tpl_addVar(vars, TPLAPPEND, "STATUS", expired);
if(nrclients > 1)
{
tpl_printf(vars, TPLADD, "UNOTIFY", "%d", nrclients);
tpl_addVar(vars, TPLADDONCE, "CLIENTCOUNTNOTIFIER", tpl_getTpl(vars, "CLIENTCOUNTNOTIFIERBIT"));
}
//Expirationdate
struct tm timeinfo;
cs_gmtime_r(&account->expirationdate, &timeinfo);
char buf [80];
strftime(buf, 80, "%Y-%m-%d", &timeinfo);
if(strcmp(buf, "1970-01-01")) { tpl_addVar(vars, TPLADD, "EXPDATE", buf); }
else { tpl_addVar(vars, TPLADD, "EXPDATE", ""); }
// append row to table template
if(!apicall)
{ tpl_addVar(vars, TPLAPPEND, "USERCONFIGS", tpl_getTpl(vars, "USERCONFIGLISTBIT")); }
else if(!filter || strcmp(filter, account->usr) == 0 || strcmp(filter, "all") == 0 || cs_strlen(filter) == 0)
{
if(apicall == 1){
tpl_addVar(vars, TPLAPPEND, "APIUSERCONFIGS", tpl_getTpl(vars, "APIUSERCONFIGLISTBIT"));
} else if (apicall == 2){
tpl_printf(vars, TPLADD, "JSONDELIMITER", "%s", (total_users > 1)?",":"");
tpl_addVar(vars, TPLAPPEND, "APIUSERCONFIGS", tpl_getTpl(vars, "JSONUSERBIT"));
}
++clientcount;
}
}
tpl_printf(vars, TPLADD, "TOTAL_USERS", "%d", total_users);
tpl_printf(vars, TPLADD, "TOTAL_DISABLED", "%d", disabled_users);
tpl_printf(vars, TPLADD, "TOTAL_EXPIRED", "%d", expired_users);
tpl_printf(vars, TPLADD, "TOTAL_ACTIVE", "%d", total_users - expired_or_disabled_users);
tpl_printf(vars, TPLADD, "TOTAL_CONNECTED", "%d", connected_users);
tpl_printf(vars, TPLADD, "TOTAL_ONLINE", "%d", online_users);
//CM info
tpl_addVar(vars, TPLADD, "DISPLAYREADERINFO", "hidden"); // no readerinfo in users
set_ecm_info(vars);
if(!apicall)
{ return tpl_getTpl(vars, "USERCONFIGLIST"); }
else
{
if(!filter || clientcount > 0)
{
return tpl_getTpl(vars, (apicall==1)?"APIUSERCONFIGLIST":"JSONUSER");
}
else
{
tpl_printf(vars, TPLADD, "APIERRORMESSAGE", "Invalid client %s", xml_encode(vars, filter));
return tpl_getTpl(vars, "APIERROR");
}
}
}
#define ENTITLEMENT_PAGE_SIZE 500
#ifdef MODULE_CCCAM
static void print_cards(struct templatevars *vars, struct uriparams *params, struct cc_card **cardarray, int32_t cardsize,
int8_t show_global_list, struct s_reader *rdr, int32_t offset, int32_t apicall)
{
if(cardarray)
{
uint8_t serbuf[8];
int32_t i, count = 0;
char provname[83];
struct cc_card *card;
int32_t cardcount = 0;
int32_t providercount = 0;
int32_t nodecount = 0;
char *provider = "";
// @todo alno: sort by click, 0=ascending, 1=descending (maybe two buttons or reverse on second click)
for(i = offset; i < cardsize; ++i)
{
card = cardarray[i];
if(count == ENTITLEMENT_PAGE_SIZE)
{ break; }
count++;
if(!apicall)
{
if(show_global_list)
{ rdr = card->origin_reader; }
if(rdr)
{ tpl_printf(vars, TPLADD, "HOST", "%s:%d", xml_encode(vars, rdr->device), rdr->r_port); }
tpl_printf(vars, TPLADD, "CAID", "%04X", card->caid);
tpl_printf(vars, TPLADD, "CARDTYPE", "%02X", card->card_type);
}
else
{
tpl_printf(vars, TPLADD, "APICARDNUMBER", "%d", cardcount);
tpl_printf(vars, TPLADD, "APICAID", "%04X", card->caid);
tpl_printf(vars, TPLADD, "APICARDTYPE", "%02X", card->card_type);
}
if(cc_UA_valid(card->hexserial)) //Add UA:
{
cc_UA_cccam2oscam(card->hexserial, serbuf, card->caid);
char tmp[20];
tpl_printf(vars, TPLAPPEND, "HOST", "<BR>\nUA_OSCam:%s", cs_hexdump(0, serbuf, 8, tmp, 20));
tpl_printf(vars, TPLAPPEND, "HOST", "<BR>\nUA_CCcam:%s", cs_hexdump(0, card->hexserial, 8, tmp, 20));
}
if(!apicall)
{
int32_t n;
char channame[CS_SERVICENAME_SIZE];
int8_t sidname = 0;
LL_ITER its = ll_iter_create(card->goodsids);
struct cc_srvid *srv;
n = 0;
if(strcmp(getParam(params, "button"), "Show detail list") == 0)
{ sidname = 1; }
tpl_addVar(vars, TPLADD, "SERVICESGOOD", "");
while((srv = ll_iter_next(&its)))
{
if(sidname)
{
tpl_printf(vars, TPLAPPEND, "SERVICESGOOD", "%04X - %s<BR>", srv->sid, xml_encode(vars, get_servicename(cur_client(), srv->sid, 0, card->caid, channame, sizeof(channame))));
} else {
tpl_printf(vars, TPLAPPEND, "SERVICESGOOD", "%04X%s", srv->sid, ++n % 10 == 0 ? "<BR>\n" : " ");
}
}
its = ll_iter_create(card->badsids);
n = 0;
tpl_addVar(vars, TPLADD, "SERVICESBAD", "");
while((srv = ll_iter_next(&its)))
{
if(sidname)
{
tpl_printf(vars, TPLAPPEND, "SERVICESBAD", "%04X - %s<BR>", srv->sid, xml_encode(vars, get_servicename(cur_client(), srv->sid, 0, card->caid, channame, sizeof(channame))));
} else {
tpl_printf(vars, TPLAPPEND, "SERVICESBAD", "%04X%s", srv->sid, ++n % 10 == 0 ? "<BR>\n" : " ");
}
}
}
tpl_addVar(vars, TPLADD, "SYSTEM", get_cardsystem_desc_by_caid(card->caid));
tpl_printf(vars, TPLADD, "SHAREID", "%08X", card->id);
tpl_printf(vars, TPLADD, "REMOTEID", "%08X", card->remote_id);
tpl_printf(vars, TPLADD, "UPHOPS", "%d", card->hop);
tpl_printf(vars, TPLADD, "MAXDOWN", "%d", card->reshare);
LL_ITER pit = ll_iter_create(card->providers);
struct cc_provider *prov;
providercount = 0;
if(!apicall)
{ tpl_addVar(vars, TPLADD, "PROVIDERS", ""); }
else
{ tpl_addVar(vars, TPLADD, "PROVIDERLIST", ""); }
while((prov = ll_iter_next(&pit)))
{
provider = xml_encode(vars, get_provider(prov->prov, card->caid, provname, sizeof(provname)));
if(!apicall)
{
if(prov->sa[0] || prov->sa[1] || prov->sa[2] || prov->sa[3])
{
tpl_printf(vars, TPLAPPEND, "PROVIDERS", "%s SA:%02X%02X%02X%02X<BR>\n", provider, prov->sa[0], prov->sa[1], prov->sa[2], prov->sa[3]);
}
else
{
tpl_printf(vars, TPLAPPEND, "PROVIDERS", "%s<BR>\n", provider);
}
}
else
{
if(prov->sa[0] || prov->sa[1] || prov->sa[2] || prov->sa[3])
{ tpl_printf(vars, TPLADD, "APIPROVIDERSA", "%02X%02X%02X%02X", prov->sa[0], prov->sa[1], prov->sa[2], prov->sa[3]); }
else
{ tpl_addVar(vars, TPLADD, "APIPROVIDERSA", ""); }
tpl_printf(vars, TPLADD, "APIPROVIDERCAID", "%04X", card->caid);
tpl_printf(vars, TPLADD, "APIPROVIDERPROVID", "%06X", prov->prov);
tpl_printf(vars, TPLADD, "APIPROVIDERNUMBER", "%d", providercount);
tpl_addVar(vars, TPLADD, "APIPROVIDERNAME", xml_encode(vars, provider));
tpl_addVar(vars, TPLAPPEND, "PROVIDERLIST", tpl_getTpl(vars, "APICCCAMCARDPROVIDERBIT"));
}
providercount++;
tpl_printf(vars, TPLADD, "APITOTALPROVIDERS", "%d", providercount);
}
LL_ITER nit = ll_iter_create(card->remote_nodes);
uint8_t *node;
nodecount = 0;
if(!apicall) { tpl_addVar(vars, TPLADD, "NODES", ""); }
else { tpl_addVar(vars, TPLADD, "NODELIST", ""); }
while((node = ll_iter_next(&nit)))
{
if(!apicall)
{
tpl_printf(vars, TPLAPPEND, "NODES", "%02X%02X%02X%02X%02X%02X%02X%02X<BR>\n",
node[0], node[1], node[2], node[3], node[4], node[5], node[6], node[7]);
}
else
{
tpl_printf(vars, TPLADD, "APINODE", "%02X%02X%02X%02X%02X%02X%02X%02X", node[0], node[1], node[2], node[3], node[4], node[5], node[6], node[7]);
tpl_printf(vars, TPLADD, "APINODENUMBER", "%d", nodecount);
tpl_addVar(vars, TPLAPPEND, "NODELIST", tpl_getTpl(vars, "APICCCAMCARDNODEBIT"));
}
nodecount++;
tpl_printf(vars, TPLADD, "APITOTALNODES", "%d", nodecount);
}
if(!apicall)
{ tpl_addVar(vars, TPLAPPEND, "CCCAMSTATSENTRY", tpl_getTpl(vars, "ENTITLEMENTCCCAMENTRYBIT")); }
else
{ tpl_addVar(vars, TPLAPPEND, "CARDLIST", tpl_getTpl(vars, "APICCCAMCARDBIT")); }
cardcount++;
}
// set previous Link if needed
if(offset >= ENTITLEMENT_PAGE_SIZE)
{
tpl_printf(vars, TPLAPPEND, "CONTROLS", "<A HREF=\"entitlements.html?offset=%d&globallist=%s&amp;label=%s\"> << PREVIOUS < </A>",
offset - ENTITLEMENT_PAGE_SIZE,
getParam(params, "globallist"),
urlencode(vars, getParam(params, "label")));
}
// set next link if needed
if(cardsize > count && offset < cardsize)
{
tpl_printf(vars, TPLAPPEND, "CONTROLS", "<A HREF=\"entitlements.html?offset=%d&globallist=%s&amp;label=%s\"> > NEXT >> </A>",
offset + ENTITLEMENT_PAGE_SIZE,
getParam(params, "globallist"),
urlencode(vars, getParam(params, "label")));
}
if(!apicall)
{
tpl_printf(vars, TPLADD, "TOTALS", "card count=%d", cardsize);
tpl_addVar(vars, TPLADD, "ENTITLEMENTCONTENT", tpl_getTpl(vars, "ENTITLEMENTCCCAMBIT"));
}
else
{
tpl_printf(vars, TPLADD, "APITOTALCARDS", "%d", cardsize);
}
}
else
{
if(!apicall)
{
tpl_addVar(vars, TPLADD, "ENTITLEMENTCONTENT", tpl_getTpl(vars, "ENTITLEMENTGENERICBIT"));
tpl_addVar(vars, TPLADD, "LOGHISTORY", "no cards found<BR>\n");
}
else
{
tpl_printf(vars, TPLADD, "APITOTALCARDS", "%d", 0);
}
}
}
#endif
static char *send_oscam_entitlement(struct templatevars *vars, struct uriparams *params, int32_t apicall)
{
if(!apicall) { setActiveMenu(vars, MNU_READERS); }
char *reader_ = getParam(params, "label");
#ifdef MODULE_CCCAM
#ifdef MODULE_CCCSHARE
char *sharelist_ = getParam(params, "globallist");
int32_t show_global_list = sharelist_ && sharelist_[0] == '1';
#else
int32_t show_global_list = 0;
#endif
struct s_reader *rdr = get_reader_by_label(getParam(params, "label"));
if(show_global_list || cs_strlen(reader_) || (rdr && rdr->typ == R_CCCAM))
{
if(show_global_list || (rdr && rdr->typ == R_CCCAM && rdr->enable))
{
if(show_global_list)
{
tpl_addVar(vars, TPLADD, "READERNAME", "GLOBAL");
tpl_addVar(vars, TPLADD, "APIHOST", "GLOBAL");
tpl_addVar(vars, TPLADD, "APIHOSTPORT", "GLOBAL");
}
else
{
tpl_addVar(vars, TPLADD, "READERNAME", xml_encode(vars, rdr->label));
tpl_addVar(vars, TPLADD, "APIHOST", xml_encode(vars, rdr->device));
tpl_printf(vars, TPLADD, "APIHOSTPORT", "%d", rdr->r_port);
}
int32_t offset = atoi(getParam(params, "offset")); //should be 0 if parameter is missed on very first call
int32_t cardsize;
#ifdef MODULE_CCCSHARE
if(show_global_list)
{
int32_t i;
LLIST **sharelist = get_and_lock_sharelist();
LLIST *sharelist2 = ll_create("web-sharelist");
for(i = 0; i < CAID_KEY; i++)
{
if(sharelist[i])
{ ll_putall(sharelist2, sharelist[i]); }
}
unlock_sharelist();
struct cc_card **cardarray = get_sorted_card_copy(sharelist2, 0, &cardsize);
ll_destroy(&sharelist2);
print_cards(vars, params, cardarray, cardsize, 1, NULL, offset, apicall);
NULLFREE(cardarray);
}
else
#endif
{
struct s_client *rc = rdr->client;
struct cc_data *rcc = (rc) ? rc->cc : NULL;
if(rcc && rcc->cards)
{
struct cc_card **cardarray = get_sorted_card_copy(rcc->cards, 0, &cardsize);
print_cards(vars, params, cardarray, cardsize, 0, rdr, offset, apicall);
NULLFREE(cardarray);
}
}
}
else
{
#else
if(cs_strlen(reader_))
{
{
struct s_reader *rdr;
#endif
tpl_addVar(vars, TPLADD, "LOGHISTORY", "->");
// normal non-cccam reader
rdr = get_reader_by_label(reader_);
if(rdr)
{
struct s_client *cl = rdr->client;
if(rdr->ll_entitlements)
{
time_t now = time((time_t *)0);
struct tm start_t, end_t;
LL_ITER itr = ll_iter_create(rdr->ll_entitlements);
S_ENTITLEMENT *item;
tpl_addVar(vars, TPLAPPEND, "LOGHISTORY", "<BR><BR>New Structure:<BR>");
char tbuffer[83];
#ifdef WITH_EMU
char keyBuffer[1024];
#endif
int jsondelimiter = 0;
while((item = ll_iter_next(&itr)))
{
#ifdef WITH_EMU
if(item->isKey)
{
tpl_addVar(vars, TPLADD, "ENTSTARTDATE", "");
tpl_addVar(vars, TPLADD, "ENTENDDATE", "");
cs_hexdump(0, item->key, item->keyLength, keyBuffer, sizeof(keyBuffer));
tpl_addVar(vars, TPLADD, "ENTEXPIERED", "e_valid");
tpl_printf(vars, TPLADD, "ENTCAID", "%04X", item->caid);
tpl_printf(vars, TPLADD, "ENTPROVID", "%08X", item->provid);
tpl_addVar(vars, TPLADD, "ENTID", item->name);
tpl_addVar(vars, TPLADD, "ENTCLASS", keyBuffer);
if(item->isData) { tpl_addVar(vars, TPLADD, "ENTTYPE", "data"); }
else { tpl_addVar(vars, TPLADD, "ENTTYPE", "key"); }
tpl_addVar(vars, TPLADD, "ENTRESNAME", "");
if((strcmp(getParam(params, "hideexpired"), "1") != 0) || (item->end > now))
{ tpl_addVar(vars, TPLAPPEND, "READERENTENTRY", tpl_getTpl(vars, "ENTITLEMENTITEMBIT")); }
continue;
}
#endif
localtime_r(&item->start, &start_t);
localtime_r(&item->end, &end_t);
if(!apicall)
{ strftime(tbuffer, 30, "%Y-%m-%d", &start_t); }
else
{ strftime(tbuffer, 30, "%Y-%m-%dT%H:%M:%S%z", &start_t); }
tpl_addVar(vars, TPLADD, "ENTSTARTDATE", tbuffer);
if(!apicall)
{ strftime(tbuffer, 30, "%Y-%m-%d", &end_t); }
else
{ strftime(tbuffer, 30, "%Y-%m-%dT%H:%M:%S%z", &end_t); }
tpl_addVar(vars, TPLADD, "ENTENDDATE", tbuffer);
tpl_addVar(vars, TPLADD, "ENTEXPIERED", item->end > now ? "e_valid" : "e_expired");
tpl_printf(vars, TPLADD, "ENTCAID", "%04X", item->caid);
tpl_printf(vars, TPLADD, "ENTPROVID", "%06X", item->provid);
tpl_printf(vars, TPLADD, "ENTID", "%08X%08X", (uint32_t)(item->id >> 32), (uint32_t)item->id);
tpl_printf(vars, TPLADD, "ENTCLASS", "%08X", item->class);
tpl_addVar(vars, TPLADD, "ENTTYPE", entitlement_type[item->type]);
char *entresname;
entresname = xml_encode(vars, get_tiername((uint16_t)(item->id & 0xFFFF), item->caid, tbuffer));
if(!tbuffer[0])
{ entresname = xml_encode(vars, get_provider(item->provid, item->caid, tbuffer, sizeof(tbuffer))); }
tpl_addVar(vars, TPLADD, "ENTRESNAME", entresname);
if((strcmp(getParam(params, "hideexpired"), "1") != 0) || (item->end > now))
{ tpl_addVar(vars, TPLAPPEND, "READERENTENTRY", tpl_getTpl(vars, "ENTITLEMENTITEMBIT")); }
if(apicall==2)
{
tpl_printf(vars, TPLAPPEND, "APIENTITLEMENTLIST","%s%s",jsondelimiter?",":"", tpl_getTpl(vars, "JSONENTITLEMENTBIT"));
jsondelimiter++;
}
}
}
if(cl && cl->typ)
{ tpl_printf(vars, TPLADD, "READERTYPE", "%c", cl->typ); }
else
{ tpl_addVar(vars, TPLADD, "READERTYPE", "null"); }
tpl_addVar(vars, TPLADD, "READERNAME", xml_encode(vars, rdr->label));
int8_t i, j;
for(i = 0; i < 15; i++) { tpl_printf(vars, TPLAPPEND, "READERROM", "%c", rdr->rom[i]); }
if(rdr->hexserial[0] || rdr->hexserial[1]) { i = 0; }
else { i = 2; }
if(rdr->hexserial[6] || rdr->hexserial[7]) { j = 8; }
else { j = 6; }
for(; i < j; i++) { tpl_printf(vars, TPLAPPEND, "READERSERIAL", "%02X%s", rdr->hexserial[i], i < j - 1 ? " " : ""); }
for(i = 0; i < rdr->nprov; i++)
{
for(j = 0; j < 4; j++) { tpl_printf(vars, TPLAPPEND, "READERPROVIDS", "%02X ", rdr->prid[i][j]); }
tpl_addVar(vars, TPLAPPEND, "READERPROVIDS", i == 0 ? "(sysid)<BR>\n" : "&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;<BR>\n");
}
#ifdef READER_VIDEOGUARD
//CountryCode Vg card
char add_nds_line = 0;
if(rdr->VgCountryC[0])
{
for(i = 0; i < 3; i++) { tpl_printf(vars, TPLAPPEND, "READERCOUNTRYC", "%c", rdr->VgCountryC[i]); }
add_nds_line = 1;
}
else
{
tpl_addVar(vars, TPLADD, "READERCOUNTRYC", "n/a");
}
//regional code for Vg card
if(rdr->VgRegionC[0])
{
for(i = 0; i < 8; i++) { tpl_printf(vars, TPLAPPEND, "READER_RCODE", "%c", rdr->VgRegionC[i]); }
add_nds_line = 1;
}
else
{
tpl_addVar(vars, TPLADD, "READER_RCODE", "n/a");
}
//Pin Vg card
if(rdr->VgPin)
{
tpl_printf(vars, TPLAPPEND, "READERPIN", "%04i", rdr->VgPin);
add_nds_line = 1;
}
else
{
tpl_addVar(vars, TPLADD, "READERPIN", "n/a");
}
//Fuse Vg card
if(rdr->VgFuse)
{
tpl_printf(vars, TPLAPPEND, "READERFUSE", "%02X", rdr->VgFuse);
add_nds_line = 1;
}
if(caid_is_videoguard(rdr->caid))
{
tpl_printf(vars, TPLAPPEND, "READERPAYLOAD", "%02X %02X %02X %02X %02X %02X", rdr->VgLastPayload[0],
rdr->VgLastPayload[1], rdr->VgLastPayload[2], rdr->VgLastPayload[3], rdr->VgLastPayload[4], rdr->VgLastPayload[5]);
add_nds_line = 1;
}
//credit on Vg card
if(rdr->VgCredit)
{
tpl_printf(vars, TPLAPPEND, "READERCREDIT", "%i", rdr->VgCredit);
add_nds_line = 1;
}
else
{
tpl_addVar(vars, TPLADD, "READERCREDIT", "n/a");
}
#endif
if(rdr->card_valid_to)
{
struct tm vto_t;
char vtobuffer[30];
localtime_r(&rdr->card_valid_to, &vto_t);
strftime(vtobuffer, 30, "%Y-%m-%d", &vto_t);
tpl_addVar(vars, TPLADD, "READERCARDVALIDTO", vtobuffer);
}
else
{
tpl_addVar(vars, TPLADD, "READERCARDVALIDTO", "n/a");
}
if(rdr->irdId[0])
{
for(i = 0; i < 4; i++) { tpl_printf(vars, TPLAPPEND, "READERIRDID", "%02X ", rdr->irdId[i]); }
}
else
{
tpl_addVar(vars, TPLADD, "READERIRDID", "n/a");
}
if(rdr->card_atr_length)
for(i = 0; i < rdr->card_atr_length; i++) { tpl_printf(vars, TPLAPPEND, "READERATR", "%02X ", rdr->card_atr[i]); }
if(caid_is_seca(rdr->caid) || caid_is_viaccess(rdr->caid))
{
if(rdr->maturity == 0xF)
{
tpl_printf(vars, TPLAPPEND, "READERMATURITY", "%s ", "no limit");
}
else
{
tpl_printf(vars, TPLAPPEND, "READERMATURITY", "%d+", rdr->maturity);
}
}
else
{
tpl_printf(vars, TPLAPPEND, "READERMATURITY", "%s ", "n/a");
}
if (rdr->csystem)
tpl_addVar(vars, TPLADD, "READERCSYSTEM", rdr->csystem->desc);
#ifdef READER_VIDEOGUARD
if(add_nds_line)
{
tpl_addVar(vars, TPLADD, "ENTITLEMENTCONTENTNDS", tpl_getTpl(vars, "ENTITLEMENTBITNDS"));
}
#endif
tpl_addVar(vars, TPLADD, "ENTITLEMENTCONTENT", tpl_getTpl(vars, "ENTITLEMENTBIT"));
}
else
{
tpl_addMsg(vars, "Reader does not exist or is not started!");
}
}
}
else
{
tpl_addVar(vars, TPLADD, "ENTITLEMENTCONTENT", tpl_getTpl(vars, "ENTITLEMENTGENERICBIT"));
}
if(!apicall)
{ return tpl_getTpl(vars, "ENTITLEMENTS"); }
else
{
if(apicall==1)
{ return tpl_getTpl(vars, "APICCCAMCARDLIST"); }
else
{ return tpl_getTpl(vars, "JSONENTITLEMENTS"); }
}
}
#ifdef WEBIF_LIVELOG
static char *send_oscam_logpoll(struct templatevars * vars, struct uriparams * params)
{
uint64_t lastid = 0;
#ifdef WITH_DEBUG
tpl_addVar(vars, TPLADD, "LOG_DEBUGMENU", tpl_getTpl(vars, "LOGDEBUGMENU"));
#endif
tpl_addVar(vars, TPLADD, "LOG_SIZEMENU", tpl_getTpl(vars, "LOGSIZEMENU"));
tpl_addVar(vars, TPLADD, "TITLEADD1", "Move mouse over log-window to stop scroll");
if(strcmp(getParam(params, "lastid"), "start") == 0){
setActiveMenu(vars, MNU_LIVELOG);
return tpl_getTpl(vars, "LOGPAGE");
}
else
{
lastid = strtoull(getParam(params, "lastid"), NULL, 10);
}
char *dot = ""; //Delimiter
#ifdef WITH_DEBUG
char *debuglvl = getParam(params, "debug");
if(cs_strlen(debuglvl) > 0) {
int32_t dblvl = atoi(debuglvl);
if(cs_dblevel != dblvl) {
if(dblvl >= 0 && dblvl <= 65535) { cs_dblevel = dblvl; }
cs_log("%s debug_level=%d", "all", cs_dblevel);
}
}
tpl_printf(vars, TPLAPPEND, "DATA","%s\"debug\":\"%d\"", dot, cs_dblevel);
dot = ",";
tpl_printf(vars, TPLAPPEND, "DATA","%s\"maxdebug\":\"%d\"",dot, MAX_DEBUG_LEVELS);
#endif
if(cfg.loghistorylines == 0){
tpl_printf(vars, TPLAPPEND, "DATA","%s\"logdisabled\":\"1\"",dot);
return tpl_getTpl(vars, "POLL");
}
if(log_history)
{
LL_ITER it = ll_iter_create(log_history);
struct s_log_history *hist;
tpl_printf(vars, TPLAPPEND, "DATA", "%s\"lines\":[", dot);
dot = "";
while((hist = (struct s_log_history*)ll_iter_next(&it)))
{
char p_usr[32];
size_t pos1 = strcspn(hist->txt, "\t") + 1;
cs_strncpy(p_usr, hist->txt , pos1 > sizeof(p_usr) ? sizeof(p_usr) : pos1);
char *p_txt = hist->txt + pos1;
pos1 = strcspn(p_txt, "\n") + 1;
char str_out[pos1];
cs_strncpy(str_out, p_txt, pos1);
uint64_t id = hist->counter;
size_t b64_str_in = cs_strlen(xml_encode(vars, str_out));
size_t b64_str_out = 32 + BASE64_LENGTH(b64_str_in);
char *b64_str_out_buf;
if(!cs_malloc(&b64_str_out_buf, b64_str_out))
{ continue; }
base64_encode(xml_encode(vars, str_out), b64_str_in, b64_str_out_buf, b64_str_out);
if(id > lastid){
tpl_printf(vars, TPLAPPEND, "DATA","%s{\"id\":\"%" PRIu64 "\",\"usr\":\"%s\",\"line\":\"%s\"}",
dot,
id,
urlencode(vars, xml_encode(vars, p_usr)),
b64_str_out_buf);
dot = ","; // next in Array with leading delimiter
}
NULLFREE(b64_str_out_buf);
}
}
tpl_addVar(vars, TPLAPPEND, "DATA", "]");
return tpl_getTpl(vars, "POLL");
}
#endif
static char *send_oscam_status(struct templatevars * vars, struct uriparams * params, int32_t apicall)
{
const char *usr;
int32_t lsec, isec, chsec, con, cau = 0;
time_t now = time((time_t *)0);
struct tm lt;
int delimiter=0;
if(!apicall)
{
setActiveMenu(vars, MNU_STATUS);
if (config_enabled(WITH_LB))
tpl_addVar(vars, TPLADD, "STATUSCOL14HEAD","LB Value/");
}
if(strcmp(getParam(params, "action"), "kill") == 0)
{
char *cptr = getParam(params, "threadid");
struct s_client *cl = NULL;
if(cs_strlen(cptr) > 1)
{ sscanf(cptr, "%p", (void **)(void *)&cl); }
if(cl && is_valid_client(cl))
{
if(is_dvbapi_usr(cl->account->usr))
{
cs_log("WebIF from %s requests to kill dvbapi client %s -> ignoring!", cs_inet_ntoa(GET_IP()), cl->account->usr);
}
else
{
kill_thread(cl);
cs_log("Client %s killed by WebIF from %s", cl->account->usr, cs_inet_ntoa(GET_IP()));
}
}
}
if(strcmp(getParam(params, "action"), "resetuserstats") == 0)
{
clear_info_clients_stats();
}
if(strcmp(getParam(params, "action"), "resetreaderstats") == 0)
{
clear_info_readers_stats();
}
if(strcmp(getParam(params, "action"), "restart") == 0)
{
struct s_reader *rdr = get_reader_by_label(getParam(params, "label"));
if(rdr)
{
if(rdr->typ != R_GBOX)
{
add_job(rdr->client, ACTION_READER_RESTART, NULL, 0);
}
#ifdef MODULE_GBOX
else
{
restart_gbox_peer(rdr->label, 0, 0);
}
#endif
cs_log("Reader %s restarted by WebIF from %s", rdr->label, cs_inet_ntoa(GET_IP()));
}
}
char *debuglvl = getParam(params, "debug");
if(cs_strlen(debuglvl) > 0)
{
#ifndef WITH_DEBUG
cs_log("*** Warning: Debug Support not compiled in ***");
#else
int32_t dblvl = atoi(debuglvl);
if(dblvl >= 0 && dblvl <= 65535) { cs_dblevel = dblvl; }
cs_log("%s debug_level=%d", "all", cs_dblevel);
#endif
}
char *hide = getParam(params, "hide");
if(cs_strlen(hide) > 0)
{
struct s_client *hideidx = NULL;
sscanf(hide, "%p", (void **)(void *)&hideidx);
if(hideidx && is_valid_client(hideidx))
{ hideidx->wihidden = 1; }
}
char *hideidle = getParam(params, "hideidle");
if(cs_strlen(hideidle) > 0)
{
if(atoi(hideidle) == 2)
{
struct s_client *cl;
for(cl = first_client; cl ; cl = cl->next)
{
if(cl->typ == 's' || cl->typ == 'h' || cl->typ == 'm'){
cl->wihidden = 0;
}
}
}
else if(atoi(hideidle) == 3)
{
struct s_client *cl;
for(cl = first_client; cl ; cl = cl->next)
{
if(cl->typ == 'r'){
cl->wihidden = 0;
}
}
}
else if(atoi(hideidle) == 4)
{
struct s_client *cl;
for(cl = first_client; cl ; cl = cl->next)
{
if(cl->typ == 'p'){
cl->wihidden = 0;
}
}
}
else if(atoi(hideidle) == 5)
{
struct s_client *cl;
for(cl = first_client; cl ; cl = cl->next)
{
if(cl->typ == 'c'){
cl->wihidden = 0;
}
}
}
else
{
int32_t oldval = cfg.http_hide_idle_clients;
config_set("webif", "httphideidleclients", hideidle);
if(oldval != cfg.http_hide_idle_clients)
{
refresh_oscam(REFR_SERVER);
}
}
}
if(cfg.http_hide_idle_clients > 0) { tpl_addVar(vars, TPLADD, "HIDEIDLECLIENTSSELECTED1", "selected"); }
else { tpl_addVar(vars, TPLADD, "HIDEIDLECLIENTSSELECTED0", "selected"); }
tpl_addVar(vars, TPLADD, "SRVIDFILE", use_srvid2 ? "oscam.srvid2" : "oscam.srvid");
int32_t user_count_all = 0, user_count_shown = 0, user_count_active = 0;
int32_t reader_count_all = 0, reader_count_conn = 0, reader_count_off = 0;
int32_t proxy_count_all = 0, proxy_count_conn = 0, proxy_count_off = 0;
int32_t server_count_all = 0, server_count_shown = 0, server_count_hidden = 0;
int32_t monitor_count_all = 0, monitor_count_shown = 0;
int32_t shown;
int32_t total_readers = 0;
int32_t active_readers = 0;
int32_t disabled_readers = 0;
int32_t connected_readers = 0;
struct s_client *cl;
int8_t filtered;
cs_readlock(__func__, &readerlist_lock);
cs_readlock(__func__, &clientlist_lock);
for(cl = first_client; cl ; cl = cl->next)
{
#ifdef CS_CACHEEX_AIO
#if defined(MODULE_CCCAM) && defined(CS_CACHEEX)
struct cc_data *cc = cl->cc;
#endif
#endif
if(cl->kill) { continue; }
#ifdef CS_CACHEEX
if(get_module(cl)->listenertype != LIS_CSPUDP)
{
#endif
// Reset template variables
tpl_addVar(vars, TPLADD, "LBLRPSTRVALUE", "");
tpl_addVar(vars, TPLADD, "CLIENTLBVALUE", "");
tpl_addVar(vars, TPLADD, "LASTREADER", "");
tpl_addVar(vars, TPLADD, "CLIENTPROTO", "");
tpl_addVar(vars, TPLADD, "CLIENTDESCRIPTION", "");
tpl_addVar(vars, TPLADD, "CLIENTLASTRESPONSETIME", "");
tpl_addVar(vars, TPLADD, "CLIENTLASTRESPONSETIMEHIST", "");
tpl_addVar(vars, TPLADD, "UPICMISSING" , "");
tpl_addVar(vars, TPLADD, "ENTITLEMENTS", "");
if(cl->typ == 'c')
{ user_count_all++; }
else if(cl->typ == 'p')
{
proxy_count_all++;
if((cl->reader->typ == R_GBOX && cl->reader->card_status != CARD_INSERTED && cl->reader->card_status != NO_CARD) ||
(cl->reader->typ != R_GBOX && cl->reader->card_status != CARD_INSERTED))
{ proxy_count_off++; }
}
else if(cl->typ == 'r')
{ reader_count_all++; if(cl->reader->card_status != CARD_INSERTED) { reader_count_off++; } }
else if(cl->typ == 's' || cl->typ == 'h')
{ server_count_all++; if(cl->wihidden) {server_count_hidden++;} }
else if(cl->typ == 'm')
{ monitor_count_all++; }
shown = 0;
if(cl->wihidden != 1)
{
filtered = !(cfg.http_hide_idle_clients != 1 || cl->typ != 'c' || (now - cl->lastecm) <= cfg.hideclient_to);
if(!filtered && cfg.http_hide_type)
{
char *p = cfg.http_hide_type;
while(*p && !filtered)
{
char type = *p++;
#ifdef CS_CACHEEX
filtered = (type == cl->typ) || (type == 'x' && (cl->typ == 'p' || cl->typ == 'r') && (cl->reader && cl->reader->cacheex.mode));
#else
filtered = (type == cl->typ);
#endif
#ifdef WITH_EMU
if(type == 'e' && cl->typ == 'r' && cl->reader->typ == R_EMU) filtered = 1;
#endif
}
}
if(!filtered)
{
if(cl->typ == 'c')
{
user_count_shown++;
if(cfg.http_hide_idle_clients != 1 && cfg.hideclient_to > 0 && (now - cl->lastecm) <= cfg.hideclient_to)
{
user_count_active++;
}
}
else if(cl->typ == 's' || cl->typ == 'h')
{
server_count_shown++;
}
else if(cl->typ == 'm')
{
monitor_count_shown++;
}
else if(cl->typ == 'r')
{
reader_count_conn++;
}
else if(cl->typ == 'p')
{
proxy_count_conn++;
}
if(cl->typ == 'c' || cl->typ == 'r' || cl->typ == 'p')
{
if(cl->lastecm >= cl->login && cl->lastecm >= cl->logout) { isec = now - cl->lastecm; }
else if(cl->logout >= cl->login) { isec = now - cl->logout; }
else { isec = now - cl->login; }
}
else { isec = now - cl->last; }
shown = 1;
lsec = now - cl->login;
chsec = now - cl->lastswitch;
usr = username(cl);
if((cl->typ == 'r') || (cl->typ == 'p')) { usr = cl->reader->label; }
if(cl->dup) { con = 2; }
else if((cl->tosleep) && (now - cl->lastswitch > cl->tosleep)) { con = 1; }
else { con = 0; }
// no AU reader == 0 / AU ok == 1 / Last EMM > aulow == -1
if(cl->typ == 'c' || cl->typ == 'p' || cl->typ == 'r')
{
if((cl->typ == 'c' && ll_count(cl->aureader_list) == 0) || ((cl->typ == 'p' || cl->typ == 'r') && cl->reader->audisabled)) { cau = 0; }
else if((now - cl->lastemm) / 60 > cfg.aulow) { cau = -1; }
else { cau = 1; }
if(cau == 0)
{
tpl_addVar(vars, TPLADD, "CLIENTCAUHTTP", "OFF");
}
else
{
if(cau == -1)
{ tpl_addVar(vars, TPLADD, "CLIENTCAUHTTP", !apicall?"<A HREF=\"#\" CLASS=\"tooltip\">ON":""); }
else
{ tpl_addVar(vars, TPLADD, "CLIENTCAUHTTP", !apicall?"<A HREF=\"#\" CLASS=\"tooltip\">ACTIVE":""); }
tpl_addVar(vars, TPLAPPEND, "CLIENTCAUHTTP", !apicall?"<SPAN>":"");
if(cl->typ == 'c')
{
struct s_reader *rdr;
LL_ITER itr = ll_iter_create(cl->aureader_list);
while((rdr = ll_iter_next(&itr)))
{
if(rdr->audisabled)
{ tpl_printf(vars, TPLAPPEND, "CLIENTCAUHTTP", "(%s)<BR>", xml_encode(vars, rdr->label)); }
else
{ tpl_printf(vars, TPLAPPEND, "CLIENTCAUHTTP", "%s<BR>", xml_encode(vars, rdr->label)); }
}
}
else { tpl_addVar(vars, TPLAPPEND, "CLIENTCAUHTTP", xml_encode(vars, cl->reader->label)); }
tpl_addVar(vars, TPLAPPEND, "CLIENTCAUHTTP", !apicall?"</SPAN></A>":"");
}
}
else
{
cau = 0;
tpl_addVar(vars, TPLADD, "CLIENTCAUHTTP", "");
}
localtime_r(&cl->login, &lt);
if(cl->typ == 'c' || cl->typ == 'm')
{
if(cl->account && cl->account->description)
tpl_printf(vars, TPLADD, "CLIENTDESCRIPTION","%s(%s)",!apicall?"&#13;":"",xml_encode(vars, cl->account->description));
else
tpl_addVar(vars, TPLADD, "CLIENTDESCRIPTION", "");
}
else if(cl->typ == 'p' || cl->typ == 'r')
{
if(cl->reader && cl->reader->description)
tpl_printf(vars, TPLADD, "CLIENTDESCRIPTION","%s(%s)",!apicall?"&#13;":"",xml_encode(vars, cl->reader->description));
else
tpl_addVar(vars, TPLADD, "CLIENTDESCRIPTION", "");
}
if(!apicall)
{
tpl_addVar(vars, TPLADD, "LBL", xml_encode(vars, usr));
tpl_printf(vars, TPLADD, "CID", "%p", cl);
if(cl->typ == 'c' || cl->typ == 'm')
{
tpl_addVar(vars, TPLADD, "TARGET", "User");
tpl_addVar(vars, TPLADD, "CSIDX", tpl_getTpl(vars, "STATUSKBUTTON"));
}
else if(cl->typ == 'p')
{
tpl_addVar(vars, TPLADD, "TARGET", "Proxy");
tpl_addVar(vars, TPLADD, "LBLENC", urlencode(vars, usr));
tpl_addVar(vars, TPLADD, "CSIDX", tpl_getTpl(vars, "STATUSRBUTTON"));
}
else if(cl->typ == 'r')
{
tpl_addVar(vars, TPLADD, "TARGET", "Reader");
tpl_addVar(vars, TPLADD, "LBLENC", urlencode(vars, usr));
tpl_addVar(vars, TPLADD, "CSIDX", tpl_getTpl(vars, "STATUSRBUTTON"));
}
else if (cl->typ == 'h' || cl->typ == 's')
{
tpl_addVar(vars, TPLADD, "TARGET", "Server");
tpl_addVar(vars, TPLADD, "CSIDX", "");
}
tpl_addVar(vars, TPLADD, "HIDEIDX", tpl_getTpl(vars, "STATUSHBUTTON"));
tpl_printf(vars, TPLADD, "CSIDXPLAIN", "id_%p", cl);
}
else
{
tpl_printf(vars, TPLADD, "HIDEIDX", "%p", cl);
tpl_printf(vars, TPLADD, "CSIDX", "id_%p", cl);
}
tpl_printf(vars, TPLADD, "CLIENTTYPE", "%c", cl->typ);
tpl_printf(vars, TPLADD, "CLIENTCNR", "%d", get_threadnum(cl));
tpl_addVar(vars, TPLADD, "CLIENTUSER", xml_encode(vars, usr));
tpl_addVar(vars, TPLADD, "STATUSUSERICON", xml_encode(vars, usr));
if (cl->typ == 'c' || cl->typ == 'm')
{
tpl_addVar(vars, TPLADD, "USERNAME", xml_encode(vars, usr));
tpl_addVar(vars, TPLADD, "USERENC", urlencode(vars, usr));
}
else if (cl->typ == 'p' || cl->typ == 'r')
{
tpl_addVar(vars, TPLADD, "READERNAME", xml_encode(vars, usr));
tpl_addVar(vars, TPLADD, "READERNAMEENC", urlencode(vars, usr));
}
bool picon_shown = false;
const char *status_user_icon_tpl = NULL;
char picon_name[32];
if(cfg.http_showpicons)
{
if(picon_exists(xml_encode(vars, usr)))
{
switch (cl->typ)
{
case 'm': // Fall through
case 'c': status_user_icon_tpl = "SUSERICON"; picon_shown = true; break;
case 'p': // Fall through
case 'r': status_user_icon_tpl = "SREADERICON"; picon_shown = true; break;
}
}
else
tpl_printf(vars, TPLADD, "UPICMISSING", "%smissing icon: IC_%s.tpl",!apicall?"&#13;":"",xml_encode(vars, usr));
}
if (!picon_shown)
{
switch (cl->typ)
{
case 'm': // Fall through
case 'c': status_user_icon_tpl = "SUSER"; break;
case 'p': // Fall through
case 'r': status_user_icon_tpl = "SREADER"; break;
}
}
if (status_user_icon_tpl)
tpl_addVar(vars, TPLADD, "STATUSUSERICON", tpl_getTpl(vars, status_user_icon_tpl));
tpl_printf(vars, TPLADD, "CLIENTCAU", "%d", cau);
if(!apicall)
{
if(cl->typ == 'c' || cl->typ == 'p' || cl->typ == 'r')
{
if(cl->crypted) { tpl_addVar(vars, TPLADD, "CLIENTCRYPTED", "ON"); }
else { tpl_addVar(vars, TPLADD, "CLIENTCRYPTED", "OFF"); }
}
else { tpl_addVar(vars, TPLADD, "CLIENTCRYPTED", ""); }
}
else { tpl_printf(vars, TPLADD, "CLIENTCRYPTED", "%d", cl->crypted); }
if(cl->typ == 'r' && cl->reader && !is_network_reader(cl->reader))
{ tpl_addVar(vars, TPLADD, "CLIENTIP", "local"); }
else if(IP_ISSET(cl->ip))
{ tpl_addVar(vars, TPLADD, "CLIENTIP", cs_inet_ntoa(cl->ip)); }
else if((cl->typ == 'p' || cl->typ == 'r') && cl->reader && cl->reader->tcp_connected)
{ tpl_addVar(vars, TPLADD, "CLIENTIP", "camd.socket"); }
else if(cl->typ == 'c' && cl->login > cl->logout)
{ tpl_addVar(vars, TPLADD, "CLIENTIP", "camd.socket"); }
else
{ tpl_addVar(vars, TPLADD, "CLIENTIP", ""); }
tpl_printf(vars, TPLADD, "CLIENTPORT", "%d", cl->port);
const char *proto = client_get_proto(cl);
#ifdef CS_CACHEEX_AIO
if(cl &&
( (cl->typ == 'c' && cl->account && cl->account->cacheex.feature_bitfield)
#if defined(MODULE_CAMD35) || defined (MODULE_CAMD35_TCP)
|| (cl->c35_extmode > 1)
#endif
#ifdef MODULE_CCCAM
|| (cc && cc->extended_lg_flagged_cws)
#endif
|| (cl->typ == 'p' && cl->reader && cl->reader->cacheex.feature_bitfield))
)
{
const char *aio_suffix = " (cx-aio)";
char *new_proto;
if(cs_malloc(&new_proto, cs_strlen(proto)+cs_strlen(aio_suffix)+1)) {
new_proto[0] = '\0';
if (!cs_strncat(new_proto, (char *)proto, cs_strlen(proto)+cs_strlen(aio_suffix)+1)) {
cs_log("FIXME!");
}
if (cs_strncat(new_proto, (char *)aio_suffix, cs_strlen(proto)+cs_strlen(aio_suffix)+1)) {
webif_add_client_proto(vars, cl, (const char*)new_proto, apicall);
} else {
cs_log("FIXME!");
}
free(new_proto);
}
} else {
#endif
webif_add_client_proto(vars, cl, proto, apicall);
#ifdef CS_CACHEEX_AIO
}
#endif
if(!apicall)
{
if((cl->typ != 'p' && cl->typ != 'r') || cl->reader->card_status == CARD_INSERTED)
{
tpl_printf(vars, TPLADD, "CLIENTLOGINDATE", "%02d.%02d.%02d<BR>%02d:%02d:%02d", lt.tm_mday, lt.tm_mon + 1, lt.tm_year % 100, lt.tm_hour, lt.tm_min, lt.tm_sec);
tpl_addVar(vars, TPLADD, "CLIENTLOGINSECS", sec2timeformat(vars, lsec));
}
else
{
tpl_addVar(vars, TPLADD, "CLIENTLOGINDATE", "");
tpl_addVar(vars, TPLADD, "CLIENTLOGINSECS", "");
}
}
else
{
tpl_printf(vars, TPLADD, "CLIENTLOGINDATEFMT", "%02d.%02d.%02d %02d:%02d:%02d", lt.tm_mday, lt.tm_mon + 1, lt.tm_year % 100, lt.tm_hour, lt.tm_min, lt.tm_sec);
char tbuffer [30];
strftime(tbuffer, 30, "%Y-%m-%dT%H:%M:%S%z", &lt);
tpl_addVar(vars, TPLADD, "CLIENTLOGINDATE", tbuffer);
tpl_printf(vars, TPLADD, "CLIENTLOGINSECS", "%d", lsec);
}
//load historical values from ringbuffer
char *value = get_ecm_historystring(cl);
tpl_addVar(vars, TPLADD, "CLIENTLASTRESPONSETIMEHIST", value);
free_mk_t(value);
if((isec < cfg.hideclient_to || cfg.hideclient_to == 0 || is_dvbapi_usr(cl->account->usr)) && (cl->typ == 'c' || cl->typ == 'p' || cl->typ == 'r'))
{
if(((cl->typ == 'c')) && (cl->lastreader[0]))
{
tpl_printf(vars, TPLADD, "MSVALUE", PRINTF_LOCAL_D, cl->cwlastresptime);
if(apicall)
{
tpl_addVar(vars, TPLADD, "LASTREADER", (cl->last_caid == NO_CAID_VALUE || isec > cfg.hideclient_to ) ? "" : cl->lastreader);
tpl_addVar(vars, TPLADD, "READERNAMEENC", urlencode(vars, cl->lastreader));
}
else
{
#ifdef WITH_LB
tpl_addVar(vars, TPLADD, "LBLVALUE", xml_encode(vars, cl->lastreader));
if(strstr(cl->lastreader, " (cache)"))
{
char lastreader_tmp[cs_strlen(cl->lastreader) - 8];
tpl_addVar(vars, TPLADD, "CLIENTLBVALUE", tpl_getVar(vars, "LBLRPSTRVALUE"));
cs_strncpy(lastreader_tmp, cl->lastreader, sizeof(lastreader_tmp));
tpl_addVar(vars, TPLADD, "LBLVALUEENC", urlencode(vars, lastreader_tmp));
tpl_addVar(vars, TPLADD, "LBLVALUETITLE", xml_encode(vars, lastreader_tmp));
}
else
{
tpl_addVar(vars, TPLADD, "LBLVALUEENC", urlencode(vars, cl->lastreader));
tpl_addVar(vars, TPLADD, "LBLVALUETITLE", xml_encode(vars, cl->lastreader));
}
tpl_addVar(vars, TPLAPPEND, "CLIENTLBVALUE", tpl_getTpl(vars, "CLIENTLBLVALUEBIT"));
#else
tpl_printf(vars, TPLAPPEND, "CLIENTLBVALUE", "%s (%'d ms)", xml_encode(vars, cl->lastreader), cl->cwlastresptime);
#endif
}
if(cl->last_caid == NO_CAID_VALUE || isec > cfg.hideclient_to) tpl_addVar(vars, TPLADD, "CLIENTLBVALUE", "");
}
if(cl->last_caid != NO_CAID_VALUE || cl->last_srvid != NO_SRVID_VALUE)
{
char channame[CS_SERVICENAME_SIZE];
const char *lastprovidername;
get_servicename_or_null(cl, cl->last_srvid, cl->last_provid, cl->last_caid, channame, sizeof(channame));
if(channame[0] == '\0')
{
cs_strncpy(channame, "unknown", sizeof(channame));
}
lastprovidername = get_cl_lastprovidername(cl);
tpl_printf(vars, TPLADD, "CLIENTCAID", "%04X", cl->last_caid);
tpl_printf(vars, TPLADD, "CLIENTPROVID", "%06X", cl->last_provid);
tpl_printf(vars, TPLADD, "CLIENTSRVID", "%04X", cl->last_srvid);
tpl_printf(vars, TPLADD, "CLIENTSRVPROVIDER", "%s%s%s", (lastprovidername[0] != '\0' && strcmp(lastprovidername, " ")) ? " [" : "", xml_encode(vars, lastprovidername), (lastprovidername[0] != '\0' && strcmp(lastprovidername, " ")) ? "]" : "");
tpl_addVar(vars, TPLADD, "CLIENTSRVNAME", xml_encode(vars, channame));
tpl_printf(vars, TPLADD, "CLIENTLASTRESPONSETIME", "%d", cl->cwlastresptime ? cl->cwlastresptime : 1);
tpl_addVar(vars, TPLADD, "CLIENTSRVTYPE", cl->last_srvidptr && cl->last_srvidptr->type ? xml_encode(vars, cl->last_srvidptr->type) : "");
tpl_addVar(vars, TPLADD, "CLIENTSRVDESCRIPTION", cl->last_srvidptr && cl->last_srvidptr->desc ? xml_encode(vars, cl->last_srvidptr->desc) : "");
tpl_addVar(vars, TPLADD, "CLIENTTIMEONCHANNEL", sec2timeformat(vars, chsec));
if(cfg.http_showpicons && cl->last_srvid)
{
char picon_channame[30];
int8_t picon_ok = 0;
get_picon_servicename_or_null(cl, cl->last_srvid, cl->last_provid, cl->last_caid, picon_channame, sizeof(picon_channame));
if(picon_channame[0])
{
snprintf(picon_name, sizeof(picon_name) / sizeof(char) - 1, "%s", picon_channame);
picon_ok = picon_exists(picon_name);
if(picon_ok) tpl_addVar(vars, TPLADD, "PICONNAME", picon_name);
if(!picon_ok && picon_servicename_remve_hd(picon_channame, sizeof(picon_channame)))
{
snprintf(picon_name, sizeof(picon_name) / sizeof(char) - 1, "%s", picon_channame);
picon_ok = picon_exists(picon_name);
if(picon_ok) tpl_addVar(vars, TPLADD, "PICONNAME", picon_name);
}
}
if(!picon_ok)
{
snprintf(picon_name, sizeof(picon_name) / sizeof(char) - 1, "%04X_%06X_%04X", cl->last_caid, cl->last_provid, cl->last_srvid);
tpl_addVar(vars, TPLADD, "PICONNAME", picon_name);
picon_ok = picon_exists(picon_name);
}
if(!picon_ok)
{
snprintf(picon_name, sizeof(picon_name) / sizeof(char) - 1, "%04X_%04X", cl->last_caid, cl->last_srvid);
picon_ok = picon_exists(picon_name);
if(picon_ok) tpl_addVar(vars, TPLADD, "PICONNAME", picon_name);
}
if(!picon_ok)
{
snprintf(picon_name, sizeof(picon_name) / sizeof(char) - 1, "0000_%04X", cl->last_srvid);
picon_ok = picon_exists(picon_name);
if(picon_ok) tpl_addVar(vars, TPLADD, "PICONNAME", picon_name);
}
if(picon_ok)
{
tpl_addVar(vars, TPLADDONCE, "CURRENTPICON", tpl_getTpl(vars, "CLIENTCURRENTCHANNELPIC"));
}
else
{
tpl_addVar(vars, TPLADDONCE, "CURRENTPICON", tpl_getTpl(vars, "CLIENTCURRENTCHANNEL"));
tpl_addVar(vars, TPLADD, "PICONNAME", "");
}
}
else
{
tpl_addVar(vars, TPLADDONCE, "CURRENTPICON", tpl_getTpl(vars, "CLIENTCURRENTCHANNELBIT"));
tpl_addVar(vars, TPLADD, "PICONNAME", "0000_0000");
}
}
else
{
tpl_addVar(vars, TPLADD, "CLIENTCAID", "0000");
tpl_addVar(vars, TPLADD, "CLIENTPROVID", "000000");
tpl_printf(vars, TPLADD, "CLIENTSRVID", "0000");
}
}
else
{
tpl_addVar(vars, TPLADD, "CLIENTCAID", "0000");
tpl_addVar(vars, TPLADD, "CLIENTPROVID", "000000");
tpl_addVar(vars, TPLADD, "CLIENTSRVID", "0000");
tpl_addVar(vars, TPLADD, "CURRENTPICON", "");
tpl_addVar(vars, TPLADD, "CLIENTSRVPROVIDER", "");
tpl_addVar(vars, TPLADD, "CLIENTSRVNAME", "");
tpl_addVar(vars, TPLADD, "CLIENTSRVTYPE", "");
tpl_addVar(vars, TPLADD, "CLIENTSRVDESCRIPTION", "");
tpl_addVar(vars, TPLADD, "CLIENTLBVALUE", "");
tpl_addVar(vars, TPLADD, "CLIENTTIMEONCHANNEL", "");
}
if(!apicall)
{
tpl_addVar(vars, TPLADD, "CLIENTIDLESECS", sec2timeformat(vars, isec));
if((cl->typ != 'p' && cl->typ != 'r') || cl->reader->card_status == CARD_INSERTED)
{ tpl_addVar(vars, TPLADD, "CLIENTIDLESECSCLASS", "idlesec_normal"); }
else
{ tpl_addVar(vars, TPLADD, "CLIENTIDLESECSCLASS", "idlesec_alert"); }
}
else
{
tpl_printf(vars, TPLADD, "CLIENTIDLESECS", "%d", isec);
}
if(con == 2) { tpl_addVar(vars, TPLADD, "CLIENTCON", "Duplicate"); }
else if(con == 1) { tpl_addVar(vars, TPLADD, "CLIENTCON", "Sleep"); }
else
{
struct s_reader *rdr = cl->reader;
char *txt = "OK";
if(!rdr && (cl->typ == 'r' || cl->typ == 'p')) { txt = "UNKNOWN"; }
else if(cl->typ == 'r' || cl->typ == 'p') //reader or proxy
{
#ifdef WITH_LB
if(rdr->lbvalue)
{
tpl_printf(vars, TPLADD, "LBLRPSTRVALUE", "%d", rdr->lbvalue);
}
else
{
tpl_addVar(vars, TPLADD, "LBLRPSTRVALUE", "no data");
}
tpl_addVar(vars, TPLADD, "LBLRPVALUE", rdr->label);
tpl_addVar(vars, TPLADD, "LBLRPVALUEENC", urlencode(vars, rdr->label));
tpl_addVar(vars, TPLADD, "CLIENTLBVALUE", tpl_getTpl(vars, "CLIENTLBLVALUERP"));
#else
tpl_addVar(vars, TPLADD, "CLIENTLBVALUE", tpl_getVar(vars, "LBLRPSTRVALUE"));
#endif
switch(rdr->card_status)
{
case NO_CARD:
if(rdr->typ == R_GBOX)
{ txt = "ONL"; }
else
{ txt = "OFF"; }
break;
case UNKNOWN:
txt = "UNKNOWN";
break;
case READER_DEVICE_ERROR:
txt = "READER DEVICE ERROR";
break;
case CARD_NEED_INIT:
if(rdr->typ == R_GBOX)
{ txt = "OFFLINE"; }
#ifdef CS_CACHEEX
else if (cl->reader->cacheex.mode > 0)
{ txt = "CCcam CacheEX"; }
#endif
else
{ txt = "NEEDINIT"; }
break;
case CARD_INSERTED:
if(cl->typ == 'p')
{
if(rdr->typ == R_GBOX)
{ txt = "ONL"; }
else
{ txt = "CONNECTED"; }
}
else
{ txt = "CARDOK"; }
break;
case CARD_FAILURE:
txt = "ERROR";
break;
default:
txt = "UNDEF";
}
#ifdef MODULE_GBOX
if(rdr->typ == R_GBOX)
{
struct gbox_peer *peer = cl->gbox;
char gbx_txt[45];
memset(gbx_txt, 0, sizeof(gbx_txt));
if(!strcmp(txt, "OFFLINE"))
{
snprintf(gbx_txt, sizeof(gbx_txt), "%s | ID: %04X", txt, peer->gbox.id);
}
else
{
snprintf(gbx_txt, sizeof(gbx_txt), "%s | crd: %d | ID: %04X", txt, gbox_count_peer_cards(peer->gbox.id), peer->gbox.id);
}
txt = gbx_txt;
}
#endif
}
tpl_addVar(vars, TPLADD, "CLIENTCON", txt);
if(rdr && (cl->typ == 'r')) //reader
{
if(rdr->ll_entitlements)
{
LL_ITER itr = ll_iter_create(rdr->ll_entitlements);
S_ENTITLEMENT *ent;
uint16_t total_ent = 0;
uint16_t active_ent = 0;
struct tm end_t;
tpl_addVar(vars, TPLADD, "TMPSPAN", "<SPAN>");
while((ent = ll_iter_next(&itr)))
{
total_ent++;
if((ent->end > now) && (ent->type != 7))
{
if(active_ent) {tpl_addVar(vars, TPLAPPEND, "TMPSPAN", "<BR><BR>");}
active_ent++;
localtime_r(&ent->end, &end_t);
tpl_printf(vars, TPLAPPEND, "TMPSPAN", "%04X@%06X<BR>exp:%04d/%02d/%02d",
ent->caid, ent->provid,
end_t.tm_year + 1900, end_t.tm_mon + 1, end_t.tm_mday);
tpl_printf(vars, TPLAPPEND, "ENTITLEMENTS", "%s{\"caid\":\"%04X\",\"provid\":\"%06X\",\"exp\":\"%04d/%02d/%02d\"}",
active_ent > 1 ? ",": "",
ent->caid, ent->provid,
end_t.tm_year + 1900, end_t.tm_mon + 1, end_t.tm_mday);
}
}
tpl_printf(vars, TPLADD, "TOTENTITLEMENTS", "%d", total_ent);
if(((total_ent) && (active_ent == 0)) || (total_ent == 0))
{
tpl_addVar(vars, TPLAPPEND, "TMPSPAN", "No active entitlements found");
}
tpl_addVar(vars, TPLAPPEND, "TMPSPAN", "</SPAN>");
if(active_ent)
{
tpl_printf(vars, TPLADD, "TMP", "(%d entitlement%s)", active_ent, (active_ent != 1) ? "s" : "");
}
else
{
tpl_addVar(vars, TPLADD, "TMP", "(no entitlements)");
}
tpl_addVar(vars, TPLADD, "ENTLABEL", urlencode(vars, cl->reader->label));
tpl_addVar(vars, TPLADD, "ENTVALUE", active_ent > 0 ? "" : "1");
if (!apicall) tpl_addVar(vars, TPLAPPEND, "CLIENTCON", tpl_getTpl(vars, "FOUNDENTITLEMENTS"));
}
else
{
tpl_addVar(vars, TPLADD, "ENTLABEL", urlencode(vars, cl->reader->label));
if (!apicall) tpl_addVar(vars, TPLAPPEND, "CLIENTCON", tpl_getTpl(vars, "NOENTITLEMENTS"));
}
}
#ifdef MODULE_CCCAM
if(!apicall || apicall == 2)
{
if(rdr && (cl->typ == 'r' || cl->typ == 'p') && strncmp(proto, "cccam", 5) == 0 && rdr->tcp_connected && rdr->card_status != CARD_FAILURE)
{
struct cc_data *rcc = cl->cc;
if(rcc)
{
LLIST *cards = rcc->cards;
if(cards)
{
int32_t cnt = ll_count(cards);
int32_t locals = rcc->num_hop1;
if(!apicall)
{
tpl_printf(vars, TPLADD, "TMP", "(%d of %d card%s)", locals, cnt, (cnt > 1) ? "s" : "");
tpl_printf(vars, TPLADD, "CCCOUNT", "%d", cnt);
tpl_printf(vars, TPLADD, "CCCHOP1", "%d", rcc->num_hop1);
tpl_printf(vars, TPLADD, "CCCHOP2", "%d", rcc->num_hop2);
tpl_printf(vars, TPLADD, "CCCHOPX", "%d", rcc->num_hopx);
tpl_printf(vars, TPLADD, "CCCCURR", "%d", cl->reader->currenthops);
tpl_printf(vars, TPLADD, "CCCRES0", "%d", rcc->num_reshare0);
tpl_printf(vars, TPLADD, "CCCRES1", "%d", rcc->num_reshare1);
tpl_printf(vars, TPLADD, "CCCRES2", "%d", rcc->num_reshare2);
tpl_printf(vars, TPLADD, "CCCRESX", "%d", rcc->num_resharex);
tpl_addVar(vars, TPLADD, "TMPSPAN", tpl_getTpl(vars, "CCENTITLEMENTS"));
tpl_addVar(vars, TPLADD, "CCCLABEL", urlencode(vars, cl->reader->label));
tpl_addVar(vars, TPLADD, "CCCRESHARE", rcc->num_reshare0 > 0 ? "1" : "");
tpl_addVar(vars, TPLADD, "CCCTMP", tpl_getVar(vars, "TMP"));
tpl_addVar(vars, TPLADD, "CCCTMPSPAN", tpl_getVar(vars, "TMPSPAN"));
tpl_addVar(vars, TPLAPPEND, "CLIENTCON", tpl_getTpl(vars, "CCENTITLETOOLTIP"));
}
if (apicall == 2)
{
tpl_addVar(vars, TPLADD, "READERNAMEENC", urlencode(vars, cl->reader->label));
tpl_printf(vars, TPLADD, "ENTITLEMENTS", "{\"locals\":\"%d\",\"cccount\":\"%d\",\"ccchop1\":\"%d\",\"ccchop2\":\"%d\",\"ccchopx\":\"%d\",\"ccccurr\":\"%d\",\"cccres0\":\"%d\",\"cccres1\":\"%d\",\"cccres2\":\"%d\",\"cccresx\":\"%d\",\"cccreshare\":\"%s\"}",
locals, cnt, rcc->num_hop1, rcc->num_hop2, rcc->num_hopx, cl->reader->currenthops,
rcc->num_reshare0, rcc->num_reshare1, rcc->num_reshare2, rcc->num_resharex,
rcc->num_reshare0 > 0 ? "1" : "");
}
cs_log_dbg(D_TRACE, "Reader %s has total %d local%s hop1 %d hopx %d from total of %d card%s", cl->reader->label, locals, (locals > 1) ? "s" : "", rcc->num_hop2, rcc->num_hopx, cnt, (cnt > 1) ? "s" : "");
}
}
}
}
#endif
}
}
}
if(!apicall)
{
// select right suborder
if(cl->typ == 'c')
{
if(shown) { tpl_addVar(vars, TPLAPPEND, "CLIENTSTATUS", tpl_getTpl(vars, "CLIENTSTATUSBIT")); }
if(cfg.http_hide_idle_clients != 1 && cfg.hideclient_to > 0)
{
tpl_printf(vars, TPLADD, "UCAC", "%d", user_count_active);
tpl_printf(vars, TPLADD, "CFGH", "%d", cfg.hideclient_to);
tpl_addVar(vars, TPLADD, "CHEADADD", tpl_getTpl(vars, "CLIENTHEADLINEADD"));
}
tpl_printf(vars, TPLADD, "UCS", "%d", user_count_shown);
tpl_printf(vars, TPLADD, "UCA", "%d", user_count_all);
tpl_addVar(vars, TPLADD, "HIDEIDLE", "5");
tpl_addVar(vars, TPLADD, "SHOWHIDDEN", "User");
tpl_addVar(vars, TPLADD, "XHEAD", tpl_getTpl(vars, "CLIENTHEADLINE"));
tpl_addVar(vars, TPLADD, "CLIENTOPTIONADD", tpl_getTpl(vars, "CLIENTHEADLINEBIT"));
tpl_addVar(vars, TPLADD, "CLIENTHEADLINE", tpl_getTpl(vars, "STATUSHEADLINE"));
tpl_addVar(vars, TPLADD, "CLIENTOPTIONADD", "");
}
else if(cl->typ == 'r')
{
if(shown)
{
tpl_addVar(vars, TPLAPPEND, "READERSTATUS", tpl_getTpl(vars, "CLIENTSTATUSBIT"));
}
tpl_printf(vars, TPLADD, "RCC", "%d", reader_count_conn);
tpl_printf(vars, TPLADD, "RCA", "%d", reader_count_all);
if(reader_count_off)
{
tpl_printf(vars, TPLADD, "RCO", "%d", reader_count_all-reader_count_off);
tpl_addVar(vars, TPLADD, "RHEADADD", tpl_getTpl(vars, "CLIENTRHEADLINEADD"));
}
tpl_addVar(vars, TPLADD, "HIDEIDLE", "3");
tpl_addVar(vars, TPLADD, "SHOWHIDDEN", "Reader");
tpl_addVar(vars, TPLADD, "XHEAD", tpl_getTpl(vars, "CLIENTRHEADLINE"));
tpl_addVar(vars, TPLADD, "READERHEADLINE", tpl_getTpl(vars, "STATUSHEADLINE"));
}
else if(cl->typ == 'p')
{
if(shown)
{
tpl_addVar(vars, TPLAPPEND, "PROXYSTATUS", tpl_getTpl(vars, "CLIENTSTATUSBIT"));
}
tpl_printf(vars, TPLADD, "PCC", "%d", proxy_count_conn);
tpl_printf(vars, TPLADD, "PCA", "%d", proxy_count_all);
if(proxy_count_off)
{
tpl_printf(vars, TPLADD, "PCO", "%d", proxy_count_all-proxy_count_off);
tpl_addVar(vars, TPLADD, "PHEADADD", tpl_getTpl(vars, "CLIENTPHEADLINEADD"));
}
tpl_addVar(vars, TPLADD, "HIDEIDLE", "4");
tpl_addVar(vars, TPLADD, "SHOWHIDDEN", "Proxy");
tpl_addVar(vars, TPLADD, "XHEAD", tpl_getTpl(vars, "CLIENTPHEADLINE"));
tpl_addVar(vars, TPLADD, "PROXYHEADLINE", tpl_getTpl(vars, "STATUSHEADLINE"));
}
else if(cl->typ == 'm' || cl->typ == 's' || cl->typ == 'h')
{
if(shown)
{
tpl_addVar(vars, TPLAPPEND, "SERVERSTATUS", tpl_getTpl(vars, "CLIENTSTATUSBIT"));
}
tpl_addVar(vars, TPLADD, "HIDEIDLE", "2");
tpl_addVar(vars, TPLADD, "SHOWHIDDEN", "Server");
if(cl->typ == 's' || cl->typ == 'h')
{
tpl_printf(vars, TPLADD, "SCS", "%d", server_count_shown);
tpl_printf(vars, TPLADD, "SCA", "%d", server_count_all);
tpl_addVar(vars, TPLADD, "XHEAD", tpl_getTpl(vars, "CLIENTSHEADLINE"));
if(shown || cl->wihidden)
{
tpl_addVar(vars, TPLADD, "SERVERHEADLINE", tpl_getTpl(vars, "STATUSHEADLINE"));
}
}
else
{
tpl_addVar(vars, TPLADD, "SHOWHIDDENADD", " & Monitors");
tpl_printf(vars, TPLADD, "MCS", "%d", monitor_count_shown);
tpl_printf(vars, TPLADD, "MCA", "%d", monitor_count_all);
tpl_addVar(vars, TPLADD, "MHEADADD", tpl_getTpl(vars, "CLIENTMHEADLINE"));
tpl_addVar(vars, TPLADD, "XHEAD", tpl_getTpl(vars, "CLIENTSHEADLINE"));
tpl_addVar(vars, TPLADD, "SERVERHEADLINE", tpl_getTpl(vars, "STATUSHEADLINE"));
tpl_addVar(vars, TPLADD, "SHOWHIDDENADD", "");
}
}
}
else
{
if(shown)
{
if(apicall == 1)
{
tpl_addVar(vars, TPLAPPEND, "APISTATUSBITS", tpl_getTpl(vars, "APISTATUSBIT"));
}
if(apicall == 2)
{
tpl_addVar(vars, TPLADD, "JSONARRAYDELIMITER", delimiter?",":"");
tpl_addVar(vars, TPLAPPEND, "JSONSTATUSBITS", tpl_getTpl(vars, "JSONSTATUSBIT"));
delimiter++;
}
}
}
#ifdef CS_CACHEEX
}
#endif
}
LL_ITER itr = ll_iter_create(configured_readers);
struct s_reader *rdrr;
while((rdrr = ll_iter_next(&itr)))
{
if(rdrr->label[0] && rdrr->typ)
{
total_readers += 1;
if(rdrr->enable) { active_readers += 1; }
else { disabled_readers += 1; }
if(rdrr->tcp_connected) { connected_readers += 1; }
}
}
cs_readunlock(__func__, &clientlist_lock);
cs_readunlock(__func__, &readerlist_lock);
if(cfg.http_status_log || (apicall == 1 && strcmp(getParam(params, "appendlog"), "1") == 0))
{
if(cfg.loghistorylines && log_history)
{
LL_ITER it = ll_iter_create(log_history);
struct s_log_history *hist;
while((hist = (struct s_log_history*)ll_iter_next(&it)))
{
char p_usr[32];
size_t pos1 = strcspn(hist->txt, "\t") + 1;
cs_strncpy(p_usr, hist->txt , pos1 > sizeof(p_usr) ? sizeof(p_usr) : pos1);
char *p_txt = hist->txt + pos1;
if(!apicall)
{
if(p_txt[0]) tpl_printf(vars, TPLAPPEND, "LOGHISTORY","\t\t<SPAN CLASS=\"%s\">%s\t\t</SPAN><BR>\n", xml_encode(vars, p_usr), xml_encode(vars, p_txt));
}
else
{
tpl_addVar(vars, TPLAPPEND, "LOGHISTORY", p_txt);
}
}
}
else
{
tpl_addVar(vars, TPLADD, "LOGHISTORY", "loghistorylines is set to 0 in your configuration");
}
}
#ifdef CS_CACHEEX
char *getting = "<IMG SRC=\"image?i=ICARRL\" ALT=\"Getting\">";
char *pushing = "<IMG SRC=\"image?i=ICARRR\" ALT=\"Pushing\">";
float cachesum = first_client ? first_client->cwcacheexgot : 1;
if(cachesum < 1)
{
cachesum = 1;
}
tpl_printf(vars, TPLADD, "TOTAL_CACHEXPUSH", "%d", first_client ? first_client->cwcacheexpush : 0);
tpl_addVar(vars, TPLADD, "TOTAL_CACHEXPUSH_IMG", pushing);
tpl_printf(vars, TPLADD, "TOTAL_CACHEXGOT", "%d", first_client ? first_client->cwcacheexgot : 0);
tpl_addVar(vars, TPLADD, "TOTAL_CACHEXGOT_IMG", getting);
tpl_printf(vars, TPLADD, "TOTAL_CACHEXHIT", "%d", first_client ? first_client->cwcacheexhit : 0);
tpl_printf(vars, TPLADD, "TOTAL_CACHESIZE", "%d", cache_size());
#ifdef CS_CACHEEX_AIO
tpl_printf(vars, TPLADD, "TOTAL_CACHESIZE_LG", "%d", cache_size_lg());
#endif
tpl_printf(vars, TPLADD, "REL_CACHEXHIT", "%.2f", (first_client ? first_client->cwcacheexhit : 0) * 100 / cachesum);
tpl_addVar(vars, TPLADD, "CACHEEXSTATS", tpl_getTpl(vars, "STATUSCACHEX"));
#endif
//User info
struct s_auth *account;
int32_t total_users = 0;
int32_t disabled_users = 0;
int32_t expired_users = 0;
int32_t expired_or_disabled_users = 0;
int32_t connected_users = 0;
int32_t online_users = 0;
for(account = cfg.account; (account); account = account->next)
{
total_users++;
if(account->expirationdate && account->expirationdate < now)
{
expired_users++;
}
if(account->disabled != 0)
{
disabled_users++;
}
if((account->expirationdate && account->expirationdate < now)||account->disabled != 0)
{
expired_or_disabled_users++;
}
int32_t latestactivity = 0;
struct s_client *latestclient = NULL;
for(cl = first_client->next; cl ; cl = cl->next)
{
if(cl->account && !strcmp(cl->account->usr, account->usr))
{
if(cl->lastecm > latestactivity || cl->login > latestactivity)
{
if(cl->lastecm > cl->login) { latestactivity = cl->lastecm; }
else { latestactivity = cl->login; }
latestclient = cl;
}
}
}
if(latestclient != NULL)
{
connected_users++;
if(latestactivity > 0)
{
if((now - latestactivity) < cfg.hideclient_to)
{
if(latestclient->cwfound + latestclient->cwnot + latestclient->cwcache > 0)
{
online_users++;
}
}
}
}
}
tpl_printf(vars, TPLADD, "TOTAL_USERS", "%d", total_users);
tpl_printf(vars, TPLADD, "TOTAL_ACTIVE", "%d", total_users - expired_or_disabled_users);
tpl_printf(vars, TPLADD, "TOTAL_EXPIRED", "%d", expired_users);
tpl_printf(vars, TPLADD, "TOTAL_DISABLED", "%d", disabled_users);
tpl_printf(vars, TPLADD, "TOTAL_ONLINE", "%d", online_users);
tpl_printf(vars, TPLADD, "TOTAL_CONNECTED", "%d", connected_users);
tpl_printf(vars, TPLADD, "TOTAL_READERS", "%d", total_readers);
tpl_printf(vars, TPLADD, "TOTAL_DISABLED_READERS", "%d", disabled_readers);
tpl_printf(vars, TPLADD, "TOTAL_ACTIVE_READERS", "%d", active_readers);
tpl_printf(vars, TPLADD, "TOTAL_CONNECTED_READERS", "%d", connected_readers);
//CM info
set_ecm_info(vars);
//copy struct to p_stat_old for cpu_usage calculation
p_stat_old = p_stat_cur;
/*
* check_available Bit mapping
* mem 0 total, 1 used & free, 2 buff, cached & free incl. buff, 3 share
* swap 4 total, 5 used & free,
* proc 6 count
* cpu 7 load
* oscam 8 vsize & rssize, 9 cpu user, 10 cpu sys, 11 cpu sum, 12 cpu refreshtime
* unused 13 - 15
*/
//Memory-CPU Info for linux based systems
#if defined(__linux__)
//get actual stats
if(!get_stats_linux(getpid(),&p_stat_cur)){
if(p_stat_old.cpu_total_time != 0){
calc_cpu_usage_pct(&p_stat_cur, &p_stat_old);
}
}
else{
//something went wrong, so fill with "N/A"
p_stat_cur.check_available = 65535;
}
#else // if not linux, fill with "N/A" but probably in future gets filled also for other platforms
p_stat_cur.check_available = 65535;
#endif
set_status_info(vars, p_stat_cur);
if(cfg.http_showmeminfo || cfg.http_showuserinfo || cfg.http_showreaderinfo || cfg.http_showloadinfo || cfg.http_showecminfo || (cfg.http_showcacheexinfo && config_enabled(CS_CACHEEX)) || (cfg.http_showcacheexinfo && config_enabled(CS_CACHEEX_AIO))){
tpl_addVar(vars, TPLADD, "DISPLAYINFO", "visible");
}
else{
tpl_addVar(vars, TPLADD, "DISPLAYINFO", "hidden");
}
tpl_addVar(vars, TPLADD, "DISPLAYSYSINFO", cfg.http_showmeminfo ? "visible" : "hidden");
tpl_addVar(vars, TPLADD, "DISPLAYUSERINFO", cfg.http_showuserinfo ? "visible" : "hidden");
tpl_addVar(vars, TPLADD, "DISPLAYREADERINFO", cfg.http_showreaderinfo ? "visible" : "hidden");
tpl_addVar(vars, TPLADD, "DISPLAYLOADINFO", cfg.http_showloadinfo ?"visible" : "hidden");
tpl_addVar(vars, TPLADD, "DISPLAYECMINFO", cfg.http_showecminfo ? "visible" : "hidden");
tpl_addVar(vars, TPLADD, "DISPLAYECMINFO_READERS", cfg.http_showecminfo ? "visible" : "hidden");
if(cfg.http_showcacheexinfo == 1 && config_enabled(CS_CACHEEX))
{
if (config_enabled(CS_CACHEEX_AIO))
{
tpl_addVar(vars, TPLADD, "DISPLAYCACHEEXINFO", "hidden");
tpl_addVar(vars, TPLADD, "DISPLAYCACHEEXAIOINFO", "visible");
}
else
{
tpl_addVar(vars, TPLADD, "DISPLAYCACHEEXINFO", "visible");
tpl_addVar(vars, TPLADD, "DISPLAYCACHEEXAIOINFO", "hidden");
}
}
else{
tpl_addVar(vars, TPLADD, "DISPLAYCACHEEXINFO", "hidden");
tpl_addVar(vars, TPLADD, "DISPLAYCACHEEXAIOINFO", "hidden");
}
#ifdef WITH_DEBUG
if(cfg.http_status_log)
{
// Debuglevel Selector
int32_t i, lvl;
for(i = 0; i < MAX_DEBUG_LEVELS; i++)
{
lvl = 1 << i;
tpl_printf(vars, TPLADD, "TMPC", "DCLASS%d", lvl);
tpl_printf(vars, TPLADD, "TMPV", "DEBUGVAL%d", lvl);
if(cs_dblevel & lvl)
{
tpl_addVar(vars, TPLADD, tpl_getVar(vars, "TMPC"), "debugls");
tpl_printf(vars, TPLADD, tpl_getVar(vars, "TMPV"), "%d", cs_dblevel - lvl);
}
else
{
tpl_addVar(vars, TPLADD, tpl_getVar(vars, "TMPC"), "debugl");
tpl_printf(vars, TPLADD, tpl_getVar(vars, "TMPV"), "%d", cs_dblevel + lvl);
}
}
if(cs_dblevel == D_ALL_DUMP)
{ tpl_addVar(vars, TPLADD, "DCLASS65535", "debugls"); }
else
{ tpl_addVar(vars, TPLADD, "DCLASS65535", "debugl"); }
tpl_addVar(vars, TPLADD, "NEXTPAGE", "status.html");
tpl_addVar(vars, TPLADD, "DCLASS", "debugl"); //default
tpl_printf(vars, TPLADD, "ACTDEBUG", "%d", cs_dblevel);
#ifdef CS_CACHEEX_AIO
tpl_addVar(vars, TPLADD, "SDEBUG", tpl_getTpl(vars, "DEBUGSELECTAIO"));
#else
tpl_addVar(vars, TPLADD, "SDEBUG", tpl_getTpl(vars, "DEBUGSELECT"));
#endif
}
#endif
if(cfg.http_status_log)
tpl_addVar(vars, TPLADDONCE, "LOG_HISTORY", tpl_getTpl(vars, "LOGHISTORYBIT"));
if(apicall)
{
if(apicall == 1)
{ return tpl_getTpl(vars, "APISTATUS"); }
if(apicall == 2)
{
tpl_printf(vars, TPLADD, "UCS", "%d", user_count_shown);
tpl_printf(vars, TPLADD, "UCA", "%d", user_count_all);
if(cfg.http_hide_idle_clients != 1 && cfg.hideclient_to > 0){
tpl_printf(vars, TPLADD, "UCAC", "%d", user_count_active);
}
tpl_printf(vars, TPLADD, "CFGH", "%d", cfg.hideclient_to);
tpl_printf(vars, TPLADD, "MCS", "%d", monitor_count_shown);
tpl_printf(vars, TPLADD, "MCA", "%d", monitor_count_all);
tpl_printf(vars, TPLADD, "SCS", "%d", server_count_shown);
tpl_printf(vars, TPLADD, "SCH", "%d", server_count_hidden);
tpl_printf(vars, TPLADD, "SCA", "%d", server_count_all);
tpl_printf(vars, TPLADD, "RCC", "%d", reader_count_conn);
tpl_printf(vars, TPLADD, "RCO", "%d", reader_count_off);
tpl_printf(vars, TPLADD, "RCA", "%d", reader_count_all);
tpl_printf(vars, TPLADD, "PCC", "%d", proxy_count_conn);
tpl_printf(vars, TPLADD, "PCO", "%d", proxy_count_off);
tpl_printf(vars, TPLADD, "PCA", "%d", proxy_count_all);
tpl_printf(vars, TPLADD, "PICONENABLED", "%d", cfg.http_showpicons?1:0);
tpl_printf(vars, TPLADD, "SRVIDFILE", "%s", use_srvid2 ? "oscam.srvid2" : "oscam.srvid");
return tpl_getTpl(vars, "JSONSTATUS");
}
}
return tpl_getTpl(vars, "STATUS");
}
static char *send_oscam_services_edit(struct templatevars * vars, struct uriparams * params)
{
struct s_sidtab *sidtab, *ptr;
char label[sizeof(cfg.sidtab->label)];
int32_t i;
setActiveMenu(vars, MNU_SERVICES);
cs_strncpy(label, strtolower(getParam(params, "service")), sizeof(label));
++cfg_sidtab_generation;
for(sidtab = cfg.sidtab; sidtab != NULL && strcmp(label, sidtab->label) != 0; sidtab = sidtab->next) { ; }
if(sidtab == NULL)
{
i = 1;
while(cs_strlen(label) < 1)
{
snprintf(label, sizeof(label) / sizeof(char) - 1, "newservice%d", i);
for(sidtab = cfg.sidtab; sidtab != NULL && strcmp(label, sidtab->label) != 0; sidtab = sidtab->next) { ; }
if(sidtab != NULL) { label[0] = '\0'; }
++i;
}
if(!cs_malloc(&sidtab, sizeof(struct s_sidtab))) { return "0"; }
if(cfg.sidtab == NULL) { cfg.sidtab = sidtab; }
else
{
for(ptr = cfg.sidtab; ptr != NULL && ptr->next != NULL; ptr = ptr->next) { ; }
ptr->next = sidtab;
}
cs_strncpy((char *)sidtab->label, label, sizeof(sidtab->label));
++cfg_sidtab_generation;
tpl_addMsg(vars, "New service has been added");
// Adding is uncritical as the new service is appended to sidtabs.ok/sidtabs.no and accounts/clients/readers have zeros there
if(write_services() != 0) { tpl_addMsg(vars, "Writing services to disk failed!"); }
}
if(strcmp(getParam(params, "action"), "Save") == 0)
{
for(i = 0; i < (*params).paramcount; i++)
{
if((strcmp((*params).params[i], "action")) && (strcmp((*params).params[i], "service")))
{
chk_sidtab((*params).params[i], (*params).values[i], sidtab);
}
}
++cfg_sidtab_generation;
tpl_addMsg(vars, "Services updated");
// We don't need any refresh here as accounts/clients/readers sidtabs.ok/sidtabs.no are unaffected!
if(write_services() != 0) { tpl_addMsg(vars, "Write Config failed!"); }
for(sidtab = cfg.sidtab; sidtab != NULL && strcmp(label, sidtab->label) != 0; sidtab = sidtab->next) { ; }
}
if(sidtab)
{
tpl_addVar(vars, TPLADD, "LABEL", xml_encode(vars, sidtab->label));
tpl_addVar(vars, TPLADD, "LABELENC", urlencode(vars, sidtab->label));
#ifdef CS_CACHEEX_AIO
tpl_addVar(vars, TPLADD, "DCRCCHECKED", (sidtab->disablecrccws_only_for_exception == 1) ? "checked" : "" );
tpl_addVar(vars, TPLADD, "NWCHECKED", (sidtab->no_wait_time == 1) ? "checked" : "" );
tpl_addVar(vars, TPLADD, "LGOECHECKED", (sidtab->lg_only_exception == 1) ? "checked" : "" );
#endif
for(i = 0; i < sidtab->num_caid; i++)
{
if(i == 0) { tpl_printf(vars, TPLADD, "CAIDS", "%04X", sidtab->caid[i]); }
else { tpl_printf(vars, TPLAPPEND, "CAIDS", ",%04X", sidtab->caid[i]); }
}
for(i = 0; i < sidtab->num_provid; i++)
{
if(i == 0) { tpl_printf(vars, TPLADD, "PROVIDS", "%06X", sidtab->provid[i]); }
else { tpl_printf(vars, TPLAPPEND, "PROVIDS", ",%06X", sidtab->provid[i]); }
}
for(i = 0; i < sidtab->num_srvid; i++)
{
if(i == 0) { tpl_printf(vars, TPLADD, "SRVIDS", "%04X", sidtab->srvid[i]); }
else { tpl_printf(vars, TPLAPPEND, "SRVIDS", ",%04X", sidtab->srvid[i]); }
}
}
#ifdef CS_CACHEEX_AIO
return tpl_getTpl(vars, "SERVICEEDITAIO");
#else
return tpl_getTpl(vars, "SERVICEEDIT");
#endif
}
static void delete_from_SIDTABBITS(SIDTABBITS * orgsidtab, int32_t position, int32_t sidtablength)
{
if(*orgsidtab)
{
int32_t i;
SIDTABBITS newsidtab = 0;
for(i = 0; i < position; ++i)
{
if(*orgsidtab & ((SIDTABBITS)1 << i))
{ newsidtab |= ((SIDTABBITS)1 << i); }
}
for(; i < sidtablength; ++i)
{
if(*orgsidtab & ((SIDTABBITS)1 << (i + 1)))
{ newsidtab |= ((SIDTABBITS)1 << i); }
}
*orgsidtab = newsidtab;
}
}
static char *send_oscam_services(struct templatevars * vars, struct uriparams * params)
{
struct s_sidtab *sidtab;
char *service = getParam(params, "service");
char channame[CS_SERVICENAME_SIZE];
int32_t i, counter = 0;
setActiveMenu(vars, MNU_SERVICES);
if(strcmp(getParam(params, "action"), "delete") == 0)
{
if(cfg.http_readonly)
{
tpl_addMsg(vars, "Sorry, Webif is in readonly mode. No deletion will be made!");
}
else
{
struct s_sidtab *sidtab_prev = NULL;
int32_t sidtablength = -1;
int32_t position = 0;
// Calculate sidtablength before deletion so that updating sidtabs is faster
for(sidtab = cfg.sidtab; sidtab; sidtab = sidtab->next)
{ ++sidtablength; }
for(sidtab = cfg.sidtab; sidtab; sidtab = sidtab->next)
{
if(strcmp(sidtab->label, service) == 0)
{
struct s_auth *account;
struct s_client *cl;
struct s_reader *rdr;
if(!sidtab_prev)
{ cfg.sidtab = sidtab->next; }
else
{ sidtab_prev->next = sidtab->next; }
for(account = cfg.account; (account); account = account->next)
{
delete_from_SIDTABBITS(&account->sidtabs.ok, position, sidtablength);
delete_from_SIDTABBITS(&account->sidtabs.no, position, sidtablength);
for(cl = first_client->next; cl ; cl = cl->next)
{
if(account == cl->account)
{
cl->sidtabs.ok = account->sidtabs.ok;
cl->sidtabs.no = account->sidtabs.no;
}
}
}
LL_ITER itr = ll_iter_create(configured_readers);
while((rdr = ll_iter_next(&itr)))
{
delete_from_SIDTABBITS(&rdr->sidtabs.ok, position, sidtablength);
delete_from_SIDTABBITS(&rdr->sidtabs.no, position, sidtablength);
delete_from_SIDTABBITS(&rdr->lb_sidtabs.ok, position, sidtablength);
}
free_sidtab(sidtab);
++counter;
break;
}
sidtab_prev = sidtab;
position++;
}
if(counter > 0)
{
++cfg_sidtab_generation;
tpl_addMsg(vars, "Service has been deleted!");
if(write_services() != 0) { tpl_addMsg(vars, "Writing services to disk failed!"); }
}
else { tpl_addMsg(vars, "Sorry but the specified service doesn't exist. No deletion will be made!"); }
}
}
sidtab = cfg.sidtab;
// Show List
counter = 0;
while(sidtab != NULL)
{
tpl_addVar(vars, TPLADD, "SID", "");
tpl_printf(vars, TPLADD, "SERVICENUM", "%d", counter + 1);
if((strcmp(getParam(params, "service"), sidtab->label) == 0) && (strcmp(getParam(params, "action"), "list") == 0))
{
tpl_addVar(vars, TPLADD, "SIDCLASS", "sidlist");
tpl_addVar(vars, TPLAPPEND, "SID", "<DIV CLASS=\"sidlistclose\"><A HREF=\"services.html\">X</A></DIV>");
for(i = 0; i < sidtab->num_srvid; i++)
{
tpl_printf(vars, TPLAPPEND, "SID", "%04X : %s<BR>", sidtab->srvid[i],
xml_encode(vars, get_servicename(cur_client(), sidtab->srvid[i], sidtab->num_provid ? sidtab->provid[0] : 0,
sidtab->num_caid ? sidtab->caid[0] : 0, channame, sizeof(channame))));
}
}
else
{
tpl_addVar(vars, TPLADD, "SIDCLASS", "");
tpl_printf(vars, TPLADD, "SID", "<A HREF=\"services.html?service=%s&amp;action=list\">Show Services</A>", urlencode(vars, sidtab->label));
}
tpl_addVar(vars, TPLADD, "LABELENC", urlencode(vars, sidtab->label));
tpl_addVar(vars, TPLADD, "LABEL", xml_encode(vars, sidtab->label));
tpl_addVar(vars, TPLADD, "SIDLIST", tpl_getTpl(vars, "SERVICECONFIGSIDBIT"));
tpl_addVar(vars, TPLAPPEND, "SERVICETABS", tpl_getTpl(vars, "SERVICECONFIGLISTBIT"));
sidtab = sidtab->next;
counter++;
}
if(counter >= MAX_SIDBITS)
{
tpl_addVar(vars, TPLADD, "BTNDISABLED", "DISABLED");
tpl_addMsg(vars, "Maximum Number of Services is reached");
}
return tpl_getTpl(vars, "SERVICECONFIGLIST");
}
static char *send_oscam_savetpls(struct templatevars * vars)
{
if(cfg.http_tpl)
{
tpl_printf(vars, TPLADD, "CNT", "%d", tpl_saveIncludedTpls(cfg.http_tpl));
tpl_addVar(vars, TPLADD, "PATH", cfg.http_tpl);
}
else { tpl_addVar(vars, TPLADD, "CNT", "0"); }
return tpl_getTpl(vars, "SAVETEMPLATES");
}
static char *send_oscam_shutdown(struct templatevars * vars, FILE * f, struct uriparams * params, int8_t apicall, int8_t *keepalive, char *extraheader)
{
if(!apicall) { setActiveMenu(vars, MNU_SHUTDOWN); }
if(strcmp(strtolower(getParam(params, "action")), "shutdown") == 0)
{
*keepalive = 0;
if(!apicall)
{
char *CSS = tpl_getUnparsedTpl("CSS", 1, "");
tpl_addVar(vars, TPLADD, "STYLESHEET", CSS);
NULLFREE(CSS);
tpl_printf(vars, TPLADD, "REFRESHTIME", "%d", SHUTDOWNREFRESH);
tpl_addVar(vars, TPLADD, "REFRESH", tpl_getTpl(vars, "REFRESH"));
tpl_printf(vars, TPLADD, "SECONDS", "%d", SHUTDOWNREFRESH);
char *result = tpl_getTpl(vars, "SHUTDOWN");
send_headers(f, 200, "OK", extraheader, "text/html", 0, cs_strlen(result), NULL, 0);
webif_write(result, f);
cs_log("Shutdown requested by WebIF from %s", cs_inet_ntoa(GET_IP()));
}
else
{
tpl_addVar(vars, TPLADD, "APICONFIRMMESSAGE", "shutdown");
cs_log("Shutdown requested by XMLApi from %s", cs_inet_ntoa(GET_IP()));
}
cs_exit_oscam();
if(!apicall)
{ return "1"; }
else
{ return tpl_getTpl(vars, "APICONFIRMATION"); }
}
else if(strcmp(strtolower(getParam(params, "action")), "restart") == 0)
{
*keepalive = 0;
if(!apicall)
{
char *CSS = tpl_getUnparsedTpl("CSS", 1, "");
tpl_addVar(vars, TPLADD, "STYLESHEET", CSS);
NULLFREE(CSS);
tpl_addVar(vars, TPLADD, "REFRESHTIME", "5");
tpl_addVar(vars, TPLADD, "REFRESH", tpl_getTpl(vars, "REFRESH"));
tpl_addVar(vars, TPLADD, "SECONDS", "5");
char *result = tpl_getTpl(vars, "SHUTDOWN");
send_headers(f, 200, "OK", extraheader, "text/html", 0, cs_strlen(result), NULL, 0);
webif_write(result, f);
cs_log("Restart requested by WebIF from %s", cs_inet_ntoa(GET_IP()));
}
else
{
tpl_addVar(vars, TPLADD, "APICONFIRMMESSAGE", "restart");
cs_log("Restart requested by XMLApi from %s", cs_inet_ntoa(GET_IP()));
}
cs_restart_oscam();
if(!apicall)
{ return "1"; }
else
{ return tpl_getTpl(vars, "APICONFIRMATION"); }
}
else
{
return tpl_getTpl(vars, "PRESHUTDOWN");
}
}
static char *send_oscam_script(struct templatevars * vars, struct uriparams * params)
{
setActiveMenu(vars, MNU_SCRIPT);
tpl_printf(vars, TPLADD, "SCRIPTOPTIONS", "<option value=\"\">----select script----</option>\n");
if(!cfg.http_readonly && cfg.http_script)
{
struct dirent **namelist;
int count, i;
count = scandir(cfg.http_script, &namelist, 0, alphasort );
if( count >= 0 )
{
for( i = 0 ; i < count; i++ )
{
if(is_ext(namelist[i]->d_name, ".script") || is_ext(namelist[i]->d_name, ".sh"))
{
tpl_printf(vars, TPLAPPEND, "SCRIPTOPTIONS", "<option value=\"script.html?scriptname=%s\">%s</option>\n",namelist[i]->d_name,namelist[i]->d_name);
}
free( namelist[i] );
}
free(namelist);
}
char *scriptname = getParam(params, "scriptname");
char *scriptparam = getParam(params, "scriptparam");
char system_str[256];
struct stat s;
snprintf(system_str, sizeof(system_str), "%s/%s", cfg.http_script, scriptname);
if(!stat(system_str,&s))
{
if(s.st_mode & S_IFREG)
{
if(s.st_mode & S_IXUSR)
{
int32_t rc = -1;
FILE *fp;
char buf[256];
if((scriptparam != NULL) && (sizeof(scriptparam) > 0))
{
cs_strncat(system_str, " ", sizeof(system_str));
cs_strncat(system_str, scriptparam, sizeof(system_str));
}
fp = popen(system_str, "r");
if (fp)
{
while (fgets(buf, sizeof(buf), fp) != NULL)
{
tpl_addVar(vars, TPLAPPEND, "SCRIPTRESULTOUT", buf);
}
rc = pclose(fp)/256;
}
tpl_printf(vars, TPLAPPEND, "CODE", "returncode: %d", rc);
tpl_printf(vars, TPLADD, "SCRIPTNAME", "scriptname: %s", scriptname);
}
else
{
tpl_printf(vars, TPLADD, "SCRIPTRESULTOUT", "[Error]: Script \"%s\" not executable!", scriptname);
}
}
}
else
{
tpl_printf(vars, TPLADD, "SCRIPTRESULTOUT", "[Error]: Script \"%s\" not found!", scriptname);
}
}
return tpl_getTpl(vars, "SCRIPT");
}
static char *send_oscam_scanusb(struct templatevars * vars)
{
setActiveMenu(vars, MNU_READERS);
#if !defined(__CYGWIN__)
FILE *fp;
char line[100];
uint8_t i;
char header[11], txt[13], entry[10], bit[8], scan[12], error[120];
// key tool package match command
char *elems[] = { "USB", "lsusb", "usbutils", "Bus ", "lsusb -v | egrep '^Bus|^ *iSerial|^ *iProduct'"
, "UDEV", "/dev/serial/by-id", "udev", "<-", "find /dev/serial/by-id -type l -exec readlink -fn {} ';' -exec echo ' <- {}' \\;"
#ifdef CARDREADER_PCSC
, "PCSC", "pcsc_scan", "pcsc-tools", ":", "pcsc_scan -r"
#endif
};
for (i = 0; i < (sizeof(elems) / sizeof(elems[0])); i+=5)
{
snprintf(header, sizeof(header), "%sHEADER", elems[i]); //key
snprintf(txt, sizeof(txt), "%s Devices", elems[i]); //key
snprintf(entry, sizeof(entry), "%sENTRY", elems[i]); //key
snprintf(bit, sizeof(bit), "%sBIT", elems[i]); //key
snprintf(scan, sizeof(scan), "SCAN%sBIT", elems[i]); //key
snprintf(error, sizeof(error), "<b>%s:</b> Failed to run or %s package not installed!", elems[i + 1], elems[i + 2]); //tool + package
tpl_addVar(vars, TPLADD, header, txt);
fp = popen(elems[i + 4], "r"); //command
if(!fp || !fgets(line, sizeof(line) - 1, fp))
{
tpl_addVar(vars, TPLADD, entry, error);
tpl_addVar(vars, TPLADD, bit, tpl_getTpl(vars, scan));
}
else
{
do{
tpl_addVar(vars, TPLADD, "USBENTRYCLASS", "");
if(strstr(line, elems[i + 3])) //match
{
tpl_addVar(vars, TPLADD, entry, line);
tpl_addVar(vars, TPLADD, "USBENTRYCLASS", "CLASS=\"scanusbsubhead\"");
}
else
{
tpl_printf(vars, TPLADD, entry, "&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;%s", line);
}
tpl_addVar(vars, TPLAPPEND, bit, tpl_getTpl(vars, scan));
}
while(fgets(line, sizeof(line) - 1, fp) != NULL);
pclose(fp);
}
}
#else
tpl_addMsg(vars, "Function not supported in CYGWIN environment");
#endif
return tpl_getTpl(vars, "SCANUSB");
}
static void webif_process_logfile(struct templatevars * vars, struct uriparams * params, char *targetfile, size_t targetfile_len)
{
snprintf(targetfile, targetfile_len, "%s", cfg.logfile);
if(strcmp(getParam(params, "clear"), "logfile") == 0)
{
if(cs_strlen(targetfile) > 0)
{
FILE *file = fopen(targetfile, "w");
if (file != NULL)
{
fclose(file);
}
}
}
#ifdef WITH_DEBUG
// Debuglevel Selector
int32_t i, lvl;
for(i = 0; i < MAX_DEBUG_LEVELS; i++)
{
lvl = 1 << i;
tpl_printf(vars, TPLADD, "TMPC", "DCLASS%d", lvl);
tpl_printf(vars, TPLADD, "TMPV", "DEBUGVAL%d", lvl);
if(cs_dblevel & lvl)
{
tpl_addVar(vars, TPLADD, tpl_getVar(vars, "TMPC"), "debugls");
tpl_printf(vars, TPLADD, tpl_getVar(vars, "TMPV"), "%d", cs_dblevel - lvl);
}
else
{
tpl_addVar(vars, TPLADD, tpl_getVar(vars, "TMPC"), "debugl");
tpl_printf(vars, TPLADD, tpl_getVar(vars, "TMPV"), "%d", cs_dblevel + lvl);
}
}
if(cs_dblevel == D_ALL_DUMP)
{ tpl_addVar(vars, TPLADD, "DCLASS65535", "debugls"); }
else
{ tpl_addVar(vars, TPLADD, "DCLASS65535", "debugl"); }
tpl_addVar(vars, TPLADD, "CUSTOMPARAM", "&file=logfile");
tpl_printf(vars, TPLADD, "ACTDEBUG", "%d", cs_dblevel);
#ifdef CS_CACHEEX_AIO
tpl_addVar(vars, TPLADD, "SDEBUG", tpl_getTpl(vars, "DEBUGSELECTAIO"));
#else
tpl_addVar(vars, TPLADD, "SDEBUG", tpl_getTpl(vars, "DEBUGSELECT"));
#endif
tpl_addVar(vars, TPLADD, "NEXTPAGE", "files.html");
#endif
if(!cfg.disablelog)
{
tpl_printf(vars, TPLADD, "SWITCH", "%d", 1);
tpl_addVar(vars, TPLADD, "TEXT", "Stop Log");
tpl_addVar(vars, TPLADD, "LOGMENU", tpl_getTpl(vars, "LOGMENUDISABLELOG"));
}
else
{
tpl_printf(vars, TPLADD, "SWITCH", "%d", 0);
tpl_addVar(vars, TPLADD, "TEXT", "Start Log");
tpl_addVar(vars, TPLADD, "LOGMENU", tpl_getTpl(vars, "LOGMENUDISABLELOG"));
}
tpl_addVar(vars, TPLAPPEND, "LOGMENU", tpl_getTpl(vars, "CLEARLOG"));
return;
}
static void webif_process_userfile(struct templatevars * vars, struct uriparams * params, char *targetfile, size_t targetfile_len)
{
snprintf(targetfile, targetfile_len, "%s", cfg.usrfile);
if(strcmp(getParam(params, "clear"), "usrfile") == 0)
{
if(cs_strlen(targetfile) > 0)
{
FILE *file = fopen(targetfile, "w");
if (file)
{
fclose(file);
}
}
}
if(!cfg.disableuserfile)
{
tpl_printf(vars, TPLADD, "SWITCH", "%d", 1);
tpl_addVar(vars, TPLADD, "TEXT", "Stop Log");
tpl_addVar(vars, TPLADD, "LOGMENU", tpl_getTpl(vars, "LOGMENUONOFF"));
}
else
{
tpl_printf(vars, TPLADD, "SWITCH", "%d", 0);
tpl_addVar(vars, TPLADD, "TEXT", "Start Log");
tpl_addVar(vars, TPLADD, "LOGMENU", tpl_getTpl(vars, "LOGMENUONOFF"));
}
tpl_addVar(vars, TPLAPPEND, "LOGMENU", tpl_getTpl(vars, "CLEARUSERLOG"));
tpl_addVar(vars, TPLADD, "FFVAL", "all");
tpl_addVar(vars, TPLADD, "FILTERFORMOPTIONS", tpl_getTpl(vars, "LOGMENUFILTERFORM"));
struct s_auth *account;
for(account = cfg.account; account; account = account->next)
{
tpl_addVar(vars, TPLADD, "FFVAL", xml_encode(vars, account->usr));
tpl_addVar(vars, TPLADD, "FFSEL", strcmp(getParam(params, "filter"), account->usr) ? "" : "selected");
tpl_addVar(vars, TPLAPPEND, "FILTERFORMOPTIONS", tpl_getTpl(vars, "LOGMENUFILTERFORM"));
}
tpl_addVar(vars, TPLADD, "FILTERFORM", tpl_getTpl(vars, "FILTERFORM"));
}
enum file_types { FTYPE_CONFIG, FTYPE_VERSION, FTYPE_ANTICASC, FTYPE_LOGFILE, FTYPE_USERFILE , FTYPE_GBOX };
struct files
{
char *file;
int menu_id;
enum file_types type;
};
static char *send_oscam_files(struct templatevars * vars, struct uriparams * params, int8_t apicall)
{
bool writable = false;
const struct files *entry;
static struct files config_files[] =
{
// id are used
// new entry after last entry before first ifdef entry
// ifdef must be add to end
{ "oscam.version", MNU_CFG_FVERSION, FTYPE_VERSION }, // id 0
{ "oscam.conf", MNU_CFG_FCONF, FTYPE_CONFIG }, // id 1
{ "oscam.user", MNU_CFG_FUSER, FTYPE_CONFIG }, // id 2
{ "oscam.server", MNU_CFG_FSERVER, FTYPE_CONFIG }, // id 3
{ "oscam.srvid", MNU_CFG_FSRVID, FTYPE_CONFIG }, // id 4
{ "oscam.srvid2", MNU_CFG_FSRVID2, FTYPE_CONFIG }, // id 5
{ "logfile", MNU_CFG_FLOGFILE, FTYPE_LOGFILE }, // id 6
{ "userfile", MNU_CFG_FUSERFILE, FTYPE_USERFILE }, // id 7
{ "css_file_name", MNU_CFG_FCSS, FTYPE_CONFIG }, // id 8
{ "oscam.services", MNU_CFG_FSERVICES, FTYPE_CONFIG }, // id 9
{ "oscam.provid", MNU_CFG_FPROVID, FTYPE_CONFIG }, // id 10
{ "oscam.tiers", MNU_CFG_FTIERS, FTYPE_CONFIG }, // id 11
{ "oscam.ratelimit", MNU_CFG_FRATELIMIT,FTYPE_CONFIG }, // id 12
{ "oscam.whitelist", MNU_CFG_FWHITELIST,FTYPE_CONFIG }, // id 13
#ifdef HAVE_DVBAPI
{ "oscam.dvbapi", MNU_CFG_FDVBAPI, FTYPE_CONFIG }, // id 14
#endif
#ifdef CS_CACHEEX
{ "oscam.fakecws", MNU_CFG_FFAKECWS, FTYPE_CONFIG }, // id 15
#endif
#ifdef CS_ANTICASC
{ "anticasc", MNU_CFG_FACLOG, FTYPE_ANTICASC }, // id 16
#endif
#ifdef MODULE_SERIAL
{ "oscam.twin", MNU_CFG_FTWIN, FTYPE_CONFIG }, // id 17
#endif
#ifdef MODULE_CONSTCW
{ "constant.cw", MNU_CFG_FKEYCW, FTYPE_CONFIG }, // id 18
#endif
#ifdef MODULE_GBOX
{ "sc.info", MNU_GBX_FSCINF, FTYPE_GBOX }, // id 19
{ "share.info", MNU_GBX_FSHRINF, FTYPE_GBOX }, // id 20
{ "share.onl", MNU_GBX_FSHRONL, FTYPE_GBOX }, // id 21
{ "gbox.ver", MNU_GBX_FVERS, FTYPE_GBOX }, // id 22
{ "attack.txt", MNU_GBX_FATTACK, FTYPE_GBOX }, // id 23
{ "gsms.log", MNU_GBX_FSMSLOG, FTYPE_GBOX }, // id 24
{ "gsms.ack", MNU_GBX_FSMSACK, FTYPE_GBOX }, // id 25
{ "gsms.nack", MNU_GBX_FSMSNACK, FTYPE_GBOX }, // id 26
{ "stats.info", MNU_GBX_FSTAINF, FTYPE_GBOX }, // id 27
{ "expired.info", MNU_GBX_FEXPINF, FTYPE_GBOX }, // id 28
{ "info.log", MNU_GBX_INFOLOG, FTYPE_GBOX }, // id 29
#endif
#ifdef WITH_EMU
{ "SoftCam.Key", MNU_CFG_FSOFTCAMKEY,FTYPE_CONFIG }, // id 30
#endif
{ NULL, 0, 0 },
};
if(use_srvid2)
{
config_files[4].menu_id = MNU_CFG_FSRVID2;
config_files[5].menu_id = MNU_CFG_FSRVID;
}
if(cfg.http_css)
{
if(strchr(cfg.http_css,'/'))
config_files[8].file = strrchr(cfg.http_css, '/')+1;
else if(strchr(cfg.http_css,'\\'))
config_files[8].file = strrchr(cfg.http_css, '\\')+1;
else
config_files[8].file = cfg.http_css;
tpl_addVar(vars, TPLADD, "FILE_USER_CSS", xml_encode(vars, config_files[8].file));
}
if(!apicall) { setActiveMenu(vars, MNU_FILES); }
tpl_addVar(vars, TPLADD, "APIFILENAME", "null");
tpl_addVar(vars, TPLADD, "APIWRITABLE", "0");
if(use_srvid2)
{
tpl_printf(vars, TPLADD, "SRVID", "%s", "oscam.srvid2");
tpl_printf(vars, TPLADD, "SRVIDSUB", "%s", "oscam.srvid");
}
else
{
tpl_printf(vars, TPLADD, "SRVID", "%s", "oscam.srvid");
tpl_printf(vars, TPLADD, "SRVIDSUB", "%s", "oscam.srvid2");
}
char *stoplog = getParam(params, "stoplog");
if(cs_strlen(stoplog) > 0)
{ cs_disable_log(atoi(stoplog)); }
char *stopusrlog = getParam(params, "stopusrlog");
if(cs_strlen(stopusrlog) > 0)
{ cfg.disableuserfile = atoi(stopusrlog); }
char *debuglvl = getParam(params, "debug");
if(cs_strlen(debuglvl) > 0)
{
#ifndef WITH_DEBUG
cs_log("*** Warning: Debug Support not compiled in ***");
#else
int32_t dblvl = atoi(debuglvl);
if(dblvl >= 0 && dblvl <= 65535) { cs_dblevel = dblvl; }
cs_log("%s debug_level=%d", "all", cs_dblevel);
#endif
}
// Process config files
char *file = getParam(params, "file");
char targetfile[256] = { 0 };
int menu_id = 0;
for(entry = config_files; entry->file; entry++)
{
if(streq(file, entry->file))
{
if(!apicall) { setActiveSubMenu(vars, entry->menu_id); }
menu_id = entry->menu_id;
tpl_addVar(vars, TPLADD, "APIWRITABLE", writable ? "1" : "0");
switch(entry->type)
{
case FTYPE_CONFIG:
writable = 1;
get_config_filename(targetfile, sizeof(targetfile), entry->file);
break;
case FTYPE_VERSION:
get_tmp_dir_filename(targetfile, sizeof(targetfile), entry->file);
break;
case FTYPE_ANTICASC:
#ifdef CS_ANTICASC
if(!apicall) { snprintf(targetfile, sizeof(targetfile), "%s", ESTR(cfg.ac_logfile)); }
#endif
break;
case FTYPE_LOGFILE:
if(!apicall) { webif_process_logfile(vars, params, targetfile, sizeof(targetfile)); }
break;
case FTYPE_USERFILE:
if(!apicall) { webif_process_userfile(vars, params, targetfile, sizeof(targetfile)); }
break;
case FTYPE_GBOX:
#ifdef MODULE_GBOX
get_gbox_filename(targetfile, sizeof(targetfile), entry->file);
#endif
break;
}
tpl_addVar(vars, TPLADD, "APIFILENAME", entry->file);
break;
}
}
if(cfg.http_css)
{
tpl_addVar(vars, TPLADD, "VIEW_FILEMENUCSS", tpl_getTpl(vars, "FILEMENUCSS"));
}
if(!strstr(targetfile, "/dev/"))
{
if(strcmp(getParam(params, "action"), "Save") == 0)
{
if((cs_strlen(targetfile) > 0) /*&& (file_exists(targetfile) == 1)*/)
{
FILE *fpsave;
char *fcontent = getParam(params, "filecontent");
if((fpsave = fopen(targetfile, "w")))
{
int32_t i, lastpos = 0, len = cs_strlen(fcontent) + 1;
//write submitted file line by line to disk and remove windows linebreaks
for(i = 0; i < len; ++i)
{
char tmp = fcontent[i];
if(tmp == '\r' || tmp == '\n' || tmp == 0)
{
fcontent[i] = 0;
fprintf(fpsave, "%s%s", fcontent + lastpos, tmp == 0 ? "" : "\n");
if(tmp == '\r' && fcontent[i + 1] == '\n') { ++i; }
lastpos = i + 1;
}
}
fclose(fpsave);
tpl_addVar(vars, TPLAPPEND, "APIFILENAME", " saved!");
// Reinit on save
switch(menu_id)
{
case MNU_CFG_FSRVID:
case MNU_CFG_FSRVID2:
init_srvid();
break;
case MNU_CFG_FPROVID:
init_provid();
break;
case MNU_CFG_FUSER:
cs_accounts_chk();
break;
case MNU_CFG_FDVBAPI:
dvbapi_read_priority();
break;
case MNU_CFG_FWHITELIST:
global_whitelist_read();
break;
case MNU_CFG_FFAKECWS:
init_fakecws();
break;
default:
break;
}
}
}
}
if((cs_strlen(targetfile) > 0) && (file_exists(targetfile) == 1))
{
FILE *fp;
char buffer[256];
if((fp = fopen(targetfile, "r")) == NULL) { return "0"; }
while(fgets(buffer, sizeof(buffer), fp) != NULL)
if(!strcmp(getParam(params, "filter"), "all"))
{ tpl_addVar(vars, TPLAPPEND, "FILECONTENT", buffer); }
else if(strstr(buffer, getParam(params, "filter")))
{ tpl_addVar(vars, TPLAPPEND, "FILECONTENT", buffer); }
fclose(fp);
}
else
{
tpl_addVar(vars, TPLAPPEND, "FILECONTENT", "File does not exist or no file selected!");
}
}
else
{
tpl_addVar(vars, TPLAPPEND, "FILECONTENT", "File not valid!");
}
tpl_addVar(vars, TPLADD, "PART", file);
if(!writable)
{
tpl_addVar(vars, TPLADD, "WRITEPROTECTION", tpl_getTpl(vars, "WRITEPROTECTION"));
tpl_addVar(vars, TPLADD, "BTNDISABLED", "DISABLED");
}
if(!apicall)
{ return tpl_getTpl(vars, "FILE"); }
else
{ return tpl_getTpl(vars, "APIFILE"); }
}
static char *send_oscam_failban(struct templatevars * vars, struct uriparams * params, int8_t apicall)
{
IN_ADDR_T ip2delete;
set_null_ip(&ip2delete);
LL_ITER itr = ll_iter_create(cfg.v_list);
V_BAN *v_ban_entry;
//int8_t apicall = 0; //remove before flight
if(!apicall) { setActiveMenu(vars, MNU_FAILBAN); }
if(strcmp(getParam(params, "action"), "delete") == 0)
{
if(strcmp(getParam(params, "intip"), "all") == 0)
{
// clear whole list
while(ll_iter_next(&itr))
{
ll_iter_remove_data(&itr);
}
}
else
{
//we have a single IP
cs_inet_addr(getParam(params, "intip"), &ip2delete);
while((v_ban_entry = ll_iter_next(&itr)))
{
if(IP_EQUAL(v_ban_entry->v_ip, ip2delete))
{
ll_iter_remove_data(&itr);
break;
}
}
}
}
ll_iter_reset(&itr);
struct timeb now;
cs_ftime(&now);
while((v_ban_entry = ll_iter_next(&itr)))
{
tpl_printf(vars, TPLADD, "IPADDRESS", "%s@%d", cs_inet_ntoa(v_ban_entry->v_ip), v_ban_entry->v_port);
tpl_addVar(vars, TPLADD, "VIOLATIONUSER", v_ban_entry->info ? v_ban_entry->info : "unknown");
struct tm st ;
localtime_r(&v_ban_entry->v_time.time, &st); // fix me, we need walltime!
if(!apicall)
{
tpl_printf(vars, TPLADD, "VIOLATIONDATE", "%02d.%02d.%02d %02d:%02d:%02d",
st.tm_mday, st.tm_mon + 1,
st.tm_year % 100, st.tm_hour,
st.tm_min, st.tm_sec);
}
else
{
char tbuffer [30];
strftime(tbuffer, 30, "%Y-%m-%dT%H:%M:%S%z", &st);
tpl_addVar(vars, TPLADD, "VIOLATIONDATE", tbuffer);
}
tpl_printf(vars, TPLADD, "VIOLATIONCOUNT", "%d", v_ban_entry->v_count);
int64_t gone = comp_timeb(&now, &v_ban_entry->v_time);
if(!apicall)
{
if(!v_ban_entry->acosc_entry)
{ tpl_addVar(vars, TPLADD, "LEFTTIME", sec2timeformat(vars, (cfg.failbantime * 60) - (gone / 1000))); }
else
{ tpl_addVar(vars, TPLADD, "LEFTTIME", sec2timeformat(vars, v_ban_entry->acosc_penalty_dur - (gone / 1000))); }
}
else
{
if(!v_ban_entry->acosc_entry)
{ tpl_printf(vars, TPLADD, "LEFTTIME", "%"PRId64"", (cfg.failbantime * 60) - (gone / 1000)); }
else
{ tpl_printf(vars, TPLADD, "LEFTTIME", "%"PRId64"", v_ban_entry->acosc_penalty_dur - (gone / 1000)); }
}
tpl_addVar(vars, TPLADD, "INTIP", cs_inet_ntoa(v_ban_entry->v_ip));
if(!apicall)
{ tpl_addVar(vars, TPLAPPEND, "FAILBANROW", tpl_getTpl(vars, "FAILBANBIT")); }
else
{ tpl_addVar(vars, TPLAPPEND, "APIFAILBANROW", tpl_getTpl(vars, "APIFAILBANBIT")); }
}
if(!apicall)
{ return tpl_getTpl(vars, "FAILBAN"); }
else
{ return tpl_getTpl(vars, "APIFAILBAN"); }
}
static bool send_EMM(struct s_reader * rdr, uint16_t caid, const struct s_cardsystem *csystem, const uint8_t *emmhex, uint32_t len)
{
if(NULL != rdr && NULL != emmhex && 0 != len)
{
EMM_PACKET *emm_pack = NULL;
if(cs_malloc(&emm_pack, sizeof(EMM_PACKET)))
{
struct s_client *webif_client = cur_client();
webif_client->grp = 0xFF; /* to access to all readers */
memset(emm_pack, '\0', sizeof(EMM_PACKET));
emm_pack->client = webif_client;
emm_pack->emmlen = len;
memcpy(emm_pack->emm, emmhex, len);
emm_pack->caid[0] = (caid >> 8) & 0xFF;
emm_pack->caid[1] = caid & 0xFF;
if(csystem && csystem->get_emm_type)
{
if(!csystem->get_emm_type(emm_pack, rdr))
{
rdr_log_dbg(rdr, D_EMM, "get_emm_type() returns error");
}
}
cs_log_dbg(D_EMM, "emm is being sent to reader %s.", rdr->label);
add_job(rdr->client, ACTION_READER_EMM, emm_pack, sizeof(EMM_PACKET));
return true;
}
}
return false;
}
static bool process_single_emm(struct templatevars * vars, struct s_reader * rdr, uint16_t caid, const struct s_cardsystem *csystem, const char *ep)
{
if(NULL != vars && NULL != rdr && NULL != ep)
{
char emmdata[1025] = {'\0'}; /*1024 + '\0'*/
uint8_t emmhex[513] = {'\0'};
char buff[7] = {'\0'};
uint16_t len = 0;
cs_strncpy(emmdata, ep, sizeof(emmdata));
remove_white_chars(emmdata);
if('\0' != emmdata[0])
{
len = cs_strlen(emmdata);
tpl_addVar(vars, TPLADD, "EP", strtoupper(emmdata));
if(key_atob_l(emmdata, emmhex, len))
{
tpl_addMsg(vars, "Single EMM has not been sent due to wrong value!");
}
else
{
len /= 2;
snprintf(buff, sizeof(buff), "0x%02X", len);
tpl_addVar(vars, TPLADD, "EP", strtoupper(emmdata));
tpl_addVar(vars, TPLADD, "SIZE", buff);
if(send_EMM(rdr, caid, csystem, emmhex, len))
{
tpl_addMsg(vars, "Single EMM has been sent.");
return true;
}
}
}
}
tpl_addVar(vars, TPLADD, "SIZE", "0x00");
return false;
}
static bool process_emm_file(struct templatevars * vars, struct s_reader * rdr, uint16_t caid, const struct s_cardsystem *csystem, const char *sFilePath)
{
bool bret = false;
uint32_t fsize = 0;
uint32_t rlines = 0;
uint32_t wemms = 0;
uint32_t errsize = 0;
char numerrl[256] = {'\0'};
char buff[20] = {'\0'};
if(NULL != rdr && NULL != sFilePath && '\0' != sFilePath[0])
{
char sMessage[128] = {0};
if(true == file_exists(sFilePath))
{
FILE *fp;
if((fp = fopen(sFilePath, "r")))
{
char line[2048] = {'\0'};
uint8_t emmhex[513] = {'\0'};
uint32_t len = 0;
tpl_addMsg(vars, "EMM file has been processed.");
while(fgets(line, sizeof(line), fp))
{
++rlines;
len = cs_strlen(remove_white_chars(line));
// wrong emm
if(len > (sizeof(emmhex) * 2) ||
key_atob_l(line, emmhex, len))
{
errsize += snprintf(numerrl + errsize, sizeof(numerrl) - errsize, "%d, ", rlines);
continue;
}
len /= 2;
if(send_EMM(rdr, caid, csystem, emmhex, len))
{
++wemms;
int32_t jcount = ll_count(rdr->client->joblist);
if (jcount > 200)
{
/* Give more time to process EMMs */
cs_sleepms(1000);
}
rdr_log_dbg(rdr, D_READER, "pending emm jobs: %i, processed emms: %i", jcount, wemms);
}
}
fsize = ftell(fp);
fclose(fp);
}
else
{
snprintf(sMessage, sizeof(sMessage), "Cannot open file '%s' (errno=%d: %s)\n", sFilePath, errno, strerror(errno));
tpl_addMsg(vars, sMessage);
}
}
else
{
snprintf(sMessage, sizeof(sMessage), "FILE \"%s\" not found!", sFilePath);
tpl_addMsg(vars, sMessage);
}
bret = true;
}
snprintf(buff, sizeof(buff), "%d bytes", fsize);
tpl_addVar(vars, TPLADD, "FSIZE", buff);
snprintf(buff, sizeof(buff), "%d", rlines);
tpl_addVar(vars, TPLADD, "NUMRLINE", buff);
snprintf(buff, sizeof(buff), "%d", wemms);
tpl_addVar(vars, TPLADD, "NUMWEMM", buff);
tpl_addVar(vars, TPLADD, "ERRLINE", numerrl);
return bret;
}
static char *send_oscam_EMM_running(struct templatevars * vars, struct uriparams * params)
{
struct s_reader *rdr = NULL;
setActiveMenu(vars, MNU_READERS);
tpl_addVar(vars, TPLADD, "READER", getParam(params, "label"));
tpl_addVar(vars, TPLADD, "FNAME", getParam(params, "emmfile"));
rdr = get_reader_by_label(getParam(params, "label"));
if(rdr)
{
int32_t tcaid = dyn_word_atob(getParam(params, "emmcaid"));
uint16_t caid = (-1 != tcaid) ? (uint16_t)tcaid : 0;
char buff[7] = "";
const struct s_cardsystem *csystem = NULL;
int32_t proxy = is_cascading_reader(rdr);
if((proxy || !rdr->csystem_active) && caid) // network reader (R_CAMD35 R_NEWCAMD R_CS378X R_CCCAM)
{
if(proxy && !rdr->ph.c_send_emm)
{
tpl_addMsg(vars, "The reader does not support EMM's!");
return tpl_getTpl(vars, "EMM_RUNNING");
}
csystem = get_cardsystem_by_caid(caid);
if(!csystem)
{
rdr_log_dbg(rdr, D_EMM, "unable to find cardsystem for caid %04X", caid);
caid = 0;
}
}
else if(!proxy && rdr->csystem_active) // local active reader
{
csystem = rdr->csystem;
if(rdr->typ != R_EMU)
{
caid = rdr->caid;
}
}
if(csystem)
{
tpl_addVar(vars, TPLADD, "SYSTEM", csystem->desc);
}
else
{
tpl_addVar(vars, TPLADD, "SYSTEM", "unknown");
}
if(caid)
{
snprintf(buff, sizeof(buff), "0x%04X", caid);
tpl_addVar(vars, TPLADD, "CAID", buff);
}
else
{
tpl_addVar(vars, TPLADD, "CAID", "unknown");
}
process_single_emm(vars, rdr, caid, csystem, getParam(params, "ep"));
process_emm_file(vars, rdr, caid, csystem, getParam(params, "emmfile"));
}
else
{
char sMessage[128] = {0};
snprintf(sMessage, sizeof(sMessage), "READER \"%s\" not found!", getParam(params, "label"));
tpl_addMsg(vars, sMessage);
tpl_addVar(vars, TPLADD, "READER", "reader not found");
}
return tpl_getTpl(vars, "EMM_RUNNING");
}
static char *send_oscam_EMM(struct templatevars * vars, struct uriparams * params)
{
setActiveMenu(vars, MNU_READERS);
tpl_addVar(vars, TPLADD, "READER", getParam(params, "label"));
struct s_reader *rdr = NULL;
rdr = get_reader_by_label(getParam(params, "label"));
if(rdr && rdr->caid)
{
char buff[5] = "";
snprintf(buff, sizeof(buff), "%04X", rdr->caid);
tpl_addVar(vars, TPLADD, "CAID", buff);
if(!is_cascading_reader(rdr))
{
tpl_addVar(vars, TPLADD, "READONLY", "readonly=\"readonly\"");
}
}
FILE *fp;
struct stat sb;
char buffer[1024];
char emm_hex[1024];
char filename[128];
char targetfile[256];
char tmpstr[20];
char emm_txt[32];
char emm_title[32];
char *emm_path;
char *emm_types[] = { "unique_emm", "shared_emm", "global_emm" };
char *emm_names[] = { "RDREMMUNIQUE", "RDREMMSHARED", "RDREMMGLOBAL" };
char *emm_cfg_names[] = { "httpemmuclean", "httpemmsclean", "httpemmgclean" };
int32_t emm_max_size[] = { cfg.http_emmu_clean, cfg.http_emms_clean, cfg.http_emmg_clean };
int num_emm_types = 3;
int i;
emm_path = cfg.emmlogdir ? cfg.emmlogdir : cs_confdir;
for( i = 0 ; i < num_emm_types; i++ )
{
snprintf(filename, sizeof(filename), "%s%s%s%s", getParam(params, "label"), "_", emm_types[i], ".log");
snprintf(targetfile, sizeof(targetfile), "%s%s%s", emm_path, emm_path[cs_strlen(emm_path) - 1] == '/' ? "" : "/", filename);
snprintf(emm_txt, sizeof(emm_txt), "%s_TXT", emm_names[i]);
tpl_addVar(vars, TPLADD, emm_txt, filename);
if((fp = fopen(targetfile, "r")) != NULL)
{
stat(targetfile, &sb);
tpl_printf(vars, TPLAPPEND, emm_txt, " (Size: %'.2f kB)", (double)sb.st_size/1024);
if(emm_max_size[i]>=0)
{
snprintf(emm_title, sizeof(emm_title), "%s_TITLE", emm_names[i]);
tpl_addVar(vars, TPLADD, emm_title, "title=\"Click Line to Copy EMM in Single EMM Write Field\"");
}
if(emm_max_size[i]>0)
{
uint32_t emms=0, emm_d, emmrs=0;
char *ptr, *saveptr1 = NULL;
while(fgets(buffer, sizeof(buffer), fp) != NULL)
{
emms++;
snprintf(tmpstr, sizeof(tmpstr), "LINE_%d", emms);
tpl_addVar(vars, TPLADD, tmpstr, buffer);
}
for(emm_d=emms;emm_d>0;--emm_d)
{
snprintf(tmpstr, sizeof(tmpstr), "LINE_%d", emm_d);
if(sscanf(tpl_getVar(vars, tmpstr), "%*s %*s %*s %s", &emm_hex[0])==1)
{
if(strstr(tpl_getVar(vars, "EMM_TMP"),emm_hex)==0)
{ tpl_addVar(vars, TPLAPPEND, "EMM_TMP", tpl_getVar(vars, tmpstr)); }
tpl_addVar(vars, TPLADD, tmpstr, "");
}
}
for(ptr = strtok_r(tpl_getVar(vars, "EMM_TMP"),"\n", &saveptr1); ptr; ptr = strtok_r(NULL,"\n", &saveptr1))
{
emmrs++;
snprintf(tmpstr, sizeof(tmpstr), "LINE_%d", emmrs);
tpl_addVar(vars, TPLADD, tmpstr, ptr);
}
tpl_addVar(vars, TPLADD, "EMM_TMP", "");
tpl_printf(vars, TPLAPPEND, emm_txt, ": %'d different EMM's from a total of %'d Entries", emmrs,emms);
for(emm_d=emmrs;emm_d>0;--emm_d)
{
snprintf(tmpstr, sizeof(tmpstr), "LINE_%d", emm_d);
tpl_printf(vars, TPLAPPEND, emm_names[i], "<a class=\"tosingleemm\" href=\"#\">%s</a>\n", tpl_getVar(vars, tmpstr));
if(sb.st_size>emm_max_size[i]*1024) { tpl_printf(vars, TPLAPPEND, "EMM_TMP", "%s\n", tpl_getVar(vars, tmpstr)); }
tpl_addVar(vars, TPLADD, tmpstr, "");
}
if(sb.st_size>emm_max_size[i]*1024)
{
char orgfile[268];
int f=0;
do {
snprintf(orgfile, sizeof(orgfile), "%s.%d", targetfile, f);
f++;
} while(access(orgfile, F_OK) != -1);
if(rename(targetfile, orgfile) == 0)
{
FILE *fs = fopen(targetfile, "w");
if (fs)
{
fprintf(fs, "%s", tpl_getVar(vars, "EMM_TMP"));
fclose(fs);
}
tpl_printf(vars, TPLAPPEND, emm_txt, "<br><b>New reduced File created!</b> Size of Original File is higher as %d kB, saved to %s", emm_max_size[i], orgfile);
}
}
tpl_addVar(vars, TPLADD, "EMM_TMP", "");
}
else if (emm_max_size[i]==0)
{
while(fgets(buffer, sizeof(buffer), fp) != NULL)
{
tpl_printf(vars, TPLAPPEND, emm_names[i], "<a class=\"tosingleemm\" href=\"#\">%s</a>", buffer);
}
}
else
{
tpl_printf(vars, TPLADD, emm_names[i],"Viewing of EMM File deactivated.<br>Set %s in Config Webif to 0 or higher for viewing or filtering EMM File.", emm_cfg_names[i]);
}
fclose(fp);
}
if(strcmp(tpl_getVar(vars, emm_names[i]),"")==0) { tpl_addVar(vars, TPLADD, emm_names[i],"no saved EMM's"); }
}
return tpl_getTpl(vars, "ASKEMM");
}
#ifdef CS_CACHEEX
static uint64_t get_cacheex_node(struct s_client * cl)
{
uint64_t node = 0x00;
#if defined(MODULE_CCCAM) || defined(MODULE_CAMD35) || defined(MODULE_CAMD35_TCP)
struct s_module *module = (cl->reader ? &cl->reader->ph : get_module(cl));
#endif
#ifdef MODULE_CCCAM
if(module->num == R_CCCAM && cl->cc)
{
struct cc_data *cc = cl->cc;
memcpy(&node, cc->peer_node_id, 8);
}
else
#endif
#ifdef MODULE_CAMD35
if(module->num == R_CAMD35)
{
memcpy(&node, cl->ncd_skey, 8);
}
else
#endif
#ifdef MODULE_CAMD35_TCP
if(module->num == R_CS378X)
{
memcpy(&node, cl->ncd_skey, 8);
}
else
#endif
{}
return node;
}
static char *send_oscam_cacheex(struct templatevars * vars, struct uriparams * params, int8_t apicall)
{
if(!apicall) { setActiveMenu(vars, MNU_CACHEEX); }
if(strcmp(getParam(params, "x"), "x") == 0)
{
// avoid compilerwarning unused vars
}
char *level[] = {"NONE", "CACHE PULL", "CACHE PUSH", "REVERSE CACHE PUSH"};
char *getting = "<IMG SRC=\"image?i=ICARRL\" ALT=\"Getting\">";
char *pushing = "<IMG SRC=\"image?i=ICARRR\" ALT=\"Pushing\">";
char *rowvariable = "";
int16_t written = 0;
#ifdef CS_CACHEEX_AIO
int16_t i = 0;
#endif
struct s_client *cl;
time_t now = time((time_t *)0);
int delimiter=0;
if(!apicall)
{
if(strcmp(getParam(params, "action"), "resetallcacheexstats") == 0)
{
cacheex_clear_all_stats();
}
}
tpl_printf(vars, TPLADD, "OWN_CACHEEX_NODEID", "%" PRIu64 "X", cacheex_node_id(cacheex_peer_id));
const char *cacheex_name_link_tpl = NULL;
tpl_addVar(vars, TPLADD, "CLIENTDESCRIPTION", "");
for(cl = first_client; cl ; cl = cl->next)
{
#ifdef CS_CACHEEX_AIO
i++;
char classname[9];
snprintf(classname, 8, "class%02d", i) < 0 ? abort() : (void)0;
classname[8] = '\0';
tpl_addVar(vars, TPLADD, "CLASSNAME", classname);
#endif
if(cl->typ == 'c' && cl->account && cl->account->cacheex.mode)
{
cacheex_name_link_tpl = "SUSER";
tpl_addVar(vars, TPLADD, "TYPE", "Client");
if(!apicall)
{
tpl_addVar(vars, TPLADD, "USERNAME", xml_encode(vars, cl->account->usr));
tpl_addVar(vars, TPLADD, "USERENC", urlencode(vars, cl->account->usr));
if(cl->account->description)
{
tpl_printf(vars, TPLADD, "CLIENTDESCRIPTION","%s(%s)",!apicall?"&#13;":"",xml_encode(vars, cl->account->description));
}
else
{
tpl_addVar(vars, TPLADD, "CLIENTDESCRIPTION", "");
}
}
else
{
tpl_addVar(vars, TPLADD, "NAME", cl->account->usr);
if(cl->account->description)
{
tpl_addVar(vars, TPLADD, "CLIENTDESCRIPTION", cl->account->description);
}
else
{
tpl_addVar(vars, TPLADD, "CLIENTDESCRIPTION", "");
}
}
{ tpl_addVar(vars, TPLADD, "IP", ""); }
// --- NODEID changer detection (persistent, based on cxnodeid_changer_detected) ---
int idx = cl ? cl->idx : 0;
if(idx < 0 || idx >= WEBIF_MAX_NODES) idx = 0;
uint64_t current_node = get_cacheex_node(cl);
if(cl && cl->cxnodeid_changer_detected)
{
tpl_printf(vars, TPLADD, "NODE", "<font color=\"red\"><b>%" PRIu64 "X</b></font>", current_node);
}
else
{
tpl_printf(vars, TPLADD, "NODE", "%" PRIu64 "X", current_node);
}
// zapamiętaj nodeid w RAM (optional)
webif_last_nodeid[idx] = current_node;
// --- end detection ---
#ifdef CS_CACHEEX_AIO
tpl_printf(vars, TPLADD, "GOTLG", "%d", cl->account->cwcacheexgotlg);
tpl_printf(vars, TPLADD, "PUSHLG", "%d", cl->account->cwcacheexpushlg);
tpl_printf(vars, TPLADD, "REL_CACHEXHITGOT", "%.2f", (double)(cl->account->cwcacheexhit ? (double)cl->account->cwcacheexhit : 0) * 100 / (double)(cl->account->cwcacheexgot ? cl->account->cwcacheexgot : 1));
#endif
tpl_addVar(vars, TPLADD, "DIRECTIONIMG", (cl->account->cacheex.mode == 3) ? getting : pushing);
rowvariable = "TABLECLIENTROWS";
written = 1;
}
else if((cl->typ == 'p' || cl->typ == 'r') && (cl->reader && cl->reader->cacheex.mode))
{
cacheex_name_link_tpl = "SREADER";
tpl_addVar(vars, TPLADD, "TYPE", "Reader");
if(!apicall)
{
tpl_addVar(vars, TPLADD, "READERNAME", xml_encode(vars, cl->reader->label));
tpl_addVar(vars, TPLADD, "READERNAMEENC", urlencode(vars, cl->reader->label));
if(cl->reader->description)
{
tpl_printf(vars, TPLADD, "CLIENTDESCRIPTION","%s(%s)",!apicall?"&#13;":"",xml_encode(vars, cl->reader->description));
}
else
{
tpl_addVar(vars, TPLADD, "CLIENTDESCRIPTION", "");
}
}
else
{
tpl_addVar(vars, TPLADD, "NAME", cl->reader->label);
if(cl->reader->description)
{
tpl_addVar(vars, TPLADD, "CLIENTDESCRIPTION", cl->reader->description);
}
else
{
tpl_addVar(vars, TPLADD, "CLIENTDESCRIPTION", "");
}
}
if(IP_ISSET(cl->ip))
{ tpl_addVar(vars, TPLADD, "IP", cs_inet_ntoa(cl->ip)); }
else if(cl->reader && cl->reader->tcp_connected)
{ tpl_addVar(vars, TPLADD, "IP", "camd.socket"); }
else
{ tpl_addVar(vars, TPLADD, "IP", ""); }
uint64_t current_node = get_cacheex_node(cl);
if(cl && cl->cxnodeid_changer_detected)
{
tpl_printf(vars, TPLADD, "NODE", "<font color=\"red\"><b>%" PRIu64 "X</b></font>", current_node);
}
else
{
tpl_printf(vars, TPLADD, "NODE", "%" PRIu64 "X", current_node);
}
tpl_addVar(vars, TPLADD, "LEVEL", level[cl->reader->cacheex.mode]);
tpl_printf(vars, TPLADD, "PUSH", "%d", cl->cwcacheexpush);
tpl_printf(vars, TPLADD, "CWCINFO", "%d", cl->cwc_info);
tpl_printf(vars, TPLADD, "GOT", "%d", cl->cwcacheexgot);
tpl_printf(vars, TPLADD, "CWCINFO", "%d", cl->cwc_info);
tpl_printf(vars, TPLADD, "HIT", "%d", cl->cwcacheexhit);
tpl_printf(vars, TPLADD, "ERR", "%d", cl->cwcacheexerr);
tpl_printf(vars, TPLADD, "ERRCW", "%d", cl->cwcacheexerrcw);
#ifdef CS_CACHEEX_AIO
tpl_printf(vars, TPLADD, "GOTLG", "%d", cl->cwcacheexgotlg);
tpl_printf(vars, TPLADD, "PUSHLG", "%d", cl->cwcacheexpushlg);
tpl_printf(vars, TPLADD, "REL_CACHEXHITGOT", "%.2f", (double)(cl->cwcacheexhit ? (double)cl->cwcacheexhit : 0) * 100 / (double)(cl->cwcacheexgot ? cl->cwcacheexgot : 1));
#endif
tpl_addVar(vars, TPLADD, "DIRECTIONIMG", (cl->reader->cacheex.mode == 3) ? pushing : getting);
rowvariable = "TABLEREADERROWS";
written = 1;
}
else if(get_module(cl)->listenertype == LIS_CSPUDP)
{
cacheex_name_link_tpl = "SREADER";
tpl_addVar(vars, TPLADD, "TYPE", "csp");
if(!apicall)
{
tpl_addVar(vars, TPLADD, "READERNAME", "csp");
tpl_addVar(vars, TPLADD, "READERNAMEENC", "csp");
}
else
{
tpl_addVar(vars, TPLADD, "NAME", "csp");
}
if(IP_ISSET(cl->ip))
{ tpl_addVar(vars, TPLADD, "IP", cs_inet_ntoa(cl->ip)); }
else if(cl->login > cl->logout)
{ tpl_addVar(vars, TPLADD, "IP", "camd.socket"); }
else
{ tpl_addVar(vars, TPLADD, "IP", ""); }
tpl_addVar(vars, TPLADD, "NODE", "csp");
if(cl->cwcacheexping)
{
tpl_printf(vars, TPLADD, "LEVEL", "csp (%d ms)", cl->cwcacheexping);
}
else
{
tpl_addVar(vars, TPLADD, "LEVEL", "csp");
}
tpl_printf(vars, TPLADD, "PUSH", "%d", cl->cwcacheexpush);
tpl_printf(vars, TPLADD, "GOT", "%d", cl->cwcacheexgot);
tpl_printf(vars, TPLADD, "HIT", "%d", cl->cwcacheexhit);
tpl_printf(vars, TPLADD, "ERR", "%d", cl->cwcacheexerr);
tpl_printf(vars, TPLADD, "ERRCW", "%d", cl->cwcacheexerrcw);
#ifdef CS_CACHEEX_AIO
tpl_printf(vars, TPLADD, "GOTLG", "%d", cl->cwcacheexgotlg);
tpl_printf(vars, TPLADD, "PUSHLG", "%d", cl->cwcacheexpushlg);
tpl_printf(vars, TPLADD, "REL_CACHEXHITGOT", "%.2f", (double)(cl->cwcacheexhit ? (double)cl->cwcacheexhit : 0) * 100 / (double)(cl->cwcacheexgot ? cl->cwcacheexgot : 1));
#endif
tpl_addVar(vars, TPLADD, "DIRECTIONIMG", getting);
rowvariable = "TABLECLIENTROWS";
written = 1;
}
if(written)
{
if(!apicall)
{
tpl_addVar(vars, TPLADD, "NAME", tpl_getTpl(vars, cacheex_name_link_tpl));
}
#ifdef CS_CACHEEX_AIO
tpl_addVar(vars, TPLAPPEND, rowvariable, tpl_getTpl(vars, "CACHEEXAIOTABLEROW"));
#else
tpl_addVar(vars, TPLAPPEND, rowvariable, tpl_getTpl(vars, "CACHEEXTABLEROW"));
#endif
if(cl->ll_cacheex_stats)
{
LL_ITER itr = ll_iter_create(cl->ll_cacheex_stats);
S_CACHEEX_STAT_ENTRY *cacheex_stats_entry;
while((cacheex_stats_entry = ll_iter_next(&itr)))
{
tpl_addVar(vars, TPLADD, "DIRECTIONIMG", "");
if(now - cacheex_stats_entry->cache_last < 20)
{ tpl_addVar(vars, TPLADD, "TYPE", cacheex_stats_entry->cache_direction == 0 ? pushing : getting); }
else
{ tpl_addVar(vars, TPLADD, "TYPE", ""); }
tpl_printf(vars, TPLADD, "NAME", "%04X@%06X:%04X", cacheex_stats_entry->cache_caid,
cacheex_stats_entry->cache_prid,
cacheex_stats_entry->cache_srvid);
if(cacheex_stats_entry->cache_direction == 0)
{
tpl_printf(vars, TPLADD, "PUSH", "%d", cacheex_stats_entry->cache_count);
#ifdef CS_CACHEEX_AIO
tpl_printf(vars, TPLADD, "PUSHLG", "%d", cacheex_stats_entry->cache_count_lg);
#endif
tpl_addVar(vars, TPLADD, "GOT", "");
}
else
{
tpl_printf(vars, TPLADD, "GOT", "%d", cacheex_stats_entry->cache_count);
#ifdef CS_CACHEEX_AIO
tpl_printf(vars, TPLADD, "GOTLG", "%d", cacheex_stats_entry->cache_count_lg);
#endif
tpl_addVar(vars, TPLADD, "PUSH", "");
}
tpl_addVar(vars, TPLADD, "HIT", "");
char channame[CS_SERVICENAME_SIZE];
char *lastchan = xml_encode(vars, get_servicename(cl, cacheex_stats_entry->cache_srvid, cacheex_stats_entry->cache_prid, cacheex_stats_entry->cache_caid, channame, sizeof(channame)));
tpl_addVar(vars, TPLADD, "LEVEL", lastchan);
if (apicall == 2)
{
tpl_printf(vars, TPLADD, "JSONDELIMITER", "%s", (delimiter > 1)?",":"");
#ifdef CS_CACHEEX_AIO
tpl_addVar(vars, TPLAPPEND, "JSONCACHEEXBITS", tpl_getTpl(vars, "JSONCACHEEXAIOBIT"));
#else
tpl_addVar(vars, TPLAPPEND, "JSONCACHEEXBITS", tpl_getTpl(vars, "JSONCACHEEXBIT"));
#endif
delimiter++;
}
else
{
#ifdef CS_CACHEEX_AIO
tpl_addVar(vars, TPLAPPEND, rowvariable, tpl_getTpl(vars, "CACHEEXAIOTABLEROWSTATS"));
#else
tpl_addVar(vars, TPLAPPEND, rowvariable, tpl_getTpl(vars, "CACHEEXTABLEROW"));
#endif
}
}
}
written = 0;
}
}
float cachesum = first_client ? first_client->cwcacheexgot : 1;
if(cachesum < 1)
{
cachesum = 1;
}
tpl_printf(vars, TPLADD, "TOTAL_CACHEXPUSH", PRINTF_LOCAL_D, first_client ? first_client->cwcacheexpush : 0);
tpl_addVar(vars, TPLADD, "TOTAL_CACHEXPUSH_IMG", pushing);
tpl_printf(vars, TPLADD, "TOTAL_CACHEXGOT", PRINTF_LOCAL_D, first_client ? first_client->cwcacheexgot : 0);
tpl_addVar(vars, TPLADD, "TOTAL_CACHEXGOT_IMG", getting);
tpl_printf(vars, TPLADD, "TOTAL_CACHEXHIT", PRINTF_LOCAL_D, first_client ? first_client->cwcacheexhit : 0);
tpl_printf(vars, TPLADD, "TOTAL_CACHESIZE", "%d", cache_size());
#ifdef CS_CACHEEX_AIO
tpl_printf(vars, TPLADD, "TOTAL_CACHESIZE_LG", "%d", cache_size_lg());
#endif
tpl_printf(vars, TPLADD, "REL_CACHEXHIT", "%.2f", (first_client ? first_client->cwcacheexhit : 0) * 100 / cachesum);
if(!apicall)
{
#ifdef CS_CACHEEX_AIO
return tpl_getTpl(vars, "CACHEEXAIOPAGE");
#else
return tpl_getTpl(vars, "CACHEEXPAGE");
#endif
}
else
{
return tpl_getTpl(vars, "JSONCACHEEX");
}
}
#endif
#ifdef WEBIF_WIKI
static char *send_oscam_wiki(struct templatevars *vars, struct uriparams *params)
{
const char *config = getParam(params, "config");
const char *section = getParam(params, "section");
const char *param = getParam(params, "param");
if(!config || !param || !config[0] || !param[0])
{
return tpl_getTpl(vars, "WIKIERROR");
}
const char *text = wiki_get_help(config, section, param);
if(text)
{
tpl_addVar(vars, TPLADD, "WIKIPARAM", param);
tpl_addVar(vars, TPLADD, "WIKICONFIG", config);
tpl_addVar(vars, TPLADD, "WIKITEXT", json_encode(vars, text));
return tpl_getTpl(vars, "WIKIJSON");
}
else
{
tpl_addVar(vars, TPLADD, "WIKIPARAM", param);
tpl_addVar(vars, TPLADD, "WIKICONFIG", config);
return tpl_getTpl(vars, "WIKINOTFOUND");
}
}
static char *send_oscam_wiki_status(struct templatevars *vars, struct uriparams *params)
{
const char *config = getParam(params, "config");
if(!config || !config[0])
{
return tpl_getTpl(vars, "WIKIERROR");
}
/* Build JSON object with status for all params in this config */
char json_buf[8192];
int pos = 0;
int32_t i, count = wiki_count();
bool first = true;
pos += snprintf(json_buf + pos, sizeof(json_buf) - pos, "{");
#ifdef COMPRESSED_WIKI
/* For compressed wiki, we iterate through entries but access via decompressed data */
const struct wiki_entry *entries = wiki_get_entries();
char *wiki_data = wiki_get_decompressed_data();
if(wiki_data)
{
for(i = 0; i < count && pos < (int)sizeof(json_buf) - 100; i++)
{
const char *e_config = wiki_data + entries[i].config_ofs;
const char *e_param = wiki_data + entries[i].param_ofs;
if(strcmp(e_config, config) != 0)
{ continue; }
/* Section filter is optional - include all params from this config */
if(!first)
{ pos += snprintf(json_buf + pos, sizeof(json_buf) - pos, ","); }
first = false;
const char *status_str = "ok";
if(entries[i].status == 1) status_str = "review";
else if(entries[i].status == 2) status_str = "missing";
pos += snprintf(json_buf + pos, sizeof(json_buf) - pos, "\"%s\":\"%s\"",
e_param, status_str);
}
}
#else
const struct wiki_entry *entries = wiki_get_entries();
for(i = 0; i < count && pos < (int)sizeof(json_buf) - 100; i++)
{
if(strcmp(entries[i].config, config) != 0)
{ continue; }
/* Section filter is optional - include all params from this config */
if(!first)
{ pos += snprintf(json_buf + pos, sizeof(json_buf) - pos, ","); }
first = false;
const char *status_str = "ok";
if(entries[i].status == 1) status_str = "review";
else if(entries[i].status == 2) status_str = "missing";
pos += snprintf(json_buf + pos, sizeof(json_buf) - pos, "\"%s\":\"%s\"",
entries[i].param, status_str);
}
#endif
pos += snprintf(json_buf + pos, sizeof(json_buf) - pos, "}");
tpl_addVar(vars, TPLADD, "WIKISTATUS", json_buf);
return tpl_getTpl(vars, "WIKISTATUSJSON");
}
#endif
static char *send_oscam_api(struct templatevars * vars, FILE * f, struct uriparams * params, int8_t *keepalive, int8_t apicall, char *extraheader)
{
if(strcmp(getParam(params, "part"), "status") == 0)
{
return send_oscam_status(vars, params, apicall);
}
else if(strcmp(getParam(params, "part"), "userstats") == 0)
{
return send_oscam_user_config(vars, params, apicall);
}
else if(strcmp(getParam(params, "part"), "failban") == 0)
{
return send_oscam_failban(vars, params, apicall);
}
#ifdef CS_CACHEEX
else if(strcmp(getParam(params, "part"), "cacheex") == 0)
{
return send_oscam_cacheex(vars, params, apicall);
}
#endif
else if(strcmp(getParam(params, "part"), "files") == 0)
{
return send_oscam_files(vars, params, apicall);
}
else if(strcmp(getParam(params, "part"), "readerlist") == 0)
{
return send_oscam_reader(vars, params, apicall);
}
else if(strcmp(getParam(params, "part"), "serverconfig") == 0)
{
//Send Errormessage
tpl_addVar(vars, TPLADD, "APIERRORMESSAGE", "serverconfig not yet avail");
return tpl_getTpl(vars, "APIERROR");
}
else if(strcmp(getParam(params, "part"), "userconfig") == 0)
{
if(((strcmp(getParam(params, "action"), "Save") == 0) ||
(strcmp(getParam(params, "action"), "Save As") == 0)) && cfg.http_readonly == 1)
{
//Send Errormessage
tpl_addVar(vars, TPLADD, "APIERRORMESSAGE", "API is in readonly mode");
return tpl_getTpl(vars, "APIERROR");
}
else
{
struct s_auth *account = get_account_by_name(getParam(params, "user"));
if(!account && strcmp(getParam(params, "action"), "Save"))
{
//Send Errormessage
tpl_addVar(vars, TPLADD, "APIERRORMESSAGE", "user not exist");
return tpl_getTpl(vars, "APIERROR");
}
else
{
return send_oscam_user_config_edit(vars, params, apicall);
}
}
}
else if(strcmp(getParam(params, "part"), "entitlement") == 0)
{
if(strcmp(getParam(params, "label"), ""))
{
struct s_reader *rdr = get_reader_by_label(getParam(params, "label"));
if(rdr)
{
if(rdr->enable == 1)
{
return send_oscam_entitlement(vars, params, apicall);
}
else
{
//Send Errormessage
tpl_addVar(vars, TPLADD, "APIERRORMESSAGE", "no cccam reader or disabled");
return tpl_getTpl(vars, "APIERROR");
}
}
else
{
//Send Errormessage
tpl_addVar(vars, TPLADD, "APIERRORMESSAGE", "reader not exist");
return tpl_getTpl(vars, "APIERROR");
}
}
else
{
//Send Errormessage
tpl_addVar(vars, TPLADD, "APIERRORMESSAGE", "no reader selected");
return tpl_getTpl(vars, "APIERROR");
}
}
else if(strcmp(getParam(params, "part"), "ecmhistory") == 0)
{
int32_t isec;
int32_t shown;
time_t now = time((time_t *)0);
const char *usr;
struct s_client *cl;
for(cl = first_client; cl ; cl = cl->next)
{
if(cl->wihidden != 1)
{
isec = now - cl->lastecm;
usr = username(cl);
shown = 0;
if(strcmp(getParam(params, "label"), "") == 0)
{
if(strcmp(getParam(params, "type"), "servers") == 0)
{
if(cl->typ == 'p' || cl->typ == 'r')
{ shown = 1; }
}
else if(strcmp(getParam(params, "type"), "users") == 0)
{
if(cl->typ == 'c')
{ shown = 1; }
}
else
{
shown = 1;
}
}
else if(strcmp(getParam(params, "label"), usr) == 0)
{
shown = 1;
}
if(shown == 1)
{
tpl_printf(vars, TPLADD, "CLIENTTYPE", "%c", cl->typ);
tpl_addVar(vars, TPLADD, "CLIENTUSER", xml_encode(vars, usr));
if(cl->typ == 'c' || cl->typ == 'm')
{
tpl_addVar(vars, TPLADD, "CLIENTDESCRIPTION", xml_encode(vars, cl->account->description ? cl->account->description : ""));
}
else if(cl->typ == 'p' || cl->typ == 'r')
{
tpl_addVar(vars, TPLADD, "CLIENTDESCRIPTION", xml_encode(vars, cl->reader->description ? cl->reader->description : ""));
}
tpl_printf(vars, TPLADD, "CLIENTLASTRESPONSETIME", "%d", cl->cwlastresptime ? cl->cwlastresptime : -1);
tpl_printf(vars, TPLADD, "CLIENTIDLESECS", "%d", isec);
//load historical values from ringbuffer
char *value = get_ecm_fullhistorystring(cl);
tpl_addVar(vars, TPLADD, "CLIENTLASTRESPONSETIMEHIST", value);
free_mk_t(value);
tpl_addVar(vars, TPLAPPEND, "APISTATUSBITS", tpl_getTpl(vars, "APISTATUSBIT"));
}
}
}
return tpl_getTpl(vars, "APISTATUS");
}
else if(strcmp(getParam(params, "part"), "readerstats") == 0)
{
if(strcmp(getParam(params, "label"), ""))
{
struct s_reader *rdr = get_reader_by_label(getParam(params, "label"));
if(rdr)
{
return send_oscam_reader_stats(vars, params, apicall);
}
else
{
//Send Errormessage
tpl_addVar(vars, TPLADD, "APIERRORMESSAGE", "reader not exist");
return tpl_getTpl(vars, "APIERROR");
}
}
else
{
//Send Errormessage
tpl_addVar(vars, TPLADD, "APIERRORMESSAGE", "no reader selected");
return tpl_getTpl(vars, "APIERROR");
}
}
#ifdef READER_VIDEOGUARD
else if(strcmp(getParam(params, "part"), "sendcmd") == 0)
{
if(strcmp(getParam(params, "label"), ""))
{
struct s_reader *rdr = get_reader_by_label(getParam(params, "label"));
if(rdr)
{
char api_msg[150];
CMD_PACKET *cmd_pack = NULL;
if(!cs_malloc(&cmd_pack, sizeof(CMD_PACKET)))
{
tpl_addVar(vars, TPLADD, "APIERRORMESSAGE", "cs_malloc failed!");
return tpl_getTpl(vars, "APIERROR");
}
struct s_client *webif_client = cur_client();
webif_client->grp = 0xFF; // access all readers
memset(cmd_pack, '0', sizeof(CMD_PACKET));
cmd_pack->client = webif_client;
cmd_pack->cmdlen = strlen(getParam(params, "cmd")) / 2;
if(cmd_pack->cmdlen > 0 && (unsigned long)abs(cmd_pack->cmdlen) <= sizeof(cmd_pack->cmd))
{
if(key_atob_l(getParam(params, "cmd"), cmd_pack->cmd, cmd_pack->cmdlen*2))
{
tpl_addVar(vars, TPLADD, "APIERRORMESSAGE", "'cmd' has not been sent due to wrong value!");
return tpl_getTpl(vars, "APIERROR");
}
}
else
{
if(cmd_pack->cmdlen)
{
snprintf(api_msg, sizeof(api_msg), "Command would exceed %lu bytes!", (long unsigned int)sizeof(cmd_pack->cmd));
tpl_addVar(vars, TPLADD, "APIERRORMESSAGE", api_msg);
}
else
{
tpl_addVar(vars, TPLADD, "APIERRORMESSAGE", "Missing parameter 'cmd'!");
}
return tpl_getTpl(vars, "APIERROR");
}
struct s_client *cl = rdr->client;
if(rdr->enable == 1 && cl && cl->typ == 'r')
{
add_job(cl, ACTION_READER_SENDCMD, cmd_pack, sizeof(CMD_PACKET));
tpl_addVar(vars, TPLADD, "APICONFIRMMESSAGE", "command sent");
return tpl_getTpl(vars, "APICONFIRMATION");
}
else
{
snprintf(api_msg, sizeof(api_msg), "Reader '%s' is not suitable!", xml_encode(vars, rdr->label));
tpl_addVar(vars, TPLADD, "APIERRORMESSAGE", api_msg);
return tpl_getTpl(vars, "APIERROR");
}
}
else
{
//Send Errormessage
tpl_addVar(vars, TPLADD, "APIERRORMESSAGE", "no such reader");
return tpl_getTpl(vars, "APIERROR");
}
}
else
{
//Send Errormessage
tpl_addVar(vars, TPLADD, "APIERRORMESSAGE", "no reader selected");
return tpl_getTpl(vars, "APIERROR");
}
}
#endif
else if(strcmp(getParam(params, "part"), "shutdown") == 0)
{
if((strcmp(strtolower(getParam(params, "action")), "restart") == 0) ||
(strcmp(strtolower(getParam(params, "action")), "shutdown") == 0))
{
if(!cfg.http_readonly)
{
return send_oscam_shutdown(vars, f, params, apicall, keepalive, extraheader);
}
else
{
tpl_addVar(vars, TPLADD, "APIERRORMESSAGE", "webif readonly mode");
return tpl_getTpl(vars, "APIERROR");
}
}
else
{
tpl_addVar(vars, TPLADD, "APIERRORMESSAGE", "missing parameter action");
return tpl_getTpl(vars, "APIERROR");
}
}
else
{
tpl_addVar(vars, TPLADD, "APIERRORMESSAGE", "part not found");
return tpl_getTpl(vars, "APIERROR");
}
}
static char *send_oscam_image(struct templatevars * vars, FILE * f, struct uriparams * params, char *image, time_t modifiedheader, uint32_t etagheader, char *extraheader)
{
char *wanted;
if(image == NULL) { wanted = getParam(params, "i"); }
else { wanted = image; }
if(cs_strlen(wanted) > 3 && wanted[0] == 'I' && wanted[1] == 'C')
{
if(etagheader == 0)
{
int8_t disktpl = 0;
char *tpl_path;
tpl_path = cfg.http_piconpath? cfg.http_piconpath : cfg.http_tpl;
if(tpl_path)
{
char path[255];
if(cs_strlen(tpl_getTplPath(wanted, tpl_path, path, 255)) > 0 && file_exists(path))
{
struct stat st;
disktpl = 1;
stat(path, &st);
if((time_t)st.st_mtime < modifiedheader)
{
send_header304(f, extraheader);
return "1";
}
}
}
if(disktpl == 0 && first_client->login < modifiedheader)
{
send_header304(f, extraheader);
return "1";
}
}
char *header = strstr(tpl_getTpl(vars, wanted), "data:");
if(header != NULL)
{
char *ptr = header + 5;
while(ptr[0] != ';' && ptr[0] != '\0') { ++ptr; }
if(ptr[0] != '\0' && ptr[1] != '\0') { ptr[0] = '\0'; }
else { return "0"; }
ptr = strstr(ptr + 1, "base64,");
if(ptr != NULL)
{
int32_t len = b64decode((uint8_t *)ptr + 7);
if(len > 0)
{
if((uint32_t)crc32(0L, (uint8_t *)ptr + 7, len) == etagheader)
{
send_header304(f, extraheader);
}
else
{
send_headers(f, 200, "OK", extraheader, header + 5, 1, len, ptr + 7, 0);
webif_write_raw(ptr + 7, f, len);
}
return "1";
}
}
}
}
// Return file not found
const char *not_found = "File not found.\n";
send_headers(f, 404, "Not Found", extraheader, "text/plain", 0, cs_strlen(not_found), (char *)not_found, 0);
webif_write_raw((char *)not_found, f, cs_strlen(not_found));
return "1";
}
static char *send_oscam_robots_txt(FILE * f)
{
const char *content = "User-agent: *\nDisallow: /\n";
send_headers(f, 200, "OK", NULL, "text/plain", 0, cs_strlen(content), (char *)content, 0);
webif_write_raw((char *)content, f, cs_strlen(content));
return "1";
}
static char *send_oscam_graph(struct templatevars * vars)
{
return tpl_getTpl(vars, "GRAPH");
}
#ifdef MODULE_GHTTP
static bool ghttp_autoconf(struct templatevars * vars, struct uriparams * params)
{
int8_t i = 0;
struct s_reader *rdr;
char *name = getParam(params, "gacname");
if(cs_strlen(name) < 3)
{
tpl_addMsg(vars, "Invalid host name!");
return false;
}
LL_ITER itr = ll_iter_create(configured_readers);
while((rdr = ll_iter_next(&itr)))
if(rdr->ph.num == R_GHTTP) { i++; } // count existing ghttp readers
while(i < 3) // if less than 3, add more
{
char lbl[128];
snprintf(lbl, sizeof(lbl), "%s%d", "ghttp", i + 1);
cs_log("GHttp autoconf: adding reader %s", lbl);
struct s_reader *newrdr;
if(!cs_malloc(&newrdr, sizeof(struct s_reader)))
{
tpl_addMsg(vars, "Create reader failed!");
return false;
};
newrdr->typ = R_GHTTP;
cs_strncpy(newrdr->label, lbl, sizeof(newrdr->label));
module_reader_set(newrdr);
reader_set_defaults(newrdr);
newrdr->enable = 0;
newrdr->grp = 1;
ll_append(configured_readers, newrdr);
i++;
}
uint16_t port = 0;
char *str = strstr(name, ":");
if(str)
{
port = atoi(str + 1);
str[0] = '\0';
}
i = 0;
itr = ll_iter_create(configured_readers);
while((rdr = ll_iter_next(&itr)))
{
if(rdr->ph.num == R_GHTTP)
{
if(i > 2) // remove superflous
{
cs_log("GHttp autoconf: removing reader %s", rdr->label);
inactivate_reader(rdr);
ll_iter_remove(&itr);
free_reader(rdr);
}
else // reconfigure the 3 first ghttp readers
{
cs_log("GHttp autoconf: reconfiguring reader %s", rdr->label);
snprintf(rdr->label, sizeof(rdr->label), "%s%d", "ghttp", i + 1);
rdr->r_port = port;
rdr->enable = 1;
rdr->ghttp_use_ssl = 0;
#ifdef WITH_SSL
rdr->ghttp_use_ssl = 1;
#endif
if(rdr->grp < 1) { rdr->grp = 1; }
cs_strncpy(rdr->r_usr, getParam(params, "gacuser"), sizeof(rdr->r_usr));
cs_strncpy(rdr->r_pwd, getParam(params, "gacpasswd"), sizeof(rdr->r_pwd));
if(i == 0) { cs_strncpy(rdr->device, name, sizeof(rdr->device)); }
else
{
if(!strstr(name, "."))
{ snprintf(rdr->device, sizeof(rdr->device), "%s%d", name, i); } // name, name1, name2
else { cs_strncpy(rdr->device, name, sizeof(rdr->device)); }
// . in the name = assume full hostname = use same for all 3 readers
}
if(i == 2) { rdr->fallback = 1; }
else { rdr->fallback = 0; }
i++;
}
}
}
cs_log("GHttp autoconf: Saving %d readers", i);
if(write_server() != 0) { tpl_addMsg(vars, "Write Config failed!"); }
itr = ll_iter_create(configured_readers);
while((rdr = ll_iter_next(&itr)))
{
if(rdr->ph.num == R_GHTTP)
{ restart_cardreader(rdr, 1); }
}
return true;
}
static char *send_oscam_ghttp(struct templatevars * vars, struct uriparams * params, int8_t apicall)
{
if(strcmp(strtolower(getParam(params, "action")), "autoconf") == 0)
{
if(!apicall)
{
bool missing = false;
if(cs_strlen(getParam(params, "gacuser")) == 0)
{
tpl_addVar(vars, TPLADD, "USERREQ", "<FONT COLOR='red'>(Required)</FONT>");
missing = true;
}
else { tpl_addVar(vars, TPLADD, "GACUSER", getParam(params, "gacuser")); }
if(cs_strlen(getParam(params, "gacpasswd")) == 0)
{
tpl_addVar(vars, TPLADD, "PWDREQ", "<FONT COLOR='red'>(Required)</FONT>");
missing = true;
}
else { tpl_addVar(vars, TPLADD, "GACPASSWD", getParam(params, "gacpasswd")); }
if(cs_strlen(getParam(params, "gacname")) == 0)
{
tpl_addVar(vars, TPLADD, "NAMEREQ", "<FONT COLOR='red'>(Required)</FONT>");
missing = true;
}
else { tpl_addVar(vars, TPLADD, "GACNAME", getParam(params, "gacname")); }
if(missing) { return tpl_getTpl(vars, "PREAUTOCONF"); }
cs_log("GHttp autoconf requested by WebIF from %s", cs_inet_ntoa(GET_IP()));
}
else
{
tpl_addVar(vars, TPLADD, "APICONFIRMMESSAGE", "autoconf");
cs_log("GHttp autoconf requested by XMLApi from %s", cs_inet_ntoa(GET_IP()));
}
if(ghttp_autoconf(vars, params))
{
tpl_printf(vars, TPLADD, "REFRESHTIME", "%d", 3);
tpl_addVar(vars, TPLADD, "REFRESH", tpl_getTpl(vars, "REFRESH"));
tpl_printf(vars, TPLADD, "SECONDS", "%d", 3);
if(apicall) { return tpl_getTpl(vars, "APICONFIRMATION"); }
else { return tpl_getTpl(vars, "AUTOCONF"); }
}
else { return tpl_getTpl(vars, "PREAUTOCONF"); } // something failed
}
else
{
if(cs_strlen(getParam(params, "token")) > 0) // parse autoconf token
{
char *token = getParam(params, "token");
int32_t len = b64decode((uint8_t *)token);
if(len > 0)
{
struct uriparams tokenprms;
tokenprms.paramcount = 0;
parseParams(&tokenprms, token);
if(cs_strlen(getParam(&tokenprms, "u")) > 0)
{
tpl_addVar(vars, TPLADD, "GACUSER", getParam(&tokenprms, "u"));
tpl_addVar(vars, TPLADD, "USERRDONLY", "readonly");
}
if(cs_strlen(getParam(&tokenprms, "p")) > 0)
{
tpl_addVar(vars, TPLADD, "GACPASSWD", getParam(&tokenprms, "p"));
tpl_addVar(vars, TPLADD, "PWDRDONLY", "readonly");
}
if(cs_strlen(getParam(&tokenprms, "n")) > 0)
{
tpl_addVar(vars, TPLADD, "GACNAME", getParam(&tokenprms, "n"));
tpl_addVar(vars, TPLADD, "NAMERDONLY", "readonly");
}
}
}
return tpl_getTpl(vars, "PREAUTOCONF");
}
}
#endif
static int8_t check_httpip(IN_ADDR_T addr)
{
int8_t i = 0;
// check all previously dyndns resolved addresses
for(i = 0; i < MAX_HTTP_DYNDNS; i++)
{
if(IP_ISSET(cfg.http_dynip[i]) && IP_EQUAL(cfg.http_dynip[i], addr))
{ return 1; }
}
return 0;
}
static int8_t check_httpdyndns(IN_ADDR_T addr)
{
// check all previously dyndns resolved addresses
if(check_httpip(addr))
{ return 1; }
// we are not ok, so resolve all dyndns entries into IP's - maybe outdated IP's
if(cfg.http_dyndns[0][0])
{
int8_t i = 0;
for(i = 0; i < MAX_HTTP_DYNDNS; i++)
{
if(cfg.http_dyndns[i][0])
{
cs_resolve((const char *)cfg.http_dyndns[i], &cfg.http_dynip[i], NULL, NULL);
cs_log_dbg(D_TRACE, "WebIf: httpdyndns [%d] resolved %s to %s ", i, (char *)cfg.http_dyndns[i], cs_inet_ntoa(cfg.http_dynip[i]));
}
}
}
else
{
cs_log_dbg(D_TRACE, "WebIf: No dyndns addresses found");
return 0;
}
// again check all dyndns resolved addresses
if(check_httpip(addr))
{ return 1; }
return 0;
}
static int8_t check_valid_origin(IN_ADDR_T addr)
{
// check whether requesting IP is in allowed IP ranges
if(check_ip(cfg.http_allowed, addr))
{ return 1; }
// we havn't found the requesting IP in allowed range. So we check for allowed httpdyndns as last chance
if(cfg.http_dyndns[0][0])
{
int8_t ok;
ok = check_httpdyndns(addr);
return ok;
}
return 0;
}
static int8_t check_request(char *result, int32_t readen)
{
if(readen < 50) { return 0; }
result[readen] = '\0';
int8_t method;
if(strncmp(result, "POST", 4) == 0) { method = 1; }
else { method = 0; }
char *headerEnd = strstr(result, "\r\n\r\n");
if(headerEnd == NULL) { return 0; }
else if(method == 0) { return 1; }
else
{
char *ptr = strstr(result, "Content-Length: ");
if(ptr != NULL)
{
ptr += 16;
if(ptr < result + readen)
{
uint32_t length = atoi(ptr);
if(cs_strlen(headerEnd + 4) >= length) { return 1; }
}
}
}
return 0;
}
static int32_t readRequest(FILE * f, IN_ADDR_T in, char **result, int8_t forcePlain)
{
int32_t n, bufsize = 0, errcount = 0;
char buf2[1024];
struct pollfd pfd2[1];
#ifdef WITH_SSL
int8_t is_ssl = 0;
if(ssl_active && !forcePlain)
{ is_ssl = 1; }
#endif
while(1)
{
errno = 0;
if(forcePlain)
{ n = read(fileno(f), buf2, sizeof(buf2)); }
else
{ n = webif_read(buf2, sizeof(buf2), f); }
if(n <= 0)
{
if((errno == 0 || errno == EINTR))
{
if(errcount++ < 10)
{
cs_sleepms(5);
continue;
}
else { return -1; }
}
#ifdef WITH_SSL
if(is_ssl)
{
if(errno != ECONNRESET)
{
int32_t errcode = ERR_peek_error();
char errstring[128];
ERR_error_string_n(errcode, errstring, sizeof(errstring) - 1);
cs_log_dbg(D_TRACE, "WebIf: read error ret=%d (%d%s%s)", n, SSL_get_error(cur_ssl(), n), errcode ? " " : "", errcode ? errstring : "");
}
return -1;
}
#else
if(errno != ECONNRESET)
{ cs_log_dbg(D_TRACE, "WebIf: read error ret=%d (errno=%d %s)", n, errno, strerror(errno)); }
#endif
return -1;
}
if(!cs_realloc(result, bufsize + n + 1))
{
send_error500(f);
NULLFREE(*result);
return -1;
}
memcpy(*result + bufsize, buf2, n);
bufsize += n;
#ifdef WITH_EMU
if(bufsize > 204800) // max request size 200kb
#else
if(bufsize > 102400) // max request size 100kb
#endif
{
cs_log("error: too much data received from %s", cs_inet_ntoa(in));
NULLFREE(*result);
*result = NULL;
return -1;
}
#ifdef WITH_SSL
if(ssl_active && !forcePlain)
{
int32_t len = 0;
len = SSL_pending((SSL *)f);
if(len > 0)
{ continue; }
pfd2[0].fd = SSL_get_fd((SSL *)f);
}
else
#endif
pfd2[0].fd = fileno(f);
pfd2[0].events = (POLLIN | POLLPRI);
int32_t rc = poll(pfd2, 1, 100);
if(rc > 0 || !check_request(*result, bufsize))
{ continue; }
else
{ break; }
}
return bufsize;
}
static int32_t process_request(FILE * f, IN_ADDR_T in)
{
int32_t ok = 0;
int8_t *keepalive = (int8_t *)pthread_getspecific(getkeepalive);
IN_ADDR_T addr = GET_IP();
do
{
#ifdef WITH_SSL
if(!ssl_active && *keepalive) { fflush(f); }
#else
if(*keepalive) { fflush(f); }
#endif
// at this point we do all checks related origin IP, ranges and dyndns stuff
ok = check_valid_origin(addr);
cs_log_dbg(D_TRACE, "WebIf: Origin checked. Result: access from %s => %s", cs_inet_ntoa(addr), (!ok) ? "forbidden" : "allowed");
// based on the failed origin checks we send a 403 to calling browser
if(!ok)
{
send_error(f, 403, "Forbidden", NULL, "Access denied.", 0);
cs_log("unauthorized access from %s - invalid ip or dyndns", cs_inet_ntoa(addr));
return 0;
}
int32_t authok = 0;
char expectednonce[(MD5_DIGEST_LENGTH * 2) + 1], opaque[(MD5_DIGEST_LENGTH * 2) + 1];
char authheadertmp[sizeof(AUTHREALM) + sizeof(expectednonce) + sizeof(opaque) + 100];
char *method, *path, *protocol, *str1, *saveptr1 = NULL, *authheader = NULL, *extraheader = NULL, *filebuf = NULL;
char *pch, *tmp, *buf, *nameInUrl, subdir[32];
/* List of possible pages */
char *pages[] =
{
"/config.html",
"/readers.html",
"/entitlements.html",
"/status.html",
"/userconfig.html",
"/readerconfig.html",
"/services.html",
"/user_edit.html",
"/site.css",
"/services_edit.html",
"/savetemplates.html",
"/shutdown.html",
"/script.html",
"/scanusb.html",
"/files.html",
"/readerstats.html",
"/failban.html",
"/oscam.js",
"/oscamapi.html",
"/image",
"/favicon.ico",
"/graph.svg",
"/oscamapi.xml",
"/cacheex.html",
"/oscamapi.json",
"/emm.html",
"/emm_running.html",
"/robots.txt",
"/ghttp.html",
"/logpoll.html",
"/jquery.js",
#ifdef WEBIF_WIKI
"/wiki.json",
"/wiki_status.json",
#endif
};
int32_t pagescnt = sizeof(pages) / sizeof(char *); // Calculate the amount of items in array
int32_t i, bufsize, len, pgidx = -1;
uint32_t etagheader = 0;
struct uriparams params;
params.paramcount = 0;
time_t modifiedheader = 0;
bufsize = readRequest(f, in, &filebuf, 0);
if(!filebuf || bufsize < 1)
{
if(!*keepalive) { cs_log_dbg(D_CLIENT, "WebIf: No data received from client %s. Closing connection.", cs_inet_ntoa(addr)); }
return -1;
}
buf = filebuf;
if((method = strtok_r(buf, " ", &saveptr1)) != NULL)
{
if((path = strtok_r(NULL, " ", &saveptr1)) != NULL)
{
if((protocol = strtok_r(NULL, "\r", &saveptr1)) == NULL)
{
NULLFREE(filebuf);
return -1;
}
}
else
{
NULLFREE(filebuf);
return -1;
}
}
else
{
NULLFREE(filebuf);
return -1;
}
tmp = protocol + cs_strlen(protocol) + 2;
pch = path;
/* advance pointer to beginning of query string */
while(pch[0] != '?' && pch[0] != '\0') { ++pch; }
if(pch[0] == '?')
{
pch[0] = '\0';
++pch;
}
nameInUrl = pch - 1;
while(nameInUrl != path && nameInUrl[0] != '/') { --nameInUrl; }
/* allow only alphanumeric sub-folders */
int32_t subdirLen = nameInUrl - path;
subdir[0] = '\0';
if(subdirLen > 0 && subdirLen < 32)
{
cs_strncpy(subdir, path + 1, subdirLen);
int32_t invalidSubdir = 0;
for(i = 0; i < subdirLen - 1; i++)
{
if(!((subdir[i] >= '0' && subdir[i] <= '9')
|| (subdir[i] >= 'a' && subdir[i] <= 'z')
|| (subdir[i] >= 'A' && subdir[i] <= 'Z')))
{
invalidSubdir = 1;
subdir[0] = '\0';
break;
}
}
if(!invalidSubdir)
{
subdir[subdirLen] = '\0';
#ifdef WIN32
subdir[subdirLen - 1] = '\\';
#else
subdir[subdirLen - 1] = '/';
#endif
}
}
/* Map page to our static page definitions */
for(i = 0; i < pagescnt; i++)
{
if(!strcmp(nameInUrl, pages[i])) { pgidx = i; }
}
parseParams(&params, pch);
if(!cfg.http_user || !cfg.http_pwd)
{ authok = 1; }
for(str1 = strtok_r(tmp, "\n", &saveptr1); str1; str1 = strtok_r(NULL, "\n", &saveptr1))
{
len = cs_strlen(str1);
if(str1[len - 1] == '\r')
{
str1[len - 1] = '\0';
--len;
}
if(len == 0)
{
if(strcmp(method, "POST") == 0)
{
parseParams(&params, str1 + 2);
}
break;
}
if(!authok && len > 50 && strncasecmp(str1, "Authorization:", 14) == 0 && strstr(str1, "Digest") != NULL)
{
if(cs_dblevel & D_CLIENT)
{
if(cs_realloc(&authheader, len + 1))
{ cs_strncpy(authheader, str1, len); }
}
authok = check_auth(str1, method, path, addr, expectednonce, opaque);
}
else if(len > 40 && strncasecmp(str1, "If-Modified-Since:", 18) == 0)
{
modifiedheader = parse_modifiedsince(str1);
}
else if(len > 20 && strncasecmp(str1, "If-None-Match:", 14) == 0)
{
for(pch = str1 + 14; pch[0] != '"' && pch[0] != '\0'; ++pch) { ; }
if(cs_strlen(pch) > 5) { etagheader = (uint32_t)strtoul(++pch, NULL, 10); }
}
else if(len > 12 && strncasecmp(str1, "Connection: Keep-Alive", 22) == 0 && strcmp(method, "POST"))
{
*keepalive = 1;
}
}
if(cfg.http_user && cfg.http_pwd)
{
if (!authok || cs_strlen(opaque) != MD5_DIGEST_LENGTH * 2)
{
calculate_opaque(addr, opaque);
}
if (authok != 2)
{
if(!authok)
{
if(authheader)
{
cs_log_dbg(D_CLIENT, "WebIf: Received wrong auth header from %s:", cs_inet_ntoa(addr));
cs_log_dbg(D_CLIENT, "%s", authheader);
}
else
{ cs_log_dbg(D_CLIENT, "WebIf: Received no auth header from %s.", cs_inet_ntoa(addr)); }
}
calculate_nonce(NULL, expectednonce, opaque);
}
if (authok != 1)
{
snprintf(authheadertmp, sizeof(authheadertmp), "WWW-Authenticate: Digest algorithm=\"MD5\", realm=\"%s\", qop=\"auth\", opaque=\"%s\", nonce=\"%s\"", AUTHREALM, opaque, expectednonce);
if (authok == 2)
{
if (!cs_strncat(authheadertmp, ", stale=true", sizeof(authheadertmp))) {
cs_log("WARNING, bug here!");
}
}
}
else
{
snprintf(authheadertmp, sizeof(authheadertmp), "Authentication-Info: nextnonce=\"%s\"", expectednonce);
}
extraheader = authheadertmp;
if (authok != 1)
{
char *msg = "Access denied.\n";
send_headers(f, 401, "Unauthorized", extraheader, "text/html", 0, cs_strlen(msg), msg, 0);
webif_write(msg, f);
NULLFREE(authheader);
NULLFREE(filebuf);
if (*keepalive) {
continue;
} else {
return 0;
}
}
}
else
{
NULLFREE(authheader);
}
/*build page*/
if(pgidx == 8)
{
send_file(f, "CSS", subdir, modifiedheader, etagheader, extraheader);
}
else if(pgidx == 17)
{
send_file(f, "JS", subdir, modifiedheader, etagheader, extraheader);
}
else if(pgidx == 30)
{
send_file(f, "JQ", subdir, modifiedheader, etagheader, extraheader);
}
else
{
time_t t;
struct templatevars *vars = tpl_create();
if(vars == NULL)
{
send_error500(f);
NULLFREE(filebuf);
return 0;
}
tpl_addVar(vars, TPLADD, "SUBDIR", subdir);
struct tm lt, st;
time(&t);
localtime_r(&t, &lt);
tpl_addVar(vars, TPLADD, "SCM_URL", SCM_URL);
tpl_addVar(vars, TPLADD, "WIKI_URL", WIKI_URL);
tpl_addVar(vars, TPLADD, "BOARD_URL", BOARD_URL);
#ifdef WEBIF_WIKI
tpl_addVar(vars, TPLADD, "WIKIINTERNALVAR", "\t\tvar wikiInternal = true;");
#else
tpl_addVar(vars, TPLADD, "WIKIINTERNALVAR", "");
#endif
tpl_addVar(vars, TPLADD, "CS_VERSION", CS_VERSION);
tpl_addVar(vars, TPLADD, "CS_GIT_COMMIT", CS_GIT_COMMIT);
tpl_addVar(vars, TPLADD, "CS_TARGET", CS_TARGET);
tpl_addVar(vars, TPLADD, "HTTPOSCAMLABEL", xml_encode(vars,cfg.http_oscam_label));
if (!boxtype_is("generic"))
{
if (!boxname_is("generic"))
tpl_printf(vars, TPLADD, "DETECTED_BOXTYPE", "%s (%s)", boxtype_get(), boxname_get());
else
tpl_addVar(vars, TPLADD, "DETECTED_BOXTYPE", boxtype_get());
}
if(cfg.http_locale){
float decimal_point = 0.0;
setlocale(LC_ALL, cfg.http_locale);
tpl_printf(vars, TPLADD, "TMP_DECPOINT", "%.1f", decimal_point);
tpl_addVar(vars, TPLADD, "LOCALE_DECPOINT", strstr(tpl_getVar(vars, "TMP_DECPOINT"), ",") ? ",": ".");
}
tpl_addVar(vars, TPLADD, "HTTP_CHARSET", "UTF-8");
if(cfg.http_picon_size > 0)
{
tpl_printf(vars, TPLADD, "HTTPPICONSIZEINS", "img.statususericon, img.protoicon, img.usericon, img.readericon {height:%dpx !important;max-height:%dpx !important;}", cfg.http_picon_size, cfg.http_picon_size);
}
if(cfg.poll_refresh > 0)
{
tpl_printf(vars, TPLADD, "POLLREFRESHTIME", "%d", cfg.poll_refresh);
}
if ( cfg.http_refresh > 0 &&
((( pgidx == 1 || pgidx == 4 ) && !cfg.poll_refresh ) ||
( pgidx == 3 && ( cfg.http_status_log || !cfg.poll_refresh )) ||
pgidx == 15 || pgidx == 23 || pgidx == -1 )) // wenn polling bei cachex.html eingeführt wird muss die 23 => 2 zeilen höher
{
tpl_printf(vars, TPLADD, "REFRESHTIME", "%d", cfg.http_refresh);
tpl_addVar(vars, TPLADD, "WITHQUERY", pgidx == 15 ? "1" : "0");
tpl_addVar(vars, TPLADD, "REFRESH", tpl_getTpl(vars, "REFRESH"));
}
#ifdef WEBIF_JQUERY
tpl_printf(vars, TPLADD, "SRCJQUERY", "jquery.js?v=%s", CS_VERSION);
#else
tpl_addVar(vars, TPLADD, "SRCJQUERY", cfg.http_extern_jquery);
#endif
if(picon_exists("LOGO")||cs_strlen(tpl_getTpl(vars, "IC_LOGO"))>3)
{
tpl_addVar(vars, TPLADD, "LOGO_INS", tpl_getTpl(vars, "LOGOBITIMG"));
}
else
{
tpl_addVar(vars, TPLADD, "LOGO_INS", tpl_getTpl(vars, "LOGOBITSVG"));
}
tpl_addVar(vars, TPLADD, "LOGO", tpl_getTpl(vars, "LOGOBIT"));
tpl_printf(vars, TPLADD, "CURDATE", "%02d.%02d.%02d", lt.tm_mday, lt.tm_mon + 1, lt.tm_year % 100);
tpl_printf(vars, TPLADD, "CURTIME", "%02d:%02d:%02d", lt.tm_hour, lt.tm_min, lt.tm_sec);
localtime_r(&first_client->login, &st);
tpl_printf(vars, TPLADD, "STARTDATE", "%02d.%02d.%02d", st.tm_mday, st.tm_mon + 1, st.tm_year % 100);
tpl_printf(vars, TPLADD, "STARTTIME", "%02d:%02d:%02d", st.tm_hour, st.tm_min, st.tm_sec);
tpl_printf(vars, TPLADD, "PROCESSID", "%d", getppid());
tpl_addVar(vars, TPLADD, "RUNAS", urlencode(vars, username(first_client)));
time_t now = time((time_t *)0);
// XMLAPI
if(pgidx == 18 || pgidx == 22 || pgidx == 24)
{
char tbuffer [30];
strftime(tbuffer, 30, "%Y-%m-%dT%H:%M:%S%z", &st);
tpl_addVar(vars, TPLADD, "APISTARTTIME", tbuffer);
tpl_printf(vars, TPLADD, "APIRUNTIME", "%" PRId64, (int64_t)now - first_client->login);
tpl_printf(vars, TPLADD, "APIREADONLY", "%d", cfg.http_readonly);
if(strcmp(getParam(&params, "callback"), ""))
{
tpl_printf(vars, TPLADD, "CALLBACK", "%s%s", getParam(&params, "callback"), "(");
tpl_addVar(vars, TPLADD, "ENDBRACKET", ")");
}
}
if (config_enabled(WITH_LB))
tpl_addVar(vars, TPLADD, "LBISDEFINED", "1");
// language code in helplink
tpl_addVar(vars, TPLADD, "LANGUAGE", cfg.http_help_lang);
tpl_addVar(vars, TPLADD, "RUNTIME", sec2timeformat(vars, (now - first_client->login)));
time_t uptime = oscam_get_uptime();
if(uptime > 0){
tpl_printf(vars, TPLADD, "UPTIMETXT", "%s Up Time: ", strcmp(tpl_getVar(vars, "DETECTED_BOXTYPE"),"")==0 ? "System" : tpl_getVar(vars, "DETECTED_BOXTYPE"));
tpl_addVar(vars, TPLADD, "UPTIME", sec2timeformat(vars, uptime));
}
tpl_addVar(vars, TPLADD, "CURIP", cs_inet_ntoa(addr));
if(cfg.http_readonly)
{ tpl_addVar(vars, TPLAPPEND, "BTNDISABLED", "DISABLED"); }
i = ll_count(cfg.v_list);
if(i > 0) { tpl_printf(vars, TPLADD, "FAILBANNOTIFIER", "<SPAN CLASS=\"span_notifier\">%d</SPAN>", i); }
tpl_printf(vars, TPLADD, "FAILBANNOTIFIERPOLL", "%d", i);
char *result = NULL;
// WebIf allows modifying many things. Thus, all pages except images/css/static are expected to be non-threadsafe!
if(pgidx != 19 && pgidx != 20 && pgidx != 21 && pgidx != 27) { cs_writelock(__func__, &http_lock); }
switch(pgidx)
{
case 0:
tpl_addVar(vars, TPLADD, "CONFIG_CONTENT", send_oscam_config(vars, &params));
result = tpl_getTpl(vars, "CONFIGCONTENT");
break;
case 1:
result = send_oscam_reader(vars, &params, 0);
break;
case 2:
result = send_oscam_entitlement(vars, &params, 0);
break;
case 3:
result = send_oscam_status(vars, &params, 0);
break;
case 4:
result = send_oscam_user_config(vars, &params, 0);
break;
case 5:
result = send_oscam_reader_config(vars, &params);
break;
case 6:
result = send_oscam_services(vars, &params);
break;
case 7:
result = send_oscam_user_config_edit(vars, &params, 0);
break;
//case 8: css file
case 9:
result = send_oscam_services_edit(vars, &params);
break;
case 10:
result = send_oscam_savetpls(vars);
break;
case 11:
result = send_oscam_shutdown(vars, f, &params, 0, keepalive, extraheader);
break;
case 12:
result = send_oscam_script(vars, &params);
break;
case 13:
result = send_oscam_scanusb(vars);
break;
case 14:
result = send_oscam_files(vars, &params, 0);
break;
case 15:
result = send_oscam_reader_stats(vars, &params, 0);
break;
case 16:
result = send_oscam_failban(vars, &params, 0);
break;
//case 17: js file
case 18:
result = send_oscam_api(vars, f, &params, keepalive, 1, extraheader);
break; //oscamapi.html
case 19:
result = send_oscam_image(vars, f, &params, NULL, modifiedheader, etagheader, extraheader);
break;
case 20:
result = send_oscam_image(vars, f, &params, "ICMAI", modifiedheader, etagheader, extraheader);
break;
case 21:
result = send_oscam_graph(vars);
break;
case 22:
result = send_oscam_api(vars, f, &params, keepalive, 1, extraheader);
break; //oscamapi.xml
#ifdef CS_CACHEEX
case 23:
result = send_oscam_cacheex(vars, &params, 0);
break;
#endif
case 24:
result = send_oscam_api(vars, f, &params, keepalive, 2, extraheader);
break; //oscamapi.json
case 25:
result = send_oscam_EMM(vars, &params);
break; //emm.html
case 26:
result = send_oscam_EMM_running(vars, &params);
break; //emm_running.html
case 27:
result = send_oscam_robots_txt(f);
break; //robots.txt
#ifdef MODULE_GHTTP
case 28:
result = send_oscam_ghttp(vars, &params, 0);
break;
#endif
#ifdef WEBIF_LIVELOG
case 29:
result = send_oscam_logpoll(vars, &params);
break;
//case 30: jquery.js
#endif
#ifdef WEBIF_WIKI
case 31:
result = send_oscam_wiki(vars, &params);
break;
case 32:
result = send_oscam_wiki_status(vars, &params);
break;
#endif
default:
result = send_oscam_status(vars, &params, 0);
break;
}
if(pgidx != 19 && pgidx != 20 && pgidx != 21 && pgidx != 27) { cs_writeunlock(__func__, &http_lock); }
if(result == NULL || !strcmp(result, "0") || cs_strlen(result) == 0) { send_error500(f); }
else if(strcmp(result, "1"))
{
//it doesn't make sense to check for modified etagheader here as standard template has timestamp in output and so site changes on every request
if(pgidx == 18)
{ send_headers(f, 200, "OK", extraheader, "text/xml", 0, cs_strlen(result), NULL, 0); }
else if(pgidx == 21)
{ send_headers(f, 200, "OK", extraheader, "image/svg+xml", 0, cs_strlen(result), NULL, 0); }
else if(pgidx == 24)
{ send_headers(f, 200, "OK", extraheader, "text/javascript", 0, cs_strlen(result), NULL, 0); }
#ifdef WEBIF_WIKI
else if(pgidx == 31)
{ send_headers(f, 200, "OK", extraheader, "application/json", 0, cs_strlen(result), NULL, 0); }
#endif
else
{ send_headers(f, 200, "OK", extraheader, "text/html", 0, cs_strlen(result), NULL, 0); }
webif_write(result, f);
}
tpl_clear(vars);
}
NULLFREE(filebuf);
}
while(*keepalive == 1 && !exit_oscam);
return 0;
}
static void *serve_process(void *conn)
{
struct s_connection *myconn = (struct s_connection *)conn;
int32_t s = myconn->socket;
struct s_client *cl = myconn->cl;
IN_ADDR_T in;
IP_ASSIGN(in, myconn->remote);
set_thread_name(__func__);
#ifdef WITH_SSL
SSL *ssl = myconn->ssl;
SAFE_SETSPECIFIC(getssl, ssl);
#endif
NULLFREE(myconn);
SAFE_SETSPECIFIC(getip, &in);
SAFE_SETSPECIFIC(getclient, cl);
int8_t keepalive = 0;
SAFE_SETSPECIFIC(getkeepalive, &keepalive);
#ifdef WITH_SSL
if(ssl_active)
{
if(SSL_set_fd(ssl, s))
{
int32_t ok = (SSL_accept(ssl) != -1);
if(!ok)
{
int8_t tries = 100;
while(!ok && tries--)
{
int32_t err = SSL_get_error(ssl, -1);
if(err != SSL_ERROR_WANT_READ && err != SSL_ERROR_WANT_WRITE)
{ break; }
else
{
struct pollfd pfd;
pfd.fd = s;
pfd.events = POLLIN | POLLPRI;
int32_t rc = poll(&pfd, 1, -1);
if(rc < 0)
{
if(errno == EINTR || errno == EAGAIN) { continue; }
break;
}
if(rc == 1)
{ ok = (SSL_accept(ssl) != -1); }
}
}
}
if(ok)
{
process_request((FILE *)ssl, in);
}
else
{
FILE *f;
f = fdopen(s, "r+");
if(f != NULL)
{
char *ptr, *filebuf = NULL, *host = NULL;
int32_t bufsize = readRequest(f, in, &filebuf, 1);
if(filebuf)
{
filebuf[bufsize] = '\0';
host = strstr(filebuf, "Host: ");
if(host)
{
host += 6;
ptr = strchr(host, '\r');
if(ptr) { ptr[0] = '\0'; }
}
}
if(host)
{
char extra[cs_strlen(host) + 20];
snprintf(extra, sizeof(extra), "Location: https://%s", host);
send_error(f, 301, "Moved Permanently", extra, "This web server is running in SSL mode.", 1);
}
else
{ send_error(f, 200, "Bad Request", NULL, "This web server is running in SSL mode.", 1); }
fflush(f);
fclose(f);
NULLFREE(filebuf);
}
else
{
cs_log_dbg(D_TRACE, "WebIf: fdopen(%d) failed. (errno=%d %s)", s, errno, strerror(errno));
}
}
}
else { cs_log("WebIf: Error calling SSL_set_fd()."); }
SSL_shutdown(ssl);
close(s);
SSL_free(ssl);
}
else
#endif
{
FILE *f;
f = fdopen(s, "r+");
if(f != NULL)
{
process_request(f, in);
fflush(f);
fclose(f);
}
else
{
cs_log_dbg(D_TRACE, "WebIf: fdopen(%d) failed. (errno=%d %s)", s, errno, strerror(errno));
}
shutdown(s, SHUT_WR);
close(s);
}
return NULL;
}
/* Creates a random string with specified length. Note that dst must be one larger than size to hold the trailing \0*/
static void create_rand_str(char *dst, int32_t size)
{
int32_t i;
for(i = 0; i < size; ++i)
{
dst[i] = (rand() % 94) + 32;
}
dst[i] = '\0';
}
static void *http_server(void *UNUSED(d))
{
struct s_client *cl = create_client(first_client->ip);
if(cl == NULL) { return NULL; }
SAFE_SETSPECIFIC(getclient, cl);
cl->typ = 'h';
int32_t s, reuse = 1;
struct s_connection *conn;
set_thread_name(__func__);
/* Create random string for nonce value generation */
create_rand_str(noncekey, 32);
/* Prepare base64 decoding array */
b64prepare();
webif_tpls_prepare();
#ifdef WEBIF_WIKI
webif_wiki_prepare();
#endif
tpl_checkDiskRevisions();
cs_lock_create(__func__, &http_lock, "http_lock", 10000);
init_noncelocks();
memset(&p_stat_cur, 0x0, sizeof(p_stat_cur));
if(pthread_key_create(&getip, NULL))
{
cs_log("Could not create getip");
return NULL;
}
if(pthread_key_create(&getkeepalive, NULL))
{
cs_log("Could not create getkeepalive");
return NULL;
}
struct SOCKADDR sin;
socklen_t len = 0;
memset(&sin, 0, sizeof(sin));
bool do_ipv6 = config_enabled(IPV6SUPPORT);
#ifdef IPV6SUPPORT
if(do_ipv6)
{
len = sizeof(struct sockaddr_in6);
if((sock = socket(AF_INET6, SOCK_STREAM, IPPROTO_TCP)) < 0)
{
cs_log("HTTP Server: ERROR: Creating IPv6 socket failed! (errno=%d %s)", errno, strerror(errno));
cs_log("HTTP Server: Falling back to IPv4.");
do_ipv6 = false;
}
else if (IP_ISSET(cfg.http_srvip) && (IN6_IS_ADDR_V4MAPPED(&cfg.http_srvip) || IN6_IS_ADDR_V4COMPAT(&cfg.http_srvip))) // ipv4 set as http_srvip
{
do_ipv6 = false;
}
else
{
struct sockaddr_in6 *ia = (struct sockaddr_in6 *)&sin;
if (IP_ISSET(cfg.http_srvip))
{
IP_ASSIGN(SIN_GET_ADDR(sin), cfg.http_srvip);
}
else if (IP_ISSET(cfg.srvip))
{
IP_ASSIGN(SIN_GET_ADDR(sin), cfg.srvip);
}
else
{
ia->sin6_addr = in6addr_any;
}
ia->sin6_family = AF_INET6;
ia->sin6_port = htons(cfg.http_port);
}
}
#endif
if(!do_ipv6)
{
len = sizeof(struct sockaddr_in);
if((sock = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP)) < 0)
{
cs_log("HTTP Server: ERROR: Creating socket failed! (errno=%d %s)", errno, strerror(errno));
return NULL;
}
SIN_GET_FAMILY(sin) = AF_INET;
if(IP_ISSET(cfg.http_srvip))
{ IP_ASSIGN(SIN_GET_ADDR(sin), cfg.http_srvip); }
else if(IP_ISSET(cfg.srvip))
{ IP_ASSIGN(SIN_GET_ADDR(sin), cfg.srvip); }
// The default is INADDR_ANY (0)
SIN_GET_PORT(sin) = htons(cfg.http_port);
}
if(setsockopt(sock, SOL_SOCKET, SO_REUSEADDR, &reuse, sizeof(reuse)) < 0)
{
cs_log("HTTP Server: Setting SO_REUSEADDR via setsockopt failed! (errno=%d %s)", errno, strerror(errno));
}
if(bind(sock, (struct sockaddr *)&sin, len) < 0)
{
cs_log("HTTP Server couldn't bind on port %d (errno=%d %s). Not starting HTTP!", cfg.http_port, errno, strerror(errno));
close(sock);
return NULL;
}
if(listen(sock, SOMAXCONN) < 0)
{
cs_log("HTTP Server: Call to listen() failed! (errno=%d %s)", errno, strerror(errno));
close(sock);
return NULL;
}
#ifdef WITH_SSL
if(pthread_key_create(&getssl, NULL))
{
cs_log("Could not create getssl");
}
SSL_CTX *ctx = NULL;
if(cfg.http_use_ssl)
{
ctx = SSL_Webif_Init();
if(ctx == NULL)
{ cs_log("SSL could not be initialized. Starting WebIf in plain mode."); }
else { ssl_active = 1; }
}
else { ssl_active = 0; }
cs_log("HTTP%s Server running. ip=%s port=%d (%s)", ssl_active ? "S" : "", cs_inet_ntoa(SIN_GET_ADDR(sin)), cfg.http_port, ssl_active ? OPENSSL_VERSION_TEXT : "no SSL");
#else
cs_log("HTTP Server running. ip=%s port=%d", cs_inet_ntoa(SIN_GET_ADDR(sin)), cfg.http_port);
#endif
struct SOCKADDR remote;
memset(&remote, 0, sizeof(remote));
while(!exit_oscam)
{
if((s = accept(sock, (struct sockaddr *) &remote, &len)) < 0)
{
if(exit_oscam)
{ break; }
if(errno != EAGAIN && errno != EINTR)
{
cs_log("HTTP Server: Error calling accept() (errno=%d %s)", errno, strerror(errno));
cs_sleepms(100);
}
else { cs_sleepms(5); }
continue;
}
else
{
getpeername(s, (struct sockaddr *) &remote, &len);
if(!cs_malloc(&conn, sizeof(struct s_connection)))
{
close(s);
continue;
}
setTCPTimeouts(s);
cur_client()->last = time((time_t *)0); //reset last busy time
conn->cl = cur_client();
#ifdef IPV6SUPPORT
if(do_ipv6)
{
struct sockaddr_in6 *ra = (struct sockaddr_in6 *)&remote;
memcpy(&conn->remote, &ra->sin6_addr, sizeof(struct in6_addr));
}
else
{
struct sockaddr_in *fba = (struct sockaddr_in *)&remote;
struct in6_addr taddr;
memset(&taddr, 0, sizeof(taddr));
taddr.s6_addr32[3] = fba->sin_addr.s_addr;
memcpy(&conn->remote, &taddr, sizeof(struct in6_addr));
}
#else
memcpy(&conn->remote, &remote.sin_addr, sizeof(struct in_addr));
#endif
conn->socket = s;
#ifdef WITH_SSL
conn->ssl = NULL;
if(ssl_active)
{
conn->ssl = SSL_new(ctx);
if(conn->ssl == NULL)
{
close(s);
cs_log("WebIf: Error calling SSL_new().");
continue;
}
}
#endif
int32_t ret = start_thread("webif workthread", serve_process, (void *)conn, NULL, 1, 1);
if(ret)
{
NULLFREE(conn);
}
}
}
// Wait a bit so that we don't close ressources while http threads are active
cs_sleepms(300);
#ifdef WITH_SSL
SSL_CTX_free(ctx);
CRYPTO_set_dynlock_create_callback(NULL);
CRYPTO_set_dynlock_lock_callback(NULL);
CRYPTO_set_dynlock_destroy_callback(NULL);
CRYPTO_set_locking_callback(NULL);
CRYPTO_set_id_callback(NULL);
OPENSSL_free(lock_cs);
lock_cs = NULL;
#endif
cs_log("HTTP Server stopped");
free_client(cl);
close(sock);
return NULL;
}
void webif_client_reset_lastresponsetime(struct s_client * cl)
{
int32_t i;
for(i = 0; i < CS_ECM_RINGBUFFER_MAX; i++)
{
cl->cwlastresptimes[i].duration = 0;
cl->cwlastresptimes[i].timestamp = time((time_t *)0);
cl->cwlastresptimes[i].rc = 0;
}
cl->cwlastresptimes_last = 0;
}
void webif_client_add_lastresponsetime(struct s_client * cl, int32_t ltime, time_t timestamp, int32_t rc)
{
int32_t last = cl->cwlastresptimes_last = (cl->cwlastresptimes_last + 1) & (CS_ECM_RINGBUFFER_MAX - 1);
cl->cwlastresptimes[last].duration = ltime > 9999 ? 9999 : ltime;
cl->cwlastresptimes[last].timestamp = timestamp;
cl->cwlastresptimes[last].rc = rc;
}
void webif_client_init_lastreader(struct s_client * client, ECM_REQUEST * er, struct s_reader * er_reader, const char *stxt[])
{
if(er_reader)
{
if(er->rc == E_FOUND)
{ cs_strncpy(client->lastreader, er_reader->label, sizeof(client->lastreader)); }
else if(er->rc == E_CACHEEX)
#ifdef CS_CACHEEX_AIO
{
if (cfg.cacheex_srcname_webif)
{
cs_strncpy(client->lastreader, username(er_reader->client), sizeof(client->lastreader));
}
else
{
cs_strncpy(client->lastreader, "cache3", sizeof(client->lastreader));
}
}
#else
{ cs_strncpy(client->lastreader, "cache3", sizeof(client->lastreader)); }
#endif
else if(er->rc < E_NOTFOUND)
{ snprintf(client->lastreader, sizeof(client->lastreader) - 1, "%.54s (cache)", er_reader->label); }
else
{ cs_strncpy(client->lastreader, stxt[er->rc], sizeof(client->lastreader)); }
}
else
{
cs_strncpy(client->lastreader, stxt[er->rc], sizeof(client->lastreader));
}
}
void webif_init(void)
{
char buf[8], fname[256];
snprintf(buf, 8, "%'d", 7);
if(strcmp(buf, "7"))
{
useLocal = 0;
}
if(cfg.http_port == 0)
{
cs_log("http disabled");
return;
}
get_config_filename(fname, sizeof(fname), "oscam.srvid2");
use_srvid2 = file_exists(fname);
if(start_thread("http", http_server, NULL, &httpthread, 0, 1) == 0)
{
httpthread_running = 1;
}
}
void webif_close(void)
{
if(!sock)
{ return; }
shutdown(sock, 2);
close(sock);
if(httpthread_running)
{ SAFE_THREAD_JOIN(httpthread, NULL); }
}
#endif