<?php use EesyPHP\App; use EesyPHP\Db; use EesyPHP\Hook; use EesyPHP\Log; use Unidecode\Unidecode; $db = new Db( App::get('db.dsn', null, 'string'), App::get('db.user', null, 'string'), App::get('db.password', null, 'string'), App::get('db.options', array(), 'array'), App::get('db.date_format', null, 'string'), App::get('db.datetime_format', null, 'string'), ); /* * Methods to handle items */ function get_items($orderby='id', $raw_values=false) { global $db; try { $query = $db -> fpdo -> from('item') -> orderBy($orderby); $result = $query -> execute(); if ($result !== false) { $info = $result -> fetchAll(); if ($info === false) return null; if ($raw_values) return $info; $items = array(); foreach ($info as $item) $items[$item['id']] = $db -> format_row_info($item, array('date')); return $items; } } catch (Exception $e) { Log :: error("Error retreiving items info from database : ".$e->getMessage()); } return false; } function get_item($id, $raw_values=false) { global $db; try { $query = $db -> fpdo -> from('item') -> where('id', $id); $result = $query -> execute(); if ($result !== false) { $info = $result -> fetch(); if ($info === false) return null; if ($raw_values) return $info; return $db -> format_row_info($info, array('date')); } } catch (Exception $e) { Log :: error("Error retreiving item #$id info from database : ".$e->getMessage()); } return false; } function add_item($values) { global $db; $values['date'] = $db -> time2datetime(time()); try { $result = $db -> fpdo -> insertInto('item') -> values($values) -> execute(); if ($result !== false) { $item = get_item($result); Log :: info("New item #$result added"); Hook :: trigger('item_added', $item); return $item; } } catch (Exception $e) { Log :: error("Error creating item in database : ".$e->getMessage()); } return false; } function update_item($id, $changes) { global $db; if (!is_array($changes)) return false; if (empty($changes)) return true; $item = get_item($id, true); if (!is_array($item)) return false; if (isset($changes['date']) && $changes['date']) $changes['date'] = $db -> time2datetime($changes['date']); try { $result = $db -> fpdo -> update('item') -> set($changes) -> where('id', $id) -> execute(); if ($result !== false) { Log :: info("Item #$id updated"); Hook :: trigger('item_updated', $item); return true; } } catch (Exception $e) { Log :: error("Error updating item #$id in database : ".$e->getMessage()); } return false; } function change_item_status($id, $status) { if (update_item($id, array('status' => $status))) { Log :: info("Status of item #$id changed to $status."); return true; } return false; } function archive_item($id) { return change_item_status($id, 'archived'); } function delete_item($id) { global $db; try { $result = $db -> fpdo -> deleteFrom('item') -> where('id', $id) -> execute(); if ($result !== false) { Log :: info("Item #$id deleted"); return True; } } catch (Exception $e) { Log :: error("Error deleting item #$id from database : ".$e->getMessage()); } return false; } function search_items($params) { global $db; // Detect PgSQL backend $is_pgsql = (strpos(App::get('db.dsn', '', 'string'), "pgsql:") === 0); $where = array(); if (isset($params['status']) && $params['status'] && $params['status'] != 'all') $where['status'] = $params['status']; $patterns_where = array(); if (isset($params['pattern']) && $params['pattern']) { foreach(preg_split('/\s+/', trim($params['pattern'])) as $word) { if (!$word) continue; $patterns_word=array(); // If numeric pattern ? if (preg_match('/^[0-9]+$/', $word)) { // Numeric pattern $word = intval($word); $patterns_word['id = ?'] = $word; } else { // Text pattern foreach (array('name', 'description') as $field) { if ($is_pgsql) { $word = Unidecode::unidecode($word); $patterns_word["unaccent($field) ILIKE ?"] = "%$word%"; } else $patterns_word["$field LIKE ?"] = "%$word%"; } } $patterns_where[] = $patterns_word; } } $order='id'; $orders=array('id', 'name', 'date', 'status', 'description'); if (isset($params['order'])) { if (!in_array($order, $orders)) return -1; $order=$params['order']; } $order_direction='DESC'; if (isset($params['order_direction']) && $params['order_direction']) { if (!in_array($params['order_direction'], array('ASC', 'DESC'))) return -1; $order_direction=$params['order_direction']; } $orderby="$order $order_direction"; $limit = ""; $page = 1; $nb_by_page = 10; $offset = 0; if (!isset($params['all'])) { if (isset($params['page']) && $params['page']>0) { if (isset($params['nb_by_page']) && $params['nb_by_page']>0) { $nb_by_page = intval($params['nb_by_page']); } $page = intval($params['page']); } $offset = ($page-1)*$nb_by_page; $limit = $nb_by_page; } try { $query = $db -> fpdo -> from('item'); if (!empty($where)) $query -> where($where); foreach ($patterns_where as $patterns_word) call_user_func_array( array($query, 'where'), array_merge( array('('.implode(' OR ', array_keys($patterns_word)).')'), array_values($patterns_word) ) ); $result = $query -> orderBy($orderby) -> limit($limit) -> offset($offset) -> execute(); if ($result === false) { Log :: error('search_items() : search in DB return false'); return false; } $rows = $result -> fetchAll(); $items = array(); foreach ($rows as $row) { $items[] = $db -> format_row_info($row, array('date')); } if (isset($params['all'])) { return array( 'count' => count($items), 'first' => 1, 'last' => count($items), 'nb_pages' => 1, 'page' => 1, 'items' => $items ); } $query_count = $db -> fpdo -> from('item') -> select(null) -> select('count(*) as count'); if (!empty($where)) $query_count -> where($where); foreach ($patterns_where as $patterns_word) call_user_func_array( array($query_count, 'where'), array_merge( array('('.implode(' OR ', array_keys($patterns_word)).')'), array_values($patterns_word) ) ); $result_count = $query_count -> execute(); if ($result_count === false) { Log :: debug('search_items() : search for count in DB return false'); return False; } $count = $result_count -> fetch(); return array( 'count' => $count['count'], 'first' => $offset+1, 'last' => ( $offset+$nb_by_page<$count['count']? $offset+$nb_by_page:$count['count']), 'nb_pages' => ceil($count['count']/$nb_by_page), 'page' => $page, 'items' => $items, ); } catch (Exception $e) { Log :: exception( $e, "An exception occured searching items with params %s infos from database : ", preg_replace("/\n[ \t]*/", " ", print_r($params, true)) ); } return false; } function export_items($fd=null) { if (!$fd) $fd = fopen('php://output', 'w'); fputcsv( $fd, array ( 'id', 'name', 'date', 'status', 'description', ) ); $items = get_items(); foreach($items as $item) { fputcsv( $fd, array( $item['id'], $item['name'], $item['date'], $item['status'], $item['description'], ) ); } return True; } function restore_items($fd=null) { global $db; if (!$fd) $fd = fopen('php://stdin', 'r'); try { $result = $db -> fpdo -> deleteFrom('item') -> execute(); if ($result === false) { Log :: error("An unknown error occured truncating item table in database."); return false; } } catch (Exception $e) { Log :: error("Error truncating item table in database : ".$e->getMessage()); return false; } $first_row = false; $line = 0; $restored = 0; $error = false; $datetime_fields = array ( 'date', ); // Map fields to support hold CSV files format $mapping = array ( 'creation_date' => 'date', 'desc' => 'description', ); while (($row = fgetcsv($fd)) !== FALSE) { $line++; if ($first_row === false) { $first_row = $row; continue; } try { $values = array(); for ($i=0; $i < count($first_row); $i++) { if (!$row[$i]) continue; $field = ( array_key_exists($first_row[$i], $mapping)? $mapping[$first_row[$i]]: $first_row[$i]); $value = (in_array($field, $datetime_fields)?$db -> time2datetime($row[$i]):$row[$i]); $values[$field] = $value; } $result = $db -> fpdo -> insertInto('item') -> values($values) -> execute(); if ($result !== false) { $restored++; } else { Log :: error("Unkwown error occured restoring item from line #$line :\n".print_r($values, true)); $error = true; } } catch (Exception $e) { Log :: error( "Error restoring item from line #$line : ".$e->getMessage()."\n".print_r($values, true)); $error = true; } } Log :: info("$restored items restored"); // Trigger hooks Hook :: trigger('items_restored'); return !$error; } # vim: tabstop=2 shiftwidth=2 softtabstop=2 expandtab