🌱
Sprout
Github
  • About
  • Roadmap to Sprout v1.0
  • Migration from Sprig
  • Introduction
    • Getting Started
    • Templating Conventions
  • Features
    • Loader System (Registry)
    • Loader System (Registry Group)
    • Function Aliases
    • Function Notices
    • Safe Functions
  • Registries
    • List of all registries
    • Backward
    • Checksum
    • Conversion
    • Crypto
    • Encoding
    • Env
    • Filesystem
    • Maps
    • Numeric
    • Network
    • Random
    • Reflect
    • Regexp
    • SemVer
    • Slices
    • Std
    • Strings
    • Time
    • Uniqueid
  • Groups
    • List of all registry groups
    • All
    • Hermetic
  • Advanced
    • How to create a handler
    • How to create a registry
    • How to create a registry group
  • Links
    • GitHub repository
Powered by GitBook
On this page
  • Introduction
  • Key Differences
  • 1. Registry System
  • 2. Handler and Function Management
  • 3. Error Handling
  • 4. Function Aliases
  • 5. Following the Go Template Convention
  • How to Transition
  • Step 1: Identify and Organize Functions
  • Step 2: Refactor Templates
  • Step 3: Register Functions and Handlers
  • Step 4: Test and Validate
  • How to Transition for your end-users
  • Migrating Common Functions
  • Example: Simple Function Migration
  • Example: Using Aliases
  • Panicking Functions
  • Function-Specific Changes
  • Deprecated Features
  • Conclusion
Edit on GitHub

Migration from Sprig

Coming from Sprig and looking to use Sprout? You're in the right place for a complete guide on making the transition. This guideline will help you navigate the differences.

PreviousRoadmap to Sprout v1.0NextGetting Started

Last updated 6 months ago

This page evolves with each version based on modifications and community feedback. Having trouble following this guide?

to get help, and contribute to improving this guide for future users.

Introduction

Sprout is a modern templating engine inspired by but with enhanced features and a more modular approach. Migrating from Sprig to Sprout involves understanding these differences and adjusting your templates and code accordingly.

Key Differences

1. Registry System

  • Sprig: Functions are globally available.

  • Sprout: Functions are grouped into registries for modularity. Each registry can be added to the handler as needed.

Migration Tip

List the registries needed for your project and register them with the handler. If you're unsure, you can safely register all built-in registries.

2. Handler and Function Management

  • Sprig: Functions are accessed directly.

  • Sprout: Functions are managed by a handler, allowing for better control over function availability, error handling, and logging.

Migration Top

Nothing to do, using the new handler is enough.

3. Error Handling

  • Sprig: Limited and inconsistent error handling, with some functions causing panics (see Panicking Functions), and not fully adhering to Go template standards.

  • Sprout: Offers configurable error handling strategies, including returning default values, or return error to stop template generation (default), providing a more consistent and flexible approach.

Migration Tip

You can learn mote about the safe strategy here: Safe Functions

4. Function Aliases

  • Sprig: Functions are accessed by a single name, deprecated functions are duplicated in code.

  • Sprout: Supports function aliases, allowing multiple names for the same function.

Migration Tip

Use WithAlias or WithAliases to set up function aliases as needed when creating your handler.

5. Following the Go Template Convention

  • Sprig: No error for naming convention are followed.

  • Sprout: Design to adhere closely to Go's native template conventions, ensuring compatibility and a consistent experience for developers familiar with Go templates. This adherence helps reduce surprises and ensures that templates behave as expected according to Go's standard practices.

Migration Tip Takes a look over renamed functions to change it in your template or use aliases.

How to Transition

Step 1: Identify and Organize Functions

Review your existing templates and identify the functions you rely on from Sprig. Determine whether these functions have direct equivalents in Sprout or if they require using the sprigin package for backward compatibility.

Step 2: Refactor Templates

Update your templates to replace Sprig function calls with their Sprout equivalents. If a direct replacement is unavailable, consider using aliases or the sprigin package to maintain functionality.

Step 3: Register Functions and Handlers

Set up your Sprout environment by creating a handler and registering the necessary function registries. This step ensures that your templates have access to the required functions.

Step 4: Test and Validate

Thoroughly test your migrated templates to ensure that all functions behave as expected. Pay particular attention to error handling and any deprecated functions that may require adjustments.

How to Transition for your end-users

You use sprig or sprout for end-users and want to migrate ? Transitioning from Sprig to Sprout can be smooth with proper guidance. Here's a detailed plan to ensure confidence during the migration:


  1. Communicate the Purpose of the Migration Explain the reasons for switching to Sprout, emphasizing improvements such as better performance, modular function registries, enhanced error handling, and new features like function notices and safe functions. Providing context will minimize resistance and help users understand the long-term benefits.

  2. Attach Your Logger to Sprout Starting from Sprout v1.0.0, deprecated functions are flagged through the logger. This ensures end-users are informed of upcoming deprecations without breaking the existing code.

  3. Update Frequently We follow a no-breaking-change policy for five minor versions. For example, a function deprecated in v1.2 will be removed in v1.7. End-of-life notices for functions will be automatically logged, so you don’t need to intervene.

  4. Provide Feedback If you or your end-users encounter any feedback during the migration—whether it’s about missing functions, bugs, or positive experiences—reach out to the maintainers or contribute to the project on GitHub. Your input helps improve the library for everyone.


Migrating Common Functions

Many functions in Sprig have direct equivalents in Sprout, but they might be organized differently or require registration in a handler.

Example: Simple Function Migration

Sprig:

{{ upper "hello" }}

Sprout:

{{ toUpper "hello" }}

Example: Using Aliases

Sprig:

{{ upper "hello" }}

Sprout:

handler := sprout.New(
    sprout.WithAlias("toUpper", "upper"),
)
funcs := handler.Build()

You can continue to use the same function name inside your template

{{ upper "hello" }}

Panicking Functions

In Sprig, errors within certain functions cause a panic. In contrast, Sprout opts for returning nil or an empty value, improving safety and predictability.

Old Behavior (Sprig): Triggers a panic on error

if err != nil {
  panic("deepCopy error: " + err.Error())
}

New Behavior (Sprout): Returns nil or an empty value on error

if err != nil {
  return nil, err
}

Methods that previously caused a panic in Sprig :

  • DeepCopy

  • MustDeepCopy

  • ToRawJson

  • Append

  • Prepend

  • Concat

  • Chunk

  • Uniq

  • Compact

  • Slice

  • Without

  • Rest

  • Initial

  • Reverse

  • First

  • Last

  • Has

  • Dig

  • RandAlphaNumeric

  • RandAlpha

  • RandAscii

  • RandNumeric

  • RandBytes

Function-Specific Changes

MustDeepCopy

  • Sprig: Accepts nil input, causing an internal panic.

  • Sprout: Returns nil if input is nil, avoiding panic.

Rand Functions

  • Sprig: Causes an internal panic if the length parameter is zero.

  • Sprout: Returns an empty string if the length is zero, ensuring stability.

DateAgo

  • Sprig: Does not support int32 and *time.Time; returns "0s".

  • Sprout: Supports int32 and *time.Time and returns the correct duration.

DateRound

  • Sprig: Returns a corrected duration in positive form, even for negative inputs.

  • Sprout: Accurately returns the duration, preserving the sign of the input.

Base32Decode / Base64Decode

  • Sprig: Decoding functions return the error string when the input is not a valid base64 encoded string.

  • Sprout: Decoding functions return an empty string if the input is not a valid base64 encoded string, simplifying error handling.

Dig

Consider the example dictionary defined as follows:

dict := map[string]any{
  "a": map[string]any{
    "b": 2,
  },
}
  • Sprig: Previously, the dig function would return the last map in the access chain.

{{ $dict | dig "a" "b" }} // Output: map[b:2]
  • Sprout: Now, the dig function returns the final object in the chain, regardless of its type (map, array, string, etc.).

{{ $dict | dig "a" "b" }} // Output: 2

ToCamelCase / ToPascalCase

  • Sprig: The toCamelCase return value are in PascalCase. No toPascalCase function is available.

  • Sprout: The toCamelCase function returns camelCase strings, while the toPascalCase function returns PascalCase strings.

Merge / MergeOverwrite

  • Sprig: The merge and mergeOverwrite functions does dereference when second value are the default golang value (example: 0 for int).

  • Sprout: The merge and mergeOverwrite functions does not dereference and keep the second value as is (example: 0 for int).

Deprecated Features

Sprout has deprecated certain features for better security and performance. For example, direct cryptographic operations in templates are discouraged.

Migration Tip

Review your template functions and avoid using deprecated features. Move critical operations outside of templates to maintain security.

All deprecated features are flagged with // ! DEPRECATED in codebase. A complete list will be available here when the v1 of Sprout are released.

Conclusion

You need more information now, contact maintainers or open .

As a library developer, you can extend Sprout by creating your . This allows you to add custom functions to meet specific use cases. Additionally, you can use to inform your end-users about important updates or issues during template execution, such as deprecations or debugging hints. These notices can enhance the user experience by providing real-time feedback, ensuring a smoother transition to the new system, and allowing users to handle function changes more efficiently.

Our maintainers and collaborators can assist you if you have question, don't hesitate to !

Perform cryptographic operations (listed in crypto package) outside of templates. the will be drop in few versions.

Migrating from Sprig to Sprout offers significant benefits, including improved error handling, modular function management, and enhanced compatibility with modern Go practices. While the sprigin package provides a bridge for backward compatibility, fully embracing Sprout’s native capabilities will lead to a more stable and maintainable codebase. For further details on Sprout’s features and API, consult the .

🌱
💜
Open an issue
Sprig
a discussion on the repository
own function registry
notices
open a discussion on GitHub
cryptoregistry
official Sprout documentation