mirror of
				https://github.com/invoiceninja/invoiceninja.git
				synced 2025-10-24 22:19:22 -04:00 
			
		
		
		
	Merge pull request #9362 from turbo124/v5-develop
Handle edge case with Nordigen where account appears active
This commit is contained in:
		
						commit
						f5386f00dd
					
				| @ -97,11 +97,11 @@ class Nordigen | |||||||
|             return $it->transform($out); |             return $it->transform($out); | ||||||
| 
 | 
 | ||||||
|         } catch (\Exception $e) { |         } catch (\Exception $e) { | ||||||
|             if (strpos($e->getMessage(), "Invalid Account ID") !== false) { |  | ||||||
|                 return false; |  | ||||||
|             } |  | ||||||
| 
 | 
 | ||||||
|             throw $e; |             nlog("Nordigen getAccount() failed => {$account_id} => " . $e->getMessage()); | ||||||
|  |              | ||||||
|  |             return false; | ||||||
|  | 
 | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -44,6 +44,7 @@ class StoreProjectRequest extends Request | |||||||
| 
 | 
 | ||||||
|         $rules['name'] = 'required'; |         $rules['name'] = 'required'; | ||||||
|         $rules['client_id'] = 'required|exists:clients,id,company_id,'.$user->company()->id; |         $rules['client_id'] = 'required|exists:clients,id,company_id,'.$user->company()->id; | ||||||
|  |         $rules['budgeted_hours'] = 'sometimes|numeric'; | ||||||
| 
 | 
 | ||||||
|         if (isset($this->number)) { |         if (isset($this->number)) { | ||||||
|             $rules['number'] = Rule::unique('projects')->where('company_id', $user->company()->id); |             $rules['number'] = Rule::unique('projects')->where('company_id', $user->company()->id); | ||||||
| @ -74,6 +75,9 @@ class StoreProjectRequest extends Request | |||||||
|             $input['color'] = ''; |             $input['color'] = ''; | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|  |         if(array_key_exists('budgeted_hours', $input) && empty($input['budgeted_hours'])) | ||||||
|  |             $input['budgeted_hours'] = 0; | ||||||
|  | 
 | ||||||
|         $this->replace($input); |         $this->replace($input); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -45,6 +45,8 @@ class UpdateProjectRequest extends Request | |||||||
|             $rules['number'] = Rule::unique('projects')->where('company_id', $user->company()->id)->ignore($this->project->id); |             $rules['number'] = Rule::unique('projects')->where('company_id', $user->company()->id)->ignore($this->project->id); | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|  |         $rules['budgeted_hours'] = 'sometimes|numeric'; | ||||||
|  | 
 | ||||||
|         if ($this->file('documents') && is_array($this->file('documents'))) { |         if ($this->file('documents') && is_array($this->file('documents'))) { | ||||||
|             $rules['documents.*'] = $this->file_validation; |             $rules['documents.*'] = $this->file_validation; | ||||||
|         } elseif ($this->file('documents')) { |         } elseif ($this->file('documents')) { | ||||||
| @ -73,6 +75,10 @@ class UpdateProjectRequest extends Request | |||||||
|         if (array_key_exists('color', $input) && is_null($input['color'])) { |         if (array_key_exists('color', $input) && is_null($input['color'])) { | ||||||
|             $input['color'] = ''; |             $input['color'] = ''; | ||||||
|         } |         } | ||||||
|  |          | ||||||
|  |         if(array_key_exists('budgeted_hours', $input) && empty($input['budgeted_hours'])) { | ||||||
|  |             $input['budgeted_hours'] = 0; | ||||||
|  |         } | ||||||
| 
 | 
 | ||||||
|         $this->replace($input); |         $this->replace($input); | ||||||
|     } |     } | ||||||
|  | |||||||
| @ -114,23 +114,26 @@ class ProcessBankTransactionsNordigen implements ShouldQueue | |||||||
| 
 | 
 | ||||||
|     private function updateAccount() |     private function updateAccount() | ||||||
|     { |     { | ||||||
|         if (!$this->nordigen->isAccountActive($this->bank_integration->nordigen_account_id)) { |         $is_account_active = $this->nordigen->isAccountActive($this->bank_integration->nordigen_account_id); | ||||||
|  |         $account = $this->nordigen->getAccount($this->bank_integration->nordigen_account_id); | ||||||
|  | 
 | ||||||
|  |         if (!$is_account_active || !$account) { | ||||||
|             $this->bank_integration->disabled_upstream = true; |             $this->bank_integration->disabled_upstream = true; | ||||||
|             $this->bank_integration->save(); |             $this->bank_integration->save(); | ||||||
|             $this->stop_loop = false; |             $this->stop_loop = false; | ||||||
|  | 
 | ||||||
|             nlog("Nordigen: account inactive: " . $this->bank_integration->nordigen_account_id); |             nlog("Nordigen: account inactive: " . $this->bank_integration->nordigen_account_id); | ||||||
|             // @turbo124 @todo send email for expired account
 |  | ||||||
| 
 | 
 | ||||||
|             $this->nordigen->disabledAccountEmail($this->bank_integration); |             $this->nordigen->disabledAccountEmail($this->bank_integration); | ||||||
| 
 | 
 | ||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         $this->nordigen_account = $this->nordigen->getAccount($this->bank_integration->nordigen_account_id); |         $this->nordigen_account = $account; | ||||||
| 
 | 
 | ||||||
|         $this->bank_integration->disabled_upstream = false; |         $this->bank_integration->disabled_upstream = false; | ||||||
|         $this->bank_integration->bank_account_status = $this->nordigen_account['account_status']; |         $this->bank_integration->bank_account_status = $account['account_status']; | ||||||
|         $this->bank_integration->balance = $this->nordigen_account['current_balance']; |         $this->bank_integration->balance = $account['current_balance']; | ||||||
| 
 | 
 | ||||||
|         $this->bank_integration->save(); |         $this->bank_integration->save(); | ||||||
|     } |     } | ||||||
|  | |||||||
| @ -49,6 +49,12 @@ class ProjectTransformer extends EntityTransformer | |||||||
| 
 | 
 | ||||||
|     public function includeClient(Project $project): \League\Fractal\Resource\Item |     public function includeClient(Project $project): \League\Fractal\Resource\Item | ||||||
|     { |     { | ||||||
|  | 
 | ||||||
|  |         if (!$project->client) { | ||||||
|  |             nlog("Project {$project->hashed_id} does not have a client attached - this project is in a bad state"); | ||||||
|  |             return null; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|         $transformer = new ClientTransformer($this->serializer); |         $transformer = new ClientTransformer($this->serializer); | ||||||
| 
 | 
 | ||||||
|         return $this->includeItem($project->client, $transformer, Client::class); |         return $this->includeItem($project->client, $transformer, Client::class); | ||||||
|  | |||||||
| @ -460,7 +460,7 @@ $lang = array( | |||||||
|     'edit_token' => 'Edit Token', |     'edit_token' => 'Edit Token', | ||||||
|     'delete_token' => 'Delete Token', |     'delete_token' => 'Delete Token', | ||||||
|     'token' => 'Token', |     'token' => 'Token', | ||||||
|     'add_gateway' => 'Add Gateway', |     'add_gateway' => 'Add Payment Gateway', | ||||||
|     'delete_gateway' => 'Delete Gateway', |     'delete_gateway' => 'Delete Gateway', | ||||||
|     'edit_gateway' => 'Edit Gateway', |     'edit_gateway' => 'Edit Gateway', | ||||||
|     'updated_gateway' => 'Successfully updated gateway', |     'updated_gateway' => 'Successfully updated gateway', | ||||||
| @ -5251,6 +5251,9 @@ $lang = array( | |||||||
|     'payment_type_help' => 'The default payment type to be used for payments', |     'payment_type_help' => 'The default payment type to be used for payments', | ||||||
|     'quote_valid_until_help' => 'The number of days that the quote is valid for', |     'quote_valid_until_help' => 'The number of days that the quote is valid for', | ||||||
|     'expense_payment_type_help' => 'The default expense payment type to be used', |     'expense_payment_type_help' => 'The default expense payment type to be used', | ||||||
|  |     'paylater' => 'Pay in 4', | ||||||
|  |     'payment_provider' => 'Payment Provider', | ||||||
|  |      | ||||||
| ); | ); | ||||||
| 
 | 
 | ||||||
| return $lang; | return $lang; | ||||||
|  | |||||||
| @ -29,6 +29,8 @@ class ProjectApiTest extends TestCase | |||||||
|     use DatabaseTransactions; |     use DatabaseTransactions; | ||||||
|     use MockAccountData; |     use MockAccountData; | ||||||
| 
 | 
 | ||||||
|  |     protected $faker; | ||||||
|  | 
 | ||||||
|     protected function setUp() :void |     protected function setUp() :void | ||||||
|     { |     { | ||||||
|         parent::setUp(); |         parent::setUp(); | ||||||
| @ -42,6 +44,110 @@ class ProjectApiTest extends TestCase | |||||||
|         Model::reguard(); |         Model::reguard(); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     public function testProjectValidationForBudgetedHoursPut() | ||||||
|  |     { | ||||||
|  | 
 | ||||||
|  |         $data = $this->project->toArray(); | ||||||
|  |         $data['budgeted_hours'] = "aa"; | ||||||
|  | 
 | ||||||
|  |         $response = $this->withHeaders([ | ||||||
|  |             'X-API-SECRET' => config('ninja.api_secret'), | ||||||
|  |             'X-API-TOKEN' => $this->token, | ||||||
|  |         ])->putJson("/api/v1/projects/{$this->project->hashed_id}", $data); | ||||||
|  | 
 | ||||||
|  |         $response->assertStatus(422); | ||||||
|  | 
 | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public function testProjectValidationForBudgetedHoursPutNull() | ||||||
|  |     { | ||||||
|  | 
 | ||||||
|  |         $data = $this->project->toArray(); | ||||||
|  |         $data['budgeted_hours'] = null; | ||||||
|  | 
 | ||||||
|  |         $response = $this->withHeaders([ | ||||||
|  |             'X-API-SECRET' => config('ninja.api_secret'), | ||||||
|  |             'X-API-TOKEN' => $this->token, | ||||||
|  |         ])->putJson("/api/v1/projects/{$this->project->hashed_id}", $data); | ||||||
|  | 
 | ||||||
|  |         $response->assertStatus(200); | ||||||
|  | 
 | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  |     public function testProjectValidationForBudgetedHoursPutEmpty() | ||||||
|  |     { | ||||||
|  | 
 | ||||||
|  |         $data = $this->project->toArray(); | ||||||
|  |         $data['budgeted_hours'] = ""; | ||||||
|  | 
 | ||||||
|  |         $response = $this->withHeaders([ | ||||||
|  |             'X-API-SECRET' => config('ninja.api_secret'), | ||||||
|  |             'X-API-TOKEN' => $this->token, | ||||||
|  |         ])->putJson("/api/v1/projects/{$this->project->hashed_id}", $data); | ||||||
|  | 
 | ||||||
|  |         $response->assertStatus(200); | ||||||
|  | 
 | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  |     public function testProjectValidationForBudgetedHours() | ||||||
|  |     { | ||||||
|  | 
 | ||||||
|  |         $data = [ | ||||||
|  |             'name' => $this->faker->firstName(), | ||||||
|  |             'client_id' => $this->client->hashed_id, | ||||||
|  |             'number' => 'duplicate', | ||||||
|  |             'budgeted_hours' => null | ||||||
|  |         ]; | ||||||
|  | 
 | ||||||
|  |         $response = $this->withHeaders([ | ||||||
|  |             'X-API-SECRET' => config('ninja.api_secret'), | ||||||
|  |             'X-API-TOKEN' => $this->token, | ||||||
|  |         ])->postJson('/api/v1/projects', $data); | ||||||
|  | 
 | ||||||
|  |         $response->assertStatus(200); | ||||||
|  | 
 | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public function testProjectValidationForBudgetedHours2() | ||||||
|  |     { | ||||||
|  | 
 | ||||||
|  |         $data = [ | ||||||
|  |             'name' => $this->faker->firstName(), | ||||||
|  |             'client_id' => $this->client->hashed_id, | ||||||
|  |             'number' => 'duplicate', | ||||||
|  |             'budgeted_hours' => "a" | ||||||
|  |         ]; | ||||||
|  | 
 | ||||||
|  |         $response = $this->withHeaders([ | ||||||
|  |             'X-API-SECRET' => config('ninja.api_secret'), | ||||||
|  |             'X-API-TOKEN' => $this->token, | ||||||
|  |         ])->postJson('/api/v1/projects', $data); | ||||||
|  | 
 | ||||||
|  |         $response->assertStatus(422); | ||||||
|  | 
 | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public function testProjectValidationForBudgetedHours3() | ||||||
|  |     { | ||||||
|  | 
 | ||||||
|  |         $data = [ | ||||||
|  |             'name' => $this->faker->firstName(), | ||||||
|  |             'client_id' => $this->client->hashed_id, | ||||||
|  |             'number' => 'duplicate', | ||||||
|  |             'budgeted_hours' => "" | ||||||
|  |         ]; | ||||||
|  | 
 | ||||||
|  |         $response = $this->withHeaders([ | ||||||
|  |             'X-API-SECRET' => config('ninja.api_secret'), | ||||||
|  |             'X-API-TOKEN' => $this->token, | ||||||
|  |         ])->postJson('/api/v1/projects', $data); | ||||||
|  | 
 | ||||||
|  |         $response->assertStatus(200); | ||||||
|  | 
 | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|     public function testProjectGetFilter() |     public function testProjectGetFilter() | ||||||
|     { |     { | ||||||
|         $response = $this->withHeaders([ |         $response = $this->withHeaders([ | ||||||
|  | |||||||
| @ -72,6 +72,11 @@ trait MockAccountData | |||||||
|     use MakesHash; |     use MakesHash; | ||||||
|     use GeneratesCounter; |     use GeneratesCounter; | ||||||
| 
 | 
 | ||||||
|  |     /** | ||||||
|  |      * @var | ||||||
|  |      */ | ||||||
|  |     public $project; | ||||||
|  | 
 | ||||||
|     /** |     /** | ||||||
|      * @var |      * @var | ||||||
|      */ |      */ | ||||||
|  | |||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user