mirror of
				https://codeberg.org/forgejo/forgejo.git
				synced 2025-10-25 11:33:11 +00:00 
			
		
		
		
	* Dropped unused codekit config * Integrated dynamic and static bindata for public * Ignore public bindata * Add a general generate make task * Integrated flexible public assets into web command * Updated vendoring, added all missiong govendor deps * Made the linter happy with the bindata and dynamic code * Moved public bindata definition to modules directory * Ignoring the new bindata path now * Updated to the new public modules import path * Updated public bindata command and drop the new prefix
		
			
				
	
	
		
			131 lines
		
	
	
	
		
			2.1 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			131 lines
		
	
	
	
		
			2.1 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| package zk
 | |
| 
 | |
| import (
 | |
| 	"errors"
 | |
| 	"fmt"
 | |
| 	"strconv"
 | |
| 	"strings"
 | |
| )
 | |
| 
 | |
| var (
 | |
| 	ErrDeadlock  = errors.New("zk: trying to acquire a lock twice")
 | |
| 	ErrNotLocked = errors.New("zk: not locked")
 | |
| )
 | |
| 
 | |
| type Lock struct {
 | |
| 	c        *Conn
 | |
| 	path     string
 | |
| 	acl      []ACL
 | |
| 	lockPath string
 | |
| 	seq      int
 | |
| }
 | |
| 
 | |
| func NewLock(c *Conn, path string, acl []ACL) *Lock {
 | |
| 	return &Lock{
 | |
| 		c:    c,
 | |
| 		path: path,
 | |
| 		acl:  acl,
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func parseSeq(path string) (int, error) {
 | |
| 	parts := strings.Split(path, "-")
 | |
| 	return strconv.Atoi(parts[len(parts)-1])
 | |
| }
 | |
| 
 | |
| func (l *Lock) Lock() error {
 | |
| 	if l.lockPath != "" {
 | |
| 		return ErrDeadlock
 | |
| 	}
 | |
| 
 | |
| 	prefix := fmt.Sprintf("%s/lock-", l.path)
 | |
| 
 | |
| 	path := ""
 | |
| 	var err error
 | |
| 	for i := 0; i < 3; i++ {
 | |
| 		path, err = l.c.CreateProtectedEphemeralSequential(prefix, []byte{}, l.acl)
 | |
| 		if err == ErrNoNode {
 | |
| 			// Create parent node.
 | |
| 			parts := strings.Split(l.path, "/")
 | |
| 			pth := ""
 | |
| 			for _, p := range parts[1:] {
 | |
| 				pth += "/" + p
 | |
| 				_, err := l.c.Create(pth, []byte{}, 0, l.acl)
 | |
| 				if err != nil && err != ErrNodeExists {
 | |
| 					return err
 | |
| 				}
 | |
| 			}
 | |
| 		} else if err == nil {
 | |
| 			break
 | |
| 		} else {
 | |
| 			return err
 | |
| 		}
 | |
| 	}
 | |
| 	if err != nil {
 | |
| 		return err
 | |
| 	}
 | |
| 
 | |
| 	seq, err := parseSeq(path)
 | |
| 	if err != nil {
 | |
| 		return err
 | |
| 	}
 | |
| 
 | |
| 	for {
 | |
| 		children, _, err := l.c.Children(l.path)
 | |
| 		if err != nil {
 | |
| 			return err
 | |
| 		}
 | |
| 
 | |
| 		lowestSeq := seq
 | |
| 		prevSeq := 0
 | |
| 		prevSeqPath := ""
 | |
| 		for _, p := range children {
 | |
| 			s, err := parseSeq(p)
 | |
| 			if err != nil {
 | |
| 				return err
 | |
| 			}
 | |
| 			if s < lowestSeq {
 | |
| 				lowestSeq = s
 | |
| 			}
 | |
| 			if s < seq && s > prevSeq {
 | |
| 				prevSeq = s
 | |
| 				prevSeqPath = p
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		if seq == lowestSeq {
 | |
| 			// Acquired the lock
 | |
| 			break
 | |
| 		}
 | |
| 
 | |
| 		// Wait on the node next in line for the lock
 | |
| 		_, _, ch, err := l.c.GetW(l.path + "/" + prevSeqPath)
 | |
| 		if err != nil && err != ErrNoNode {
 | |
| 			return err
 | |
| 		} else if err != nil && err == ErrNoNode {
 | |
| 			// try again
 | |
| 			continue
 | |
| 		}
 | |
| 
 | |
| 		ev := <-ch
 | |
| 		if ev.Err != nil {
 | |
| 			return ev.Err
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	l.seq = seq
 | |
| 	l.lockPath = path
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| func (l *Lock) Unlock() error {
 | |
| 	if l.lockPath == "" {
 | |
| 		return ErrNotLocked
 | |
| 	}
 | |
| 	if err := l.c.Delete(l.lockPath, -1); err != nil {
 | |
| 		return err
 | |
| 	}
 | |
| 	l.lockPath = ""
 | |
| 	l.seq = 0
 | |
| 	return nil
 | |
| }
 |