mirror of
				https://github.com/firefly-iii/firefly-iii.git
				synced 2025-10-26 05:26:17 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			256 lines
		
	
	
		
			9.3 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			256 lines
		
	
	
		
			9.3 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
| <?php
 | |
| /**
 | |
|  * AppendsLocationData.php
 | |
|  * Copyright (c) 2020 james@firefly-iii.org
 | |
|  *
 | |
|  * This file is part of Firefly III (https://github.com/firefly-iii).
 | |
|  *
 | |
|  * This program is free software: you can redistribute it and/or modify
 | |
|  * it under the terms of the GNU Affero General Public License as
 | |
|  * published by the Free Software Foundation, either version 3 of the
 | |
|  * License, or (at your option) any later version.
 | |
|  *
 | |
|  * This program is distributed in the hope that it will be useful,
 | |
|  * but WITHOUT ANY WARRANTY; without even the implied warranty of
 | |
|  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 | |
|  * GNU Affero General Public License for more details.
 | |
|  *
 | |
|  * You should have received a copy of the GNU Affero General Public License
 | |
|  * along with this program.  If not, see <https://www.gnu.org/licenses/>.
 | |
|  */
 | |
| 
 | |
| declare(strict_types=1);
 | |
| 
 | |
| namespace FireflyIII\Support\Request;
 | |
| 
 | |
| /**
 | |
|  * Trait AppendsLocationData
 | |
|  */
 | |
| trait AppendsLocationData
 | |
| {
 | |
|     /**
 | |
|      * Abstract method.
 | |
|      *
 | |
|      * @param mixed $key
 | |
|      *
 | |
|      * @return bool
 | |
|      */
 | |
|     abstract public function has($key);
 | |
| 
 | |
|     /**
 | |
|      * Abstract method.
 | |
|      *
 | |
|      * @return string
 | |
|      */
 | |
|     abstract public function method();
 | |
| 
 | |
|     /**
 | |
|      * Abstract method.
 | |
|      *
 | |
|      * @param mixed ...$patterns
 | |
|      *
 | |
|      * @return mixed
 | |
|      */
 | |
|     abstract public function routeIs(...$patterns);
 | |
| 
 | |
|     /**
 | |
|      * Abstract method stolen from "InteractsWithInput".
 | |
|      *
 | |
|      * @param null $key
 | |
|      * @param bool $default
 | |
|      *
 | |
|      * @return mixed
 | |
|      *
 | |
|      * @SuppressWarnings(PHPMD.BooleanArgumentFlag)
 | |
|      */
 | |
|     abstract public function boolean($key = null, $default = false);
 | |
| 
 | |
|     public function addFromromTransactionStore(array $information, array $return): array
 | |
|     {
 | |
|         $return['store_location'] = false;
 | |
|         if (true === $information['store_location']) {
 | |
|             $long = array_key_exists('longitude', $information) ? $information['longitude'] : null;
 | |
|             $lat  = array_key_exists('latitude', $information) ? $information['latitude'] : null;
 | |
|             if (null !== $long && null !== $lat && $this->validLongitude($long) && $this->validLatitude($lat)) {
 | |
|                 $return['store_location'] = true;
 | |
|                 $return['longitude']      = $information['longitude'];
 | |
|                 $return['latitude']       = $information['latitude'];
 | |
|                 $return['zoom_level']     = $information['zoom_level'];
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         return $return;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Read the submitted Request data and add new or updated Location data to the array.
 | |
|      */
 | |
|     protected function appendLocationData(array $data, ?string $prefix): array
 | |
|     {
 | |
|         app('log')->debug(sprintf('Now in appendLocationData("%s")', $prefix), $data);
 | |
|         $data['store_location']  = false;
 | |
|         $data['update_location'] = false;
 | |
|         $data['remove_location'] = false;
 | |
|         $data['longitude']       = null;
 | |
|         $data['latitude']        = null;
 | |
|         $data['zoom_level']      = null;
 | |
| 
 | |
|         $longitudeKey            = $this->getLocationKey($prefix, 'longitude');
 | |
|         $latitudeKey             = $this->getLocationKey($prefix, 'latitude');
 | |
|         $zoomLevelKey            = $this->getLocationKey($prefix, 'zoom_level');
 | |
|         $isValidPOST             = $this->isValidPost($prefix);
 | |
|         $isValidPUT              = $this->isValidPUT($prefix);
 | |
|         $isValidEmptyPUT         = $this->isValidEmptyPUT($prefix);
 | |
| 
 | |
|         // for a POST (store), all fields must be present and not NULL.
 | |
|         if ($isValidPOST) {
 | |
|             app('log')->debug('Method is POST and all fields present and not NULL.');
 | |
|             $data['store_location'] = true;
 | |
|             $data['longitude']      = $this->convertString($longitudeKey);
 | |
|             $data['latitude']       = $this->convertString($latitudeKey);
 | |
|             $data['zoom_level']     = $this->convertString($zoomLevelKey);
 | |
|         }
 | |
| 
 | |
|         // for a PUT (api update) or POST update (UI)
 | |
|         if ($isValidPUT) {
 | |
|             app('log')->debug('Method is PUT and all fields present and not NULL.');
 | |
|             $data['update_location'] = true;
 | |
|             $data['longitude']       = $this->convertString($longitudeKey);
 | |
|             $data['latitude']        = $this->convertString($latitudeKey);
 | |
|             $data['zoom_level']      = $this->convertString($zoomLevelKey);
 | |
|         }
 | |
|         if ($isValidEmptyPUT) {
 | |
|             app('log')->debug('Method is PUT and all fields present and NULL.');
 | |
|             $data['remove_location'] = true;
 | |
|         }
 | |
|         app('log')->debug(sprintf('Returning longitude: "%s", latitude: "%s", zoom level: "%s"', $data['longitude'], $data['latitude'], $data['zoom_level']));
 | |
|         app('log')->debug(
 | |
|             sprintf(
 | |
|                 'Returning actions: store: %s, update: %s, delete: %s',
 | |
|                 var_export($data['store_location'], true),
 | |
|                 var_export($data['update_location'], true),
 | |
|                 var_export($data['remove_location'], true),
 | |
|             )
 | |
|         );
 | |
| 
 | |
|         return $data;
 | |
|     }
 | |
| 
 | |
|     private function validLongitude(string $longitude): bool
 | |
|     {
 | |
|         $number = (float) $longitude;
 | |
| 
 | |
|         return $number >= -180 && $number <= 180;
 | |
|     }
 | |
| 
 | |
|     private function validLatitude(string $latitude): bool
 | |
|     {
 | |
|         $number = (float) $latitude;
 | |
| 
 | |
|         return $number >= -90 && $number <= 90;
 | |
|     }
 | |
| 
 | |
|     private function getLocationKey(?string $prefix, string $key): string
 | |
|     {
 | |
|         if (null === $prefix) {
 | |
|             return $key;
 | |
|         }
 | |
| 
 | |
|         return sprintf('%s_%s', $prefix, $key);
 | |
|     }
 | |
| 
 | |
|     private function isValidPost(?string $prefix): bool
 | |
|     {
 | |
|         app('log')->debug('Now in isValidPost()');
 | |
|         $longitudeKey   = $this->getLocationKey($prefix, 'longitude');
 | |
|         $latitudeKey    = $this->getLocationKey($prefix, 'latitude');
 | |
|         $zoomLevelKey   = $this->getLocationKey($prefix, 'zoom_level');
 | |
|         $hasLocationKey = $this->getLocationKey($prefix, 'has_location');
 | |
|         // fields must not be null:
 | |
|         if (null !== $this->get($longitudeKey) && null !== $this->get($latitudeKey) && null !== $this->get($zoomLevelKey)) {
 | |
|             app('log')->debug('All fields present');
 | |
|             // if is POST and route contains API, this is enough:
 | |
|             if ('POST' === $this->method() && $this->routeIs('api.v1.*')) {
 | |
|                 app('log')->debug('Is API location');
 | |
| 
 | |
|                 return true;
 | |
|             }
 | |
|             // if is POST and route does not contain API, must also have "has_location" = true
 | |
|             if ('POST' === $this->method() && $this->routeIs('*.store') && !$this->routeIs('api.v1.*') && '' !== $hasLocationKey) {
 | |
|                 app('log')->debug('Is POST + store route.');
 | |
|                 $hasLocation = $this->boolean($hasLocationKey);
 | |
|                 if (true === $hasLocation) {
 | |
|                     app('log')->debug('Has form form location');
 | |
| 
 | |
|                     return true;
 | |
|                 }
 | |
|                 app('log')->debug('Does not have form location');
 | |
| 
 | |
|                 return false;
 | |
|             }
 | |
|             app('log')->debug('Is not POST API or POST form');
 | |
| 
 | |
|             return false;
 | |
|         }
 | |
|         app('log')->debug('Fields not present');
 | |
| 
 | |
|         return false;
 | |
|     }
 | |
| 
 | |
|     private function isValidPUT(?string $prefix): bool
 | |
|     {
 | |
|         $longitudeKey   = $this->getLocationKey($prefix, 'longitude');
 | |
|         $latitudeKey    = $this->getLocationKey($prefix, 'latitude');
 | |
|         $zoomLevelKey   = $this->getLocationKey($prefix, 'zoom_level');
 | |
|         $hasLocationKey = $this->getLocationKey($prefix, 'has_location');
 | |
|         app('log')->debug('Now in isValidPUT()');
 | |
| 
 | |
|         // all fields must be set:
 | |
|         if (null !== $this->get($longitudeKey) && null !== $this->get($latitudeKey) && null !== $this->get($zoomLevelKey)) {
 | |
|             app('log')->debug('All fields present.');
 | |
|             // must be PUT and API route:
 | |
|             if ('PUT' === $this->method() && $this->routeIs('api.v1.*')) {
 | |
|                 app('log')->debug('Is API location');
 | |
| 
 | |
|                 return true;
 | |
|             }
 | |
|             // if POST and not API route, must also have "has_location"
 | |
|             // if is POST and route does not contain API, must also have "has_location" = true
 | |
|             if ('POST' === $this->method() && $this->routeIs('*.update') && !$this->routeIs('api.v1.*') && '' !== $hasLocationKey) {
 | |
|                 app('log')->debug('Is POST + store route.');
 | |
|                 $hasLocation = $this->boolean($hasLocationKey);
 | |
|                 if (true === $hasLocation) {
 | |
|                     app('log')->debug('Has form location data + has_location');
 | |
| 
 | |
|                     return true;
 | |
|                 }
 | |
|                 app('log')->debug('Does not have form location');
 | |
| 
 | |
|                 return false;
 | |
|             }
 | |
|             app('log')->debug('Is not POST API or POST form');
 | |
| 
 | |
|             return false;
 | |
|         }
 | |
|         app('log')->debug('Fields not present');
 | |
| 
 | |
|         return false;
 | |
|     }
 | |
| 
 | |
|     private function isValidEmptyPUT(?string $prefix): bool
 | |
|     {
 | |
|         $longitudeKey = $this->getLocationKey($prefix, 'longitude');
 | |
|         $latitudeKey  = $this->getLocationKey($prefix, 'latitude');
 | |
|         $zoomLevelKey = $this->getLocationKey($prefix, 'zoom_level');
 | |
| 
 | |
|         return (
 | |
|             null === $this->get($longitudeKey)
 | |
|             && null === $this->get($latitudeKey)
 | |
|             && null === $this->get($zoomLevelKey))
 | |
|                && (
 | |
|                    'PUT' === $this->method()
 | |
|                    || ('POST' === $this->method() && $this->routeIs('*.update'))
 | |
|                );
 | |
|     }
 | |
| }
 |