| 
									
										
										
										
											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; | 
					
						
							|  |  |  | 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-04-05 18:20:06 +02:00
										 |  |  | use Illuminate\Support\Collection; | 
					
						
							| 
									
										
										
										
											2015-02-25 15:19:14 +01:00
										 |  |  | use Log; | 
					
						
							| 
									
										
										
										
											2015-03-03 17:40:17 +01:00
										 |  |  | use Navigation; | 
					
						
							| 
									
										
										
										
											2015-02-25 15:19:14 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | /** | 
					
						
							|  |  |  |  * Class BillRepository | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * @package FireflyIII\Repositories\Bill | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | class BillRepository implements BillRepositoryInterface | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2015-04-05 18:20:06 +02:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * @param Bill $bill | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @return mixed | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     public function destroy(Bill $bill) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         return $bill->delete(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * @return Collection | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     public function getBills() | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         return Auth::user()->bills()->orderBy('name', 'ASC')->get(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * @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') | 
					
						
							|  |  |  |                     ->get(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * @param Bill $bill | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @return Collection | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     public function getPossiblyRelatedJournals(Bill $bill) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         $set = \DB::table('transactions')->where('amount', '>', 0)->where('amount', '>=', $bill->amount_min)->where('amount', '<=', $bill->amount_max)->get( | 
					
						
							|  |  |  |             ['transaction_journal_id'] | 
					
						
							|  |  |  |         ); | 
					
						
							|  |  |  |         $ids = []; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         /** @var Transaction $entry */ | 
					
						
							|  |  |  |         foreach ($set as $entry) { | 
					
						
							|  |  |  |             $ids[] = intval($entry->transaction_journal_id); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         $journals = new Collection; | 
					
						
							|  |  |  |         if (count($ids) > 0) { | 
					
						
							|  |  |  |             $journals = Auth::user()->transactionjournals()->whereIn('id', $ids)->get(); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         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 | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @return Carbon | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     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. | 
					
						
							|  |  |  |          */ | 
					
						
							|  |  |  |         $today = Navigation::addPeriod(new Carbon, $bill->repeat_freq, 0); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $skip  = $bill->skip + 1; | 
					
						
							|  |  |  |         $start = Navigation::startOfPeriod(new Carbon, $bill->repeat_freq); | 
					
						
							|  |  |  |         /* | 
					
						
							|  |  |  |          * 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) { | 
					
						
							|  |  |  |                     $finalDate = clone $start; | 
					
						
							|  |  |  |                     break; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             // add period for next round!
 | 
					
						
							|  |  |  |             $start = Navigation::addPeriod($start, $bill->repeat_freq, 0); | 
					
						
							|  |  |  |             $counter++; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return $finalDate; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * @param Bill               $bill | 
					
						
							|  |  |  |      * @param TransactionJournal $journal | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @return bool | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     public function scan(Bill $bill, TransactionJournal $journal) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         /* | 
					
						
							|  |  |  |          * Match words. | 
					
						
							|  |  |  |          */ | 
					
						
							|  |  |  |         $wordMatch   = false; | 
					
						
							|  |  |  |         $matches     = explode(',', $bill->match); | 
					
						
							|  |  |  |         $description = strtolower($journal->description); | 
					
						
							|  |  |  |         Log::debug('Now scanning ' . $description); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         /* | 
					
						
							|  |  |  |          * Attach expense account to description for more narrow matching. | 
					
						
							|  |  |  |          */ | 
					
						
							|  |  |  |         if (count($journal->transactions) < 2) { | 
					
						
							|  |  |  |             $transactions = $journal->transactions()->get(); | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             $transactions = $journal->transactions; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         /** @var Transaction $transaction */ | 
					
						
							|  |  |  |         foreach ($transactions as $transaction) { | 
					
						
							|  |  |  |             /** @var Account $account */ | 
					
						
							|  |  |  |             $account = $transaction->account()->first(); | 
					
						
							|  |  |  |             /** @var AccountType $type */ | 
					
						
							|  |  |  |             $type = $account->accountType()->first(); | 
					
						
							|  |  |  |             if ($type->type == 'Expense account' || $type->type == 'Beneficiary account') { | 
					
						
							|  |  |  |                 $description .= ' ' . strtolower($account->name); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         Log::debug('Final description: ' . $description); | 
					
						
							|  |  |  |         Log::debug('Matches searched: ' . join(':', $matches)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $count = 0; | 
					
						
							|  |  |  |         foreach ($matches as $word) { | 
					
						
							|  |  |  |             if (!(strpos($description, strtolower($word)) === false)) { | 
					
						
							|  |  |  |                 $count++; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if ($count >= count($matches)) { | 
					
						
							|  |  |  |             $wordMatch = true; | 
					
						
							|  |  |  |             Log::debug('word match is true'); | 
					
						
							|  |  |  |         } else { | 
					
						
							| 
									
										
										
										
											2015-03-03 17:40:17 +01:00
										 |  |  |             Log::debug('Count: ' . $count . ', count(matches): ' . count($matches)); | 
					
						
							| 
									
										
										
										
											2015-02-25 15:19:14 +01:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         /* | 
					
						
							|  |  |  |          * Match amount. | 
					
						
							|  |  |  |          */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $amountMatch = false; | 
					
						
							|  |  |  |         if (count($transactions) > 1) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             $amount = max(floatval($transactions[0]->amount), floatval($transactions[1]->amount)); | 
					
						
							|  |  |  |             $min    = floatval($bill->amount_min); | 
					
						
							|  |  |  |             $max    = floatval($bill->amount_max); | 
					
						
							|  |  |  |             if ($amount >= $min && $amount <= $max) { | 
					
						
							|  |  |  |                 $amountMatch = true; | 
					
						
							|  |  |  |                 Log::debug('Amount match is true!'); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         /* | 
					
						
							|  |  |  |          * If both, update! | 
					
						
							|  |  |  |          */ | 
					
						
							|  |  |  |         if ($wordMatch && $amountMatch) { | 
					
						
							|  |  |  |             Log::debug('TOTAL match is true!'); | 
					
						
							|  |  |  |             $journal->bill()->associate($bill); | 
					
						
							|  |  |  |             $journal->save(); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * @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 | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @return Bill|static | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     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; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } |