Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Wrap the code indexer #9476

Merged
merged 5 commits into from
Dec 24, 2019
Merged
Show file tree
Hide file tree
Changes from 1 commit
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
49 changes: 38 additions & 11 deletions modules/indexer/code/indexer.go
Original file line number Diff line number Diff line change
Expand Up @@ -5,17 +5,15 @@
package code

import (
"context"
"os"
"time"

"code.gitea.io/gitea/modules/graceful"
"code.gitea.io/gitea/modules/log"
"code.gitea.io/gitea/modules/setting"
)

var (
indexer Indexer
)

// SearchResult result of performing a search in a repo
type SearchResult struct {
RepoID int64
Expand All @@ -36,28 +34,45 @@ type Indexer interface {
// Init initialize the repo indexer
func Init() {
if !setting.Indexer.RepoIndexerEnabled {
indexer.Close()
return
}

ctx, cancel := context.WithCancel(context.Background())

graceful.GetManager().RunAtTerminate(ctx, func() {
log.Debug("Closing repository indexer")
indexer.Close()
log.Info("PID: %d Repository Indexer closed", os.Getpid())
})

waitChannel := make(chan time.Duration)
go func() {
start := time.Now()
log.Info("Initializing Repository Indexer")
var created bool
var err error
indexer, created, err = NewBleveIndexer(setting.Indexer.RepoPath)
log.Info("PID: %d Initializing Repository Indexer at: %s", os.Getpid(), setting.Indexer.RepoPath)
bleveIndexer, created, err := NewBleveIndexer(setting.Indexer.RepoPath)
if err != nil {
if bleveIndexer != nil {
bleveIndexer.Close()
}
cancel()
indexer.Close()
log.Fatal("indexer.Init: %v", err)
close(waitChannel)
log.Fatal("PID: %d Unable to initialize the Repository Indexer at path: %s Error: %v", os.Getpid(), setting.Indexer.RepoPath, err)
}
indexer.set(bleveIndexer)

go processRepoIndexerOperationQueue(indexer)

if created {
go populateRepoIndexer()
}
select {
case waitChannel <- time.Since(start):
case <-graceful.GetManager().IsShutdown():
}

waitChannel <- time.Since(start)
close(waitChannel)
}()

if setting.Indexer.StartupTimeout > 0 {
Expand All @@ -67,9 +82,21 @@ func Init() {
timeout += setting.GracefulHammerTime
}
select {
case duration := <-waitChannel:
case <-graceful.GetManager().IsShutdown():
log.Warn("Shutdown before Repository Indexer completed initialization")
cancel()
indexer.Close()
case duration, ok := <-waitChannel:
if !ok {
log.Warn("Repository Indexer Initialization failed")
cancel()
indexer.Close()
return
}
log.Info("Repository Indexer Initialization took %v", duration)
case <-time.After(timeout):
cancel()
indexer.Close()
log.Fatal("Repository Indexer Initialization Timed-Out after: %v", timeout)
}
}()
Expand Down
2 changes: 0 additions & 2 deletions modules/indexer/code/queue.go
Original file line number Diff line number Diff line change
Expand Up @@ -22,8 +22,6 @@ type repoIndexerOperation struct {
var repoIndexerOperationQueue chan repoIndexerOperation

func processRepoIndexerOperationQueue(indexer Indexer) {
defer indexer.Close()
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Once the indexer is running we should close it at Terminate not Shutdown. Things running in the ShutdownContext and up to HammerTime might still need to send data to the repo indexer.

If the indexer fails to be set up then we should close ASAP as we want gitea to stop ASAP.


repoIndexerOperationQueue = make(chan repoIndexerOperation, setting.Indexer.UpdateQueueLength)
for {
select {
Expand Down
91 changes: 91 additions & 0 deletions modules/indexer/code/wrapped.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,91 @@
// Copyright 2019 The Gitea Authors. All rights reserved.
// Use of this source code is governed by a MIT-style
// license that can be found in the LICENSE file.

package code

import (
"fmt"
"sync"
)

var (
indexer = newWrappedIndexer()
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The indexer has to be wrapped to allow searches etc to hold whilst we are waiting for the indexer to boot up further we must not run functions on the indexer until it is completely ready.

If the indexer is closed before its internal element is present then we will return a simple error rather than panic.

)

// ErrWrappedIndexerClosed is the error returned if the indexer was closed before it was ready
var ErrWrappedIndexerClosed = fmt.Errorf("Indexer closed before ready")

type wrappedIndexer struct {
internal Indexer
lock sync.RWMutex
cond *sync.Cond
closed bool
}

func newWrappedIndexer() *wrappedIndexer {
w := &wrappedIndexer{}
w.cond = sync.NewCond(w.lock.RLocker())
return w
}

func (w *wrappedIndexer) set(indexer Indexer) {
w.lock.Lock()
defer w.lock.Unlock()
if w.closed {
// Too late!
indexer.Close()
}
w.internal = indexer
w.cond.Broadcast()
}

func (w *wrappedIndexer) get() (Indexer, error) {
w.lock.RLock()
defer w.lock.RUnlock()
if w.internal == nil {
if w.closed {
return nil, ErrWrappedIndexerClosed
}
w.cond.Wait()
}
return w.internal, nil
}

func (w *wrappedIndexer) Index(repoID int64) error {
indexer, err := w.get()
if err != nil {
return err
}
return indexer.Index(repoID)
}

func (w *wrappedIndexer) Delete(repoID int64) error {
indexer, err := w.get()
if err != nil {
return err
}
return indexer.Delete(repoID)
}

func (w *wrappedIndexer) Search(repoIDs []int64, keyword string, page, pageSize int) (int64, []*SearchResult, error) {
indexer, err := w.get()
if err != nil {
return 0, nil, err
}
return indexer.Search(repoIDs, keyword, page, pageSize)

}

func (w *wrappedIndexer) Close() {
w.lock.Lock()
defer w.lock.Unlock()
if w.closed {
return
}
w.closed = true
w.cond.Broadcast()
if w.internal != nil {
w.internal.Close()
}
}