Skip to content

Commit d5a81c8

Browse files
committed
Civi::url() - Allow building new URL from empty. Update comments.
1 parent 4987909 commit d5a81c8

File tree

3 files changed

+80
-17
lines changed

3 files changed

+80
-17
lines changed

Civi.php

+24-11
Original file line numberDiff line numberDiff line change
@@ -219,21 +219,35 @@ public static function settings($domainID = NULL) {
219219
}
220220

221221
/**
222-
* Construct a URL based on a logical service address. URL building follows a few rules:
222+
* Construct a URL based on a logical service address. For example:
223223
*
224-
* 1. You should initialize with a baseline URL (e.g. 'frontend://civicrm/profile/view?id=123&gid=456').
224+
* Civi::url('frontend://civicrm/user?reset=1');
225+
*
226+
* Civi::url()
227+
* ->setScheme('frontend')
228+
* ->setPath(['civicrm', 'user'])
229+
* ->setQuery(['reset' => 1])
230+
*
231+
* URL building follows a few rules:
232+
*
233+
* 1. You may initialize with a baseline URL.
225234
* 2. The scheme indicates the general type of URL ('frontend://', 'backend://', 'asset://', 'assetBuilder://').
226-
* 3. The URL object provides getters, setters, and adders (e.g. `getScheme()`, `setPath(...)`, `addQuery(...)`)
235+
* 3. The result object provides getters, setters, and adders (e.g. `getScheme()`, `setPath(...)`, `addQuery(...)`)
227236
* 4. Strings are raw. Arrays are auto-encoded. (`addQuery('name=John+Doughnut')` or `addQuery(['name' => 'John Doughnut'])`)
228237
* 5. You may use variable expressions (`id=[contact]&gid=[profile]`).
229238
* 6. The URL can be cast to string (aka `__toString()`).
230239
*
231-
* Here are several examples:
240+
* If you are converting from `CRM_Utils_System::url()` to `Civi::url()`, then be sure to:
241+
*
242+
* - Pay attention to the scheme (eg 'current://' vs 'frontend://')
243+
* - Pay attention to HTML escaping, as the behavior changed:
244+
* - Civi::url() returns plain URLs (eg "id=100&gid=200") by default
245+
* - CRM_Utils_System::url() returns HTML-escaped URLs (eg "id=100&gid=200") by default
232246
*
233-
* Ex: Link to constituent's dashboard (specifically on frontend UI)
234-
* $url = Civi::url('frontend://civicrm/user?reset=1');
247+
* Here are several examples:
235248
*
236249
* Ex: Link to constituent's dashboard (on frontend UI or backend UI -- based on the active scheme of current page-view)
250+
* $url = Civi::url('current://civicrm/user?reset=1');
237251
* $url = Civi::url('//civicrm/user?reset=1');
238252
*
239253
* Ex: Link to constituent's dashboard (with method calls - good for dynamic options)
@@ -263,13 +277,13 @@ public static function settings($domainID = NULL) {
263277
* $url = Civi::url('frontend://civicrm/ajax/api4/[entity]/[action]')
264278
* ->addVars(['entity' => 'Foo', 'action' => 'bar']);
265279
*
266-
* @param string $logicalUri
280+
* @param string|null $logicalUri
267281
* Logical URI. The scheme of the URI may be one of:
268282
* - 'frontend://' (Front-end page-route for constituents)
269283
* - 'backend://' (Back-end page-route for staff)
270-
* - 'service://` (Web-service page-route for automated integrations; aka webhooks and IPNs)
284+
* - 'service://' (Web-service page-route for automated integrations; aka webhooks and IPNs)
271285
* - 'current://' (Whichever UI is currently active)
272-
* - 'default://'(Whichever UI is recorded in the metadata)
286+
* - 'default://' (Whichever UI is recorded in the metadata)
273287
* - 'asset://' (Static asset-file; see \Civi::paths())
274288
* - 'assetBuilder://' (Dynamically-generated asset-file; see \Civi\Core\AssetBuilder)
275289
* - 'ext://' (Static asset-file provided by an extension)
@@ -282,11 +296,10 @@ public static function settings($domainID = NULL) {
282296
* - 't': text (aka `setHtmlEscape(FALSE)`)
283297
* - 's': ssl (aka `setSsl(TRUE)`)
284298
* - 'c': cache code for resources (aka Civi::resources()->addCacheCode())
285-
* FIXME: Should we have a flag for appending 'resCacheCode'?
286299
* @return \Civi\Core\Url
287300
* URL object which may be modified or rendered as text.
288301
*/
289-
public static function url(string $logicalUri, ?string $flags = NULL): \Civi\Core\Url {
302+
public static function url(?string $logicalUri = NULL, ?string $flags = NULL): \Civi\Core\Url {
290303
return new \Civi\Core\Url($logicalUri, $flags);
291304
}
292305

Civi/Core/Url.php

+24-6
Original file line numberDiff line numberDiff line change
@@ -116,17 +116,39 @@ final class Url implements \JsonSerializable {
116116
private $varsCallback;
117117

118118
/**
119-
* @param string $logicalUri
119+
* @param string|null $logicalUri
120120
* @param string|null $flags
121121
* @see \Civi::url()
122122
*/
123-
public function __construct(string $logicalUri, ?string $flags = NULL) {
123+
public function __construct(?string $logicalUri = NULL, ?string $flags = NULL) {
124+
if ($logicalUri !== NULL) {
125+
$this->useUri($logicalUri);
126+
}
127+
if ($flags !== NULL) {
128+
$this->useFlags($flags);
129+
}
130+
}
131+
132+
/**
133+
* Parse a logical URI.
134+
*
135+
* @param string $logicalUri
136+
* @return void
137+
*/
138+
protected function useUri(string $logicalUri): void {
124139
if ($logicalUri[0] === '/') {
140+
// Scheme-relative path implies a preferences to inherit current scheme.
125141
$logicalUri = 'current:' . $logicalUri;
126142
}
127143
elseif ($logicalUri[0] === '[') {
128144
$logicalUri = 'asset://' . $logicalUri;
129145
}
146+
// else: Should we fill in scheme when there is NO indicator (eg $logicalUri===`civicrm/event/info')?
147+
// It could be a little annoying to write `frontend://` everywhere. It's not hard to add this.
148+
// But it's ambiguous whether `current://` or `default://` is the better interpretation.
149+
// I'd sooner vote for something explicit but short -- eg aliases (f<=>frontend; d<=>default)
150+
// - `Civi::url('f://civicrm/event/info')`
151+
// - `Civi::url('civicrm/event/info', 'f')`.
130152

131153
$parsed = parse_url($logicalUri);
132154
$this->scheme = $parsed['scheme'] ?? NULL;
@@ -138,10 +160,6 @@ public function __construct(string $logicalUri, ?string $flags = NULL) {
138160
$fragmentParts = isset($parsed['fragment']) ? explode('?', $parsed['fragment'], 2) : [];
139161
$this->fragment = $fragmentParts[0] ?? NULL;
140162
$this->fragmentQuery = $fragmentParts[1] ?? NULL;
141-
142-
if ($flags !== NULL) {
143-
$this->useFlags($flags);
144-
}
145163
}
146164

147165
/**

tests/phpunit/Civi/Core/UrlTest.php

+32
Original file line numberDiff line numberDiff line change
@@ -153,4 +153,36 @@ public function testVars(): void {
153153
}
154154
}
155155

156+
public function testFunkyStartPoints(): void {
157+
$baseline = (string) \Civi::url('frontend://civicrm/event/info?id=1');
158+
$this->assertStringContainsString('event/info', $baseline);
159+
160+
$alternatives = [
161+
// Start with nothing!
162+
\Civi::url()
163+
->setScheme('frontend')
164+
->setPath(['civicrm', 'event', 'info'])
165+
->addQuery(['id' => 1]),
166+
167+
// Start with nothing! And build it backwards!
168+
\Civi::url()
169+
->addQuery(['id' => 1])
170+
->addPath('civicrm')->addPath('event')->addPath('info')
171+
->setScheme('frontend'),
172+
173+
// Start with just the scheme
174+
\Civi::url('frontend:')
175+
->addPath('civicrm/event/info')
176+
->addQuery('id=1'),
177+
178+
// Start with just the path
179+
\Civi::url('civicrm/event/info')
180+
->setScheme('frontend')
181+
->addQuery(['id' => 1]),
182+
];
183+
foreach ($alternatives as $key => $alternative) {
184+
$this->assertEquals($baseline, (string) $alternative, "Alternative #$key should match baseline");
185+
}
186+
}
187+
156188
}

0 commit comments

Comments
 (0)