mirror of
https://github.com/postmannen/ctrl.git
synced 2025-01-20 22:52:13 +00:00
terminating subproc publishers if inactive for given amount of time
This commit is contained in:
parent
1dadc0df64
commit
b1bcefb949
5 changed files with 96 additions and 33 deletions
|
@ -90,6 +90,8 @@ type Configuration struct {
|
||||||
// EnableDebug will also enable printing all the messages received in the errorKernel
|
// EnableDebug will also enable printing all the messages received in the errorKernel
|
||||||
// to STDERR.
|
// to STDERR.
|
||||||
EnableDebug bool
|
EnableDebug bool
|
||||||
|
// KeepPublishersAliveFor number of seconds
|
||||||
|
KeepPublishersAliveFor int
|
||||||
|
|
||||||
// Make the current node send hello messages to central at given interval in seconds
|
// Make the current node send hello messages to central at given interval in seconds
|
||||||
StartPubREQHello int
|
StartPubREQHello int
|
||||||
|
@ -171,6 +173,7 @@ type ConfigurationFromFile struct {
|
||||||
EnableAclCheck *bool
|
EnableAclCheck *bool
|
||||||
IsCentralAuth *bool
|
IsCentralAuth *bool
|
||||||
EnableDebug *bool
|
EnableDebug *bool
|
||||||
|
KeepPublishersAliveFor *int
|
||||||
|
|
||||||
StartPubREQHello *int
|
StartPubREQHello *int
|
||||||
EnableKeyUpdates *bool
|
EnableKeyUpdates *bool
|
||||||
|
@ -236,6 +239,7 @@ func newConfigurationDefaults() Configuration {
|
||||||
EnableAclCheck: false,
|
EnableAclCheck: false,
|
||||||
IsCentralAuth: false,
|
IsCentralAuth: false,
|
||||||
EnableDebug: false,
|
EnableDebug: false,
|
||||||
|
KeepPublishersAliveFor: 10,
|
||||||
|
|
||||||
StartPubREQHello: 30,
|
StartPubREQHello: 30,
|
||||||
EnableKeyUpdates: true,
|
EnableKeyUpdates: true,
|
||||||
|
@ -440,6 +444,11 @@ func checkConfigValues(cf ConfigurationFromFile) Configuration {
|
||||||
} else {
|
} else {
|
||||||
conf.EnableDebug = *cf.EnableDebug
|
conf.EnableDebug = *cf.EnableDebug
|
||||||
}
|
}
|
||||||
|
if cf.KeepPublishersAliveFor == nil {
|
||||||
|
conf.KeepPublishersAliveFor = cd.KeepPublishersAliveFor
|
||||||
|
} else {
|
||||||
|
conf.KeepPublishersAliveFor = *cf.KeepPublishersAliveFor
|
||||||
|
}
|
||||||
|
|
||||||
// --- Start pub/sub
|
// --- Start pub/sub
|
||||||
|
|
||||||
|
@ -602,6 +611,7 @@ func (c *Configuration) CheckFlags() error {
|
||||||
flag.BoolVar(&c.EnableAclCheck, "enableAclCheck", fc.EnableAclCheck, "true/false *TESTING* enable Acl checking.")
|
flag.BoolVar(&c.EnableAclCheck, "enableAclCheck", fc.EnableAclCheck, "true/false *TESTING* enable Acl checking.")
|
||||||
flag.BoolVar(&c.IsCentralAuth, "isCentralAuth", fc.IsCentralAuth, "true/false, *TESTING* is this the central auth server")
|
flag.BoolVar(&c.IsCentralAuth, "isCentralAuth", fc.IsCentralAuth, "true/false, *TESTING* is this the central auth server")
|
||||||
flag.BoolVar(&c.EnableDebug, "enableDebug", fc.EnableDebug, "true/false, will enable debug logging so all messages sent to the errorKernel will also be printed to STDERR")
|
flag.BoolVar(&c.EnableDebug, "enableDebug", fc.EnableDebug, "true/false, will enable debug logging so all messages sent to the errorKernel will also be printed to STDERR")
|
||||||
|
flag.IntVar(&c.KeepPublishersAliveFor, "keepPublishersAliveFor", fc.KeepPublishersAliveFor, "The amount of time we allow a publisher to stay alive without receiving any messages to publish")
|
||||||
|
|
||||||
// Start of Request publishers/subscribers
|
// Start of Request publishers/subscribers
|
||||||
|
|
||||||
|
|
|
@ -51,6 +51,8 @@ type Message struct {
|
||||||
ACKTimeout int `json:"ACKTimeout" yaml:"ACKTimeout"`
|
ACKTimeout int `json:"ACKTimeout" yaml:"ACKTimeout"`
|
||||||
// RetryWait specified the time in seconds to wait between retries.
|
// RetryWait specified the time in seconds to wait between retries.
|
||||||
RetryWait int `json:"retryWait" yaml:"retryWait"`
|
RetryWait int `json:"retryWait" yaml:"retryWait"`
|
||||||
|
// IsSubPublishedMsg enables timeout of publishing process, and is used together with process.isSubProcess to be able to terminate the sub processes publishers.
|
||||||
|
IsSubPublishedMsg bool `json:"isSubPublishedMsg" yaml:"isSubPublishedMsg"`
|
||||||
// Resend retries
|
// Resend retries
|
||||||
Retries int `json:"retries" yaml:"retries"`
|
Retries int `json:"retries" yaml:"retries"`
|
||||||
// The ACK timeout of the new message created via a request event.
|
// The ACK timeout of the new message created via a request event.
|
||||||
|
|
30
process.go
30
process.go
|
@ -34,6 +34,8 @@ const (
|
||||||
// process holds all the logic to handle a message type and it's
|
// process holds all the logic to handle a message type and it's
|
||||||
// method, subscription/publishin messages for a subject, and more.
|
// method, subscription/publishin messages for a subject, and more.
|
||||||
type process struct {
|
type process struct {
|
||||||
|
// isSubProcess is used to indentify subprocesses spawned by other processes.
|
||||||
|
isSubProcess bool
|
||||||
// server
|
// server
|
||||||
server *server
|
server *server
|
||||||
// messageID
|
// messageID
|
||||||
|
@ -687,12 +689,30 @@ func (p process) publishMessages(natsConn *nats.Conn) {
|
||||||
// fails in the loop we should throw an error and use `continue`
|
// fails in the loop we should throw an error and use `continue`
|
||||||
// to jump back here to the beginning of the loop and continue
|
// to jump back here to the beginning of the loop and continue
|
||||||
// with the next message.
|
// with the next message.
|
||||||
|
|
||||||
|
// Adding a timer that will be used for when to remove the sub process
|
||||||
|
// publisher. The timer is reset each time a message is published with
|
||||||
|
// the process, so the sub process publisher will not be removed until
|
||||||
|
// it have not received any messages for the given amount of time.
|
||||||
|
ticker := time.NewTicker(time.Second * time.Duration(p.configuration.KeepPublishersAliveFor))
|
||||||
|
|
||||||
for {
|
for {
|
||||||
|
|
||||||
// Wait and read the next message on the message channel, or
|
// Wait and read the next message on the message channel, or
|
||||||
// exit this function if Cancel are received via ctx.
|
// exit this function if Cancel are received via ctx.
|
||||||
select {
|
select {
|
||||||
|
case <-ticker.C:
|
||||||
|
// We only want to remove subprocesses
|
||||||
|
if p.isSubProcess {
|
||||||
|
p.ctxCancel()
|
||||||
|
|
||||||
|
p.processes.active.mu.Lock()
|
||||||
|
delete(p.processes.active.procNames, p.processName)
|
||||||
|
p.processes.active.mu.Unlock()
|
||||||
|
}
|
||||||
|
|
||||||
case m := <-p.subject.messageCh:
|
case m := <-p.subject.messageCh:
|
||||||
|
ticker.Reset(time.Second * time.Duration(p.configuration.KeepPublishersAliveFor))
|
||||||
// Sign the methodArgs, and add the signature to the message.
|
// Sign the methodArgs, and add the signature to the message.
|
||||||
m.ArgSignature = p.addMethodArgSignature(m)
|
m.ArgSignature = p.addMethodArgSignature(m)
|
||||||
// fmt.Printf(" * DEBUG: add signature, fromNode: %v, method: %v, len of signature: %v\n", m.FromNode, m.Method, len(m.ArgSignature))
|
// fmt.Printf(" * DEBUG: add signature, fromNode: %v, method: %v, len of signature: %v\n", m.FromNode, m.Method, len(m.ArgSignature))
|
||||||
|
@ -821,6 +841,16 @@ func (p process) publishAMessage(m Message, zEnc *zstd.Encoder, once sync.Once,
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// NB: vvvvvvvvvvvvvvvvvvvvv-THIS DOES NOT WORK FOR CANCELING THE PUBLISHER-vvvvvvvvv
|
||||||
|
//if m.IsSubPublishedMsg {
|
||||||
|
// p.ctxCancel()
|
||||||
|
// go func() {
|
||||||
|
// p.processes.active.mu.Lock()
|
||||||
|
// delete(p.processes.active.procNames, p.processName)
|
||||||
|
// p.processes.active.mu.Unlock()
|
||||||
|
// }()
|
||||||
|
//}
|
||||||
|
|
||||||
// Increment the counter for the next message to be sent.
|
// Increment the counter for the next message to be sent.
|
||||||
p.messageID++
|
p.messageID++
|
||||||
|
|
||||||
|
|
|
@ -183,7 +183,8 @@ func (m methodREQCopySrc) handler(proc process, message Message, node string) ([
|
||||||
sub := newSubjectNoVerifyHandler(m, node)
|
sub := newSubjectNoVerifyHandler(m, node)
|
||||||
|
|
||||||
// Create a new sub process that will do the actual file copying.
|
// Create a new sub process that will do the actual file copying.
|
||||||
copySrcSubProc := newProcess(ctx, proc.server, sub, processKindSubscriber, nil)
|
|
||||||
|
copySrcSubProc := newSubProcess(ctx, proc.server, sub, processKindSubscriber, nil)
|
||||||
|
|
||||||
// Give the sub process a procFunc so we do the actual copying within a procFunc,
|
// Give the sub process a procFunc so we do the actual copying within a procFunc,
|
||||||
// and not directly within the handler.
|
// and not directly within the handler.
|
||||||
|
@ -233,6 +234,14 @@ func (m methodREQCopySrc) handler(proc process, message Message, node string) ([
|
||||||
return ackMsg, nil
|
return ackMsg, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// newSubProcess is a wrapper around newProcess which sets the isSubProcess value to true.
|
||||||
|
func newSubProcess(ctx context.Context, server *server, subject Subject, processKind processKind, procFunc func() error) process {
|
||||||
|
p := newProcess(ctx, server, subject, processKind, procFunc)
|
||||||
|
p.isSubProcess = true
|
||||||
|
|
||||||
|
return p
|
||||||
|
}
|
||||||
|
|
||||||
// ----
|
// ----
|
||||||
|
|
||||||
type methodREQCopyDst struct {
|
type methodREQCopyDst struct {
|
||||||
|
@ -275,7 +284,7 @@ func (m methodREQCopyDst) handler(proc process, message Message, node string) ([
|
||||||
sub := newSubjectNoVerifyHandler(cia.DstMethod, node)
|
sub := newSubjectNoVerifyHandler(cia.DstMethod, node)
|
||||||
|
|
||||||
// Create a new sub process that will do the actual file copying.
|
// Create a new sub process that will do the actual file copying.
|
||||||
copyDstSubProc := newProcess(ctx, proc.server, sub, processKindSubscriber, nil)
|
copyDstSubProc := newSubProcess(ctx, proc.server, sub, processKindSubscriber, nil)
|
||||||
|
|
||||||
// Give the sub process a procFunc so we do the actual copying within a procFunc,
|
// Give the sub process a procFunc so we do the actual copying within a procFunc,
|
||||||
// and not directly within the handler.
|
// and not directly within the handler.
|
||||||
|
@ -423,11 +432,12 @@ func copySrcSubProcFunc(proc process, cia copyInitialData, cancel context.Cancel
|
||||||
// We want to send a message back to src that we are ready to start.
|
// We want to send a message back to src that we are ready to start.
|
||||||
// fmt.Printf("\n\n\n ************** DEBUG: copyDstHandler sub process sending copyReady to:%v\n ", message.FromNode)
|
// fmt.Printf("\n\n\n ************** DEBUG: copyDstHandler sub process sending copyReady to:%v\n ", message.FromNode)
|
||||||
msg := Message{
|
msg := Message{
|
||||||
ToNode: cia.DstNode,
|
ToNode: cia.DstNode,
|
||||||
FromNode: cia.SrcNode,
|
FromNode: cia.SrcNode,
|
||||||
Method: cia.DstMethod,
|
Method: cia.DstMethod,
|
||||||
ReplyMethod: REQNone,
|
ReplyMethod: REQNone,
|
||||||
Data: csaSerialized,
|
Data: csaSerialized,
|
||||||
|
IsSubPublishedMsg: true,
|
||||||
}
|
}
|
||||||
|
|
||||||
// fmt.Printf("\n ***** DEBUG: copyDstSubProcFunc: cia.SrcMethod: %v\n\n ", cia.SrcMethod)
|
// fmt.Printf("\n ***** DEBUG: copyDstSubProcFunc: cia.SrcMethod: %v\n\n ", cia.SrcMethod)
|
||||||
|
@ -479,11 +489,12 @@ func copySrcSubProcFunc(proc process, cia copyInitialData, cancel context.Cancel
|
||||||
// We want to send a message back to src that we are ready to start.
|
// We want to send a message back to src that we are ready to start.
|
||||||
// fmt.Printf("\n\n\n ************** DEBUG: copyDstHandler sub process sending copyReady to:%v\n ", message.FromNode)
|
// fmt.Printf("\n\n\n ************** DEBUG: copyDstHandler sub process sending copyReady to:%v\n ", message.FromNode)
|
||||||
msg := Message{
|
msg := Message{
|
||||||
ToNode: cia.DstNode,
|
ToNode: cia.DstNode,
|
||||||
FromNode: cia.SrcNode,
|
FromNode: cia.SrcNode,
|
||||||
Method: cia.DstMethod,
|
Method: cia.DstMethod,
|
||||||
ReplyMethod: REQNone,
|
ReplyMethod: REQNone,
|
||||||
Data: csaSerialized,
|
Data: csaSerialized,
|
||||||
|
IsSubPublishedMsg: true,
|
||||||
}
|
}
|
||||||
|
|
||||||
// fmt.Printf("\n ***** DEBUG: copyDstSubProcFunc: cia.SrcMethod: %v\n\n ", cia.SrcMethod)
|
// fmt.Printf("\n ***** DEBUG: copyDstSubProcFunc: cia.SrcMethod: %v\n\n ", cia.SrcMethod)
|
||||||
|
@ -531,11 +542,12 @@ func copyDstSubProcFunc(proc process, cia copyInitialData, message Message, canc
|
||||||
{
|
{
|
||||||
fmt.Printf("\n\n\n ************** DEBUG: copyDstHandler sub process sending copyReady to:%v\n ", message.FromNode)
|
fmt.Printf("\n\n\n ************** DEBUG: copyDstHandler sub process sending copyReady to:%v\n ", message.FromNode)
|
||||||
msg := Message{
|
msg := Message{
|
||||||
ToNode: cia.SrcNode,
|
ToNode: cia.SrcNode,
|
||||||
FromNode: cia.DstNode,
|
FromNode: cia.DstNode,
|
||||||
Method: cia.SrcMethod,
|
Method: cia.SrcMethod,
|
||||||
ReplyMethod: REQNone,
|
ReplyMethod: REQNone,
|
||||||
Data: csaSerialized,
|
Data: csaSerialized,
|
||||||
|
IsSubPublishedMsg: true,
|
||||||
}
|
}
|
||||||
|
|
||||||
fmt.Printf("\n ***** DEBUG: copyDstSubProcFunc: cia.SrcMethod: %v\n\n ", cia.SrcMethod)
|
fmt.Printf("\n ***** DEBUG: copyDstSubProcFunc: cia.SrcMethod: %v\n\n ", cia.SrcMethod)
|
||||||
|
@ -607,11 +619,12 @@ func copyDstSubProcFunc(proc process, cia copyInitialData, message Message, canc
|
||||||
|
|
||||||
// fmt.Printf("\n\n\n ************** DEBUG: copyDstHandler sub process sending copyReady to:%v\n ", message.FromNode)
|
// fmt.Printf("\n\n\n ************** DEBUG: copyDstHandler sub process sending copyReady to:%v\n ", message.FromNode)
|
||||||
msg := Message{
|
msg := Message{
|
||||||
ToNode: cia.SrcNode,
|
ToNode: cia.SrcNode,
|
||||||
FromNode: cia.DstNode,
|
FromNode: cia.DstNode,
|
||||||
Method: cia.SrcMethod,
|
Method: cia.SrcMethod,
|
||||||
ReplyMethod: REQNone,
|
ReplyMethod: REQNone,
|
||||||
Data: csaSer,
|
Data: csaSer,
|
||||||
|
IsSubPublishedMsg: true,
|
||||||
}
|
}
|
||||||
|
|
||||||
// fmt.Printf("\n ***** DEBUG: copyDstSubProcFunc: cia.SrcMethod: %v\n\n ", cia.SrcMethod)
|
// fmt.Printf("\n ***** DEBUG: copyDstSubProcFunc: cia.SrcMethod: %v\n\n ", cia.SrcMethod)
|
||||||
|
@ -633,11 +646,12 @@ func copyDstSubProcFunc(proc process, cia copyInitialData, message Message, canc
|
||||||
}
|
}
|
||||||
|
|
||||||
msg := Message{
|
msg := Message{
|
||||||
ToNode: cia.SrcNode,
|
ToNode: cia.SrcNode,
|
||||||
FromNode: cia.DstNode,
|
FromNode: cia.DstNode,
|
||||||
Method: cia.SrcMethod,
|
Method: cia.SrcMethod,
|
||||||
ReplyMethod: REQNone,
|
ReplyMethod: REQNone,
|
||||||
Data: csaSer,
|
Data: csaSer,
|
||||||
|
IsSubPublishedMsg: true,
|
||||||
}
|
}
|
||||||
|
|
||||||
sam, err := newSubjectAndMessage(msg)
|
sam, err := newSubjectAndMessage(msg)
|
||||||
|
@ -766,11 +780,12 @@ func copyDstSubProcFunc(proc process, cia copyInitialData, message Message, canc
|
||||||
// We want to send a message back to src that we are ready to start.
|
// We want to send a message back to src that we are ready to start.
|
||||||
// fmt.Printf("\n\n\n ************** DEBUG: copyDstHandler sub process sending copyReady to:%v\n ", message.FromNode)
|
// fmt.Printf("\n\n\n ************** DEBUG: copyDstHandler sub process sending copyReady to:%v\n ", message.FromNode)
|
||||||
msg := Message{
|
msg := Message{
|
||||||
ToNode: cia.SrcNode,
|
ToNode: cia.SrcNode,
|
||||||
FromNode: cia.DstNode,
|
FromNode: cia.DstNode,
|
||||||
Method: cia.SrcMethod,
|
Method: cia.SrcMethod,
|
||||||
ReplyMethod: REQNone,
|
ReplyMethod: REQNone,
|
||||||
Data: csaSerialized,
|
Data: csaSerialized,
|
||||||
|
IsSubPublishedMsg: true,
|
||||||
}
|
}
|
||||||
|
|
||||||
// fmt.Printf("\n ***** DEBUG: copyDstSubProcFunc: cia.SrcMethod: %v\n\n ", cia.SrcMethod)
|
// fmt.Printf("\n ***** DEBUG: copyDstSubProcFunc: cia.SrcMethod: %v\n\n ", cia.SrcMethod)
|
||||||
|
|
|
@ -483,7 +483,13 @@ func (s *server) routeMessagesToProcess(dbFileName string) {
|
||||||
// log.Printf("info: processNewMessages: did not find that specific subject, starting new process for subject: %v\n", subjName)
|
// log.Printf("info: processNewMessages: did not find that specific subject, starting new process for subject: %v\n", subjName)
|
||||||
|
|
||||||
sub := newSubject(sam.Subject.Method, sam.Subject.ToNode)
|
sub := newSubject(sam.Subject.Method, sam.Subject.ToNode)
|
||||||
proc := newProcess(s.ctx, s, sub, processKindPublisher, nil)
|
var proc process
|
||||||
|
switch {
|
||||||
|
case m.IsSubPublishedMsg:
|
||||||
|
proc = newSubProcess(s.ctx, s, sub, processKindPublisher, nil)
|
||||||
|
default:
|
||||||
|
proc = newProcess(s.ctx, s, sub, processKindPublisher, nil)
|
||||||
|
}
|
||||||
|
|
||||||
proc.spawnWorker()
|
proc.spawnWorker()
|
||||||
// log.Printf("info: processNewMessages: new process started, subject: %v, processID: %v\n", subjName, proc.processID)
|
// log.Printf("info: processNewMessages: new process started, subject: %v, processID: %v\n", subjName, proc.processID)
|
||||||
|
|
Loading…
Add table
Reference in a new issue