public function testBuscarProductosPorID_Sucursal() { $dir_suc = self::RandomString(25, FALSE, FALSE, FALSE); $desc = self::RandomString(10, FALSE, FALSE, FALSE); $sucursal = SucursalesController::Nueva($descripcion = $desc, $direccion = $dir_suc, $id_tarifa = 1); $this->assertInternalType("int", $sucursal['id_sucursal'], "---- 'testBuscarProductosPorID_Sucursal' 'id_sucursal' NO ES UN ENTERO"); $empresa_rfc = self::RandomString(13, FALSE, FALSE, FALSE); $empresa_razon = self::RandomString(10, FALSE, FALSE, FALSE); $contabilidad['id_moneda'] = 1; $contabilidad['ejercicio'] = "2013"; $contabilidad['periodo_actual'] = 1; $contabilidad['duracion_periodo'] = 1; $empresa = EmpresasController::Nuevo((object) $contabilidad, $direccion = array(array("calle" => "Monte Balcanes", "numero_exterior" => "107", "colonia" => "Arboledas", "id_ciudad" => 334, "codigo_postal" => "38060", "numero_interior" => null, "referencia" => "Calle cerrada", "telefono1" => "4616149974", "telefono2" => "45*451*454")), $razon_social = $empresa_razon, $rfc = $empresa_rfc); $this->assertInternalType("int", $empresa['id_empresa'], "---- 'testBuscarProductosPorID_Sucursal' 'id_empresa' NO ES UN ENTERO"); //se crea un nuevo producto $codigo_p = self::RandomString(5, FALSE, FALSE, FALSE); $nombre_p = self::RandomString(15, FALSE, FALSE, FALSE); $p = ProductosController::Nuevo($activo = true, $codigo_producto = $codigo_p, $compra_en_mostrador = true, $id_unidad_compra = 1, $metodo_costeo = "costo", $nombre_producto = $nombre_p, $visible_en_vc = true, $codigo_de_barras = null, $control_de_existencia = null, $costo_estandar = 10, $descripcion_producto = null, $foto_del_producto = null, $garantia = null, $id_categoria = null, $id_empresas = array($empresa['id_empresa']), $id_unidad = null, $impuestos = null, $precio_de_venta = 12); //TODO: En un futuro desaparecera esto ya que por instancia lo correctro es que haya una sola empresa y todas als sucursales perteneceran a la empresa ProductoEmpresaDAO::save(new ProductoEmpresa(array("id_producto" => $p["id_producto"], "id_empresa" => $empresa['id_empresa']))); SucursalEmpresaDAO::save(new SucursalEmpresa(array("id_sucursal" => $sucursal['id_sucursal'], "id_empresa" => $empresa['id_empresa']))); $res = ProductosController::Buscar($query = null, $id_producto = null, $id_sucursal = $sucursal['id_sucursal']); $this->assertInternalType("int", $res["numero_de_resultados"], "---- 'testBuscarProductosPorID_Sucursal' 'numero_de_resultados' NO ES UN ENTERO"); $this->assertGreaterThan(0, $res['numero_de_resultados'], "---- 'testBuscarProductosPorID_Sucursal' SE DEBIÓ DE ENCONTRAR ALMENOS 1 RESULTADO CON NOMBRE PRODUCTO: " . $nombre_p); }
/** * *Para poder editar un traspaso,este no tuvo que haber sido enviado aun * * @param id_traspaso int Id del traspaso a editar * @param productos json Productos a enviar con sus cantidades * @param fecha_envio_programada string Fecha de envio programada **/ public static function EditarTraspasoAlmacen($id_traspaso, $fecha_envio_programada = null, $productos = null) { Logger::log("Editando traspaso " . $id_traspaso); //valida que el traspaso exista, que no haya sido cancelado y que no se hayan efectuado operaciones sobre el. $validar = self::validarParametrosTraspaso($id_traspaso, null, null, $fecha_envio_programada); if (is_string($validar)) { Logger::error($validar); throw new Exception($validar); } $traspaso = TraspasoDAO::getByPK($id_traspaso); if ($traspaso->getCancelado()) { Logger::error("El traspaso ya ha sido cancelado, no puede ser editado"); throw new Exception("El traspaso ya ha sido cancelado, no puede ser editado"); } if ($traspaso->getCompleto() || $traspaso->getEstado() !== "Envio programado") { Logger::error("No se puede editar el traspaso de almacen pues ya se han realizado acciones sobre el"); throw new Exception("No se puede editar el traspaso de almacen pues ya se han realizado acciones sobre el"); } //Si se recibe el parametro fecha de envio se toma como actualizacion. if (!is_null($fecha_envio_programada)) { $traspaso->setFechaEnvioProgramada($fecha_envio_programada); } DAO::transBegin(); try { TraspasoDAO::save($traspaso); if (!is_null($productos)) { $productos = object_to_array($productos); if (!is_array($productos)) { throw new Exception("Los productos son invalidos", 901); } //Se actualiza la cantidad de cada producto programado para este traspaso, si el producto //no se encuentra, se verifica que su empresa concuerde con la del almacen de recibo y //se crea el nuevo registro. // //Despues, se recorren los productos que se encuentran actualmente programados a enviarse en el traspaso, //los productos que no se encuentre en la nueva lista obtenida seran eliminados. foreach ($productos as $p) { if (!array_key_exists("id_producto", $p) || !array_key_exists("id_unidad", $p) || !array_key_exists("cantidad", $p)) { throw new Exception("Los productos no tienen todos los parametros necesarios", 901); } $traspaso_producto = TraspasoProductoDAO::getByPK($id_traspaso, $p["id_producto"], $p["id_unidad"]); if (is_null($traspaso_producto)) { $almacen_recibe = AlmacenDAO::getByPK($traspaso->getIdAlmacenRecibe()); $productos_empresa = ProductoEmpresaDAO::search(new ProductoEmpresa(array("id_producto" => $p["id_producto"]))); $encontrado = false; foreach ($productos_empresa as $p_e) { if ($p_e->getIdEmpresa() == $almacen_recibe->getIdEmpresa()) { $encontrado = true; } } if (!$encontrado) { throw new Exception("Se busca enviar un producto que no es de la empresa del almacen que recibe", 901); } $traspaso_producto = new TraspasoProducto(array("id_traspaso" => $id_traspaso, "id_producto" => $p["id_producto"], "id_unidad" => $p["id_unidad"], "cantidad_recibida" => 0)); } $traspaso_producto->setCantidadEnviada($p["cantidad"]); TraspasoProductoDAO::save($traspaso_producto); } /* Fin de foreach de productos*/ $traspasos_producto = TraspasoProductoDAO::search(new TraspasoProducto(array("id_traspaso" => $id_traspaso))); foreach ($traspasos_producto as $t_p) { $encontrado = false; foreach ($productos as $p) { if ($t_p->getIdProducto() == $p["id_producto"] && $t_p->getIdUnidad() == $p["id_unidad"]) { $encontrado = true; } } if (!$encontrado) { TraspasoProductoDAO::delete($t_p); } } /* Fin de foreach de traspaso_producto*/ } /* Fin de if de productos */ } catch (Exception $e) { DAO::transRollback(); Logger::error("No se pudo editar el traspaso: " . $e); if ($e->getCode() == 901) { throw new Exception("No se pudo editar el traspaso: " . $e->getMessage(), 901); } throw new Exception("No se pudo editar el traspaso", 901); } DAO::transEnd(); Logger::log("Traspaso editado correctamente"); }
/** * * Buscar productos por codigo_producto, nombre_producto, descripcion_producto. * * @param query string Buscar productos por codigo_producto, nombre_producto, descripcion_producto. * @param id_producto int Si estoy buscando un producto del cual ya tengo conocido su id. Si se envia `id_producto` todos los demas campos seran ignorados. * @param id_sucursal int Buscar las existencias de este producto en una sucursal especifica. * @return resultados json * @return numero_de_resultados int **/ public static function Buscar($query, $id_producto = null, $id_sucursal = null) { if (!is_null($id_producto)) { Logger::log("Buscando producto por id_producto = {$id_producto}.... "); $p = ProductoDAO::getByPK($id_producto); if (is_null($p)) { return array("resultados" => array(), "numero_de_resultados" => 0); } return array("resultados" => array($p->asArray()), "numero_de_resultados" => 1); } //!is_null($id_producto) if (!is_null($id_sucursal)) { Logger::log("Buscando producto por id_sucursal = {$id_sucursal}.... "); $empresas = SucursalEmpresaDAO::search(new SucursalEmpresa(array("id_sucursal" => $id_sucursal))); if (empty($empresas)) { Logger::log("no results"); return array("resultados" => array(), "numero_de_resultados" => 0); } $results = array(); foreach ($empresas as $e) { Logger::log("suc {$id_sucursal} pertenece a empresa:" . $e->getIdEmpresa()); $productos_e = ProductoEmpresaDAO::search(new ProductoEmpresa(array("id_empresa" => $e->getIdEmpresa()))); foreach ($productos_e as $p_e) { $p = ProductoDAO::getByPK($p_e->getIdProducto()); if (is_null($p)) { throw new BusinessLogicException("el producto " . $p_e->getIdProducto() . " no existe"); } array_push($results, $p->asArray()); } } if (sizeof($results) == 0) { array_push($results, array("nombre_producto" => "lksadjfklaj", "descripcion" => "asdf")); } return array("numero_de_resultados" => sizeof($results), "resultados" => $results); } //if (!is_null($id_sucursal)) $productos = ProductoDAO::buscarProductos($query); $resultado = array(); foreach ($productos as $p) { $r = $p->asArray(); //una vez que tengo los productos vamos a agergarle sus //precios tarifarios $r["tarifas"] = TarifasController::_CalcularTarifa($p, "venta"); //buscar sus existencias $existencias = InventarioController::Existencias(null, null, $p->getIdProducto(), null); $r["existencias"] = $existencias["resultados"]; array_push($resultado, $r); } //$productos as $p if (strlen($query) > 0 && sizeof($resultado) == 0) { array_push($resultado, array("nombre_producto" => "El producto <i>{$query}</i> no se encuentra.", "descripcion" => "¿ Desea agregarlo ahora ?", "id_producto" => -99, "query" => $query)); } return array("resultados" => $resultado, "numero_de_resultados" => sizeof($resultado)); }
/** * *Registra una nueva compra fuera de caja, puede usarse para que el administrador haga directamente una compra. El usuario y al sucursal seran tomados de la sesion. La fecha sera tomada del servidor. La empresa sera tomada del almacen del cual fueron tomados los productos. * * @param descuento float Monto descontado por descuentos * @param subtotal float Total de la compra antes de impuestos y descuentos. * @param detalle json Objeto que contendr el arreglo de id productos, cantidad, precio, descuento, id de unidad y procesado que involucran esta compra. * @param impuesto float Monto agregado por impuestos * @param tipo_compra string Si la compra es a credito o de contado * @param retencion float Monto agregado por retenciones * @param id_usuario_compra int Id usuario al que se le compra, si es a una sucursal, se pone el id en negativo * @param id_empresa int Id de la empresa a nombre de la cual se hace la compra * @param total float Total de la compra * @param cheques json Si el tipo de pago es con cheque, se almacena el nombre del banco, el monto y los ultimos 4 numeros del o de los cheques * @param saldo float Cantidad pagada de la * @param tipo_de_pago string Si el pago sera en efectivo, con cheque o tarjeta * @return id_compra int Id autogenerado por la inserci�n de la compra **/ public static function Nueva($descuento, $detalle, $id_empresa, $id_usuario_compra, $impuesto, $retencion, $subtotal, $tipo_compra, $total, $cheques = null, $id_sucursal = null, $saldo = 0, $tipo_de_pago = null) { Logger::log(" ===== Creando nueva compra... ===== "); //validemos al comprador $proveedor = UsuarioDAO::getByPK($id_usuario_compra); if (is_null($proveedor)) { Logger::error("el provedor {$id_usuario_compra} no exite"); throw new InvalidDataException("El proveedor no existe"); } if ($proveedor->getActivo() == false) { throw new BusinessLogicException("No se puede comprar de un proveedor no activo."); } //validemos la empresa $empresa = EmpresaDAO::getByPK($id_empresa); if (is_null($empresa)) { Logger::error("La empresa {$id_empresa} no existe"); throw new InvalidDataException("La empresa que compra no existe."); } if ($empresa->getActivo() == false) { throw new BusinessLogicException("Una empresa inactiva no puede hacer compras."); } //validemos los valores conocidos //( 0 >= descuento > 100, subtotal > 0, total >= subtotal, etc etc) //validemos sucursal $sucursal = null; if (!is_null($id_sucursal) && strlen($id_sucursal) > 0) { $sucursal = SucursalDAO::getByPK($id_sucursal); if (is_null($sucursal)) { Logger::error("La sucursal {$id_sucursal} no existe"); //throw new InvalidDataException("La sucural que se envio no existe."); } } //validemos detalles de compra //debe traer // -id_producto // -cantidad // -precio // -lote if (!is_array($detalle)) { throw InvalidDataException("El detalle no es un arreglo"); } for ($detalleIterator = 0; $detalleIterator < sizeof($detalle); $detalleIterator++) { //por cada producto // -debe existir // -si se lo compro a un proveedor no hay pedo // si se lo compro a un cliente, debe de tener comprar_caja = 1 // -debe tener cantidad mayor a 0 // -que exista el lote a donde va a ir $p = $detalle[$detalleIterator]; if (!isset($p->precio)) { throw new InvalidArgumentException("No se envio el precio"); } if (!isset($p->id_producto)) { throw new InvalidArgumentException("No se envio el id_producto"); } if (!isset($p->cantidad)) { throw new InvalidArgumentException("No se envio la cantidad"); } if (!isset($p->lote)) { throw new InvalidArgumentException("No se envio el lote"); } $producto = ProductoDAO::getByPK($p->id_producto); if (is_null($producto)) { throw new InvalidArgumentException("El producto a comprar no existe"); } if ($p->cantidad <= 0) { throw new InvalidArgumentException("No puedes comprar 0 unidades"); } } $s = SesionController::getCurrentUser(); //terminaron las validaciones $compra = new Compra(); $compra->setIdVendedorCompra($id_usuario_compra); $compra->setTipoDeCompra($tipo_compra); $compra->setFecha(time()); $compra->setSubtotal($subtotal); $compra->setImpuesto($impuesto); $compra->setDescuento($descuento); $compra->setTotal($subtotal + $impuesto); $compra->setIdUsuario($s->getIdUsuario()); $compra->setIdEmpresa($id_empresa); $compra->setSaldo(0); $compra->setCancelada(false); $compra->setTipoDePago($tipo_de_pago); $compra->setRetencion(0); try { DAO::transBegin(); CompraDAO::save($compra); } catch (Exception $e) { DAO::transRollback(); throw InvalidDatabaseOperationException($e); } for ($detalleIterator = 0; $detalleIterator < sizeof($detalle); $detalleIterator++) { //por cada producto // --- procesos --- // -insertar en productoempresa // -insertar en loteproducto // -insertar en entradalote // -si el tipo de precio de venta es costo, actualizar // -insertar compra producto $p = $detalle[$detalleIterator]; try { ProductoEmpresaDAO::save(new ProductoEmpresa(array("id_empresa" => $id_empresa, "id_producto" => $p->id_producto))); if (is_null($p->lote)) { throw new InvalidDataException("No selecciono a que lote ira el producto " . $p->id_producto); } if (strlen($p->lote) == 0) { throw new InvalidDataException("No selecciono a que lote ira el producto " . $p->id_producto); } //busquemos el id del lote $l = LoteDAO::search(new Lote(array("folio" => $p->lote))); $l = $l[0]; //busquemos la unidad que nos mandaron $uAbreviacion = $p->id_unidad; $uResults = UnidadMedidaDAO::search(new UnidadMedida(array("abreviacion" => $uAbreviacion, "activa" => 1))); if (sizeof($uResults) != 1) { throw new InvalidDataException("La unidad de medida `" . $p->id_unidad . "` no existe, o no esta activa."); } else { $p->id_unidad = $uResults[0]->getIdUnidadMedida(); } //busequemos si este producto ya existe en este lote $lp = LoteProductoDAO::getByPK($l->getIdLote(), $p->id_producto); if (is_null($lp)) { //no existe, insertar $loteProducto = new LoteProducto(array("id_lote" => $l->getIdLote(), "id_producto" => $p->id_producto, "cantidad" => $p->cantidad, "id_unidad" => $p->id_unidad)); LoteProductoDAO::save($loteProducto); } else { //ya existe, sumar //revisemos si es de la misma unidad if ($lp->getIdUnidad() == $p->id_unidad) { //es igual, solo hay que sumar $lp->setCantidad($lp->getCantidad() + $p->cantidad); } else { //no es igual, hay que convertir try { $r = UnidadMedidaDAO::convertir($p->id_unidad, $lp->getIdUnidad(), $p->cantidad); } catch (BusinessLogicException $ide) { //no se pudo convertir porque son de //diferentes categorias throw $ide; //mostrar una excpetion mas fresona } $lp->setCantidad($lp->getCantidad() + $r); } //$lp->setCantidad( $lp->getCantidad() + $p->cantidad ); LoteProductoDAO::save($lp); } $loteEntrada = new LoteEntrada(array("id_lote" => $l->getIdLote(), "id_usuario" => $s->getIdUsuario(), "fecha_registro" => time(), "motivo" => "Compra a Proveedor")); LoteEntradaDAO::save($loteEntrada); LoteEntradaProductoDAO::save(new LoteEntradaProducto(array("id_lote_entrada" => $loteEntrada->getIdLoteEntrada(), "id_producto" => $p->id_producto, "id_unidad" => $p->id_unidad, "cantidad" => $p->cantidad))); $compraProducto = new CompraProducto(array("id_compra" => $compra->getIdCompra(), "id_producto" => $p->id_producto, "cantidad" => $p->cantidad, "precio" => $p->precio, "descuento" => 0, "impuesto" => 0, "id_unidad" => $p->id_unidad, "retencion" => 0)); CompraProductoDAO::save($compraProducto); } catch (InvalidDataException $e) { Logger::error($e); DAO::transRollback(); throw $e; } catch (exception $e) { Logger::error($e); DAO::transRollback(); throw new InvalidDatabaseOperationException($e); } } //for try { DAO::transEnd(); } catch (Exception $e) { throw InvalidDatabaseOperationException($e); } Logger::log("===== COMPRA " . $compra->getIdCompra() . " EXITOSA ===== "); return array("id_compra" => $compra->getIdCompra()); }
/** * *Metodo que surte una sucursal por parte de un proveedor. La sucursal sera tomada de la sesion actual. Update Creo que este metodo tiene que estar bajo sucursal. * * @param id_almacen int Id del almacen que se surte * @param productos json Objeto que contendr los ids de los productos, sus unidades y sus cantidades * @param motivo string Motivo del movimiento * @return id_surtido string Id generado por el registro de surtir **/ static function Entrada($id_almacen, $productos, $motivo = null) { Logger::log("Resgitrando entrada a almacen"); $entrada_almacen = new EntradaAlmacen(); //Se obtiene el usuario de la sesion $id_usuario = SesionController::getCurrentUser(); if (is_null($id_usuario)) { Logger::error("No se puede obtener al usuario de la sesion, ya inicio sesion?"); throw new Exception("No se puede obtener al usuario de la sesion, ya inicio sesion?"); } //valida que el almacen exista $almacen = AlmacenDAO::getByPK($id_almacen); if (is_null($almacen)) { Logger::error("El almacen con id: " . $id_almacen . " no existe"); throw new Exception("El almacen con id: " . $id_almacen . " no existe"); } if (!$almacen->getActivo()) { Logger::error("El almacen no esta activo, no se le pueden ingresar productos"); throw new Exception("El almacen no esta activo, no se le pueden ingresar productos"); } //valida que el motivo sea un string valido if (!is_null($motivo)) { $validar = self::validarString($motivo, 255, "motivo"); if (is_string($validar)) { Logger::error($validar); throw new Exception($validar); } } //Se inicializa el registro de la tabla entrada_almacen $entrada_almacen->setIdAlmacen($id_almacen); $entrada_almacen->setMotivo($motivo); $entrada_almacen->setIdUsuario($id_usuario); $entrada_almacen->setFechaRegistro(time()); DAO::transBegin(); try { //se guarda el registro de entrada_almacen EntradaAlmacenDAO::save($entrada_almacen); //Por cada producto recibido se crea un registro en la tabla producto_entrada_almacen. //Cada producto ingresado incrementa su cantidad en el almacen. Si aun no existe, //se crea su registro y se guarda. $producto_entrada_almacen = new ProductoEntradaAlmacen(array("id_entrada_almacen" => $entrada_almacen->getIdEntradaAlmacen())); $productos = object_to_array($productos); if (!is_array($productos)) { throw new Exception("Los productos fueron recibidos incorrectamente", 901); } foreach ($productos as $p) { if (!array_key_exists("id_producto", $p) || !array_key_exists("id_unidad", $p) || !array_key_exists("cantidad", $p)) { throw new Exception("Los productos fueron recibidos incorrectamente", 901); } //valida que el producto a ingresar pertenezca a la misma empresa que el almacen //pues un almacen solo puede ocntener producto de la empresa a la que pertenece. $productos_empresa = ProductoEmpresaDAO::search(new ProductoEmpresa(array("id_producto" => $p["id_producto"]))); $encontrado = false; foreach ($productos_empresa as $p_e) { if ($p_e->getIdEmpresa() == $almacen->getIdEmpresa()) { $encontrado = true; } } if (!$encontrado) { throw new Exception("Se quiere agregar un producto que no es de la empresa de este almacen", 901); } if (is_null(ProductoDAO::getByPK($p["id_producto"]))) { throw new Exception("El producto con id: " . $p["id_producto"] . " no existe", 901); } $producto_entrada_almacen->setIdProducto($p["id_producto"]); if (is_null(UnidadDAO::getByPK($p["id_unidad"]))) { throw new Exception("La unidad con id: " . $p["id_unidad"] . " no existe", 901); } $producto_entrada_almacen->setIdUnidad($p["id_unidad"]); $producto_entrada_almacen->setCantidad($p["cantidad"]); $producto_almacen = ProductoAlmacenDAO::getByPK($p["id_producto"], $id_almacen, $p["id_unidad"]); if (is_null($producto_almacen)) { $producto_almacen = new ProductoAlmacen(array("id_producto" => $p["id_producto"], "id_almacen" => $id_almacen, "id_unidad" => $p["id_unidad"])); } $producto_almacen->setCantidad($producto_almacen->getCantidad() + $p["cantidad"]); ProductoEntradaAlmacenDAO::save($producto_entrada_almacen); ProductoAlmacenDAO::save($producto_almacen); } } catch (Exception $e) { DAO::transRollback(); Logger::error("No se pudo registrar la entrada al almacen: " . $e); if ($e->getCode() == 901) { throw new Exception("No se pudo registrar la entrada al almacen: " . $e->getCode(), 901); } throw new Exception("No se pudo registrar la entrada al almacen", 901); } DAO::transEnd(); Logger::log("Entrada a almacen registrada exitosamente"); return array("id_surtido" => $entrada_almacen->getIdEntradaAlmacen()); }