| 
									
										
										
										
											2015-02-25 15:19:14 +01:00
										 |  |  | <?php | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace FireflyIII\Repositories\Bill; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-05 18:20:06 +02:00
										 |  |  | use Auth; | 
					
						
							| 
									
										
										
										
											2015-02-25 15:19:14 +01:00
										 |  |  | use Carbon\Carbon; | 
					
						
							| 
									
										
										
										
											2015-05-08 14:00:49 +02:00
										 |  |  | use DB; | 
					
						
							| 
									
										
										
										
											2015-12-12 10:33:19 +01:00
										 |  |  | use FireflyIII\Models\Account; | 
					
						
							| 
									
										
										
										
											2015-02-25 15:19:14 +01:00
										 |  |  | use FireflyIII\Models\Bill; | 
					
						
							| 
									
										
										
										
											2015-04-05 18:20:06 +02:00
										 |  |  | use FireflyIII\Models\Transaction; | 
					
						
							| 
									
										
										
										
											2015-02-25 15:19:14 +01:00
										 |  |  | use FireflyIII\Models\TransactionJournal; | 
					
						
							| 
									
										
										
										
											2015-12-26 09:39:35 +01:00
										 |  |  | use Illuminate\Database\Query\JoinClause; | 
					
						
							| 
									
										
										
										
											2015-04-05 18:20:06 +02:00
										 |  |  | use Illuminate\Support\Collection; | 
					
						
							| 
									
										
										
										
											2015-10-03 17:51:13 +02:00
										 |  |  | use Log; | 
					
						
							| 
									
										
										
										
											2015-03-03 17:40:17 +01:00
										 |  |  | use Navigation; | 
					
						
							| 
									
										
										
										
											2015-07-09 09:41:54 +02:00
										 |  |  | use Steam; | 
					
						
							| 
									
										
										
										
											2015-02-25 15:19:14 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | /** | 
					
						
							|  |  |  |  * Class BillRepository | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * @package FireflyIII\Repositories\Bill | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | class BillRepository implements BillRepositoryInterface | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2015-05-17 10:01:47 +02:00
										 |  |  |     /** | 
					
						
							| 
									
										
										
										
											2015-12-26 09:40:24 +01:00
										 |  |  |      * @deprecated | 
					
						
							| 
									
										
										
										
											2015-05-17 10:01:47 +02:00
										 |  |  |      * Returns the sum of all payments connected to this bill between the dates. | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @param Bill   $bill | 
					
						
							|  |  |  |      * @param Carbon $start | 
					
						
							|  |  |  |      * @param Carbon $end | 
					
						
							|  |  |  |      * | 
					
						
							| 
									
										
										
										
											2015-05-26 18:57:31 +00:00
										 |  |  |      * @return integer | 
					
						
							| 
									
										
										
										
											2015-05-17 10:01:47 +02:00
										 |  |  |      */ | 
					
						
							|  |  |  |     public function billPaymentsInRange(Bill $bill, Carbon $start, Carbon $end) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         $amount   = 0; | 
					
						
							|  |  |  |         $journals = $bill->transactionjournals()->before($end)->after($start)->get(); | 
					
						
							|  |  |  |         /** @var TransactionJournal $journal */ | 
					
						
							|  |  |  |         foreach ($journals as $journal) { | 
					
						
							|  |  |  |             $amount += $journal->amount; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return $amount; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-09 09:41:54 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-07 10:14:10 +02:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * Create a fake bill to help the chart controller. | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @param string $description | 
					
						
							|  |  |  |      * @param Carbon $date | 
					
						
							|  |  |  |      * @param float  $amount | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @return Bill | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     public function createFakeBill($description, Carbon $date, $amount) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         $bill              = new Bill; | 
					
						
							|  |  |  |         $bill->name        = $description; | 
					
						
							|  |  |  |         $bill->match       = $description; | 
					
						
							|  |  |  |         $bill->amount_min  = $amount; | 
					
						
							|  |  |  |         $bill->amount_max  = $amount; | 
					
						
							|  |  |  |         $bill->date        = $date; | 
					
						
							|  |  |  |         $bill->repeat_freq = 'monthly'; | 
					
						
							|  |  |  |         $bill->skip        = 0; | 
					
						
							|  |  |  |         $bill->automatch   = false; | 
					
						
							|  |  |  |         $bill->active      = false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return $bill; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-05 18:20:06 +02:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * @param Bill $bill | 
					
						
							|  |  |  |      * | 
					
						
							| 
									
										
										
										
											2015-05-26 18:57:31 +00:00
										 |  |  |      * @return boolean|null | 
					
						
							| 
									
										
										
										
											2015-04-05 18:20:06 +02:00
										 |  |  |      */ | 
					
						
							|  |  |  |     public function destroy(Bill $bill) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         return $bill->delete(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-07 10:14:10 +02:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * @return Collection | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     public function getActiveBills() | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         /** @var Collection $set */ | 
					
						
							| 
									
										
										
										
											2015-12-26 09:39:35 +01:00
										 |  |  |         $set = Auth::user()->bills()->where('active', 1)->get()->sortBy('name'); | 
					
						
							| 
									
										
										
										
											2015-04-07 10:14:10 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         return $set; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-05 18:20:06 +02:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * @return Collection | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     public function getBills() | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2015-04-06 09:15:59 +02:00
										 |  |  |         /** @var Collection $set */ | 
					
						
							| 
									
										
										
										
											2015-06-17 06:11:35 +02:00
										 |  |  |         $set = Auth::user()->bills()->orderBy('name', 'ASC')->get(); | 
					
						
							| 
									
										
										
										
											2015-04-06 09:15:59 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-01 10:52:42 +02:00
										 |  |  |         $set = $set->sortBy( | 
					
						
							|  |  |  |             function (Bill $bill) { | 
					
						
							| 
									
										
										
										
											2015-09-08 19:36:54 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |                 $int = $bill->active == 1 ? 0 : 1; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-09-08 19:38:04 +02:00
										 |  |  |                 return $int . strtolower($bill->name); | 
					
						
							| 
									
										
										
										
											2015-07-01 10:52:42 +02:00
										 |  |  |             } | 
					
						
							|  |  |  |         ); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-06 09:15:59 +02:00
										 |  |  |         return $set; | 
					
						
							| 
									
										
										
										
											2015-04-05 18:20:06 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-12 10:33:19 +01:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * @param Collection $accounts | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @return Collection | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     public function getBillsForAccounts(Collection $accounts) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         /** @var Collection $set */ | 
					
						
							|  |  |  |         $set = Auth::user()->bills()->orderBy('name', 'ASC')->get(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $ids = []; | 
					
						
							|  |  |  |         /** @var Account $account */ | 
					
						
							|  |  |  |         foreach ($accounts as $account) { | 
					
						
							|  |  |  |             $ids[] = $account->id; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $set = $set->filter( | 
					
						
							|  |  |  |             function (Bill $bill) use ($ids) { | 
					
						
							|  |  |  |                 // get transaction journals from or to any of the mentioned accounts.
 | 
					
						
							|  |  |  |                 // if zero, return null.
 | 
					
						
							|  |  |  |                 $journals = $bill->transactionjournals()->leftJoin('transactions', 'transactions.transaction_journal_id', '=', 'transaction_journals.id') | 
					
						
							|  |  |  |                                  ->whereIn('transactions.account_id', $ids)->count(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 return ($journals > 0); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         ); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $set = $set->sortBy( | 
					
						
							|  |  |  |             function (Bill $bill) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 $int = $bill->active == 1 ? 0 : 1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 return $int . strtolower($bill->name); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         ); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return $set; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-05 18:20:06 +02:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * @param Bill $bill | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @return Collection | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     public function getJournals(Bill $bill) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         return $bill->transactionjournals()->withRelevantData() | 
					
						
							|  |  |  |                     ->orderBy('transaction_journals.date', 'DESC') | 
					
						
							|  |  |  |                     ->orderBy('transaction_journals.order', 'ASC') | 
					
						
							|  |  |  |                     ->orderBy('transaction_journals.id', 'DESC') | 
					
						
							| 
									
										
										
										
											2015-05-23 15:42:19 +02:00
										 |  |  |                     ->get(['transaction_journals.*']); | 
					
						
							| 
									
										
										
										
											2015-04-05 18:20:06 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-07 10:14:10 +02:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * Get all journals that were recorded on this bill between these dates. | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @param Bill   $bill | 
					
						
							|  |  |  |      * @param Carbon $start | 
					
						
							|  |  |  |      * @param Carbon $end | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @return Collection | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     public function getJournalsInRange(Bill $bill, Carbon $start, Carbon $end) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         return $bill->transactionjournals()->before($end)->after($start)->get(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-05 18:20:06 +02:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * @param Bill $bill | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @return Collection | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     public function getPossiblyRelatedJournals(Bill $bill) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2015-05-08 14:00:49 +02:00
										 |  |  |         $set = DB::table('transactions')->where('amount', '>', 0)->where('amount', '>=', $bill->amount_min)->where('amount', '<=', $bill->amount_max)->get( | 
					
						
							| 
									
										
										
										
											2015-04-05 18:20:06 +02:00
										 |  |  |             ['transaction_journal_id'] | 
					
						
							|  |  |  |         ); | 
					
						
							|  |  |  |         $ids = []; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         /** @var Transaction $entry */ | 
					
						
							|  |  |  |         foreach ($set as $entry) { | 
					
						
							|  |  |  |             $ids[] = intval($entry->transaction_journal_id); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         $journals = new Collection; | 
					
						
							|  |  |  |         if (count($ids) > 0) { | 
					
						
							| 
									
										
										
										
											2015-12-09 22:39:50 -02:00
										 |  |  |             $journals = Auth::user()->transactionjournals()->transactionTypes([TransactionType::WITHDRAWAL])->whereIn('transaction_journals.id', $ids)->get( | 
					
						
							| 
									
										
										
										
											2015-10-03 17:55:08 +02:00
										 |  |  |                 ['transaction_journals.*'] | 
					
						
							|  |  |  |             ); | 
					
						
							| 
									
										
										
										
											2015-04-05 18:20:06 +02:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return $journals; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-03 17:40:17 +01:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * Every bill repeats itself weekly, monthly or yearly (or whatever). This method takes a date-range (usually the view-range of Firefly itself) | 
					
						
							|  |  |  |      * and returns date ranges that fall within the given range; those ranges are the bills expected. When a bill is due on the 14th of the month and | 
					
						
							|  |  |  |      * you give 1st and the 31st of that month as argument, you'll get one response, matching the range of your bill. | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @param Bill   $bill | 
					
						
							|  |  |  |      * @param Carbon $start | 
					
						
							|  |  |  |      * @param Carbon $end | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @return mixed | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     public function getRanges(Bill $bill, Carbon $start, Carbon $end) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         $startOfBill = $bill->date; | 
					
						
							|  |  |  |         $startOfBill = Navigation::startOfPeriod($startOfBill, $bill->repeat_freq); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // all periods of this bill up until the current period:
 | 
					
						
							|  |  |  |         $billStarts = []; | 
					
						
							|  |  |  |         while ($startOfBill < $end) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             $endOfBill = Navigation::endOfPeriod($startOfBill, $bill->repeat_freq); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             $billStarts[] = [ | 
					
						
							|  |  |  |                 'start' => clone $startOfBill, | 
					
						
							|  |  |  |                 'end'   => clone $endOfBill, | 
					
						
							|  |  |  |             ]; | 
					
						
							|  |  |  |             // actually the next one:
 | 
					
						
							|  |  |  |             $startOfBill = Navigation::addPeriod($startOfBill, $bill->repeat_freq, $bill->skip); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         // for each
 | 
					
						
							|  |  |  |         $validRanges = []; | 
					
						
							|  |  |  |         foreach ($billStarts as $dateEntry) { | 
					
						
							|  |  |  |             if ($dateEntry['end'] > $start && $dateEntry['start'] < $end) { | 
					
						
							|  |  |  |                 // count transactions for bill in this range (not relevant yet!):
 | 
					
						
							|  |  |  |                 $validRanges[] = $dateEntry; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return $validRanges; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-05 18:20:06 +02:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * @param Bill $bill | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @return Carbon|null | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     public function lastFoundMatch(Bill $bill) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         $last = $bill->transactionjournals()->orderBy('date', 'DESC')->first(); | 
					
						
							|  |  |  |         if ($last) { | 
					
						
							|  |  |  |             return $last->date; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return null; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-25 15:19:14 +01:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * @param Bill $bill | 
					
						
							|  |  |  |      * | 
					
						
							| 
									
										
										
										
											2015-05-26 20:59:16 +02:00
										 |  |  |      * @return \Carbon\Carbon | 
					
						
							| 
									
										
										
										
											2015-02-25 15:19:14 +01:00
										 |  |  |      */ | 
					
						
							|  |  |  |     public function nextExpectedMatch(Bill $bill) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2015-03-03 17:40:17 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-25 15:19:14 +01:00
										 |  |  |         $finalDate = null; | 
					
						
							|  |  |  |         if ($bill->active == 0) { | 
					
						
							|  |  |  |             return $finalDate; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         /* | 
					
						
							|  |  |  |          * $today is the start of the next period, to make sure FF3 won't miss anything | 
					
						
							|  |  |  |          * when the current period has a transaction journal. | 
					
						
							|  |  |  |          */ | 
					
						
							| 
									
										
										
										
											2015-05-05 10:23:01 +02:00
										 |  |  |         /** @var \Carbon\Carbon $obj */ | 
					
						
							| 
									
										
										
										
											2015-05-05 12:57:27 +02:00
										 |  |  |         $obj   = new Carbon; | 
					
						
							| 
									
										
										
										
											2015-05-05 10:23:01 +02:00
										 |  |  |         $today = Navigation::addPeriod($obj, $bill->repeat_freq, 0); | 
					
						
							| 
									
										
										
										
											2015-02-25 15:19:14 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |         $skip  = $bill->skip + 1; | 
					
						
							| 
									
										
										
										
											2015-05-05 10:23:01 +02:00
										 |  |  |         $start = Navigation::startOfPeriod($obj, $bill->repeat_freq); | 
					
						
							| 
									
										
										
										
											2015-02-25 15:19:14 +01:00
										 |  |  |         /* | 
					
						
							|  |  |  |          * go back exactly one month/week/etc because FF3 does not care about 'next' | 
					
						
							|  |  |  |          * bills if they're too far into the past. | 
					
						
							|  |  |  |          */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $counter = 0; | 
					
						
							|  |  |  |         while ($start <= $today) { | 
					
						
							|  |  |  |             if (($counter % $skip) == 0) { | 
					
						
							|  |  |  |                 // do something.
 | 
					
						
							|  |  |  |                 $end          = Navigation::endOfPeriod(clone $start, $bill->repeat_freq); | 
					
						
							|  |  |  |                 $journalCount = $bill->transactionjournals()->before($end)->after($start)->count(); | 
					
						
							|  |  |  |                 if ($journalCount == 0) { | 
					
						
							| 
									
										
										
										
											2015-05-27 07:58:54 +02:00
										 |  |  |                     $finalDate = new Carbon($start->format('Y-m-d')); | 
					
						
							| 
									
										
										
										
											2015-02-25 15:19:14 +01:00
										 |  |  |                     break; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             // add period for next round!
 | 
					
						
							|  |  |  |             $start = Navigation::addPeriod($start, $bill->repeat_freq, 0); | 
					
						
							|  |  |  |             $counter++; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return $finalDate; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * @param Bill               $bill | 
					
						
							|  |  |  |      * @param TransactionJournal $journal | 
					
						
							|  |  |  |      * | 
					
						
							| 
									
										
										
										
											2015-05-26 18:57:31 +00:00
										 |  |  |      * @return boolean|null | 
					
						
							| 
									
										
										
										
											2015-02-25 15:19:14 +01:00
										 |  |  |      */ | 
					
						
							|  |  |  |     public function scan(Bill $bill, TransactionJournal $journal) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2015-11-20 20:13:10 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |         /* | 
					
						
							|  |  |  |          * Can only support withdrawals. | 
					
						
							|  |  |  |          */ | 
					
						
							| 
									
										
										
										
											2015-12-09 22:39:50 -02:00
										 |  |  |         if (false === $journal->isWithdrawal()) { | 
					
						
							| 
									
										
										
										
											2015-11-20 20:13:10 +01:00
										 |  |  |             return false; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-25 15:19:14 +01:00
										 |  |  |         $matches     = explode(',', $bill->match); | 
					
						
							| 
									
										
										
										
											2015-06-06 15:36:12 +02:00
										 |  |  |         $description = strtolower($journal->description) . ' ' . strtolower($journal->destination_account->name); | 
					
						
							| 
									
										
										
										
											2015-06-07 08:13:19 +02:00
										 |  |  |         $wordMatch   = $this->doWordMatch($matches, $description); | 
					
						
							| 
									
										
										
										
											2015-10-01 07:48:50 +02:00
										 |  |  |         $amountMatch = $this->doAmountMatch($journal->amount_positive, $bill->amount_min, $bill->amount_max); | 
					
						
							| 
									
										
										
										
											2015-10-03 17:51:13 +02:00
										 |  |  |         Log::debug('Journal #' . $journal->id . ' has description "' . $description . '"'); | 
					
						
							| 
									
										
										
										
											2015-02-25 15:19:14 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-20 20:13:10 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-25 15:19:14 +01:00
										 |  |  |         /* | 
					
						
							|  |  |  |          * If both, update! | 
					
						
							|  |  |  |          */ | 
					
						
							|  |  |  |         if ($wordMatch && $amountMatch) { | 
					
						
							|  |  |  |             $journal->bill()->associate($bill); | 
					
						
							|  |  |  |             $journal->save(); | 
					
						
							| 
									
										
										
										
											2015-06-07 08:13:19 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |             return true; | 
					
						
							| 
									
										
										
										
											2015-10-03 17:51:13 +02:00
										 |  |  |         } else { | 
					
						
							|  |  |  |             Log::debug('Wordmatch: ' . (($wordMatch) ? 'true' : 'false') . ' AmountMatch: ' . (($amountMatch) ? 'true' : 'false')); | 
					
						
							| 
									
										
										
										
											2015-06-07 08:13:19 +02:00
										 |  |  |         } | 
					
						
							|  |  |  |         if ($bill->id == $journal->bill_id) { | 
					
						
							|  |  |  |             // if no match, but bill used to match, remove it:
 | 
					
						
							|  |  |  |             $journal->bill_id = null; | 
					
						
							|  |  |  |             $journal->save(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             return true; | 
					
						
							| 
									
										
										
										
											2015-02-25 15:19:14 +01:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2015-06-07 08:13:19 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-13 10:02:36 +02:00
										 |  |  |         return false; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-25 15:19:14 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * @param array $data | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @return Bill | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     public function store(array $data) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $bill = Bill::create( | 
					
						
							|  |  |  |             [ | 
					
						
							|  |  |  |                 'name'        => $data['name'], | 
					
						
							|  |  |  |                 'match'       => $data['match'], | 
					
						
							|  |  |  |                 'amount_min'  => $data['amount_min'], | 
					
						
							|  |  |  |                 'user_id'     => $data['user'], | 
					
						
							|  |  |  |                 'amount_max'  => $data['amount_max'], | 
					
						
							|  |  |  |                 'date'        => $data['date'], | 
					
						
							|  |  |  |                 'repeat_freq' => $data['repeat_freq'], | 
					
						
							|  |  |  |                 'skip'        => $data['skip'], | 
					
						
							|  |  |  |                 'automatch'   => $data['automatch'], | 
					
						
							|  |  |  |                 'active'      => $data['active'], | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             ] | 
					
						
							|  |  |  |         ); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return $bill; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * @param Bill  $bill | 
					
						
							|  |  |  |      * @param array $data | 
					
						
							|  |  |  |      * | 
					
						
							| 
									
										
										
										
											2015-05-26 18:57:31 +00:00
										 |  |  |      * @return Bill | 
					
						
							| 
									
										
										
										
											2015-02-25 15:19:14 +01:00
										 |  |  |      */ | 
					
						
							|  |  |  |     public function update(Bill $bill, array $data) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $bill->name        = $data['name']; | 
					
						
							|  |  |  |         $bill->match       = $data['match']; | 
					
						
							|  |  |  |         $bill->amount_min  = $data['amount_min']; | 
					
						
							|  |  |  |         $bill->amount_max  = $data['amount_max']; | 
					
						
							|  |  |  |         $bill->date        = $data['date']; | 
					
						
							|  |  |  |         $bill->repeat_freq = $data['repeat_freq']; | 
					
						
							|  |  |  |         $bill->skip        = $data['skip']; | 
					
						
							|  |  |  |         $bill->automatch   = $data['automatch']; | 
					
						
							|  |  |  |         $bill->active      = $data['active']; | 
					
						
							|  |  |  |         $bill->save(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return $bill; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2015-06-07 08:13:19 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * @param array $matches | 
					
						
							|  |  |  |      * @param       $description | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @return bool | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     protected function doWordMatch(array $matches, $description) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         $wordMatch = false; | 
					
						
							|  |  |  |         $count     = 0; | 
					
						
							|  |  |  |         foreach ($matches as $word) { | 
					
						
							|  |  |  |             if (!(strpos($description, strtolower($word)) === false)) { | 
					
						
							|  |  |  |                 $count++; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if ($count >= count($matches)) { | 
					
						
							|  |  |  |             $wordMatch = true; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return $wordMatch; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * @param float $amount | 
					
						
							|  |  |  |      * @param float $min | 
					
						
							|  |  |  |      * @param float $max | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @return bool | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     protected function doAmountMatch($amount, $min, $max) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         if ($amount >= $min && $amount <= $max) { | 
					
						
							|  |  |  |             return true; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2015-07-09 09:41:54 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * Gets a collection of paid bills and a collection of unpaid bills to be used | 
					
						
							|  |  |  |      * in the pie chart on the front page. | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @param Carbon $start | 
					
						
							|  |  |  |      * @param Carbon $end | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @return Collection | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     public function getBillsForChart(Carbon $start, Carbon $end) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         $paid   = new Collection; | 
					
						
							|  |  |  |         $unpaid = new Collection; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $bills = $this->getActiveBills(); | 
					
						
							|  |  |  |         /** @var Bill $bill */ | 
					
						
							|  |  |  |         foreach ($bills as $bill) { | 
					
						
							|  |  |  |             $ranges = $this->getRanges($bill, $start, $end); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             foreach ($ranges as $range) { | 
					
						
							|  |  |  |                 // paid a bill in this range?
 | 
					
						
							|  |  |  |                 $journals = $this->getJournalsInRange($bill, $range['start'], $range['end']); | 
					
						
							|  |  |  |                 if ($journals->count() == 0) { | 
					
						
							|  |  |  |                     $unpaid->push([$bill, $range['start']]); | 
					
						
							|  |  |  |                 } else { | 
					
						
							|  |  |  |                     $paid = $paid->merge($journals); | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         $set = new Collection; | 
					
						
							|  |  |  |         $set->put('paid', $paid); | 
					
						
							|  |  |  |         $set->put('unpaid', $unpaid); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return $set; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * Takes the paid/unpaid bills collection set up before and expands it using | 
					
						
							|  |  |  |      * credit cards the user might have. | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @param Collection $set | 
					
						
							|  |  |  |      * @param Carbon     $start | 
					
						
							|  |  |  |      * @param Carbon     $end | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @return Collection | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     public function getCreditCardInfoForChart(Collection $set, Carbon $start, Carbon $end) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $accounts    = app('FireflyIII\Repositories\Account\AccountRepositoryInterface'); | 
					
						
							| 
									
										
										
										
											2015-12-27 08:39:41 +01:00
										 |  |  |         $creditCards = $accounts->getCreditCards($end); | 
					
						
							| 
									
										
										
										
											2015-07-09 09:41:54 +02:00
										 |  |  |         $paid        = $set->get('paid'); | 
					
						
							|  |  |  |         $unpaid      = $set->get('unpaid'); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         foreach ($creditCards as $creditCard) { | 
					
						
							|  |  |  |             $date    = new Carbon($creditCard->getMeta('ccMonthlyPaymentDate')); | 
					
						
							| 
									
										
										
										
											2015-12-27 08:39:41 +01:00
										 |  |  |             if ($creditCard->balance < 0) { | 
					
						
							| 
									
										
										
										
											2015-07-09 09:41:54 +02:00
										 |  |  |                 // unpaid! create a fake bill that matches the amount.
 | 
					
						
							|  |  |  |                 $description = $creditCard->name; | 
					
						
							| 
									
										
										
										
											2015-12-27 08:39:41 +01:00
										 |  |  |                 $amount      = $creditCard->balance * -1; | 
					
						
							| 
									
										
										
										
											2015-07-09 09:41:54 +02:00
										 |  |  |                 $fakeBill    = $this->createFakeBill($description, $date, $amount); | 
					
						
							|  |  |  |                 unset($description, $amount); | 
					
						
							|  |  |  |                 $unpaid->push([$fakeBill, $date]); | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2015-12-27 08:39:41 +01:00
										 |  |  |             if ($creditCard->balance == 0) { | 
					
						
							| 
									
										
										
										
											2015-07-09 09:41:54 +02:00
										 |  |  |                 // find transfer(s) TO the credit card which should account for
 | 
					
						
							|  |  |  |                 // anything paid. If not, the CC is not yet used.
 | 
					
						
							|  |  |  |                 $journals = $accounts->getTransfersInRange($creditCard, $start, $end); | 
					
						
							|  |  |  |                 $paid     = $paid->merge($journals); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         $set = new Collection; | 
					
						
							|  |  |  |         $set->put('paid', $paid); | 
					
						
							|  |  |  |         $set->put('unpaid', $unpaid); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return $set; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2015-12-26 09:39:35 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * This method returns all active bills which have been paid for in the given range, | 
					
						
							|  |  |  |      * with the field "paid" indicating how much the bill was for. | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @param Carbon $start | 
					
						
							|  |  |  |      * @param Carbon $end | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @return Collection | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     public function billsPaidInRange(Carbon $start, Carbon $end) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         $set = Auth::user()->bills() | 
					
						
							|  |  |  |                    ->leftJoin('transaction_journals', 'transaction_journals.bill_id', '=', 'bills.id') | 
					
						
							|  |  |  |                    ->leftJoin( | 
					
						
							|  |  |  |                        'transactions', function (JoinClause $join) { | 
					
						
							|  |  |  |                        $join->on('transactions.transaction_journal_id', '=', 'transaction_journals.id')->where('transactions.amount', '>', 0); | 
					
						
							|  |  |  |                    } | 
					
						
							|  |  |  |                    ) | 
					
						
							|  |  |  |                    ->where('transaction_journals.date', '>=', $start->format('Y-m-d')) | 
					
						
							|  |  |  |                    ->where('transaction_journals.date', '<=', $end->format('Y-m-d')) | 
					
						
							|  |  |  |                    ->where('bills.active', 1) | 
					
						
							|  |  |  |                    ->groupBy('bills.id')->get( | 
					
						
							|  |  |  |                 ['bills.*', DB::Raw('SUM(`transactions`.`amount`) as `paid`')] | 
					
						
							|  |  |  |             ); | 
					
						
							| 
									
										
										
										
											2015-12-26 09:40:24 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-26 09:39:35 +01:00
										 |  |  |         return $set; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2015-02-25 15:19:14 +01:00
										 |  |  | } |