mirror of
				https://github.com/firefly-iii/firefly-iii.git
				synced 2025-10-25 21:16:47 +00:00 
			
		
		
		
	Can handle some mandatory fields (not all).
This commit is contained in:
		| @@ -52,7 +52,7 @@ class BankController extends Controller | |||||||
|         } |         } | ||||||
|         $importJob = $repository->create($bank); |         $importJob = $repository->create($bank); | ||||||
|  |  | ||||||
|         return redirect(route('import.bank.configure', [$bank, $importJob->key])); |         return redirect(route('import.file.configure', [$importJob->key])); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|   | |||||||
| @@ -25,7 +25,8 @@ namespace FireflyIII\Import\Configurator; | |||||||
| use FireflyIII\Exceptions\FireflyException; | use FireflyIII\Exceptions\FireflyException; | ||||||
| use FireflyIII\Models\ImportJob; | use FireflyIII\Models\ImportJob; | ||||||
| use FireflyIII\Support\Import\Configuration\ConfigurationInterface; | use FireflyIII\Support\Import\Configuration\ConfigurationInterface; | ||||||
| use FireflyIII\Support\Import\Configuration\Spectre\SelectBank; | use FireflyIII\Support\Import\Configuration\Spectre\SelectProvider; | ||||||
|  | use FireflyIII\Support\Import\Configuration\Spectre\InputMandatory; | ||||||
| use FireflyIII\Support\Import\Configuration\Spectre\SelectCountry; | use FireflyIII\Support\Import\Configuration\Spectre\SelectCountry; | ||||||
| use Log; | use Log; | ||||||
|  |  | ||||||
| @@ -97,8 +98,11 @@ class SpectreConfigurator implements ConfiguratorInterface | |||||||
|         if (!$this->job->configuration['selected-country']) { |         if (!$this->job->configuration['selected-country']) { | ||||||
|             return 'import.spectre.select-country'; |             return 'import.spectre.select-country'; | ||||||
|         } |         } | ||||||
|         if (!$this->job->configuration['selected-bank']) { |         if (!$this->job->configuration['selected-provider']) { | ||||||
|             return 'import.spectre.select-bank'; |             return 'import.spectre.select-provider'; | ||||||
|  |         } | ||||||
|  |         if (!$this->job->configuration['has-input-mandatory']) { | ||||||
|  |             return 'import.spectre.input-fields'; | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         throw new FireflyException('No view for state'); |         throw new FireflyException('No view for state'); | ||||||
| @@ -119,13 +123,14 @@ class SpectreConfigurator implements ConfiguratorInterface | |||||||
|      */ |      */ | ||||||
|     public function isJobConfigured(): bool |     public function isJobConfigured(): bool | ||||||
|     { |     { | ||||||
|         $config                     = $this->job->configuration; |         $config                        = $this->job->configuration; | ||||||
|         $config['selected-country'] = $config['selected-country'] ?? false; |         $config['selected-country']    = $config['selected-country'] ?? false; | ||||||
|         $config['selected-bank']    = $config['selected-bank'] ?? false; |         $config['selected-provider']       = $config['selected-provider'] ?? false; | ||||||
|         $this->job->configuration   = $config; |         $config['has-input-mandatory'] = $config['has-input-mandatory'] ?? false; | ||||||
|  |         $this->job->configuration      = $config; | ||||||
|         $this->job->save(); |         $this->job->save(); | ||||||
|  |  | ||||||
|         if ($config['selected-country'] && $config['selected-bank'] && false) { |         if ($config['selected-country'] && $config['selected-provider'] && $config['has-input-mandatory'] && false) { | ||||||
|             return true; |             return true; | ||||||
|         } |         } | ||||||
|  |  | ||||||
| @@ -159,9 +164,11 @@ class SpectreConfigurator implements ConfiguratorInterface | |||||||
|             case !$this->job->configuration['selected-country']: |             case !$this->job->configuration['selected-country']: | ||||||
|                 $class = SelectCountry::class; |                 $class = SelectCountry::class; | ||||||
|                 break; |                 break; | ||||||
|             case !$this->job->configuration['selected-bank']: |             case !$this->job->configuration['selected-provider']: | ||||||
|                 $class = SelectBank::class; |                 $class = SelectProvider::class; | ||||||
|                 break; |                 break; | ||||||
|  |             case !$this->job->configuration['has-input-mandatory']: | ||||||
|  |                 $class = InputMandatory::class; | ||||||
|             default: |             default: | ||||||
|                 break; |                 break; | ||||||
|         } |         } | ||||||
|   | |||||||
							
								
								
									
										100
									
								
								app/Support/Import/Configuration/Spectre/InputMandatory.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										100
									
								
								app/Support/Import/Configuration/Spectre/InputMandatory.php
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,100 @@ | |||||||
|  | <?php | ||||||
|  | declare(strict_types=1); | ||||||
|  |  | ||||||
|  | namespace FireflyIII\Support\Import\Configuration\Spectre; | ||||||
|  |  | ||||||
|  |  | ||||||
|  | use FireflyIII\Exceptions\FireflyException; | ||||||
|  | use FireflyIII\Models\ImportJob; | ||||||
|  | use FireflyIII\Models\SpectreProvider; | ||||||
|  | use FireflyIII\Support\Import\Configuration\ConfigurationInterface; | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * Class InputMandatory | ||||||
|  |  */ | ||||||
|  | class InputMandatory implements ConfigurationInterface | ||||||
|  | { | ||||||
|  |     /** @var ImportJob */ | ||||||
|  |     private $job; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * Get the data necessary to show the configuration screen. | ||||||
|  |      * | ||||||
|  |      * @return array | ||||||
|  |      */ | ||||||
|  |     public function getData(): array | ||||||
|  |     { | ||||||
|  |         $config     = $this->job->configuration; | ||||||
|  |         $providerId = $config['provider']; | ||||||
|  |         $provider   = SpectreProvider::where('spectre_id', $providerId)->first(); | ||||||
|  |         if (is_null($provider)) { | ||||||
|  |             throw new FireflyException(sprintf('Cannot find Spectre provider with ID #%d', $providerId)); | ||||||
|  |         } | ||||||
|  |         $fields    = $provider->data['required_fields'] ?? []; | ||||||
|  |         $positions = []; | ||||||
|  |         // Obtain a list of columns | ||||||
|  |         foreach ($fields as $key => $row) { | ||||||
|  |             $positions[$key] = $row['position']; | ||||||
|  |         } | ||||||
|  |         array_multisort($positions, SORT_ASC, $fields); | ||||||
|  |         $country = SelectCountry::$allCountries[$config['country']] ?? $config['country']; | ||||||
|  |  | ||||||
|  |         return compact('provider', 'country', 'fields'); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * Return possible warning to user. | ||||||
|  |      * | ||||||
|  |      * @return string | ||||||
|  |      */ | ||||||
|  |     public function getWarningMessage(): string | ||||||
|  |     { | ||||||
|  |         return ''; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * @param ImportJob $job | ||||||
|  |      * | ||||||
|  |      * @return ConfigurationInterface | ||||||
|  |      */ | ||||||
|  |     public function setJob(ImportJob $job) | ||||||
|  |     { | ||||||
|  |         $this->job = $job; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * Store the result. | ||||||
|  |      * | ||||||
|  |      * @param array $data | ||||||
|  |      * | ||||||
|  |      * @return bool | ||||||
|  |      */ | ||||||
|  |     public function storeConfiguration(array $data): bool | ||||||
|  |     { | ||||||
|  |         $config     = $this->job->configuration; | ||||||
|  |         $providerId = $config['provider']; | ||||||
|  |         $provider   = SpectreProvider::where('spectre_id', $providerId)->first(); | ||||||
|  |         if (is_null($provider)) { | ||||||
|  |             throw new FireflyException(sprintf('Cannot find Spectre provider with ID #%d', $providerId)); | ||||||
|  |         } | ||||||
|  |         $mandatory = []; | ||||||
|  |         $fields    = $provider->data['required_fields'] ?? []; | ||||||
|  |         foreach ($fields as $field) { | ||||||
|  |             $name             = $field['name']; | ||||||
|  |             $mandatory[$name] = app('crypt')->encrypt($data[$name]) ?? null; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         // store in config of job: | ||||||
|  |         $config['mandatory-fields']    = $mandatory; | ||||||
|  |         $config['has-input-mandatory'] = true; | ||||||
|  |         $this->job->configuration      = $config; | ||||||
|  |         $this->job->save(); | ||||||
|  |  | ||||||
|  |         // try to grab login for this job. See what happens? | ||||||
|  |         // fire job that creates login object. user is redirected to "wait here" page (status page). Page should | ||||||
|  |         // refresh and go back to interactive when user is supposed to enter SMS code or something. | ||||||
|  |         // otherwise start downloading stuff | ||||||
|  |  | ||||||
|  |         return true; | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -13,7 +13,7 @@ use FireflyIII\Support\Import\Configuration\ConfigurationInterface; | |||||||
|  */ |  */ | ||||||
| class SelectCountry implements ConfigurationInterface | class SelectCountry implements ConfigurationInterface | ||||||
| { | { | ||||||
|     private $allCountries |     public static $allCountries | ||||||
|         = [ |         = [ | ||||||
|             'AF' => 'Afghanistan', |             'AF' => 'Afghanistan', | ||||||
|             'AX' => 'Aland Islands', |             'AX' => 'Aland Islands', | ||||||
| @@ -282,7 +282,7 @@ class SelectCountry implements ConfigurationInterface | |||||||
|         $countries = []; |         $countries = []; | ||||||
|         /** @var SpectreProvider $provider */ |         /** @var SpectreProvider $provider */ | ||||||
|         foreach ($providers as $provider) { |         foreach ($providers as $provider) { | ||||||
|             $countries[$provider->country_code] = $this->allCountries[$provider->country_code] ?? $provider->country_code; |             $countries[$provider->country_code] = self::$allCountries[$provider->country_code] ?? $provider->country_code; | ||||||
|         } |         } | ||||||
|         asort($countries); |         asort($countries); | ||||||
|  |  | ||||||
|   | |||||||
| @@ -9,9 +9,9 @@ use FireflyIII\Models\SpectreProvider; | |||||||
| use FireflyIII\Support\Import\Configuration\ConfigurationInterface; | use FireflyIII\Support\Import\Configuration\ConfigurationInterface; | ||||||
| 
 | 
 | ||||||
| /** | /** | ||||||
|  * Class SelectBank |  * Class SelectProvider | ||||||
|  */ |  */ | ||||||
| class SelectBank implements ConfigurationInterface | class SelectProvider implements ConfigurationInterface | ||||||
| { | { | ||||||
|     /** @var ImportJob */ |     /** @var ImportJob */ | ||||||
|     private $job; |     private $job; | ||||||
| @@ -32,8 +32,9 @@ class SelectBank implements ConfigurationInterface | |||||||
|             $name                   = $provider->data['name']; |             $name                   = $provider->data['name']; | ||||||
|             $providers[$providerId] = $name; |             $providers[$providerId] = $name; | ||||||
|         } |         } | ||||||
|  |         $country = SelectCountry::$allCountries[$config['country']] ?? $config['country']; | ||||||
| 
 | 
 | ||||||
|         return compact('providers'); |         return compact('providers', 'country'); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     /** |     /** | ||||||
| @@ -66,8 +67,8 @@ class SelectBank implements ConfigurationInterface | |||||||
|     public function storeConfiguration(array $data): bool |     public function storeConfiguration(array $data): bool | ||||||
|     { |     { | ||||||
|         $config                   = $this->job->configuration; |         $config                   = $this->job->configuration; | ||||||
|         $config['bank']           = intval($data['bank_code']) ?? 0; // default to fake country.
 |         $config['provider']       = intval($data['provider_code']) ?? 0; // default to fake country.
 | ||||||
|         $config['selected-bank']  = true; |         $config['selected-provider']  = true; | ||||||
|         $this->job->configuration = $config; |         $this->job->configuration = $config; | ||||||
|         $this->job->save(); |         $this->job->save(); | ||||||
| 
 | 
 | ||||||
| @@ -3,14 +3,19 @@ declare(strict_types=1); | |||||||
|  |  | ||||||
|  |  | ||||||
| return [ | return [ | ||||||
|     'bunq_prerequisites_title'     => 'Prerequisites for an import from bunq', |     'bunq_prerequisites_title'      => 'Prerequisites for an import from bunq', | ||||||
|     'bunq_prerequisites_text'      => 'In order to import from bunq, you need to obtain an API key. You can do this through the app.', |     'bunq_prerequisites_text'       => 'In order to import from bunq, you need to obtain an API key. You can do this through the app.', | ||||||
|  |  | ||||||
|     // Spectre: |     // Spectre: | ||||||
|     'spectre_title'                => 'Import using Spectre', |     'spectre_title'                 => 'Import using Spectre', | ||||||
|     'spectre_prerequisites_title'  => 'Prerequisites for an import using Spectre', |     'spectre_prerequisites_title'   => 'Prerequisites for an import using Spectre', | ||||||
|     'spectre_prerequisites_text'   => 'In order to import data using the Spectre API, you need to prove some secrets. They can be found on the <a href="https://www.saltedge.com/clients/profile/secrets">secrets page</a>.', |     'spectre_prerequisites_text'    => 'In order to import data using the Spectre API, you need to prove some secrets. They can be found on the <a href="https://www.saltedge.com/clients/profile/secrets">secrets page</a>.', | ||||||
|     'spectre_enter_pub_key'        => 'The import will only work when you enter this public key on your <a href="https://www.saltedge.com/clients/security/edit">security page</a>.', |     'spectre_enter_pub_key'         => 'The import will only work when you enter this public key on your <a href="https://www.saltedge.com/clients/security/edit">security page</a>.', | ||||||
|     'spectre_select_country_title' => 'Select a country', |     'spectre_select_country_title'  => 'Select a country', | ||||||
|     'spectre_select_country_text'  => 'Firefly III has a large selection of banks and sites from which Spectre can download transactional data. These banks are sorted by country. Please not that there is a "Fake Country" for when you wish to test something. If you wish to import from other financial tools, please use the imaginary country called "Other financial applications". By default, Spectre only allows you to download data from fake banks. Make sure your status is "Live" on your <a href="https://www.saltedge.com/clients/dashboard">Dashboard</a> if you wish to download from real banks.', |     'spectre_select_country_text'   => 'Firefly III has a large selection of banks and sites from which Spectre can download transactional data. These banks are sorted by country. Please not that there is a "Fake Country" for when you wish to test something. If you wish to import from other financial tools, please use the imaginary country called "Other financial applications". By default, Spectre only allows you to download data from fake banks. Make sure your status is "Live" on your <a href="https://www.saltedge.com/clients/dashboard">Dashboard</a> if you wish to download from real banks.', | ||||||
|  |     'spectre_select_provider_title' => 'Select a bank', | ||||||
|  |     'spectre_select_provider_text'  => 'Spectre supports the following banks or financial services grouped under <em>:country</em>. Please pick the one you wish to import from.', | ||||||
|  |     'spectre_input_fields_title'    => 'Input mandatory fields', | ||||||
|  |     'spectre_input_fields_text'     => 'The following fields are mandated by ":provider" (from :country).', | ||||||
|  |     'spectre_instructions_english'  => 'These instructions are provided by Spectre for your convencience. They are in English:', | ||||||
| ]; | ]; | ||||||
|   | |||||||
							
								
								
									
										65
									
								
								resources/views/import/spectre/input-fields.twig
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										65
									
								
								resources/views/import/spectre/input-fields.twig
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,65 @@ | |||||||
|  | {% extends "./layout/default" %} | ||||||
|  |  | ||||||
|  | {% block breadcrumbs %} | ||||||
|  |     {{ Breadcrumbs.renderIfExists }} | ||||||
|  | {% endblock %} | ||||||
|  | {% block content %} | ||||||
|  |     <div class="row"> | ||||||
|  |         <form class="form-horizontal" action="{{ route('import.file.process-configuration', job.key) }}" method="post"> | ||||||
|  |             <input type="hidden" name="_token" value="{{ csrf_token() }}"/> | ||||||
|  |             <div class="col-lg-12 col-md-12 col-sm-12"> | ||||||
|  |                 <div class="box box-default"> | ||||||
|  |                     <div class="box-header with-border"> | ||||||
|  |                         <h3 class="box-title">{{ trans('bank.spectre_input_fields_title') }}</h3> | ||||||
|  |                     </div> | ||||||
|  |                     <div class="box-body"> | ||||||
|  |                         <div class="row"> | ||||||
|  |                             <div class="col-lg-8"> | ||||||
|  |                                 <p> | ||||||
|  |                                     {{ trans('bank.spectre_input_fields_text',{provider: data.provider.data.name, country: data.country})|raw }} | ||||||
|  |                                 </p> | ||||||
|  |                                 <p> | ||||||
|  |                                     {{ trans('bank.spectre_instructions_english') }} | ||||||
|  |                                 </p> | ||||||
|  |                                 <p> | ||||||
|  |                                     {{ data.provider.data.instruction|nl2br }} | ||||||
|  |                                 </p> | ||||||
|  |                             </div> | ||||||
|  |                         </div> | ||||||
|  |  | ||||||
|  |                         <div class="row"> | ||||||
|  |                             <div class="col-lg-8"> | ||||||
|  |                                 {% for field in data.fields %} | ||||||
|  |                                     {# text, password, select, file #} | ||||||
|  |                                     {% if field.nature == 'text' %} | ||||||
|  |                                         {{ ExpandedForm.text(field.name,null, {label: field.english_name ~ ' ('~field.localized_name~')'}) }} | ||||||
|  |                                     {% endif %} | ||||||
|  |                                     {% if field.nature == 'password' %} | ||||||
|  |                                         {{ ExpandedForm.password(field.name, {label: field.english_name ~ ' ('~field.localized_name~')'}) }} | ||||||
|  |                                     {% endif %} | ||||||
|  |                                     {% if field.nature == 'select' %} | ||||||
|  |                                         DO NOT SUPPORT | ||||||
|  |                                         {{ dump(field) }} | ||||||
|  |                                     {% endif %} | ||||||
|  |                                     {% if field.narture == 'file' %} | ||||||
|  |                                         DO NOT SUPPORT | ||||||
|  |                                         {{ dump(field) }} | ||||||
|  |                                     {% endif %} | ||||||
|  |  | ||||||
|  |                                 {% endfor %} | ||||||
|  |                             </div> | ||||||
|  |                         </div> | ||||||
|  |                         <div class="box-footer"> | ||||||
|  |                             <button type="submit" class="btn pull-right btn-success"> | ||||||
|  |                                 {{ ('submit')|_ }} | ||||||
|  |                             </button> | ||||||
|  |                         </div> | ||||||
|  |                     </div> | ||||||
|  |                 </div> | ||||||
|  |         </form> | ||||||
|  |     </div> | ||||||
|  | {% endblock %} | ||||||
|  | {% block scripts %} | ||||||
|  | {% endblock %} | ||||||
|  | {% block styles %} | ||||||
|  | {% endblock %} | ||||||
| @@ -16,7 +16,7 @@ | |||||||
|                         <div class="row"> |                         <div class="row"> | ||||||
|                             <div class="col-lg-8"> |                             <div class="col-lg-8"> | ||||||
|                                 <p> |                                 <p> | ||||||
|                                     {{ trans('bank.spectre_select_country_text') }} |                                     {{ trans('bank.spectre_select_country_text')|raw }} | ||||||
|                                 </p> |                                 </p> | ||||||
|                             </div> |                             </div> | ||||||
|                         </div> |                         </div> | ||||||
|   | |||||||
| @@ -16,14 +16,14 @@ | |||||||
|                         <div class="row"> |                         <div class="row"> | ||||||
|                             <div class="col-lg-8"> |                             <div class="col-lg-8"> | ||||||
|                                 <p> |                                 <p> | ||||||
|                                     {{ trans('bank.spectre_select_bank_title') }} |                                     {{ trans('bank.spectre_select_provider_text',{country: data.country})|raw }} | ||||||
|                                 </p> |                                 </p> | ||||||
|                             </div> |                             </div> | ||||||
|                         </div> |                         </div> | ||||||
| 
 | 
 | ||||||
|                         <div class="row"> |                         <div class="row"> | ||||||
|                             <div class="col-lg-8"> |                             <div class="col-lg-8"> | ||||||
|                                 {{ ExpandedForm.select('bank_code', data.providers, null)|raw }} |                                 {{ ExpandedForm.select('provider_code', data.providers, null)|raw }} | ||||||
|                             </div> |                             </div> | ||||||
|                         </div> |                         </div> | ||||||
|                         <div class="box-footer"> |                         <div class="box-footer"> | ||||||
		Reference in New Issue
	
	Block a user