Secure Singleton Pattern

The Secure Singleton pattern ensures a single, secure instance of a class is used throughout the application, making it ideal for shared resources like encryption services or configuration settings. Implementing access controls and thread safety prevents unauthorized access and ensures secure, concurrent use of resources.

Example: Secure Singleton for Encryption Service in C#


public sealed class EncryptionService {
    private static readonly Lazy<EncryptionService> instance = new(() => new EncryptionService());
    private static readonly object lockObject = new();

    public static EncryptionService Instance {
        get {
            lock (lockObject) {
                return instance.Value;
            }
        }
    }

    private EncryptionService() {}

    public string EncryptData(string data) {
        // Implement encryption logic here
        return Convert.ToBase64String(System.Text.Encoding.UTF8.GetBytes(data));
    }
}

// Usage
var encryptedData = EncryptionService.Instance.EncryptData("Secure Data");

This secure Singleton ensures that the `EncryptionService` is thread-safe, controlling access to a single instance while providing encryption functionality.

Data Sanitization Pattern

Data Sanitization ensures that user input is validated and cleansed to prevent injection attacks, such as SQL Injection or XSS (Cross-Site Scripting). This pattern is essential for any application accepting user input, as it prevents malicious data from compromising system integrity.

Example: Input Sanitization in JavaScript


function sanitizeInput(input) {
    const div = document.createElement("div");
    div.innerText = input;
    return div.innerHTML;
}

// Usage
const sanitizedData = sanitizeInput("<script>alert('xss')</script>");
console.log(sanitizedData); // Output: "<script>alert('xss')</script>"

In this example, `sanitizeInput` encodes potentially harmful characters, preventing XSS attacks by ensuring that user input is safely processed before displaying it in the application.

Token-Based Authentication Pattern

Token-Based Authentication secures APIs by issuing tokens that verify user identity, commonly used in applications requiring secure access control. Tokens, like JSON Web Tokens (JWT), allow stateless authentication, ensuring secure, session-based access without storing user credentials on the server.

Example: Token-Based Authentication Using JWT in Node.js


const jwt = require("jsonwebtoken");

function generateToken(user) {
    const payload = { id: user.id, role: user.role };
    return jwt.sign(payload, "secureSecretKey", { expiresIn: "1h" });
}

function verifyToken(token) {
    try {
        return jwt.verify(token, "secureSecretKey");
    } catch (error) {
        throw new Error("Invalid token");
    }
}

// Usage
const token = generateToken({ id: 1, role: "admin" });
console.log("Token:", token);
const decoded = verifyToken(token);
console.log("Decoded Token:", decoded);

This example uses JWT for authentication, generating a token upon user login and verifying it for subsequent requests, protecting access to secure endpoints.

Access Control List (ACL) Pattern

The Access Control List (ACL) pattern restricts access to resources based on user roles and permissions, ensuring that only authorized users can access specific data or functionalities. Implementing ACLs helps enforce security policies consistently across an application.

Example: Implementing ACL in Python


class Resource:
    def __init__(self, name):
        self.name = name

class ACL:
    def __init__(self):
        self.permissions = {}

    def add_permission(self, user, resource):
        if user not in self.permissions:
            self.permissions[user] = []
        self.permissions[user].append(resource)

    def has_access(self, user, resource):
        return resource in self.permissions.get(user, [])

# Usage
acl = ACL()
resource = Resource("SensitiveData")
acl.add_permission("admin", resource)

print(acl.has_access("admin", resource))  # Output: True
print(acl.has_access("guest", resource))  # Output: False

This ACL implementation restricts access to resources based on user permissions, helping prevent unauthorized access and enforcing secure access policies.

Best Practices for Security Design Patterns

  • Use Secure Secrets Management: Store sensitive information, like JWT secrets, securely and avoid hardcoding them in source code.
  • Regularly Update Dependencies: Update libraries and dependencies to prevent known vulnerabilities from compromising security.
  • Enforce Least Privilege: Apply the ACL pattern to restrict access based on user roles, limiting resource access to only what is necessary.

Conclusion

Security design patterns like Secure Singleton, Data Sanitization, Token-Based Authentication, and ACL provide practical ways to protect applications from common vulnerabilities. By implementing these patterns, developers can build secure, resilient applications that safeguard user data and prevent unauthorized access. Integrating security measures into the design phase ensures that applications remain robust against emerging threats, promoting a safe and trusted user experience.