[181c4de] | 1 | // **************************************************************************************************************************************************
|
---|
| 2 | // Libreria: ogAdmLib
|
---|
| 3 | // Autor: José Manuel Alonso (E.T.S.I.I.) Universidad de Sevilla
|
---|
| 4 | // Fecha Creación: Marzo-2010
|
---|
| 5 | // Fecha Última modificación: Marzo-2010
|
---|
| 6 | // Nombre del fichero: ogAdmLib.h
|
---|
| 7 | // Descripción: Este fichero implementa el archivo de cabecera de la libreria ogAdmLib
|
---|
| 8 | // **************************************************************************************************************************************************
|
---|
| 9 | // ________________________________________________________________________________________________________
|
---|
| 10 | // Valores definidos
|
---|
| 11 | // ________________________________________________________________________________________________________
|
---|
| 12 | #define LONSTD 1024 // Longitud de memoria estandar
|
---|
| 13 | #define LONINT 16 // Longitud de memoria estandar para un número entero
|
---|
| 14 | #define LONFIL 256 // Longitud de memoria estandar para nombres de archivo completos (incluido path)
|
---|
| 15 | #define LONIP 16 // Longitud de memoria estandar para cadenas que contiene una dirección IP
|
---|
| 16 | #define LONMAC 16 // Longitud de memoria estandar para cadenas que contiene una dirección MAC
|
---|
| 17 | #define LONSQL 4096 // Longitud de memoria estandar para una sentencia SQL
|
---|
| 18 | #define LONPRM 512 // Longitud estandar de los parámetros del fichero de configuración del servicio
|
---|
| 19 | #define LONSCP 2024 // Longitud estandar de los parámetros de las tramas
|
---|
| 20 | #define LONFUN 64 // Longitud estandar de los nombres de las funciones que procesan las tramas
|
---|
| 21 | #define LONSUC 512 // Longitud de los mensajes de sucesos
|
---|
| 22 | #define LONBLK 512 // Longitud de los paquetes de tramas leidos cada vez
|
---|
| 23 | #define MAXPRM 10 // Máximo número de parámeros del fichero de configuración del servicio
|
---|
| 24 | #define MAXPAR 16 // Máximo número de particiones manejadas por el sistema
|
---|
| 25 | #define MAXLONURL 512 // Longitud máxima de una dirección url con parámetros
|
---|
| 26 |
|
---|
| 27 | #define LONHEXPRM 5 // Longitud del campo que contiene el tamaño de la cadena de parámetros
|
---|
| 28 | #define LONGITUD_CABECERATRAMA 16 // Longitud de la cabecera de las tramas
|
---|
| 29 | #define LONGITUD_PARAMETROS 1024 // Longitud estandar de la información de la trama (parámetros)
|
---|
| 30 | #define MAXCMD_PARAMETROS 200 // Máximo número de parámetros de una trama
|
---|
| 31 |
|
---|
| 32 | #define MAXIMOS_CLIENTES 4000 // Máximo número de conexiones con ordenadores clientes
|
---|
| 33 | #define MAXIMAS_FUNCIONES LONSTD // Máximo número de funciones que procesan los mensajes entre servicio y clientes
|
---|
| 34 | #define MAXIMAS_LINEAS 3000 // Longitud máxima de lineas en un archivo de comandos
|
---|
| 35 |
|
---|
| 36 | #define AUTOINCORPORACION_OFF 0x0000 // Los ordenadores no se pueden dar de alta automáticamente
|
---|
| 37 | #define AUTOINCORPORACION_ONA 0x0001 // Los ordenadores se pueden dar de alta automáticamente si existe el aula
|
---|
| 38 | #define AUTOINCORPORACION_ONX 0x0002 // Los ordenadores se pueden dar de alta automáticamentee y si no existe el aula la crea
|
---|
| 39 |
|
---|
| 40 | #define DEBUG_BAJO 1 // Nivel de debug bajo
|
---|
| 41 | #define DEBUG_MEDIO 2 // Nivel de debug medio
|
---|
| 42 | #define DEBUG_ALTO 3 // Nivel de debug alto
|
---|
| 43 | #define DEBUG_MAXIMO 4 // Nivel de debug máximo
|
---|
| 44 |
|
---|
| 45 | #define CLIENTE_OCUPADO "BSY" // Cliente ocupado
|
---|
| 46 | #define CLIENTE_APAGADO "OFF" // Cliente apagado
|
---|
| 47 | #define CLIENTE_INICIANDO "INI" // Cliente iniciando
|
---|
| 48 |
|
---|
| 49 | #define CLIENTE_OPENGNSYS "OPG" // Cliente Opengnsys
|
---|
| 50 |
|
---|
| 51 | #define CLIENTE_WIN "WIN" // Cliente Windows genérico
|
---|
| 52 | #define CLIENTE_WNT "WNT" // Windows NT
|
---|
| 53 | #define CLIENTE_W2K "W2K" // Windows 2000
|
---|
| 54 | #define CLIENTE_WS2 "WS2" // Windows Server 2003
|
---|
| 55 | #define CLIENTE_WXP "WXP" // Cliente Windows XP
|
---|
| 56 | #define CLIENTE_W95 "W95" // Windows 95
|
---|
| 57 | #define CLIENTE_W98 "W98" // Windows 98
|
---|
| 58 | #define CLIENTE_WML "WML" // Windows Milenium
|
---|
| 59 | #define CLIENTE_MS2 "MS2" // MsDos
|
---|
| 60 | #define CLIENTE_WVI "WVI" // Cliente Windows Vista
|
---|
| 61 | #define CLIENTE_WI7 "WI7" // Cliente Windows 7
|
---|
| 62 |
|
---|
| 63 | #define CLIENTE_LNX "LNX" // Cliente Linux
|
---|
| 64 |
|
---|
| 65 | #define ACCION_SINRESULTADO 0 // Sin resultado
|
---|
| 66 | #define ACCION_EXITOSA 1 // Finalizada con éxito
|
---|
| 67 | #define ACCION_FALLIDA 2 // Finalizada con errores
|
---|
| 68 |
|
---|
| 69 | #define ACCION_INICIADA 1 // Acción activa
|
---|
| 70 | #define ACCION_DETENIDA 2 // Acción momentanemente parada
|
---|
| 71 | #define ACCION_FINALIZADA 3 // Accion finalizada
|
---|
| 72 |
|
---|
| 73 | #define EJECUCION_COMANDO 1
|
---|
| 74 | #define EJECUCION_PROCEDIMIENTO 2
|
---|
| 75 | #define EJECUCION_TAREA 3
|
---|
| 76 | #define EJECUCION_RESERVA 4
|
---|
| 77 |
|
---|
| 78 | #define AMBITO_CENTROS 0x01
|
---|
| 79 | #define AMBITO_GRUPOSAULAS 0x02
|
---|
| 80 | #define AMBITO_AULAS 0x04
|
---|
| 81 | #define AMBITO_GRUPOSORDENADORES 0x08
|
---|
| 82 | #define AMBITO_ORDENADORES 0x10
|
---|
| 83 |
|
---|
| 84 | // Código de los tipos de mensajes
|
---|
| 85 | #define MSG_COMANDO '1' // Mensaje del tipo comando
|
---|
| 86 | #define MSG_NOTIFICACION '2' // Respuesta a la ejecución un comando
|
---|
| 87 | #define MSG_PETICION '3' // Petición de cualquier actuación
|
---|
| 88 | #define MSG_RESPUESTA '4' // Respuesta a una petición
|
---|
| 89 | #define MSG_INFORMACION '5' // Envío de cualquier información sin espera de confirmación o respuesta
|
---|
| 90 |
|
---|
| 91 | #define ANNOREF 2009 // Año de referencia base
|
---|
| 92 |
|
---|
| 93 | #define LONGITUD_SCRIPTSALIDA 512 // Longitud máxima de la información devuelta por una función de interface
|
---|
| 94 | #define MAXARGS 16 // Número máximo de argumentos enviados a un scripts
|
---|
| 95 | #define MAXCNX 5 // Máximos intentos de conexión al servidor de Administración
|
---|
| 96 |
|
---|
| 97 | #define PUERTO_WAKEUP 9 // Puerto wake up
|
---|
| 98 |
|
---|
| 99 | #define MAXHARDWARE 128 // Máximos elementos hardware a detectar
|
---|
| 100 | #define MAXSOFTWARE 2048 // Máximos elementos software a detectar
|
---|
| 101 | // ________________________________________________________________________________________________________
|
---|
| 102 | // Tipos definidos
|
---|
| 103 | // ________________________________________________________________________________________________________
|
---|
| 104 | typedef unsigned long DWORD;
|
---|
| 105 | typedef unsigned short WORD;
|
---|
| 106 |
|
---|
| 107 | #ifndef __WINDOWS__
|
---|
| 108 | typedef int BOOLEAN;
|
---|
| 109 | typedef char BYTE;
|
---|
| 110 | typedef int SOCKET;
|
---|
| 111 | #endif
|
---|
| 112 |
|
---|
| 113 | typedef void* LPVOID;
|
---|
| 114 |
|
---|
| 115 | #define TRUE 1
|
---|
| 116 | #define FALSE 0
|
---|
| 117 |
|
---|
| 118 | #define SOCKET_ERROR (-1)
|
---|
| 119 | #define INVALID_SOCKET (SOCKET)(~0)
|
---|
| 120 |
|
---|
| 121 | #define LEER 0
|
---|
| 122 | #define ESCRIBIR 1
|
---|
| 123 |
|
---|
| 124 | #define CHARNULL '\0'
|
---|
| 125 |
|
---|
| 126 | // ________________________________________________________________________________________________________
|
---|
| 127 | //
|
---|
| 128 | // Variables globales
|
---|
| 129 | // ________________________________________________________________________________________________________
|
---|
| 130 | char szPathFileCfg[LONSTD],szPathFileLog[LONSTD];
|
---|
| 131 | int ndebug; // Nivel de debuger
|
---|
| 132 |
|
---|
| 133 | typedef struct{ // Estructura de las tramas
|
---|
| 134 | char arroba; // Caracter arroba siempre
|
---|
| 135 | char identificador[14]; // Identificador de la trama, siempre JMMLCAMDJ_MCDJ
|
---|
| 136 | char tipo; // Tipo de mensaje
|
---|
| 137 | int lonprm; // Longitud en hexadecimal de los parámetros
|
---|
| 138 | char *parametros; // Parámetros de la trama
|
---|
| 139 | }TRAMA;
|
---|
| 140 | // ________________________________________________________________________________________________________
|
---|
| 141 | // Tabla de errores
|
---|
| 142 | // ________________________________________________________________________________________________________
|
---|
| 143 | const char* tbErrores[]={"Se han generado errores. No se puede continuar la ejecución de este módulo",\
|
---|
| 144 | "001-El nombre del fichero de configuración del programa está vacío",\
|
---|
| 145 | "002-No existe fichero de configuración del programa",\
|
---|
| 146 | "003-No hay memoria suficiente para el buffer",\
|
---|
| 147 | "004-Error en el fichero de configuración del programa. No se ha definido el parámetro SERVIDORADM",\
|
---|
| 148 | "005-Error en el fichero de configuración del programa. No se ha definido el parámetro PUERTO",\
|
---|
| 149 | "006-Error en el fichero de configuración del programa. No se ha definido el parámetro USUARIO",\
|
---|
| 150 | "007-Error en el fichero de configuración del programa. No se ha definido el parámetro PASSWORD",\
|
---|
| 151 | "008-Error en el fichero de configuración del programa. No se ha definido el parámetro DATASOURCE",\
|
---|
| 152 | "009-Error en el fichero de configuración del programa. No se ha definido el parámetro CATALOG",\
|
---|
| 153 | "010-Error en los parámetros de ejecución del programa. Debe especificar el fichero de configuración",\
|
---|
| 154 | "011-Error en los parámetros de ejecución del programa. Debe especificar el fichero de log",\
|
---|
| 155 | "012-Error de sintaxis en los parámetros de ejecución del programa: Debe especificar -f nombre_del_fichero_de_configuración_del_programa -l nombre_del_fichero_de_log_del_programa -d nivel de debug",\
|
---|
| 156 | "013-Error al crear socket ***socket() fallo",\
|
---|
| 157 | "014-Error al enlazar socket al interface ***bind() fallo",\
|
---|
| 158 | "015-Error al acceptar conexión de clientes ***accept() fallo",\
|
---|
| 159 | "016-Error al crear hebra de cliente en módulo main()",\
|
---|
| 160 | "017-Error al recibir trama ***recv() fallo",\
|
---|
| 161 | "018-No se reconoce el mensaje enviado",\
|
---|
| 162 | "019-Trama recibida NO válida",\
|
---|
| 163 | "020-No se puede establecer conexión con la base de datos",\
|
---|
| 164 | "021-No se han podido recuperar los datos de la consulta o bien insertar, modificar o eliminar datos",\
|
---|
| 165 | "022-El cliente no se ha sido dado de alta en la base de datos del sistema. Se rechaza su petición de inclusión",\
|
---|
| 166 | "023-Ha habido algún problema en la incorporación automática del cliente",\
|
---|
| 167 | "024-Ha habido algún problema en la actualización de la configuración del cliente",\
|
---|
| 168 | "025-La tabla de clientes está llena, no pueden registrarse más clientes en el sistema",\
|
---|
| 169 | "026-Error al enviar trama ***send() fallo",\
|
---|
| 170 | "027-No se encuentra Repositorio del cliente",\
|
---|
| 171 | "028-Ha ocurrido algún error al tomar las particiones",\
|
---|
| 172 | "029-Ha ocurrido algún problema en el proceso de inclusión del cliente. Se rechaza su petición",\
|
---|
| 173 | "030-Ha ocurrido algún problema en el proceso de respuesta al comando",\
|
---|
| 174 | "031-No se ha encontrado la acción a notificar es posible que se haya eliminado el registro",\
|
---|
| 175 | "032-Ha ocurrido algún problema en el envío del comando",\
|
---|
| 176 | "033-Error en el fichero de configuración del programa. No se ha definido el parámetro PATHSCRIPTS",\
|
---|
| 177 | "034-Error en el fichero de configuración del programa. No se ha definido el parámetro URLMENU",\
|
---|
| 178 | "035-Error en el fichero de configuración del programa. No se ha definido el parámetro URLMSG",\
|
---|
| 179 | "036-No se ha podido recuperar la configuración de las particiones del disco",\
|
---|
| 180 | "037-Ha ocurrido algún problema en el proceso de inclusión del cliente",\
|
---|
| 181 | "038-No se ha podido establecer conexión con el Servidor de Administración",\
|
---|
| 182 | "039-Ha ocurrido algún problema al procesar la trama recibida",\
|
---|
| 183 | "040-Se han recibido parámetros con valores no válidos",\
|
---|
| 184 | "041-Ha ocurrido algún problema en el proceso de inclusión del cliente",\
|
---|
| 185 | "042-Ha ocurrido algún problema al enviar una petición de comandos o tareas pendientes al Servidor de Administración",\
|
---|
| 186 | "043-Ha ocurrido algún problema al enviar una petición de comandos interactivos al Servidor de Administración",\
|
---|
| 187 | "044-Ha ocurrido algún problema al enviar una respuesta de comandos al servidor",\
|
---|
| 188 | "045-Ha ocurrido algún problema al recibir una petición de comandos o tareas pendientes desde el Servidor de Administración",\
|
---|
| 189 | "046-Ha ocurrido algún problema al recibir un comando interactivo desde el Servidor de Administración",\
|
---|
| 190 | "047-El cliente no está registrado en la tabla de sockest del sistema",\
|
---|
| 191 | "048-Error al configurar opción BROADCAST para socket: setsockopt(SO_BROADCAST)",\
|
---|
| 192 | "049-Error al enviar trama magic packet",\
|
---|
| 193 | "050-Ha ocurrido algún problema al enviar un fichero por la red",\
|
---|
| 194 | "051-Error en el fichero de configuración del programa. No se ha definido el parámetro PATHLOGFIL",\
|
---|
| 195 | "052-No se puede crear archivo temporal para ejecución de Comandos",\
|
---|
| 196 | "053-Ha ocurrido algún problema al procesar el Inventario Hardware del cliente",\
|
---|
| 197 | "054-Existe un tipo de hardware que no está registrado",\
|
---|
| 198 | "055-Ha ocurrido algún problema al actualizar el hardware del cliente",\
|
---|
| 199 | "056-Error en el fichero de configuración del programa. No se ha definido el parámetro PATHINTERFACE",\
|
---|
| 200 | "057-Ha ocurrido algún problema al enviar un archivo por la red",\
|
---|
| 201 | "058-Ha ocurrido algún problema al recibir un archivo por la red",\
|
---|
| 202 | "059-Error al crear la hebra DHCP o BOOTP",\
|
---|
| 203 | "060-Error al crear la hebra TFTP",\
|
---|
| 204 | "061-Error al crear socket para servicio DHCP",\
|
---|
| 205 | "062-Error al enlazar socket con interface para servicio DHCP",\
|
---|
| 206 | "063-No hay puertos libres para la hebra del servicio",\
|
---|
| 207 | "064-Error al crear estructura de control para protocolo DHCP",\
|
---|
| 208 | "065-Error al recibir mensaje DHCP. Se para el servicio",\
|
---|
| 209 | "066-Error al crear la hebra cliente DHCP",\
|
---|
| 210 | "067-Error al crear socket para servicio BOOTP",\
|
---|
| 211 | "068-Error al enlazar socket con interface para servicio BOOTP",\
|
---|
| 212 | "069-Error al crear estructura de control para protocolo BOOTP",\
|
---|
| 213 | "070-Error al recibir mensaje BOOTP. Se para el servicio",\
|
---|
| 214 | "071-Error al crear la hebra cliente BOOTP",\
|
---|
| 215 | "072-Error al crear socket para servicio TFTP",\
|
---|
| 216 | "073-Error al enlazar socket con interface para servicio TFTP",\
|
---|
| 217 | "074-Error al crear estructura de control para protocolo TFTP",\
|
---|
| 218 | "075-Error al recibir mensaje TFTP. Se para el servicio",\
|
---|
| 219 | "076-Error al crear la hebra cliente TFTP",\
|
---|
| 220 | "077-No se encontró opción DHCP",\
|
---|
| 221 | "078-ERROR TFTP",\
|
---|
| 222 | "079-Error al recibir mensaje TFTP en hebra cliente",\
|
---|
| 223 | "080-Error al recibir mensaje DHCP",\
|
---|
| 224 | "081-Error al crear socket de usuario para hebra",\
|
---|
| 225 | "082-Ha ocurrido algún problema al procesar el Inventario software del cliente",\
|
---|
| 226 | "083-Ha ocurrido algún problema al actualizar el software del cliente",\
|
---|
| 227 | "084-Ha ocurrido algún problema al reiniciar la sesión del cliente",\
|
---|
| 228 | "085-No se ha podido recuperar la dirección IP del cliente",\
|
---|
| 229 | "086-Error al ejecutar el comando",\
|
---|
| 230 | "087-Error al leer o escribir el contenido del archivo de eco de consola remota",\
|
---|
| 231 | "088-Ha habido algún problerma al procesar la caché",\
|
---|
| 232 | "089-Error en el fichero de configuración del programa. No se ha definido el parámetro URLMENU",\
|
---|
| 233 | "090-Error en el fichero de configuración del programa. No se ha definido el parámetro URLMSG",\
|
---|
| 234 | "091-Ha habido algún problema al enviar un mensaje de tipo petición al Servidor",\
|
---|
| 235 | "092-Error en el fichero de configuración del programa. No se ha definido el parámetro IPLOCAL",\
|
---|
| 236 | "093-No se puede caragar la libraria Windows para rabajar con sockets",\
|
---|
| 237 | };
|
---|
| 238 | // ________________________________________________________________________________________________________
|
---|
| 239 | // Tabla de mensajes
|
---|
| 240 | // ________________________________________________________________________________________________________
|
---|
| 241 | const char* tbMensajes[]={"",\
|
---|
| 242 | "001-Inicio de sesion",\
|
---|
| 243 | "002-Petición de inclusión de cliente",\
|
---|
| 244 | "003-Abriendo sesión en el servidor de Administración",\
|
---|
| 245 | "004-Cliente iniciado",\
|
---|
| 246 | "005-Ejecución de archivo Autoexec",\
|
---|
| 247 | "006-Procesa comandos pendientes",\
|
---|
| 248 | "007-Acciones pendientes procesadas",\
|
---|
| 249 | "008-Ejecución del script",\
|
---|
| 250 | "009-Parámetro del script",\
|
---|
| 251 | "010-Ha ocurrido algún error en la creación del proceso hijo",\
|
---|
| 252 | "011-Aviso: La información de salida del script excede de la longitud permitida. Puede haberse truncado",\
|
---|
| 253 | "012-Información devuelta por el script",\
|
---|
| 254 | "013-Estatus de finalización del script",\
|
---|
| 255 | "014-Configuración de particiones",\
|
---|
| 256 | "015-Enviando petición de inclusión en el sistema al Servidor de Administración",\
|
---|
| 257 | "016-Recibiendo respuesta de inclusión desde el Servidor de Administración",\
|
---|
| 258 | "017-Enviando petición de comandos o tareas pendientes al Servidor de Administración",\
|
---|
| 259 | "018-Recibiendo respuesta de comandos o tareas pendientes desde el Servidor de Administración",\
|
---|
| 260 | "019-Disponibilidad de comandos activada",\
|
---|
| 261 | "020-Disponibilidad de comandos desactivada",\
|
---|
| 262 | "021-Ejecución de comando",\
|
---|
| 263 | "022-Sin eco",\
|
---|
| 264 | "023-Procesando caché",\
|
---|
| 265 | "024-Repositorio iniciado",\
|
---|
| 266 |
|
---|
| 267 | };
|
---|
| 268 | // ________________________________________________________________________________________________________
|
---|
| 269 | // Prototipo de funciones
|
---|
| 270 | // ________________________________________________________________________________________________________
|
---|
| 271 | char *desencriptar(char *,int*);
|
---|
| 272 | char *encriptar(char *,int*);
|
---|
| 273 | struct tm * tomaHora();
|
---|
| 274 | void registraLog(const char *,const char *,int );
|
---|
| 275 | void errorLog(const char *,int ,int);
|
---|
| 276 | void errorInfo(const char *,char *);
|
---|
| 277 | void infoLog(int);
|
---|
| 278 | void infoDebug(char*);
|
---|
| 279 | BOOLEAN validacionParametros(int,char**,int);
|
---|
| 280 | char* reservaMemoria(int);
|
---|
| 281 | char* ampliaMemoria(char*,int);
|
---|
| 282 | void liberaMemoria(char*);
|
---|
| 283 | BOOLEAN initParametros(TRAMA*,int);
|
---|
| 284 | int splitCadena(char **,char *, char);
|
---|
| 285 | void sustituir(char *,char ,char );
|
---|
| 286 | char* StrToUpper(char *);
|
---|
| 287 | char* StrToLower(char *);
|
---|
| 288 | void INTROaFINCAD(TRAMA*);
|
---|
| 289 | void FINCADaINTRO(TRAMA*);
|
---|
| 290 | int cuentaIPES(char*);
|
---|
| 291 | char *tomaParametro(const char*,TRAMA*);
|
---|
| 292 | char *copiaParametro(const char*,TRAMA *);
|
---|
| 293 | BOOLEAN contieneIP(char *,char *);
|
---|
| 294 | char* rTrim(char *);
|
---|
| 295 | SOCKET TCPConnect(char *,char *);
|
---|
| 296 | SOCKET abreConexion(void);
|
---|
| 297 | BOOLEAN enviaMensaje(SOCKET *,TRAMA *,char);
|
---|
| 298 | TRAMA* recibeMensaje(SOCKET *);
|
---|
| 299 | BOOLEAN mandaTrama(SOCKET*,TRAMA*);
|
---|
| 300 | BOOLEAN sendData(SOCKET *, char* ,int );
|
---|
| 301 | BOOLEAN enviaTrama(SOCKET *,TRAMA *);
|
---|
| 302 | TRAMA* recibeTrama(SOCKET*);
|
---|
| 303 | BOOLEAN recData(SOCKET *,char*,int,int*);
|
---|
| 304 | BOOLEAN sendFlag(SOCKET *, char* ,int );
|
---|
| 305 | BOOLEAN recibeFlag(SOCKET*,TRAMA*);
|
---|
| 306 | char* URLEncode(char *);
|
---|
| 307 | char* URLDecode(char *);
|
---|
| 308 | char* leeArchivo(char*);
|
---|
| 309 | int lonArchivo(char *);
|
---|
| 310 | BOOLEAN escribeArchivo(char *,char*);
|
---|
| 311 | BOOLEAN sendArchivo(SOCKET *,char *);
|
---|
| 312 | BOOLEAN recArchivo(SOCKET *,char *);
|
---|
| 313 | SOCKET TCPConnect(char *,char*);
|
---|
| 314 |
|
---|