cloud native

Cloud native refers to designing, developing, and operating applications that optimize cloud computing platforms' features and capabilities.

I plan to extensively explore the ecosystem and understand the potential trajectory of modern computing and its future:

The Projects sub-node in here hosts an index into tooling that I've enqueued for exploration.

All the major practical components of the cloud native landscape (as of 0x22AB) are being developed in Golang : initializing exploration of the same directed towards my intentions of contributing competently in the CNCF landscape.

1. The Path to Cloud Native

  • The primary incentive so far pushing the evolution of computing services has been the pressure to scale.
  • 1950s : about the mainframe computer with all the logic and data residing together as one monolith.
  • 1980s : networks of personal computers encouraged some changes. Some application logic could be off-loaded to these PCs (partioning into the Client-Server Architecture) : the first major move towards decoupled services.
  • 1990s : the dotcom rush led to explosion of (SaaS) Software as a Service. The associated logistics of operating (develop, deploy, maintain) such a service began to get complex. This further encouraged the decoupling of the business layer in the Client Server architecture into multiple Microservices.
  • 2000s : AWS popularized (IaaS) Infrastructure as a Service : yielding the initiation of the term Cloud Computing.

2. Important Terms (checkout System Design)

2.3. Resilience

4. Cloud Native Patterns

4.1. context.Context

4.2. Stability Patterns

4.2.1. Circuit Breaker

  • https://en.wikipedia.org/wiki/Circuit_breaker_design_pattern
  • automatically degrades service functions in response to a likely fault, preventing larger or cascading failures by eliminating recurring errors and providing reasonable error responses
  • 2 major components :
    • Circuit
    • Breaker
  • Consider the Scenario:->
    • User Request <-> Business Logic Layer <-> Data Layer
    • the Business Logic implements a Breaker if the Circuit (Data Layer in this case has failed)
    • will retry after sensible pauses and backoff strats
    • won't continue requesting if failures are assured.
  1. Code
    • Consider the Circuit type as, that signifies the signature of the function that's interacting with the database. An error as one of its returns is necessary.

      type Circuit func(context.Context) (string, error)
      
    • Consider the Breaker function that accepts a function of type Circuit and some failure threshold information (number of consecutive failures to be tolerated in this case, before breaking (opening) the circuit), returning a wrapped circuit with the same function signature.

      func Breaker(circuit Circuit, failureThreshold uint) Circuit {
      
              consecutiveFailures := 0
              lastAttempt := time.Now()
      
              // for closures access over breaker's administrative data
              // failure count and retry times
              var m sync.RWMutex
      
              return func(context.Context) (string,error) {
                      //see exponential backoff
                      m.RLock()
      
                      d := consecutiveFailures - int(failureThreshold)
      
                      if d >= 0 { //circuit breaking in case of failure
                              shouldRetryAt := lastAttempt.Add(time.Second*2 << d)
                              if !time.Now().After(shouldRetryAt){
                                      m.Runlock()
                                      return "", errors.New("Service unreachable")
                              }
                      }
      
                      m.Runlock()
      
                      // continuing request if no track of failures
      
                      response, err := circuit(ctx)
      
                      m.Lock() // acquiring write locks for closure's commons
                      defer m.Unlock()
      
                      lastAttempt = time.Now()
      
                      if err != nil {
                              consecutiveFailures++
                              return response, err
                      }
      
                      consecutiveFailures = 0
                      return response, nil
              }
      }
      
      

4.2.2. Debounce

  • etymological origins in electronic circuits : https://www.geeksforgeeks.org/switch-debounce-in-digital-circuits/
  • limits the frequency of a function invocation so that only the first or last in a cluster of calls is actually performed.
  • is native to javascript but can port to others as needed, will be proceeding in golang
  • 2 components:
    • Circuit : the computation to be regulated
    • Debounce : A closure over Circuit that manages the calls
  • similar logic to Circuit Breakers in that the closure maintains the rate limiting logic and state
  1. Code
    • on each call of the Debounce returned closure, regardless of the outcome, a time interval is set.
      • calls before expiry of that duration are ignored, any after the duration are passed along to the inner Circuit function.
        • this is a "function-first" : i.e cache results and ignore the latter calls
      • Alternatively, a "funciton-last" implementation will accumulate a series of requests before calling Circuit
        • This could be useful when the inner circuit needs some kick-starting corpus of inputs (think autocompletion)
        • can be employed if the response can be delayed a little and increased latency is not an issue.
    • The Core Circuit can be a function as follows

      type Circuit func(context.Context) (string, error)
      
    • The Debounce prepped closure can then be structured as follows (function-first)

      func DebounceFirst(circuit Circuit, d time.Duration) Circuit {
      
              var threshold time.Time
              var result string // result cache
              var err error
              var m sync.Mutex
      
              return func(ctx context.Context) (string, error) {
                      m.Lock()
      
                      defer func() {
                              threshold = time.Now().Add(d)
                              m.Unlock()
                      }()
      
                      if time.Now().Before(threshold){
                              //return cached result before threshold
                              return result, err
                      }
      
                      // if expired, compute and cache result
                      // in the enclosed variable result
                      result, err = circuit(ctx)
      
                      return result, err
              }
      }
      
      
      • a function-last implementation needs a little more book-keeping

        func DebounceLast(circuit Circuit, d time.Duration) Circuit {
                var threshold time.Time = time.Now()
                var ticker *time.Ticker
                var result string
                var err error
                var once sync.Once
                var m sync.Mutex
                return func(ctx context.Context) (string, error) {
                        m.Lock()
                        defer m.Unlock()
                        threshold = time.Now().Add(d)
                        once.Do(func() {
                                ticker = time.NewTicker(time.Millisecond * 100)
                                go func() {
                                        defer func() {
                                                m.Lock()
                                                ticker.Stop()
                                                once = sync.Once{}
                                                m.Unlock()
                                        }()
                                        for {
                                                select {
                                                case <-ticker.C:
                                                        m.Lock()
                                                        if time.Now().After(threshold) {
                                                                result, err = circuit(ctx)
                                                                m.Unlock()
                                                                return
                                                        }
                                                        m.Unlock()
                                                case <-ctx.Done():
                                                        m.Lock()
                                                        result, err = "", ctx.Err()
                                                        m.Unlock()
                                                        return
                                                }
                                        }
                                }()
                        })
                        return result, err
                }
        }
        

4.2.3. Retry

  1. Code
    • like the Circuit, the effector will have a function signature as follows
    type Effector func(context.Context) (string, error)
    
    • the Retry can take in paramters like the number of retries and delay durations, returning a closure of the same signature as the Effector

      func Retry(effector Effector, retries int, delay time.Duration) Effector{
              return func(ctx context.Context) (string, error) {
                      for r := 0; ; r++ {
                              response, err := effector(ctx)
                              if err == nil || r >= retries {
                                      return response, err
                              }
                              log.Printf("Attempt %d failed; retrying in %v", r + 1, delay)
      
                              select {
                              case <- time.After(delay):
                              case <- ctx.Done():
                                      return "", ctx.Err()
                              }
                      }
              }
      }
      
      • emulating an erroneous function to try out retry

        #+beginsrc go

    var count int

    func EmulateTransientError(ctx context.Context) (string, error) { count++

    if count <= 3 { return "intentional fail", errors.New("error") } else { return "success", nil } }

    func main() { r := Retry(EmulateTransientError, 5, 2*time.Second)

    res, err := r(context.Background())

    fmt.Println(res,err) } #+

4.2.4. Throttle

  • limits the frequency of a funciton call to some maximum number of invocations per minute
  • See Rate Limiting Algorithms
  • diff w/ Debounce
    • debounce collates clusters of calls (across flexible durations) into representative boundary calls
    • throttle limits the amount of calls in a relatively fixed duration
  • 2 components
    • Effector : the function being regulated
    • Throttle : the enwrapping closure over Effector : implementing the rate limiting layer
  1. Code
    type Effector func(context.Context) (string, error)
    
    func Throttle(e Effector, max uint, refill uint, d time.Duration) Effector {
            var tokens = max
            var once sync.Once
    
            return func(ctx context.Context) (string, error) {
                    if ctx.Err() != nil {
                            return "", ctx.Err()
                    }
            }
    
            once.Do(func() {
                    ticker := time.NewTicker(d)
    
                    go func() {
                            defer ticker.Stop()
    
                            for {
                                    select {
                                    case <-ctx.Done():
                                            return
    
                                    case <- ticker.C:
                                            t := tokens + refill
                                            if t > max {
                                                    t = max
                                            }
                                            tokens = t
                                    }
                            }
                    }()
            })
    
            if tokens <= 0 {
                    return "", fmt.Errorf("too many calls")
            }
    
            tokens--
    
            return e(ctx)
    }
    

4.2.5. Timeout

  • allows a process to stop waiting for an answer once it's clear that an answer may not be coming
  • 3 components:
    • client : calls a slow function
    • slowfunction : a long running function
    • timeout : wrapper over slow function
  • straightforward if a function utilizes context.Context in golang,
  1. Code
    ctx := context.Background()
    ctxt,cancel := context.WithTimeout(ctx, 10*time.Second)
    defer cancel()
    result, err := SomeFunction(ctxt)
    
    • This isn't usually the case though
      • build a closure that respects the context in such a case, followed by a select over your injected timeout context and the result of a goroutined slow function
    • will need to convert the slow function into a context respecting wrapper as follows
    type SlowFunction func(string) (string, error)
    
    type WithContext func(context.Context, string) (string, error)
    
    func Timeout(f SlowFunction) WithContext {
            return func(ctx context.Context, arg string) (string error) {
                    chres := make(chan string) //channel for results
                    cherr := make(chan string) //channel for errors
    
                    go func() {
                            res, err := f(arg) //dispatch slow function
                            chres <- res
                            cherr <- err
                    }()
    
                    select {
                    case res := <-chres://if done before timeout
                            return res, <-cherr
                    case <-ctx.Done()://in case of timeout
                            return "",ctx.Err()
                    }
            }
    }
    
    • finally, using Timeout will be like
    func main() {
            ctx := context.Background()
            ctxt, cancel := context.WithTimeout(ctx, 1*time.Second)
            defer cancel()
    
            timeout := Timeout(Slow)
            res, err := timeout(ctxt, "some input")
    
            fmt.Println(res, err)
    }
    
    • an alternative to using context.Context (context.Context is the preferred method btw), is using the time.After function : https://pkg.go.dev/time#After

4.3. Concurrency Patterns

Tags::arch:programming:cloud: