amondar-libs/php-state-flow

A lightweight PHP package for working with U.S. state codes and names.

Maintainers

Package info

github.com/amondar-libs/php-state-flow

pkg:composer/amondar-libs/php-state-flow

Statistics

Installs: 3

Dependents: 0

Suggesters: 0

Stars: 0

Open Issues: 0

1.3.3 2026-04-16 02:26 UTC

This package is auto-updated.

Last update: 2026-04-16 02:30:49 UTC


README

A lightweight PHP package for working with U.S. state codes and names.

It provides:

  • Fast in-memory lookups for state code/name conversions
  • Prebuilt regex fragments for validation/parsing
  • Optional support for custom vocabularies via backed enums

Requirements

  • PHP ^8.3

Installation

composer require amondar-libs/php-state-flow

Quick Start

<?php

use Amondar\PhpStateFlow\StateFlow;

StateFlow::getAbbreviation('New York');   // "NY"
StateFlow::getName('ny');          // "New York"

// Regex fragments
$cityRegex = StateFlow::getCityRegex();
preg_match('/^' . $cityRegex . '$/', 'Los Angeles, CA'); // 1

Core Concepts

There can be only one

This package developed with only one thing in mind:

  • any array in response based on: values for frontend and keys for search in any cases.

That's why keys are lower cased and values are normalized.

Default Vocabulary

By default, all methods use Amondar\PhpStateFlow\State, which contains 48 U.S. states.

Custom Vocabulary

Most methods accept a third argument ($vocabulary) with a class-string of a backed enum. This allows reuse of the same helpers with your own enum dataset.

API Reference

getAbbreviations(string $vocabulary = State::class): array

Returns all enum case names (state abbreviations by default), e.g. ['AL', 'AZ', ..., 'WY'].

$codes = StateFlow::getAbbreviations();

getCount(string $vocabulary = State::class): int

Returns total number of items in the vocabulary.

$count = StateFlow::getCount(); // 48

getNames(string $vocabulary = State::class): array

Returns all enum backed values (state names by default), e.g. ['Alabama', 'New York', ...].

$names = StateFlow::getNames();

getAbbreviationsRegex(string $vocabulary = State::class): string

Returns a pipe-separated regex fragment of codes, e.g. AL|AZ|AR|....

$codeRegex = StateFlow::getAbbreviationsRegex();

getNamesRegex(string $vocabulary = State::class): string

Returns a pipe-separated regex fragment of names, e.g. Alabama|Arizona|....

$labelRegex = StateFlow::getNamesRegex();

getAbbreviationByNameMap(string $vocabulary = State::class): array

Returns a map of normalized snake-case labels to enum values.

Examples:

  • new_york => NY
  • west_virginia => WV
$map = StateFlow::getAbbreviationByNameMap();

getNameByAbbreviationMap(string $vocabulary = State::class): array

Returns a map of lowercase enum values to human-readable labels.

Examples:

  • ny => New York
  • al => Alabama
$map = StateFlow::getNameByAbbreviationMap();

getAbbreviation(string $name, bool $lower = false, string $vocabulary = State::class): ?string

Converts a full state name to its short code.

Behavior:

  • Input is normalized (lower + squish + snake)
  • Returns null if state name is unknown
  • Optional lowercase output
StateFlow::getAbbreviation('New York');         // "NY"
StateFlow::getAbbreviation('  New   York  ');   // "NY"
StateFlow::getAbbreviation('New York', true);   // "ny"
StateFlow::getAbbreviation('NY');               // null

getName(string $short, bool $lower = false, string $vocabulary = State::class): ?string

Converts a short code to full state name.

Behavior:

  • Code lookup is case-insensitive
  • Returns null if code is unknown
  • Optional lowercase output
StateFlow::getName('NY');        // "New York"
StateFlow::getName('ny');        // "New York"
StateFlow::getName('NY', true);  // "new york"
StateFlow::getName('XX');        // null

getCityRegex(int $maxCityName = 30, ?string $defaultCity = null, string $vocabulary = State::class): string

Builds a regex fragment for values like:

  • City, ST (code)
  • City, State Name (label)

Supports:

  • Optional whitespace after comma
  • Configurable max city length ($maxCityName)
  • Optional exact fallback city ($defaultCity)
$regex = StateFlow::getCityRegex();
preg_match('/^' . $regex . '$/', 'New York, NY');       // 1
preg_match('/^' . $regex . '$/', 'New York, New York'); // 1
preg_match('/^' . $regex . '$/', 'Invalid City, XX');   // 0

$regexWithDefault = StateFlow::getCityRegex(30, 'Anywhere');
preg_match('/^' . $regexWithDefault . '$/', 'Anywhere'); // 1

getOriginRegex(int $maxCityName = 30, string $vocabulary = State::class): string

Builds a regex fragment that matches:

  • City, ST
  • City, State Name
  • Standalone state code
  • Standalone state label
$regex = StateFlow::getOriginRegex();

preg_match('/^' . $regex . '$/', 'NY');            // 1
preg_match('/^' . $regex . '$/', 'New York, NY');  // 1
preg_match('/^' . $regex . '$/', 'XX');            // 0

search(string $query, string $vocabulary = State::class): array

Searches states by abbreviation fragment and returns matched pairs as:

  • key: lowercase abbreviation
  • value: full state name

Behavior:

  • Query is normalized (trim + lowercase)
  • Empty/whitespace-only query returns an empty array
  • Matches if query is contained in name or equals full abbreviation
// Search all with "n" in name.
StateFlow::search('n');
// [
//   'nc' => 'North Carolina',
//   'ny' => 'New York',
//   'tn' => 'Tennessee',
//   ...
// ]

StateFlow::search('  NY  ');
// ['ny' => 'New York']

StateFlow::search('   ');
// []

getRandom(string $vocabulary = State::class): array{abbreviation: string, name: string}

Returns one random item from vocabulary as an associative pair:

  • abbreviation: uppercase abbreviation key
  • name: full state name
$random = StateFlow::getRandom();
// [
//   'abbreviation' => 'NY',
//   'name' => 'New York',
// ]

normalizeNameKey(string $name): string

Normalizes a given name key by converting it to lowercase, replacing spaces and hyphens with underscores, and trimming excess spaces.

Behavior:

  • Input is trimmed and squished (multiple spaces reduced to one)
  • Hyphens and spaces are replaced with underscores
  • Output is lowercased
StateFlow::normalizeNameKey('New York');         // "new_york"
StateFlow::normalizeNameKey('  West   Virginia-State  '); // "west_virginia_state"

Using a Custom Enum Vocabulary

<?php

enum Region: string
{
    case NORTH = 'N';
    case SOUTH = 'S';
}

use Amondar\PhpStateFlow\StateFlow;

$codes = StateFlow::getAbbreviations(Region::class); // ['NORTH', 'SOUTH']
$names = StateFlow::getNames(Region::class);         // ['N', 'S']
$name = StateFlow::getName('north', false, Region::class); // 'N'

Caching Notes

This package caches computed arrays/regex strings in static in-memory properties for the current PHP process.

  • Improves repeated lookup performance.
  • Cache lifetime is process-bound (request/worker lifecycle).
  • Fully compatible with long-lived process applications like Laravel Octane, Roadrunner, etc.

Testing

Run tests with Pest:

./vendor/bin/pest

Or run parallel tests via composer script:

composer run ppest

License

MIT. See LICENSE.md.