PEP: Add str.ensureprefix and str.ensuresuffix Methods
Abstract
This PEP proposes adding two new methods to the str class: ensureprefix and ensuresuffix. These methods will ensure that a string starts or ends with a specified prefix or suffix, respectively. If the string already has the prefix or suffix, it will remain unchanged; otherwise, the prefix or suffix will be added.
Motivation
String manipulation is a common task in Python programming. While Python provides methods like str.startswith and str.endswith to check for prefixes and suffixes, there is no built-in way to ensure that a string has a specific prefix or suffix. Developers often write custom utility functions for this purpose, which can lead to repetitive and less readable code.
For example, ensuring a string starts with a specific tag or ends with a specific delimiter requires writing code like this:
tag = "data"
if not tag.startswith("_"):
    tag = "_" + tag
text = "Hello"
if not text.endswith("!"):
    text += "!"
With the proposed methods, this can be simplified to:
tag = "data".ensureprefix("_")
text = "Hello".ensuresuffix("!")
This improves code readability and reduces the likelihood of errors.
Complementing removeprefix and removesuffix
The existing removeprefix and removesuffix methods (introduced in PEP 616) allow developers to remove a prefix or suffix from a string if it exists. For example:
tag = "_data".removeprefix("_")  # "data"
text = "Hello!".removesuffix("!")  # "Hello"
The proposed ensureprefix and ensuresuffix methods provide the inverse functionality, ensuring that a string has a specific prefix or suffix. Together, these methods form a complete and symmetric set of tools for prefix and suffix manipulation:
- removeprefix: Removes a prefix if it exists.
- ensureprefix: Adds a prefix if it does not exist.
- removesuffix: Removes a suffix if it exists.
- ensuresuffix: Adds a suffix if it does not exist.
This symmetry makes the API more intuitive and consistent.
Rationale
The addition of str.ensureprefix and str.ensuresuffix aligns with Python’s philosophy of providing clear, concise, and expressive syntax for common tasks. These methods complement the existing str.removeprefix and str.removesuffix methods, completing the set of tools for prefix and suffix manipulation.
Use Cases
- String Formatting: Ensuring strings meet specific formatting requirements (e.g., adding a dollar sign to prices).
- Data Normalization: Standardizing strings in datasets (e.g., ensuring all entries start or end with a specific pattern).
- Tagging and Labeling: Ensuring tags or labels have a consistent format (e.g., adding a prefix to identifiers).
- Delimited Data: Ensuring strings end with a specific delimiter for concatenation or parsing.
Existing Workarounds
Currently, developers use utility functions or manual checks to achieve this functionality. For example:
def ensureprefix(s: str, prefix: str) -> str:
    return s if s.startswith(prefix) else prefix + s
def ensuresuffix(s: str, suffix: str) -> str:
    return s if s.endswith(suffix) else s + suffix
While these workarounds are functional, they are less convenient and less discoverable than built-in methods.
Specification
New Methods
Two new methods will be added to the str class:
- str.ensureprefix(prefix: str) -> str
- If the string starts with prefix, return the string unchanged.
- Otherwise, return prefix + self.
- str.ensuresuffix(suffix: str) -> str
- If the string ends with suffix, return the string unchanged.
- Otherwise, return self + suffix.
Examples
# Ensure prefix
assert "data".ensureprefix("_") == "_data"
assert "_data".ensureprefix("_") == "_data"
# Ensure suffix
assert "Hello".ensuresuffix("!") == "Hello!"
assert "Hello!".ensuresuffix("!") == "Hello!"
Edge Cases
- If prefixorsuffixis an empty string, the original string is returned unchanged.
- If prefixorsuffixis longer than the string, the result will be the concatenation of the prefix/suffix and the string.
assert "hello".ensureprefix("") == "hello"
assert "hello".ensuresuffix("") == "hello"
assert "a".ensureprefix("longprefix") == "longprefixa"
assert "a".ensuresuffix("longsuffix") == "alongsuffix"
Backward Compatibility
This proposal introduces new methods to the str class and does not modify any existing behavior. As such, it is fully backward-compatible.
Rejected Alternatives
1. Using Utility Functions
- While utility functions work, they are less discoverable and require additional imports or definitions.
2. Monkey Patching
- Monkey patching the strclass is discouraged due to potential conflicts and maintenance issues.
3. Subclassing str
- Subclassing strto add these methods is possible but less convenient than having them directly available on all strings.