package app

type ID int

// Context of a function call from proxy to app.
type Context interface {
	CallerTag() string
}

// A Space contains all apps that may be available in a V2Ray runtime.
// Caller must check the availability of an app by calling HasXXX before getting its instance.
type Space interface {
	HasApp(ID) bool
	GetApp(ID) interface{}
}

type ForContextCreator func(Context, interface{}) interface{}

var (
	metadataCache = make(map[ID]ForContextCreator)
)

func RegisterApp(id ID, creator ForContextCreator) {
	// TODO: check id
	metadataCache[id] = creator
}

type contextImpl struct {
	callerTag string
}

func (this *contextImpl) CallerTag() string {
	return this.callerTag
}

type spaceImpl struct {
	cache map[ID]interface{}
	tag   string
}

func newSpaceImpl(tag string, cache map[ID]interface{}) *spaceImpl {
	space := &spaceImpl{
		tag:   tag,
		cache: make(map[ID]interface{}),
	}
	context := &contextImpl{
		callerTag: tag,
	}
	for id, object := range cache {
		creator, found := metadataCache[id]
		if found {
			space.cache[id] = creator(context, object)
		}
	}
	return space
}

func (this *spaceImpl) HasApp(id ID) bool {
	_, found := this.cache[id]
	return found
}

func (this *spaceImpl) GetApp(id ID) interface{} {
	obj, found := this.cache[id]
	if !found {
		return nil
	}
	return obj
}

// A SpaceController is supposed to be used by a shell to create Spaces. It should not be used
// directly by proxies.
type SpaceController struct {
	objectCache map[ID]interface{}
}

func NewController() *SpaceController {
	return &SpaceController{
		objectCache: make(map[ID]interface{}),
	}
}

func (this *SpaceController) Bind(id ID, object interface{}) {
	this.objectCache[id] = object
}

func (this *SpaceController) ForContext(tag string) Space {
	return newSpaceImpl(tag, this.objectCache)
}