Archivo de la categoría: PHP

Todo lo relacionado con la programación en PHP

Clase para sumar tiempos con PHP – PHP Tips

En un proyecto en el que estoy trabajando me he encontrado con que tenia que sumar tiempos y devolver el tiempo final, como no encontré ninguna función que se adaptara al 100% programe la siguiente clase. He creado un Gist para quien la pueda necesitar o interesar.


<?php
/**
* SumaTiempos.php Clase para sumar tiempos
*
* Esta clase sirve para teniendo un tiempo inicial y un Array con tiempos
* los suma y nos devuelve el tiempo final
*
* PHP Version 5.3
*
* @author Ruben Lacasa Mas <ruben@rubenlacasa.es>
* @copyright 2013 Ruben Lacasa Mas http://rubenlacasa.es
* @license http://creativecommons.org/licenses/by-nc-nd/3.0
* CC-BY-NC-ND-3.0
* @link https://gist.github.com/sbarrat/5642808
*/
class SumaTiempos
{
/**
* Inicializamos el array que almacenara los valores del tiempo
* @var Array
*/
private $tiempo = array();
/**
* Constructor de clase
*
* @param String $tiempo Tiempo que definimos
*/
public function __construct($tiempo = '00:00:00.00')
{
list(
$this->tiempo['horas'],
$this->tiempo['minutos'],
$this->tiempo['segundos']
) = explode(":", $tiempo);
}
/**
* Sumamos el tiempo a otro
*
* @param SumaTiempos $tiempo Tiempo que queremos agregar
*/
public function sumaTiempo(SumaTiempos $tiempo)
{
// Segundos
$this->tiempo['segundos']
= floatval($this->tiempo['segundos'])
+ floatval($tiempo->tiempo['segundos']);
$minutos = intval($this->tiempo['segundos'] / 60);
$this->tiempo['segundos'] -= $minutos * 60;
// Minutos
$this->tiempo['minutos']
= intval($this->tiempo['minutos'])
+ intval($tiempo->tiempo['minutos'] + $minutos);
$horas = intval($this->tiempo['minutos'] / 60);
$this->tiempo['minutos'] -= $horas * 60;
// Horas
$this->tiempo['horas']
= intval($this->tiempo['horas'])
+ intval($tiempo->tiempo['horas'] + $horas);
}
/**
* Formatea y devuelve el tiempo final
*
* @return String Tiempo final formateado
*/
public function verTiempoFinal()
{
return sprintf(
'%02s:%02s:%05s',
$this->tiempo['horas'],
$this->tiempo['minutos'],
$this->tiempo['segundos']
);
}
}
/**
* Ejemplo de uso
*/
/**
* Array de tiempos que queremos sumar
* @var Array
*/
$tiempos = array('10:21:54.35', '01:12:44.56', '20:01:12.93', '00:28:18.56');
// Inicializamos el tiempo
$tiempoInicial = new SumaTiempos();
// Recorremos los tiempos y los vamos sumando
foreach ($tiempos as $parcial) {
$tiempoInicial->sumaTiempo(new SumaTiempos($parcial));
}
// Mostramos el tiempo final
echo $tiempoInicial->verTiempoFinal(); // En este ejemplo 32:04:10.40

view raw

SumaTiempos.php

hosted with ❤ by GitHub

Crear un tunel para la depuración – Zend Tips

En mi dia a dia suelo trabajar con Zend Studio como IDE de desarrollo y el servidor es Zend Server en el cual funciona el Zend Debugger. Normalmente mientras estoy desarrollando estoy en la misma red que el servidor, por lo tanto no necesito crear un túnel para depurar las aplicaciones ya que tengo configurado el Zend Debugger para que envie los datos a mi estación de trabajo. Pero si alguna vez tengo que desarrollar fuera de la red de trabajo y necesito depurar necesito crear un túnel con el servidor para que pueda existir una comunicación entre el Zend Studio y el Zend Server. Para realizar esto creo un túnel SSH. Este se crea de la siguiente manera:


user@workstation:~> ssh -R 10137:127.0.0.1:10137 user@debugserver
user@debugserver's password: <enter user's password on the debug server>
Welcome to the Debug Server!
user@debugserver:~$
Esto sirve desde tanto para un terminal Linux com MacOS X, o cualquier *nix.
En el caso de tener que hacerlo desde un terminal Windows utilizaríamos el Putty.

Problemas con cliente SOAP – PHP Tips

En uno de los proyectos en los cuales estoy trabajando ahora mismo, necesitaba la implementación de un entorno cliente – servidor SOAP. El servidor, vamos a decir que debido a motivos de ecosistema, lo implemente con Delphi, y el cliente con PHP.

Bien, una vez el entorno implementado, y después de hacer las pruebas pertinentes entre el cliente y servidor, los que yo denomino test de «hola mundo», me puse a implementar los servicios que necesitaba, y aquí comenzaron los problemas.

Aparentemente estaba todo bien, pero el servidor me devolvia este error:

Uncaught SoapFault exception: [Client] Function function_nameis not a valid method for this service

Pues bien todo el problema venia que el cliente tenia almacenado en cache los servicios anteriores, y no los nuevos. Podemos solucionar este error de varias maneras. Una seria deshabilitando la cache wsdl en tiempo de ejecución, asi prevenimos que si implementamos servicios nuevos, el cliente no lo reconozca. Que decir que esta opción esta bien en desarrollo, ya que una vez que este todo establecido definitivamente lo podemos deshabilitar.

ini_set("soap.wsdl_cache_enabled", "0");

Otra manera de  hacerlo seria, si usamos la clase SoapClient para crear el cliente SOAP, es pasándolo como parámetro en la creación del objeto

$clienteSoap = new SoapClient('http://misitio.com?wsdl', array('cache_wsdl' => WSDL_CACHE_NONE));

Configuración SOAP en tiempo de ejecución
Clase SoapClient

Averiguar el genero de un nombre con PHP – PHP Tips

El otro día tuve un pequeño problema, tenia que diseñar un formulario, en el cual se recogen los datos desde una base de datos de nombres para luego procesarlos en otro sitio, bien pues cuando obtenía los datos quería que me apareciera el tratamiento (Don o Doña) de la persona que estaba seleccionando y esto quería hacerlo automáticamente.  Una de las primeras cosas que hago cuando me suceden estos dilemas es ponerme a programarlo por mi cuenta, gran fallo, ya que una de las cosas que tenemos que pensar es que esto es una situación habitual y seguramente hay alguna función que nos permita hacerlo de una manera sencilla.

Antes de ponerme a pensar que existiera esa función intente crear una función para solucionar este problema de una manera sencilla. En principio todos los nombres que tengo en la base de datos están en castellano,  entonces pense, a ver la inmensa mayoría de los nombres de chica finalizan con ‘a’, pues bien voy a crear una función para obtener el ultimo carácter del nombre y si es ‘a’ es una chica, si no es un chico. Esta no es una buena solución, aunque tengo que decir que en la mayoría de los casos acierta, pero tenemos excepciones, por ejemplo, en el caso de las chicas (Rocio, Isabel,..) y en el caso de los chicos (Borja). Podríamos optar por agregar una array de excepciones, y cada vez que nos ocurriera una ‘mala coincidencia’ agregar esa excepción. Esta función quedaria de esta manera.

function detectaGenero($nombre) {
    $genero = "Chico";
    $excepcionesChico = array('Borja',);
    $excepcionesChica = array('Rocio','Isabel');
    $letra = substr($nombre, -1);
    if (($letra == 'a' || in_array($nombre, $excepcionesChica)) && !in_array($nombre, $excepcionesChico)) {
        $genero = "Chica";
    }
    return $genero;
}

Si esta función la ponemos a comparar nombres en otro idioma no nos serviria de nada, y luego seria tambien un inconveniente tener que estar cada vez que encontráramos una excepción agregarla al array.

La buena noticia es que existe una extensión PECL Gender, que una vez instalada en nuestro servidor nos permite llamar a la clase Gender y poder conocer el genero de ese nombre. El inconveniente, que es una extensión PECL y si nuestra aplicación web, esta en un hosting en el cual no podemos agregar este tipo de extensiones no nos sirve de nada, a no ser que el proveedor quiera instalarla.

Esta extensión es un port del programa gender.c escrito originalmente por Joerg Michael. Su base de datos de nombres contiene mas de 40000 nombres de 54 países.
La función anterior usando Gender quedaria así:

function detectaGenero($nombre) {
    $genero = "Chico";
    $gender = new Gender;
    $result = $gender->get($nombre);
    if ($result === Gender::IS_FEMALE OR $result === Gender::IS_MOSTLY_FEMALE) {
        $genero = "Chica";
    }
    return $genero;
}

Como podemos ver es mas sencilla que la anterior, mas fiable, y no tenemos que estar agregando excepciones, y pensar en los nombres en otros idiomas.
PHP: Gender – Manual

Recuperar site offline Drupal

Si por algún motivo se nos queda nuestro sitio de Drupal en modo offline, y no podemos acceder para recuperar el estado normal, ejecutando la siguiente consulta SQL conseguiremos que nuestro sitio vuelva a estar disponible.

 

UPDATE variable SET value = 's:1:"0";' WHERE name= 'site_offline';
DELETE FROM cache WHERE cid = 'variables';

 

Capitalizando utf-8 – PHP Tips

Una de las cosas con las que nos tenemos que pelear muchas veces es el tema de la codificación de caracteres, generalmente esto lo solucionamos con el utf8_encode o el utf8_decode, pero en algún caso puede suceder que nos sirva pero que nos salgan resultados no deseados. Pongamos el caso siguiente, tenemos en la base de datos el nombre y apellidos de un cliente y lo tenemos en mayúsculas, y queremos recuperar ese dato para mostrarlo y lo queremos mostrar capitalizado, para realizar esto haríamos lo siguiente:


$string = "RUBÉN YAGÜE ESPAÑA";

$string = ucwords(strtolower(utf8_encode($string)));

echo $string; // Mostraria RubÈn YagÜe EspaÑa

Seguramente si nos vemos en este escenario lo mas rápido seria optar por mostrarlo todo en mayúsculas y ya, pero existe una manera de hacerlo bien.

Buscando por las funciones de PHP he encontrado las funciones mb_* las cuales nos van a ayudar a conseguir nuestro objetivo.

A continuación pongo la función que me he creado para realizar esto

function capitalize($string)
{
    if (mb_detect_encoding($string) === 'UTF-8') {
        $string = mb_convert_case(utf8_encode($string), MB_CASE_TITLE, 'UTF-8');
    } else {
        $string = mb_convert_case($string, MB_CASE_TITLE, 'UTF-8');
    }
    return $string;
}

Esta función me detecta si la codificación de la cadena es UTF-8, si lo es utilizando la función
mb_conver_case y codificando la cadena con utf8_encode nos creara la cadena final capitalizada
correctamente.

Diferencias entre las comillas dobles y las simples en PHP – PHP Tips

Los que nos dedicamos a programar tenemos mas o menos claro cuando usar las comillas dobles o las simples a la hora de establecer el valor de las variables,  e intentamos mantener una uniformidad en nuestra programación, yo por ejemplo use comillas simples o dobles siempre pongo las variables concatenadas y nunca las pongo dentro de las comillas. Bien, para los que se pregunten cual es la diferencia fundamental entre las dobles y las simples la respuesta es que cuando se usan comillas simples las variables no se sustituye el valor de la variable por su valor, en cambio con las comillas dobles si se sustituye, es decir:


$saludo = 'Hola Mundo';

echo 'Y digo al mundo $saludo'; //Muestra: Y digo al mundo $saludo

echo &quot;Y digo al mundo $saludo&quot;; // Muestra: Y digo al mundo Hola Mundo

Tambien tenemos que tener cuidado si en nuestra cadena ponemos caracteres especiales como el fin de párrafo o salto de linea, ya que con las comillas simples se representaran literalmente y con las comillas dobles se generaran.


$linea = 'Hola soy la primera linea\r\nHola soy la segunda'; //No realizara el saldo de linea

$linea = &quot;Hola soy la primera linea\r\nHola soy la segunda&quot;; //Realizara el salto de linea

LOAD DATA INFILE desde un terminal remoto – MySQL Tips

Siguiendo con las entradas relacionadas con mis notas mentales, hoy me he vuelto a cruzar con una situación que se me plantea de vez en cuando y siempre tengo que tirar de la documentación.

El escenario es el siguiente, tengo un programa en php que se ejecuta en terminal que me realiza una consulta e extracción de datos de una base de datos paradox que esta en un servidor, si he dicho paradox, y me genera un fichero csv el cual a continuación por medio de una consulta de load data infile se me vuelca a una base de datos mysql que tengo en otro servidor. Resulta que la consulta load data infile no he conseguido que se me ejecute, en cuando lo solucione publicare la solución así que lo que hago es una vez terminado el proceso de volcado me conecto al servidor vía terminal y ejecuto la consulta load data infile desde la terminal. Si esto lo hago de esta manera y no ejecutando la consulta directamente es debido a lo lento que funciona el paradox. El proceso de volcado de los datos de paradox al fichero oscila entre los 8-10 minutos, realizando la operación directa el proceso duraba entre 45 minutos a 1 hora.

Primero me conecto al servidor remoto mysql con un usuario y contraseña que tenga permisos para acceso remoto, añadiendo al final la opción –local-infile=1 para poder subir ficheros


#mysql -u [usuario] -p -h [host remoto] --local-infile=1

A continuación ejecuto la consulta para que me vuelque el fichero


LOAD DATA LOCAL INFILE [fichero] REPLACE INTO TABLE [db.table]     CHARACTER SET [tipo caracteres]    FIELDS TERMINATED BY  [terminación campos]     ([columna1], [columna2], ..);

En [fichero] ponemos entre comillas la ruta completa del fichero en nuestro ordenador,  en [db.table] el nombre de la base de datos y la tabla en la que vamos a volcar los datos, en tipo carácter ponemos entre comillas el tipo de codificación p.e cp1250, utf8, en [terminación campos] como están delimitados los campos, p.e ; y finalmente entre paréntesis las columnas del la tabla.

Una vez ejecutado puedo decir que en mi caso se volcaron 42673 registros en 0.53 segundos, esta misma operación la he realizado generando un fichero sql para la inserción de los datos y el tiempo era infinitamente superior a este.

 

PhpStorm 5.0: PHP Coding at Its Finest

Ya esta aqui la nueva versión de PhpStorm

PhpStorm 5.0: PHP Coding at Its Finest.

Git “Command not found” Error in Mountain Lion [Quickfix]

git icon, created for the Open Icon Library

Si usais git via comando o lo usais a traves de alguna herramienta y habeis actualizado vuestro mac a Mountain Lion, seguramente os encontrareis con este problema. En este link esta como poder solucionarlo

Git “Command not found” Error in Mountain Lion [Quickfix].