whereNot(); $name = substr($name, 3); } if (!str_contains(strtolower($name), "where")) { $method = 'where' . ucfirst($name); } else { $method = lcfirst($name); } if (method_exists($this, $method) === true) { return call_user_func_array([$that, $method], $arguments); } throw new MethodNotFoundException("Method " . self::class . '::' . $method . '() is not supported'); } /** * Validate a given criteria * @param $criteria * * @return string * @throws InvalidWhereQueryCriteriaException */ protected function validate_criteria($criteria): string { $command = strtoupper($criteria); if (str_starts_with($command, "CUSTOM ")) { return substr($criteria, 7); } if (in_array($command, $this->available_criteria) === false) { throw new InvalidWhereQueryCriteriaException("Invalid imap search criteria: $command"); } return $criteria; } /** * Register search parameters * @param mixed $criteria * @param mixed $value * * @return $this * @throws InvalidWhereQueryCriteriaException * * Examples: * $query->from("someone@email.tld")->seen(); * $query->whereFrom("someone@email.tld")->whereSeen(); * $query->where([["FROM" => "someone@email.tld"], ["SEEN"]]); * $query->where(["FROM" => "someone@email.tld"])->where(["SEEN"]); * $query->where(["FROM" => "someone@email.tld", "SEEN"]); * $query->where("FROM", "someone@email.tld")->where("SEEN"); */ public function where(mixed $criteria, mixed $value = null): WhereQuery { if (is_array($criteria)) { foreach ($criteria as $key => $value) { if (is_numeric($key)) { $this->where($value); }else{ $this->where($key, $value); } } } else { $this->push_search_criteria($criteria, $value); } return $this; } /** * Push a given search criteria and value pair to the search query * @param $criteria string * @param $value mixed * * @throws InvalidWhereQueryCriteriaException */ protected function push_search_criteria(string $criteria, mixed $value){ $criteria = $this->validate_criteria($criteria); $value = $this->parse_value($value); if ($value === '') { $this->query->push([$criteria]); } else { $this->query->push([$criteria, $value]); } } /** * @param Closure|null $closure * * @return $this */ public function orWhere(Closure $closure = null): WhereQuery { $this->query->push(['OR']); if ($closure !== null) $closure($this); return $this; } /** * @param Closure|null $closure * * @return $this */ public function andWhere(Closure $closure = null): WhereQuery { $this->query->push(['AND']); if ($closure !== null) $closure($this); return $this; } /** * @return WhereQuery * @throws InvalidWhereQueryCriteriaException */ public function whereAll(): WhereQuery { return $this->where('ALL'); } /** * @return WhereQuery * @throws InvalidWhereQueryCriteriaException */ public function whereAnswered(): WhereQuery { return $this->where('ANSWERED'); } /** * @param string $value * * @return WhereQuery * @throws InvalidWhereQueryCriteriaException */ public function whereBcc(string $value): WhereQuery { return $this->where('BCC', $value); } /** * @param mixed $value * @return WhereQuery * @throws InvalidWhereQueryCriteriaException * @throws MessageSearchValidationException */ public function whereBefore(mixed $value): WhereQuery { $date = $this->parse_date($value); return $this->where('BEFORE', $date); } /** * @param string $value * * @return WhereQuery * @throws InvalidWhereQueryCriteriaException */ public function whereBody(string $value): WhereQuery { return $this->where('BODY', $value); } /** * @param string $value * * @return WhereQuery * @throws InvalidWhereQueryCriteriaException */ public function whereCc(string $value): WhereQuery { return $this->where('CC', $value); } /** * @return WhereQuery * @throws InvalidWhereQueryCriteriaException */ public function whereDeleted(): WhereQuery { return $this->where('DELETED'); } /** * @param string $value * * @return WhereQuery * @throws InvalidWhereQueryCriteriaException */ public function whereFlagged(string $value): WhereQuery { return $this->where('FLAGGED', $value); } /** * @param string $value * * @return WhereQuery * @throws InvalidWhereQueryCriteriaException */ public function whereFrom(string $value): WhereQuery { return $this->where('FROM', $value); } /** * @param string $value * * @return WhereQuery * @throws InvalidWhereQueryCriteriaException */ public function whereKeyword(string $value): WhereQuery { return $this->where('KEYWORD', $value); } /** * @return WhereQuery * @throws InvalidWhereQueryCriteriaException */ public function whereNew(): WhereQuery { return $this->where('NEW'); } /** * @return WhereQuery * @throws InvalidWhereQueryCriteriaException */ public function whereNot(): WhereQuery { return $this->where('NOT'); } /** * @return WhereQuery * @throws InvalidWhereQueryCriteriaException */ public function whereOld(): WhereQuery { return $this->where('OLD'); } /** * @param mixed $value * * @return WhereQuery * @throws MessageSearchValidationException * @throws InvalidWhereQueryCriteriaException */ public function whereOn(mixed $value): WhereQuery { $date = $this->parse_date($value); return $this->where('ON', $date); } /** * @return WhereQuery * @throws InvalidWhereQueryCriteriaException */ public function whereRecent(): WhereQuery { return $this->where('RECENT'); } /** * @return WhereQuery * @throws InvalidWhereQueryCriteriaException */ public function whereSeen(): WhereQuery { return $this->where('SEEN'); } /** * @param mixed $value * * @return WhereQuery * @throws MessageSearchValidationException * @throws InvalidWhereQueryCriteriaException */ public function whereSince(mixed $value): WhereQuery { $date = $this->parse_date($value); return $this->where('SINCE', $date); } /** * @param string $value * * @return WhereQuery * @throws InvalidWhereQueryCriteriaException */ public function whereSubject(string $value): WhereQuery { return $this->where('SUBJECT', $value); } /** * @param string $value * * @return WhereQuery * @throws InvalidWhereQueryCriteriaException */ public function whereText(string $value): WhereQuery { return $this->where('TEXT', $value); } /** * @param string $value * * @return WhereQuery * @throws InvalidWhereQueryCriteriaException */ public function whereTo(string $value): WhereQuery { return $this->where('TO', $value); } /** * @param string $value * * @return WhereQuery * @throws InvalidWhereQueryCriteriaException */ public function whereUnkeyword(string $value): WhereQuery { return $this->where('UNKEYWORD', $value); } /** * @return WhereQuery * @throws InvalidWhereQueryCriteriaException */ public function whereUnanswered(): WhereQuery { return $this->where('UNANSWERED'); } /** * @return WhereQuery * @throws InvalidWhereQueryCriteriaException */ public function whereUndeleted(): WhereQuery { return $this->where('UNDELETED'); } /** * @return WhereQuery * @throws InvalidWhereQueryCriteriaException */ public function whereUnflagged(): WhereQuery { return $this->where('UNFLAGGED'); } /** * @return WhereQuery * @throws InvalidWhereQueryCriteriaException */ public function whereUnseen(): WhereQuery { return $this->where('UNSEEN'); } /** * @return WhereQuery * @throws InvalidWhereQueryCriteriaException */ public function whereNoXSpam(): WhereQuery { return $this->where("CUSTOM X-Spam-Flag NO"); } /** * @return WhereQuery * @throws InvalidWhereQueryCriteriaException */ public function whereIsXSpam(): WhereQuery { return $this->where("CUSTOM X-Spam-Flag YES"); } /** * Search for a specific header value * @param $header * @param $value * * @return WhereQuery * @throws InvalidWhereQueryCriteriaException */ public function whereHeader($header, $value): WhereQuery { return $this->where("CUSTOM HEADER $header $value"); } /** * Search for a specific message id * @param $messageId * * @return WhereQuery * @throws InvalidWhereQueryCriteriaException */ public function whereMessageId($messageId): WhereQuery { return $this->whereHeader("Message-ID", $messageId); } /** * Search for a specific message id * @param $messageId * * @return WhereQuery * @throws InvalidWhereQueryCriteriaException */ public function whereInReplyTo($messageId): WhereQuery { return $this->whereHeader("In-Reply-To", $messageId); } /** * @param $country_code * * @return WhereQuery * @throws InvalidWhereQueryCriteriaException */ public function whereLanguage($country_code): WhereQuery { return $this->where("Content-Language $country_code"); } /** * Get message be it UID. * * @param int|string $uid * * @return WhereQuery * @throws InvalidWhereQueryCriteriaException */ public function whereUid(int|string $uid): WhereQuery { return $this->where('UID', $uid); } /** * Get messages by their UIDs. * * @param array $uids * * @return WhereQuery * @throws InvalidWhereQueryCriteriaException */ public function whereUidIn(array $uids): WhereQuery { $uids = implode(',', $uids); return $this->where('UID', $uids); } /** * Apply the callback if the given "value" is truthy. * copied from @url https://github.com/laravel/framework/blob/8.x/src/Illuminate/Support/Traits/Conditionable.php * * @param mixed $value * @param callable $callback * @param callable|null $default * @return $this|null */ public function when(mixed $value, callable $callback, ?callable $default = null): mixed { if ($value) { return $callback($this, $value) ?: $this; } elseif ($default) { return $default($this, $value) ?: $this; } return $this; } /** * Apply the callback if the given "value" is falsy. * copied from @url https://github.com/laravel/framework/blob/8.x/src/Illuminate/Support/Traits/Conditionable.php * * @param mixed $value * @param callable $callback * @param callable|null $default * @return $this|mixed */ public function unless(mixed $value, callable $callback, ?callable $default = null): mixed { if (!$value) { return $callback($this, $value) ?: $this; } elseif ($default) { return $default($this, $value) ?: $this; } return $this; } /** * Get all available search criteria * * @return array|string[] */ public function getAvailableCriteria(): array { return $this->available_criteria; } }