WebTestCaseBase.php 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405
  1. <?php
  2. namespace WebserviceBundle\tests;
  3. use Symfony\Bundle\FrameworkBundle\Test\WebTestCase;
  4. use Symfony\Component\BrowserKit\Cookie;
  5. use Symfony\Component\HttpFoundation\Response;
  6. use Symfony\Component\Security\Core\Authentication\Token\UsernamePasswordToken;
  7. use Symfony\Bundle\FrameworkBundle\Client;
  8. use Doctrine\Common\Collections\ArrayCollection;
  9. use ReflectionClass;
  10. use Symfony\Component\Security\Core\User\User;
  11. use WebserviceBundle\Services\WebserviceMock;
  12. class WebTestCaseBase extends WebTestCase
  13. {
  14. /**
  15. * @var Client Cliente web.
  16. */
  17. private $client;
  18. /**
  19. * @var bool Me dice si estamos en el modulo base.
  20. */
  21. protected $isModuleBase;
  22. /**
  23. * @var int Contiene el id de usuario a simular
  24. */
  25. protected $idUser;
  26. /**
  27. * Funcion que crea un nuevo cliente para realizar consultas. Para cada consulta web necesito crear un nuevo cliente.
  28. * @param bool $new Me dice si tengo que crear un nuevo cliente.
  29. * @return Client Retorna el cliente.
  30. */
  31. protected function getClient($new = false)
  32. {
  33. if (!$this->client || $new) {
  34. // por defecto es environment=test y debug = true
  35. $this->client = static::createClient();
  36. }
  37. return $this->client;
  38. }
  39. /**
  40. * Funcion que busca un servicio dentro del container.
  41. * @param mixed $service Contiene el nombre del servicio.
  42. * @return mixed Retorna el servicio.
  43. */
  44. protected function get($service)
  45. {
  46. return $this->getClient()->getContainer()->get($service);
  47. }
  48. /**
  49. * @return mixed Retorna el manager de doctrine.
  50. */
  51. protected function getDoctrineManager()
  52. {
  53. return $this->get('doctrine')->getManager();
  54. }
  55. /**
  56. * @return bool Retorna TRUE si estoy en el modulo base.
  57. */
  58. public function isModuleBase()
  59. {
  60. if ($this->isModuleBase == null) {
  61. $em = $this->getDoctrineManager();
  62. if ($em == null) {
  63. $this->isModuleBase = false;
  64. } else {
  65. $this->isModuleBase = array_key_exists('BaseTenancyBundle', $this->getClient()->getContainer()->getParameter('kernel.bundles'));
  66. }
  67. }
  68. return $this->isModuleBase;
  69. }
  70. /**
  71. * Se creo este metodo para crear objeto desde un string para no tener conflictos en otros modulos.
  72. * @param string $repository Contiene el objeto del repositorio.
  73. * @return mixed Crea una instancia de una objeto.
  74. */
  75. public function getObject($repository)
  76. {
  77. $obj = $this->getDoctrineManager()->getMetadataFactory()->getMetadataFor($repository)->getName();
  78. $rc = new ReflectionClass($obj);
  79. return $rc->newInstance();
  80. }
  81. /**
  82. * Utiliza un mock del webservice original. Solo modifica el llamado de la url.
  83. * @param array $webServiceData Contiene los datos que debe retorna el webservice.
  84. */
  85. public function fakeWebservice($webServiceData = null)
  86. {
  87. $logger = $this->get('logger');
  88. if ($webServiceData != null && is_array($webServiceData)) {
  89. if ($this->getClient()->getContainer()->has('webservice')) {
  90. $original = $this->getClient()->getContainer()->get('webservice');
  91. $fake = new WebserviceMock(
  92. $original->getServiceContainer(),
  93. null, //$original->getSecurityTokenStorage(),
  94. null, //$original->getHttpClient(),
  95. $webServiceData);
  96. static::$kernel->getContainer()->set('webService', $fake);
  97. $logger->err("Seteo el web service");
  98. }
  99. }
  100. }
  101. /**
  102. * Funcion que crea el servicio de tenencias. Por defecto crea la tenencia base.
  103. * @param int $current Contiene la tenencia actual.
  104. * @param array $tenancies Contiene las tenencias. Ej. array(array('id'=>1,'name'=>'base'))
  105. */
  106. protected function fakeTenancyService($current = 1, $tenancies = array())
  107. {
  108. $service_tenancy = $this->get('base_tenancy.tenancy_service');
  109. if (count($tenancies) == 0) {
  110. $tenancies [] = array('id' => 1, 'name' => 'Tenencia Base');
  111. }
  112. if ($this->isModuleBase()) {
  113. // tengo que cargar las tenencias que no existan en la base de datos.
  114. $em = $this->getDoctrineManager();
  115. $user = $em->getRepository("BaseUserBundle:User")->findOneBy(array('id' => $this->idUser));
  116. if (!$user) {
  117. // el usuario no existe, entonces lo creo
  118. $user = $this->getObject("BaseUserBundle:User");
  119. $user->setUsername("adminpruebas");
  120. $user->setPassword("adminpass");
  121. $user->setEnabled(true);
  122. $user->setEmail("pepe@pepe.com");
  123. $em->persist($user);
  124. $em->flush();
  125. }
  126. $arrayTenancies = new ArrayCollection();
  127. foreach ($tenancies as $tenancyData) {
  128. $tenancy = $em->getRepository("BaseTenancyBundle:Tenancy")->findOneBy(array('id' => $tenancyData['id']));
  129. if (!$tenancy) {
  130. // la tenencia no existe, entonces la creo
  131. $tenancy = $this->getObject("BaseTenancyBundle:Tenancy");
  132. $tenancy->setId($tenancyData['id']);
  133. $tenancy->setName($tenancyData['name']);
  134. $tenancy->setEnabled(true);
  135. $em->persist($tenancy);
  136. $em->flush();
  137. }
  138. $ten_user = $em->createQueryBuilder()
  139. ->select('t')
  140. ->from('BaseTenancyBundle:Tenancy', 't')
  141. ->join('t.users', 'ut',
  142. \Doctrine\ORM\Query\Expr\Join::WITH,
  143. $em->getExpressionBuilder()->eq('ut.id', $this->idUser))
  144. ->where($em->getExpressionBuilder()->eq('t.id', $tenancyData['id']))
  145. ->getQuery()
  146. ->execute();
  147. if (!$ten_user) {
  148. $arrayTenancies[] = $tenancy;
  149. }
  150. }
  151. // actualizo las tenencias asociadas al usuario
  152. $user->setTenancies($arrayTenancies);
  153. $em->flush();
  154. } else {
  155. // seteo las tenencias como array en el usuario
  156. $this->get('security.token_storage')->getToken()->getUser()->setTenancies($tenancies);
  157. }
  158. $service_tenancy->setTenancy($current);
  159. }
  160. /**
  161. * Se crean los datos para hacer el fake al oauth
  162. * @param int $idUser Contiene el id de usuario a simular.
  163. * @throws \ErrorException Lanza una excepcion durante las distintas validaciones.
  164. */
  165. protected function FakeLogin($idUser = 1)
  166. {
  167. $this->idUser = $idUser;
  168. // obtengo la session
  169. $session = $this->get('session');
  170. if ($session != null) {
  171. // busco el usuario logueado
  172. $em = $this->getDoctrineManager();
  173. if ($em != null) {
  174. // FOS\UserBundle\Model\UserManager
  175. // FOS\UserBundle\Doctrine\UserManager
  176. if ($this->isModuleBase()) {
  177. $userManager = $this->get('fos_user.user_manager');
  178. if ($userManager != null) {
  179. try {
  180. $user = $userManager->findUserBy(array('id' => $this->idUser));
  181. } catch (\Throwable $t) {
  182. $user = null;
  183. }
  184. if (!$user) {
  185. // el usuario no existe por lo tanto lo creo. Deberia ver el tema de las tenencias
  186. // no tengo la oportunidad de probarlo
  187. $user = $userManager->createUser();
  188. $cargo = false;
  189. try {
  190. $em = $this->getDoctrineManager();
  191. $userDB = $em->getRepository("BaseUserBundle:User")->findOneBy(array('id' => $this->idUser));
  192. if ($userDB) {
  193. $cargo = true;
  194. $user->setEmail($userDB->email);
  195. $user->Id = $idUser;
  196. $user->setUsername($userDB->username);
  197. $user->setPlainPassword($userDB->plainPassword);
  198. $user->setEnabled(true);
  199. $user->setRoles($userDB->getRoles());
  200. // este comando ademas se setear los datos, los persiste en la base de datos.
  201. $userManager->updateUser($user);
  202. }
  203. } catch (\Throwable $t) {
  204. }
  205. if (!$cargo) {
  206. // no se cargo el usuario por algun error. Lo hago manual.
  207. $user->setEmail("pepe@pepe.com");
  208. $user->Id = $idUser;
  209. $user->setUsername('admin');
  210. $user->setPlainPassword('adminpass');
  211. $user->setEnabled(true);
  212. $user->setRoles(array("ROLE_USER"));
  213. $userManager->updateCanonicalFields($user);
  214. $userManager->updatePassword($user);
  215. }
  216. }
  217. // contiene el nombre de firewall a utilizar
  218. $firewall = 'main';
  219. // busco el token en el security
  220. $tokenStorage = $this->get('security.token_storage');
  221. if ($tokenStorage != null) {
  222. $token = null;
  223. if (!$tokenStorage->getToken()) {
  224. // no tengo el token, entonces lo creo
  225. $token = new UsernamePasswordToken($user, null, $firewall, array('ROLE_SUPER_ADMIN'));
  226. } else {
  227. // utilizo el token que yo esta creado
  228. $token = $tokenStorage->getToken();
  229. }
  230. // seteo en la session el firewall que voy a utilizar y el token
  231. $session->set('_security_' . $firewall, serialize($token));
  232. $session->save();
  233. // creo una cookie con los datos de la session.
  234. $cookie = new Cookie($session->getName(), $session->getId());
  235. $this->getClient()->getCookieJar()->set($cookie);
  236. // seteo el token en el security.token_storage
  237. $tokenStorage->setToken($token);
  238. } else {
  239. throw new \ErrorException("Error al obtener el token storage.");
  240. }
  241. } else {
  242. throw new \ErrorException("Error al obtener el fos user manager.");
  243. }
  244. } else {
  245. // no estoy en el modulo base
  246. // creo la clase de esta forma para no utilizar el "use" porque va a traer conflicto en otros modulos
  247. $class = "Base\OAuthClientBundle\Security\Core\User\CustomOAuthUser";
  248. $user = new $class('admin');
  249. // no se cargo el usuario por algun error. Lo hago manual.
  250. $user->Id = $idUser;
  251. $user->setRoles(array("ROLE_USER"));
  252. // contiene el nombre de firewall a utilizar
  253. $firewall = 'main';
  254. // busco el token en el security
  255. $tokenStorage = $this->get('security.token_storage');
  256. if ($tokenStorage != null) {
  257. $token = null;
  258. // no tengo el token, entonces lo creo
  259. $token = new UsernamePasswordToken($user, null, $firewall, array('ROLE_SUPER_ADMIN'));
  260. // seteo en la session el firewall que voy a utilizar y el token
  261. $session->set('_security_' . $firewall, serialize($token));
  262. $session->save();
  263. // creo una cookie con los datos de la session.
  264. $cookie = new Cookie($session->getName(), $session->getId());
  265. $this->getClient()->getCookieJar()->set($cookie);
  266. // seteo el token en el security.token_storage
  267. $tokenStorage->setToken($token);
  268. }
  269. }
  270. } else {
  271. throw new \ErrorException("Error al obtener doctrine.");
  272. }
  273. } else {
  274. throw new \ErrorException("Error al obtener la session.");
  275. }
  276. }
  277. /**
  278. * Crea las tenencias.
  279. * @param int $quantity Contiene la cantidad de tenencias a crear por defecto.
  280. * @return array Retorna una matriz con las tenencias a crear.
  281. * @throws \Exception Lanza un excepcion en caso de no poder crear las tenencias.
  282. */
  283. protected function generateTenancies($quantity = 0)
  284. {
  285. if ($quantity <= 0) {
  286. throw new \Exception("No se pasaron los argumentos de forma correcta.");
  287. } else {
  288. $resp = array();
  289. for ($i = 0; $i < $quantity; $i++) {
  290. if ($i == 1) {
  291. $resp[$i] = array('id' => $i, 'name' => 'Tenencia Base');
  292. } else {
  293. $resp[$i] = array('id' => $i, 'name' => 'Tenencia ' . $i);
  294. }
  295. }
  296. return $resp;
  297. }
  298. }
  299. /**
  300. * Cambia los datos para realizar una modificacion
  301. * @param array $original Contiene los datos originales.
  302. * @param array $newData Contiene los nuevos datos a setear.
  303. * @return array|string Retorna el array con los datos o el valor de la key pasada como parametro.
  304. * @throws \Exception Lanza un excepcion en caso de no encontrar la key.
  305. */
  306. protected function obtainDataChange($original, $newData)
  307. {
  308. foreach ($newData as $k => $v) {
  309. $original[$k] = $v;
  310. }
  311. return $original;
  312. }
  313. /**
  314. * Genera el filtro de busqueda filters
  315. * @param array $data Contiene los datos a buscar de la forma clave -> valor.
  316. * @param bool $qbcriteria Me dice si utiliza qb-criteria.
  317. * @return string Retorna el string generado.
  318. */
  319. protected function generateFilters($data, $qbcriteria = true)
  320. {
  321. $resp = '?';
  322. if ($qbcriteria) {
  323. $resp .= 'filters[qb-criteria]';
  324. }
  325. if ($data != null && count($data) > 0) {
  326. foreach ($data as $k => $v) {
  327. $resp .= '&filters[' . $k . ']=' . $v;
  328. }
  329. }
  330. return $resp;
  331. }
  332. /**
  333. * Funcion que busca el valor de una propiedad dentro del response.
  334. * @param Response $response Contiene el response.
  335. * @param string $property Contiene el nombre de la propiedad.
  336. * @param string $object Contiene el nombre del objeto del cual voy a obtener la propiedad.
  337. * @return null|mixed Retorna el valor de la propiedad.
  338. */
  339. protected function getProperty(Response $response, $property, $object = null)
  340. {
  341. $resp = json_decode($response->getContent(), true);
  342. if (count($resp) > 0) {
  343. if ($object != null) {
  344. // busco el objecto
  345. foreach ($resp as $k => $v) {
  346. if ($k == $object) {
  347. $resp = $v;
  348. break;
  349. }
  350. }
  351. }
  352. foreach ($resp as $v) {
  353. try {
  354. if (isset($v[$property])) {
  355. return $v[$property];
  356. }
  357. } catch (\Throwable $t) {
  358. }
  359. }
  360. return null;
  361. } else {
  362. if (isset($resp[$property])) {
  363. return $resp[$property];
  364. } else {
  365. return null;
  366. }
  367. }
  368. }
  369. /**
  370. * Funcion que inicializa por defecto las variables.
  371. * Crea el client para el request.
  372. * Hace el fake del login.
  373. * Crea 1 tenencia por defecto (TENENCIA BASE = 1).
  374. * @param array $webServiceData Contiene los datos que debe retorna el webservice.
  375. */
  376. protected function initDefault($webServiceData = null)
  377. {
  378. // creo un nuevo cliente para consultar. Lo tengo que crear para cada consulta.
  379. $this->getClient(true);
  380. // reemplaza el webservice por un mock
  381. $this->fakeWebservice($webServiceData);
  382. // hago fake del login
  383. $this->FakeLogin();
  384. // creo el servicio de tenencias
  385. $this->fakeTenancyService();
  386. }
  387. }