Lenguaje y Normas de Desarrollo

He aquí algunas normas sobre el lenguaje, la sintaxis y las norma de desarrollo en vigor para el proyecto Dolibarr:

= Versiones =


 * Dolibarr debe funcionar con:
 * 1) Todos los SO (Windows, Linux, MACOS...)
 * 2) PHP  (Debe funcionar sin ningún módulo PHP complementario, exceptuando los módulos de acceso a la base de datos).
 * 3) Mysql

= Normas de Copyright =
 * Todos los archivos PHP deben empezar con una cabezara como

Si edita un archivo existente del proyecto, debe añadir una línea de Copyright debajo de las otras.

= Normas PHP =


 * Dolibarr está escrito en PHP y soporta todas las versiones de PHP superiores a la . Todos los archivos deben de contener la extensión .php

Los otros operadores ($HTTP_SERVER_GET, ...) están siendo depreciados en el seno de PHP, no deben de ser ya utilizados. Por lo tanto, el código debe funcionar incluso cuando la opción register_long_arrays esté en off. Además de que el código debe funcionar cuando la opción de PHP register_globals esté en off (recomendado por PHP), también lo deberá hacer cuando la opción register_globals esté en on (por defecto en muchas instalaciones).
 * La llamada a las variables superglobales PHP deben pasar a través de los operadores dedicados $_COOKIES, $_SERVER, $_ENV y a través de la función dolibarr GETPOST para obtener el contenido de$_GET o $_POST...


 * Los smart tags PHP no són utilizados. Las secciones de código deben de empezar por <?php


 * No se permite el uso de la variable PHP_SELF. Utilice en su lugar $_SERVER["PHP_SELF"]

en lugar de que es menos eficiente.
 * Cuando varias variables deben ser inicializadas con el mismo valor, utilice múltiples líneas


 * las cadenas deben de ser encuadradas con comillas simples y las variables sacadas de la cadena.


 * Los comentarios deben seguir la sintaxis de C, es decir, una doble barra invertida para una línea de comentario y el uso de una barra con arterisco para abrir un bloque de varias líneas


 * Los ficheros deben ser guardados en formato Unix (LF) y no en formato Windows (CR/LF). El formato Unix es compatible con los SO Unix, Windows, Mac, mientras que el formato de archivo de texto de Windows es problemático en algunos PHP en Unix.


 * Las funciones deben devolver >0 en caso de éxito, y un número <0 en caso de error.


 * Sin código muerto (no usado) en el código principal de Dolibarr (El código usado únicamente por un módulo externo debe ser incluido en el módulo externo).


 * Usar "include_once" para la inclusión de todo archivo que contenga definiciones de funciones o de clases (archivos *.class.php y *.lib.php), usar "include" para la inclusión de archivos de tipo template o que contenga una mezcla de HTML y PHP (archivos *.inc.php et *.tpl.php).

= Normas SQL =

Tabla structura

 * Todas las tablas tienen prefijo para evitar conflictos de nombres. A día de hoy, el prefijo está determinado y no se puede cambiar. Su valor es llx_. Sin embargo, está pendiente de considerar, en una futura versión, el poder cambiarlo en el momento de la instalación.

- integer para cualquier foreign key - double(24,8) para cualquier importe - double(6,3) para las tasas de IVA - real para una cantidad - varchar para una cadena (incluidos los de longitud 1, el tipo char se está depreciando) - timestamp para fecha+hora que deba ser actualizado automáticamente - datetime para fecha+hora - date para fecha
 * En la práctica, con la finalidad de set compatible con todas las precisiones de los paises, utilizaremos los siguientes tipos:

Clave primaria
La clave primaria de una tbla se llama rowid.

Hay unas pocas tablas que incumplen la norma ahora (ej Table llx_c_actioncomm), por razones históricas. Podrá ser estudiada una evolución para una próxima versión.

Clave secundaria
El nombre de una clave secundaria empieza con el prefijo fk_ seguido por el nombre de la tabla vinculada (necesario para evitar duplicaciones globales en la base de datos, problemáticas en algunos RDBMS como postgresql), seguido del campo vinculado (para permitir diferentes claves secundarias en la misma tabla).

Ejemplo: fk_facture_fourn_fk_soc

Clave alternativa
A veces no sólamente la clave primaria debe ser única. Por lo tanto, podemos añadir un índice de clave única alternativa. Este índice es llamado por un nombre que comienza con el prefijo uk_ seguido por el nombre de la tabla (necesario para evitar la duplicación de índeces globales en la base de datos, problemáticos en algunos DBMS como postgresql) y de un sufijo que caracterice a la clave (para permitir múltiples índices en una sola tabla).

Ejemplo: uk_societe_code_client

Indices de rendimiento
Algunos campos se utilizan a menudo como un criterio para la búsqueda, clasificación o afiliación. Conviene en este caso, poner un índice de rendimiento. Tales índices se llamarán con un prefijo idx_ seguidos del nombre de la tabla y el nombre del campo en el que el índice trabaja.

Ejemplo: idx_societe_user_creat

Formato de fichero DDL
Los ficheros que contienen la definición de la estructura de la base de datos (DDL) deben de estar repartidos en dos tablas: Donde habrá un comentario junto a cada campo para explicar su significado.
 * Cada tabla se define en su propio archivo cuyo nombre es llx_mitabla.sql
 * Claves de relacionales y el índice de rendimiento o otras constraints de carácter único se definen en un archivo cuyo nombre es llx_mitable.key.sql

Estos ficheros se guardan el la carpeta install/mysql/tables para los ficheros standards ó mimodulo/tables para las tablas llevadas por un módulo.

Ejemplos Fichero tabla llx_matable.sql:

Ejemplos Fichero de claves/índices llx_matable.key.sql

SQL Coding rules
Dentro de los selects pueden utilizar alias para simplificar la redacción de las consultas: Sin embargo, no debemos utilizar este alias en las consultas de actualización porque no es compatible con mysql 3.1.
 * Uso de alias / nombres de los campos


 * ¡Los SELECT * están prohibidos! Cada Select deberá especificar la lista completa de los campos para recuperar. Esto permitirá evitar confusiones. Ejemplo:


 * En las consultas SQL, entrecomillaremos los campos, pero no los numéricos que contengan cantidades que se almacenen en los campos de tipo double o real. Los entrecomillados en los numéricos de tipo float se almacenan con un valor diferente. Por ejemplo 412.62 dentro del insert se guardara con el valor 412.61999512 en base si el campo es de tipo double(24,8). Y PHP sólo ve 412.61999512. Otras herramientas verán 412.62 dando la impresión de que no hay problemas. Y PHP tiene razón, hay un problema en base. Mediante la eliminación de las comillas en los numéricos, esto irá mejor.

Ejemplo:

Nota, el problema de los float es general y no sólamete en el acceso a la base de datos, está presente en todos los lenguajes cuando trabajamos sobre números reales, por lo que deberán ser limpiados por la función price2num con el segundo parámetro rellenado con: 'MU', 'MT' ó 'MS' según sea necesario. Consulte el capítulo "Los números reales, importes y cálculos" más abajo.


 * las funciones NOW, SYSDATE ó IFNULL están prohibidas en las órdenes de SQL. Si se necesita indicar la fecha del momento en un campo, el valor debe provenir de PHP y no del motor de la base de datos. Esto además de ofrecer una mejor portabilidad del código, también permite una gestión correcta de la zona horaria.

Reglas de migración de base de datos

 * Los scripts de migración de estructura se encuentran en el directorio install/mysql/migration/a.b.c-x.y.z.sql. En un archivo de migración de n a n+1, no destruiremos una tabla (DROP TABLE) que no sea utilizada en n+1, se realizará en la migración de la versión n+1 a n+2. Esto tiene el fin de conservar las tablas depreciadas durante una versión para poder reparar una situación de error de migración.

Particularidades de MySQL
En efecto, este formato soporta claves secundarias, las eventuales restricciones que se adjuntan y la noción de integridad de transacciones requeridas para obteter datos coherentes entre tablas en la base de datos.
 * Las tablas deben de estar en formato InnoDB.

Para activarla, añada la siguiente línea en su archivo config del servidor MySql (my.cnf o my.ini)
 * Dolibarr debe trabajar incluso con la opción strict de Mysql activa.

Particularidades de PostgreSQL
Sólo los archivos de MySQL deben mantenerse. Los archivos para otras bases de datos se convierten "al vuelo" por el gestor de la base de datos.

= Normas HTML =


 * Todos los atributos de las etiquetas HTML deben de estar *en minúscula* y entrecomillados con *doble comilla*.

Por ejemplo:
 * Los enlaces href deben de ser absolutos. para las páginas, basarse en la function dol_buildpath. Para las imágenes basarse en la llamada de la función img_picto.


 * Las tablas HTML deben tener las columnas sin valores forcados, a excepción de las columnas que contienen información cuya longitud no es variable. Por ejemplo, una columna con una imagen puede forzarse a un with="20px". En los otros casos, hay que evitar forzar la longitud de las columnas. Esto es porque en la mayoría de los casos el navegador realiza un buen trabajo al definir automáticamente la longitud de las columnas, independientemente de la resolución de la pantalla.


 * Javascript/ajax y la llamada a scripts java en las páginas php está fuera de las normas. Si de todas maneras se incluye código javascript, debe estar condicionado mediante el test "$conf->use_javascript_ajax"


 * Las ventanas emergentes no deben de ser utilizadas, exceptuando los tooltips (y quedan condicionados por el punto anterior).


 * Los scripts externos son escritos en Perl si no pueden serlo en php, la utilización de otro lenguaje no está prohibido, pero debe de ser discutido priero en la lista de correo de desarrollo.

= Normas Dolibarr y plantillas de código =

Plantillas de código
Para estandarizar el código y acelerar el desarrollo de nuevos componentes en Dolibarr, en la carpeta dev/skeletons encontraremos 4 plantillas de código preparadas.


 * 1 que sirve de ejemplo de descriptor de módulo: myModule.class.php
 * 1 que sirve de ejemplo de código para crear una nueva clase: skeleton_class.class.php
 * 1 que sirve de ejemplo de código para crear una nueva página: skeleton_page.php
 * 1 que sirve de ejemplo de código para crear una nuevo script a ejecutar en línea de comandos: skeleton_script.php

Úsenlas como ejemplo. Tenga en cuenta que estas plantillas también son utilizadas por el generador de código PHP que se describe en el capítulo de Desarrollo de módulos Dolibarr módulo para acelerar su desarrollo.

Las fechas y la Zona Horaria
Dolibarr es una aplicación multi-usuario y multi-ubicación. Por lo tanto, es necesario almacenar las fechas en el formato correcto. Para evitar problemas de conversión, deben aplicarse las siguientes reglas: Por lo tanto, el 1 de enero de 1970, a las 3 en París (TZ=+1) = las 2 en Greenwitch (TZ=0) será almacenado en memora 7200 y será enviado a la base de datos con la cadena '19700101030000' (PHP convierte en hora de su TZ y la base de datos la deconvierte con su TZ que es la misma que la de PHP).
 * Una fecha en memoria debe de estar en formato Timestamp GMT.
 * Una fecha guardada en la base de datos contiene el Timestamp GMT en relación con la fecha que figura en la consulta en hora local del servidor PHP. No se trata de las fechas de actualización de la base de datos (campo tms en la base de datos).

Los métodos select deben traducir los campos fecha al formato de cadenta TZ de la base ('19700101030000') mediante la llamada del método db->jdate para recuperar la información en memoria en formato Timestamp GMT. Y los métodos insert deben deben generar la petición convirtiendo la fecha en memoria conocida en variable, mediante el método db->idate (Ver ejemplos generados por la plantilla).


 * Las actualizaciones automáticas de la base de datos (campo tms en la base) contienen el Timestamp GMT del momento en que se ha realizado la modificación. Los métodos select recuperan directamente este dato en memoria en formato Timestamp GMT.


 * La manipulación de fechas en PHP debe realizarse usando las funciones de fechas de Dolibarr: dol_now, dol_mktime, dol_stringtotime, dol_getdate, dol_time_plus_duree. También se encuentran otras funciones listas para su uso en el archivo date.lib.php.

Codificación UTF8/ISO
Dolibarr guarda la información de la manera siguiente:


 * En la base de datos, los datos en UTF8 ó ISO. Esto depende del pagecode de la base de datos, y por lo tanto, de las opciones en la creación de esta base de datos. En todos los casos, el controlador de base de datos Dolibarr (dentro de /lib/database) lee e inserta por conversión desde y hacia UTF8.
 * Los datos en memoria se guardan en UTF8 (instancias de objetos PHP).
 * las páginas web mostradas en pantalla se encuentran en formato UTF8 (con versiones < 2.5.1, es el antiguo parámetro $character_set del fichero conf.php quien define el formato de salida).

Los números reales, importes y cálculos
Tanto en PHP como en otros lenguajes (Java, por ejemplo), los datos no enteros (float, real, double) no son fialbes. Pruebe a realizar por ejemplo No obtendrá 0, sino un número muy pequeño en potencia 10 negativo. Si obtiene cero, puede encontrar otros ejemplos que no funcionan. El problema de los float es general, una variable resultante del cálculo de números reales debe SISTEMATICAMENTE ser limpiada mediante la función price2num con el segundo parámetro rellenado con: 'MU', 'MT' ó 'MS' según el caso (ver doc de la función). Si no es un precio que se ajuste a los parámetros de MU, MT, MS debe utilizar la función round.

Creación de tablas
No crear una tabla a utilizar, es decir, al primer uso del módulo. Si crea un módulo que usa tablas que no se incluyen como estándar en el código de Dolibarr, asegúrese de seguir el tutorial Creación de un módulo que explica cómo incluir las tablas que se crean en la activación del módulo y no en su uso (ver más arriba).

Logs
Añada traces dentro de su código mediante la función

Carpetas de trabajo
Si tiene que crear una carpeta de trabajo, en el código, haga referencia a esta carpeta de la siguiente forma DOL_DATA_ROOT.'/mimodulo'

La carpeta puede crearse en desde su código mediante el código siguiente:

Si necesita una carpeta de datos temporales, esta carpeta deberá ser DOL_DATA_ROOT.'/minmodulo/temp'

= Patrones de Diseño y programación con objetos =

Patrones de Diseño Creacionales (GoF)
Design patterns definidos por el le Gang Of Four (ver la wikipedia sobre los Design patterns). No hay design patterns creacionales particulares. Hay objetos cercanos a los Singleton y creados conforme al 100% en el espíritu, pero no en la sintaxis, a fin de ser compatible con PHP 4 que no es un lenguaje orientado a objetos completo.

Design patterns Estructurales (GoF)
(ver la wikipedia sobre los Design patterns). No hay design patterns estructurales particulares.

Design patterns de Comportamiento (GoF)
(ver la wikipedia sobre los Design patterns). No hay design patterns de Comportamiento particulares.

Motivos de organización del código
Martin Fowler ha identificado 3 métodos de organización de código llamados principios: Es el antiguo principio utilizado en los lenguajes de procedimiento. Inconveniente: Redundancia de código. Necesidad de conocer el modelo físico a desarrollar. Es un modelo posible desde los lenguajes orientados a objeto. Son procedimentos funcionales (que deben de ser indentificados antes) que sirven de base para las clases objeto. Inconveniente: Motivo de mantenimiento complejo. Un intermedio entre los 2 anteriores donde se tiene una sola clase por tabla de la base de datos.
 * El Transaction Script (El código es lineal en función de una acción del usuario).
 * El Domain Model
 * El Table Module

-> Como se muestra en el código de la plantilla (ver sección anterior) Dolibarr se basa en el principio de Table Module.

Comunicación lógica de negociado - datos (ORM)
Existen 3 tipos de conexiones: Es el más simple. Creamos una tabla por clase y cada clase es un puente con la tabla correspondiente, ver una clase por línea tabla. Una instancia de la clase será pues un registro de la tabla. La clase contiene sólo el código de acceso a líneas o columnas de tablas.
 * Data Gateway a tablas y filas

Ejemplo: Es el método utilizado al utilizar ciertos Frameworks de ORM como iBatis (http://ibatis.apache.org/).

Idéntico al anterior, pero podemos añadir algunas funciones de negociado en la clase, siempre que estas funciones sean exclusivas de la tabla o la grabación.
 * El Active Record

Ejemplo: Es el método utilizado por los desarrollos de Dolibarr y por otras muchas aplicaciones en PHP que disponen de su propio propio framework y prácticas de desarrollo.

Las clases representan las entidades del problema y no los datos. Por lo tanto hay que doblar, triplicar... estas clases con clases de mapeo para acceder a los datos. Más "purista" en el papel ya que el negociado está más cercano, este método también tiene la desventaja de ser más complejo en la práctica.
 * El Data Mapper

Ejemplo: Es el utilizadao si utilizamos el Framework de ORM Propel (http://propel.phpdb.org/trac/). Lo encontraremos pues en aplicacciones más pesadas basadas en este ORM.

-> Para los desarrollos de Dolibarr, es recomendado utilizar el modo Active Record  que ofrece las ventajas de un modelo cercano de negociado sin la complejidad y sin dificultar mucho la técnica. Es en este modo mediante el cual el desarrollo, la comprensión del código y el mantenimiento técnico y/o funcional parece ser la más productivo (Sin embargo, se trata de un eterno debate entre los puristas y los pragmáticos, el debate en el que nadie puede realmente estar en lo cierto, porque todo depende de los objetivos a atender).