- xml - AJAX/Jquery XML 解析
- 具有多重继承的 XML 模式
- .net - 枚举序列化 Json 与 XML
- XML 简单类型、简单内容、复杂类型、复杂内容
我正在开发一个 slack 机器人作为我的第一个 golang 项目,虽然这个特定的机器人命令的功能很好,但它可能会随机出现 panic 并抛出错误。
我能够确定我正在进行数据竞赛,特别是我的两个 go 例程。但是,我很难确定如何修补它们,或者执行互斥锁是否是处理共享变量的正确方法。
我正在尝试确定问题是否出在我的result 变量上,该变量被分成多个 channel ,每个 channel 都由 go 例程处理,或者是否是os.exec 包使用我的缓冲区变量来运行 stdout 和 stderr 修改的命令。
下面是代码和数据竞争示例。
package reboot
import (
"bytes"
"encoding/json"
"fmt"
"io/ioutil"
"log"
"os/exec"
"regexp"
)
// to handle json keys
type rebootObject struct {
Name string
Command string
}
var (
fileNotFound = "config file not found"
cmdNotFound = "Error finding command argument"
)
// *bot.Cmd is the first arg to be passed to the bot
// bot.CmdResultV3 wants message from go routine and done = true
func Reboot(command *bot.Cmd) (result bot.CmdResultV3, err error) {
result = bot.CmdResultV3{Message: make(chan string), Done: make(chan bool, 1)}
// load json config file with names/commands
filePath := "./plugins/reboot/config.json"
file, err1 := ioutil.ReadFile(filePath)
if err1 != nil {
bot.Reply(command.Channel, fileNotFound, command.User)
}
var scriptParse []rebootObject
//userinput := "box4535345346" // faking user input
userinput := command.Args[0] // real one
err2 := json.Unmarshal(file, &scriptParse)
if err2 != nil {
fmt.Println("error:", err2)
bot.Reply(command.Channel, fileNotFound, command.User)
}
//strip numbers off input to match json key
reg, err := regexp.Compile("[^a-zA-Z]+")
if err != nil {
log.Fatal(err)
}
// loop through json file to find the match of user input to json name key
cmdFound := false
for k := range scriptParse {
newinput := reg.ReplaceAllString(userinput, "")
// keep running for loop until names do match
if scriptParse[k].Name != newinput {
continue
}
cmdFound = true
cmd := exec.Command("/bin/bash", "-c", scriptParse[k].Command)
var b bytes.Buffer
cmd.Stdout = &b
cmd.Stderr = &b
err = cmd.Start()
if err != nil {
return
}
done := false
go func() {
cmd.Wait()
done = true
result.Done <- true
}()
go func() {
for {
line, _ := b.ReadString('\n')
if line != "" {
result.Message <- line
}
if done {
close(result.Message)
break
}
}
}()
}
if cmdFound == false {
result.Done <- true
bot.Reply(command.Channel, cmdNotFound, command.User)
}
return result, nil
}
数据竞赛:
==================
WARNING: DATA RACE
Read at 0x00c420582558 by goroutine 37:
bytes.(*Buffer).readSlice()
/usr/local/go/src/bytes/buffer.go:421 +0x48
bytes.(*Buffer).ReadString()
/usr/local/go/src/bytes/buffer.go:440 +0x45
github.com/owner/gobot/plugins/reboot.Reboot.func2()
/Users/macowner/go/src/github.com/owner/gobot/plugins/reboot/reboot.go:79 +0x41
Previous write at 0x00c420582558 by goroutine 35:
bytes.(*Buffer).ReadFrom()
/usr/local/go/src/bytes/buffer.go:92 +0x608
io.copyBuffer()
/usr/local/go/src/io/io.go:386 +0x348
io.Copy()
/usr/local/go/src/io/io.go:362 +0x7e
os/exec.(*Cmd).writerDescriptor.func1()
/usr/local/go/src/os/exec/exec.go:264 +0x68
os/exec.(*Cmd).Start.func1()
/usr/local/go/src/os/exec/exec.go:380 +0x34
Goroutine 37 (running) created at:
github.com/owner/gobot/plugins/reboot.Reboot()
/Users/macowner/go/src/github.com/owner/gobot/plugins/reboot/reboot.go:77 +0x885
github.com/owner/gobot/bot.(*Bot).handleCmd()
/Users/macowner/go/src/github.com/owner/gobot/bot/cmd.go:240 +0x13b
github.com/owner/gobot/bot.(*Bot).MessageReceived()
/Users/macowner/go/src/github.com/owner/gobot/bot/bot.go:101 +0x5a8
github.com/owner/gobot/bot.Run()
/Users/macowner/go/src/github.com/owner/gobot/bot/reply.go:142 +0xdb3
main.main()
/Users/macowner/go/src/github.com/owner/gobot/main.go:46 +0x9b0
Goroutine 35 (running) created at:
os/exec.(*Cmd).Start()
/usr/local/go/src/os/exec/exec.go:379 +0xa6b
github.com/owner/gobot/plugins/reboot.Reboot()
/Users/macowner/go/src/github.com/owner/gobot/plugins/reboot/reboot.go:67 +0x7c8
github.com/owner/gobot/bot.(*Bot).handleCmd()
/Users/macowner/go/src/github.com/owner/gobot/bot/cmd.go:240 +0x13b
github.com/owner/gobot/bot.(*Bot).MessageReceived()
/Users/macowner/go/src/github.com/owner/gobot/bot/bot.go:101 +0x5a8
github.com/owner/gobot/bot.Run()
/Users/macowner/go/src/github.com/owner/gobot/bot/reply.go:142 +0xdb3
main.main()
/Users/macowner/go/src/github.com/owner/gobot/main.go:46 +0x9b0
==================
==================
WARNING: DATA RACE
Read at 0x00c420582540 by goroutine 37:
bytes.(*Buffer).readSlice()
/usr/local/go/src/bytes/buffer.go:421 +0x62
bytes.(*Buffer).ReadString()
/usr/local/go/src/bytes/buffer.go:440 +0x45
github.com/owner/gobot/plugins/reboot.Reboot.func2()
/Users/macowner/go/src/github.com/owner/gobot/plugins/reboot/reboot.go:79 +0x41
Previous write at 0x00c420582540 by goroutine 35:
bytes.(*Buffer).ReadFrom()
/usr/local/go/src/bytes/buffer.go:91 +0x5d0
io.copyBuffer()
/usr/local/go/src/io/io.go:386 +0x348
io.Copy()
/usr/local/go/src/io/io.go:362 +0x7e
os/exec.(*Cmd).writerDescriptor.func1()
/usr/local/go/src/os/exec/exec.go:264 +0x68
os/exec.(*Cmd).Start.func1()
/usr/local/go/src/os/exec/exec.go:380 +0x34
Goroutine 37 (running) created at:
github.com/owner/gobot/plugins/reboot.Reboot()
/Users/macowner/go/src/github.com/owner/gobot/plugins/reboot/reboot.go:77 +0x885
github.com/owner/gobot/bot.(*Bot).handleCmd()
/Users/macowner/go/src/github.com/owner/gobot/bot/cmd.go:240 +0x13b
github.com/owner/gobot/bot.(*Bot).MessageReceived()
/Users/macowner/go/src/github.com/owner/gobot/bot/bot.go:101 +0x5a8
github.com/owner/gobot/bot.Run()
/Users/macowner/go/src/github.com/owner/gobot/bot/reply.go:142 +0xdb3
main.main()
/Users/macowner/go/src/github.com/owner/gobot/main.go:46 +0x9b0
Goroutine 35 (running) created at:
os/exec.(*Cmd).Start()
/usr/local/go/src/os/exec/exec.go:379 +0xa6b
github.com/owner/gobot/plugins/reboot.Reboot()
/Users/macowner/go/src/github.com/owner/gobot/plugins/reboot/reboot.go:67 +0x7c8
github.com/owner/gobot/bot.(*Bot).handleCmd()
/Users/macowner/go/src/github.com/owner/gobot/bot/cmd.go:240 +0x13b
github.com/owner/gobot/bot.(*Bot).MessageReceived()
/Users/macowner/go/src/github.com/owner/gobot/bot/bot.go:101 +0x5a8
github.com/owner/gobot/bot.Run()
/Users/macowner/go/src/github.com/owner/gobot/bot/reply.go:142 +0xdb3
main.main()
/Users/macowner/go/src/github.com/owner/gobot/main.go:46 +0x9b0
==================
==================
WARNING: DATA RACE
Write at 0x00c420582560 by goroutine 37:
bytes.(*Buffer).readSlice()
/usr/local/go/src/bytes/buffer.go:429 +0x186
bytes.(*Buffer).ReadString()
/usr/local/go/src/bytes/buffer.go:440 +0x45
github.com/owner/gobot/plugins/reboot.Reboot.func2()
/Users/macowner/go/src/github.com/owner/gobot/plugins/reboot/reboot.go:79 +0x41
Previous write at 0x00c420582560 by goroutine 35:
bytes.(*Buffer).ReadFrom()
/usr/local/go/src/bytes/buffer.go:191 +0x5f
io.copyBuffer()
/usr/local/go/src/io/io.go:386 +0x348
io.Copy()
/usr/local/go/src/io/io.go:362 +0x7e
os/exec.(*Cmd).writerDescriptor.func1()
/usr/local/go/src/os/exec/exec.go:264 +0x68
os/exec.(*Cmd).Start.func1()
/usr/local/go/src/os/exec/exec.go:380 +0x34
Goroutine 37 (running) created at:
github.com/owner/gobot/plugins/reboot.Reboot()
/Users/macowner/go/src/github.com/owner/gobot/plugins/reboot/reboot.go:77 +0x885
github.com/owner/gobot/bot.(*Bot).handleCmd()
/Users/macowner/go/src/github.com/owner/gobot/bot/cmd.go:240 +0x13b
github.com/owner/gobot/bot.(*Bot).MessageReceived()
/Users/macowner/go/src/github.com/owner/gobot/bot/bot.go:101 +0x5a8
github.com/owner/gobot/bot.Run()
/Users/macowner/go/src/github.com/owner/gobot/bot/reply.go:142 +0xdb3
main.main()
/Users/macowner/go/src/github.com/owner/gobot/main.go:46 +0x9b0
Goroutine 35 (running) created at:
os/exec.(*Cmd).Start()
/usr/local/go/src/os/exec/exec.go:379 +0xa6b
github.com/owner/gobot/plugins/reboot.Reboot()
/Users/macowner/go/src/github.com/owner/gobot/plugins/reboot/reboot.go:67 +0x7c8
github.com/owner/gobot/bot.(*Bot).handleCmd()
/Users/macowner/go/src/github.com/owner/gobot/bot/cmd.go:240 +0x13b
github.com/owner/gobot/bot.(*Bot).MessageReceived()
/Users/macowner/go/src/github.com/owner/gobot/bot/bot.go:101 +0x5a8
github.com/owner/gobot/bot.Run()
/Users/macowner/go/src/github.com/owner/gobot/bot/reply.go:142 +0xdb3
main.main()
/Users/macowner/go/src/github.com/owner/gobot/main.go:46 +0x9b0
==================
==================
WARNING: DATA RACE
Read at 0x00c42014cc00 by goroutine 37:
runtime.slicebytetostring()
/usr/local/go/src/runtime/string.go:72 +0x0
bytes.(*Buffer).ReadString()
/usr/local/go/src/bytes/buffer.go:441 +0x84
github.com/owner/gobot/plugins/reboot.Reboot.func2()
/Users/macowner/go/src/github.com/owner/gobot/plugins/reboot/reboot.go:79 +0x41
Previous write at 0x00c42014cc00 by goroutine 35:
internal/race.WriteRange()
/usr/local/go/src/internal/race/race.go:49 +0x42
syscall.Read()
/usr/local/go/src/syscall/syscall_unix.go:165 +0x9a
internal/poll.(*FD).Read()
/usr/local/go/src/internal/poll/fd_unix.go:122 +0x1a0
os.(*File).read()
/usr/local/go/src/os/file_unix.go:216 +0x70
os.(*File).Read()
/usr/local/go/src/os/file.go:103 +0x8e
bytes.(*Buffer).ReadFrom()
/usr/local/go/src/bytes/buffer.go:209 +0x1dd
io.copyBuffer()
/usr/local/go/src/io/io.go:386 +0x348
io.Copy()
/usr/local/go/src/io/io.go:362 +0x7e
os/exec.(*Cmd).writerDescriptor.func1()
/usr/local/go/src/os/exec/exec.go:264 +0x68
os/exec.(*Cmd).Start.func1()
/usr/local/go/src/os/exec/exec.go:380 +0x34
Goroutine 37 (running) created at:
github.com/owner/gobot/plugins/reboot.Reboot()
/Users/macowner/go/src/github.com/owner/gobot/plugins/reboot/reboot.go:77 +0x885
github.com/owner/gobot/bot.(*Bot).handleCmd()
/Users/macowner/go/src/github.com/owner/gobot/bot/cmd.go:240 +0x13b
github.com/owner/gobot/bot.(*Bot).MessageReceived()
/Users/macowner/go/src/github.com/owner/gobot/bot/bot.go:101 +0x5a8
github.com/owner/gobot/bot.Run()
/Users/macowner/go/src/github.com/owner/gobot/bot/reply.go:142 +0xdb3
main.main()
/Users/macowner/go/src/github.com/owner/gobot/main.go:46 +0x9b0
Goroutine 35 (running) created at:
os/exec.(*Cmd).Start()
/usr/local/go/src/os/exec/exec.go:379 +0xa6b
github.com/owner/gobot/plugins/reboot.Reboot()
/Users/macowner/go/src/github.com/owner/gobot/plugins/reboot/reboot.go:67 +0x7c8
github.com/owner/gobot/bot.(*Bot).handleCmd()
/Users/macowner/go/src/github.com/owner/gobot/bot/cmd.go:240 +0x13b
github.com/owner/gobot/bot.(*Bot).MessageReceived()
/Users/macowner/go/src/github.com/owner/gobot/bot/bot.go:101 +0x5a8
github.com/owner/gobot/bot.Run()
/Users/macowner/go/src/github.com/owner/gobot/bot/reply.go:142 +0xdb3
main.main()
/Users/macowner/go/src/github.com/owner/gobot/main.go:46 +0x9b0
==================
==================
WARNING: DATA RACE
Read at 0x00c4202d2600 by goroutine 37:
runtime.slicebytetostring()
/usr/local/go/src/runtime/string.go:72 +0x0
bytes.(*Buffer).ReadString()
/usr/local/go/src/bytes/buffer.go:441 +0x84
github.com/owner/gobot/plugins/reboot.Reboot.func2()
/Users/macowner/go/src/github.com/owner/gobot/plugins/reboot/reboot.go:79 +0x41
Previous write at 0x00c4202d2600 by goroutine 35:
runtime.slicecopy()
/usr/local/go/src/runtime/slice.go:160 +0x0
bytes.(*Buffer).ReadFrom()
/usr/local/go/src/bytes/buffer.go:205 +0x4b2
io.copyBuffer()
/usr/local/go/src/io/io.go:386 +0x348
io.Copy()
/usr/local/go/src/io/io.go:362 +0x7e
os/exec.(*Cmd).writerDescriptor.func1()
/usr/local/go/src/os/exec/exec.go:264 +0x68
os/exec.(*Cmd).Start.func1()
/usr/local/go/src/os/exec/exec.go:380 +0x34
Goroutine 37 (running) created at:
github.com/owner/gobot/plugins/reboot.Reboot()
/Users/macowner/go/src/github.com/owner/gobot/plugins/reboot/reboot.go:77 +0x885
github.com/owner/gobot/bot.(*Bot).handleCmd()
/Users/macowner/go/src/github.com/owner/gobot/bot/cmd.go:240 +0x13b
github.com/owner/gobot/bot.(*Bot).MessageReceived()
/Users/macowner/go/src/github.com/owner/gobot/bot/bot.go:101 +0x5a8
github.com/owner/gobot/bot.Run()
/Users/macowner/go/src/github.com/owner/gobot/bot/reply.go:142 +0xdb3
main.main()
/Users/macowner/go/src/github.com/owner/gobot/main.go:46 +0x9b0
Goroutine 35 (running) created at:
os/exec.(*Cmd).Start()
/usr/local/go/src/os/exec/exec.go:379 +0xa6b
github.com/owner/gobot/plugins/reboot.Reboot()
/Users/macowner/go/src/github.com/owner/gobot/plugins/reboot/reboot.go:67 +0x7c8
github.com/owner/gobot/bot.(*Bot).handleCmd()
/Users/macowner/go/src/github.com/owner/gobot/bot/cmd.go:240 +0x13b
github.com/owner/gobot/bot.(*Bot).MessageReceived()
/Users/macowner/go/src/github.com/owner/gobot/bot/bot.go:101 +0x5a8
github.com/owner/gobot/bot.Run()
/Users/macowner/go/src/github.com/owner/gobot/bot/reply.go:142 +0xdb3
main.main()
/Users/macowner/go/src/github.com/owner/gobot/main.go:46 +0x9b0
==================
==================
WARNING: DATA RACE
Write at 0x00c4202008c8 by goroutine 36:
github.com/owner/gobot/plugins/reboot.Reboot.func1()
/Users/macowner/go/src/github.com/owner/gobot/plugins/reboot/reboot.go:74 +0x4d
Previous read at 0x00c4202008c8 by goroutine 37:
github.com/owner/gobot/plugins/reboot.Reboot.func2()
/Users/macowner/go/src/github.com/owner/gobot/plugins/reboot/reboot.go:83 +0x61
Goroutine 36 (running) created at:
github.com/owner/gobot/plugins/reboot.Reboot()
/Users/macowner/go/src/github.com/owner/gobot/plugins/reboot/reboot.go:72 +0x846
github.com/owner/gobot/bot.(*Bot).handleCmd()
/Users/macowner/go/src/github.com/owner/gobot/bot/cmd.go:240 +0x13b
github.com/owner/gobot/bot.(*Bot).MessageReceived()
/Users/macowner/go/src/github.com/owner/gobot/bot/bot.go:101 +0x5a8
github.com/owner/gobot/bot.Run()
/Users/macowner/go/src/github.com/owner/gobot/bot/reply.go:142 +0xdb3
main.main()
/Users/macowner/go/src/github.com/owner/gobot/main.go:46 +0x9b0
Goroutine 37 (running) created at:
github.com/owner/gobot/plugins/reboot.Reboot()
/Users/macowner/go/src/github.com/owner/gobot/plugins/reboot/reboot.go:77 +0x885
github.com/owner/gobot/bot.(*Bot).handleCmd()
/Users/macowner/go/src/github.com/owner/gobot/bot/cmd.go:240 +0x13b
github.com/owner/gobot/bot.(*Bot).MessageReceived()
/Users/macowner/go/src/github.com/owner/gobot/bot/bot.go:101 +0x5a8
github.com/owner/gobot/bot.Run()
/Users/macowner/go/src/github.com/owner/gobot/bot/reply.go:142 +0xdb3
main.main()
/Users/macowner/go/src/github.com/owner/gobot/main.go:46 +0x9b0
==================
最佳答案
第一个 goroutine,从 cmd.Wait()
开始,正在写入缓冲区 b
而另一个 goroutine,从 line, _ := b.ReadString('\n')
正在读取它。这就是数据竞赛发生的原因。修改代码,执行完成后只读取执行结果。
关于go - 使用 slack bot 处理多个数据竞争,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/48769644/
我刚刚创建了一个新的 Slack 应用程序并向其中添加了一个机器人。 现在我有 2 个可用的 token : OAuth 访问 token (xoxp) 机器人用户 OAuth 访问 token (x
我正在构建与 Slack 和客户端的消息传递集成。我需要在用户登录客户端时在相应的对话上显示徽章,以向客户端显示它有多少条未读消息。 我知道 channels.history 方法有 unreads
我有一个 slack POST api 调用,我正在从我的应用程序中调用它:- slack_client.api_call("chat.postMessage", channel=channel, t
关闭。这个问题不满足Stack Overflow guidelines .它目前不接受答案。 想改善这个问题吗?更新问题,使其成为 on-topic对于堆栈溢出。 去年关闭。 Improve this
Slack 默认头像来自哪里?这是他们自己的服务还是第三方服务? 在任何一种情况下,其他人是否有可能使用头像创建服务/系统,例如通过 slack API? 最佳答案 我 asked Slack dir
我正在编写一个脚本,以我的身份发布自动消息。我很难找到我的用户 token 。我必须创建一个应用程序吗? 最佳答案 是的,您必须创建一个应用程序才能生成用户 token 。您可以从此处创建一个应用程序
我正在为一些自动通知创建一个 slackbot,并且我收到了一个请求,让通知占据整个 slack 窗口的宽度,这看起来更干净。我发现 block 似乎占据了一个指定的宽度,并且不会水平延伸超过该点。我
我想创建一个 Slack 命令 /command do-something-with-that-file,同时用户上传文件。 到目前为止,我的命令已收到,但负载中没有文件。这可能吗? 我还在作用域中添
我一直在使用 Slack 操作 API,但不知道如何使用 response_url 在原始消息的线程中发送回复。 chat.postMessage 选项实际上不起作用,因为它需要手动将机器人添加到 c
“Slack”可以将链接展开为视频,但有些服务可以,有些则不能。 Vimeo( https://vimeo.com/ ) 可以在 Slack 中将链接展开为视频。 TED( https://www.t
我想复制一个现有的 Slack 应用程序,只更改一些小的配置。 除了从头开始创建新的 slack 应用程序并重新配置它之外,还有更简单的方法吗? 最佳答案 或者,您可以使用App Manifest 功
On slack - 我可以使用以下文本语法直接从我的两者中标记用户: hello Francois 我想对群组执行同样的操作(不幸的是,我的应用程序只知道群组名称,不知道群组 ID)。是否有这样做
我们使用通信工具 Slack。它在我的左侧边栏中显示了我最近与之进行过直接消息对话的在线和离开用户。我想在列表中看到当前在线的每个人,我们 Slack 服务器上的所有在线用户。在聊天应用程序的侧边栏中
我正在尝试构建一个在首次安装到工作区时执行某些操作的应用程序,例如:Ping 每个团队成员。 我找不到在应用安装时触发的事件类型: https://api.slack.com/events 有什么办法
我有一个发送长消息的机器人。 当它发送多条消息时,用户图标通常会滚出屏幕,因为没有人在机器人的消息之间发送过消息。 这意味着发送消息的时间和发送者不太明显。 有没有办法强制 Slack 为每条消息显示
我不完全确定这是否适合 Stack Overflow... 只是一个简短的问题,希望很多人会觉得有用。 我看到您可以使用主题标签,但它们必须与现有 channel 相对应。我有一个与一个项目相关的 c
例如,我希望在我休息的日子里每天在特定 channel 中显示提醒。就像是: /remind #channel @me will be on PTO 1/1/2016 until 1/5/2016 然
我被一位同事添加到 Slack 工作区。我一定已经接受了移动应用程序上的某些内容,因为我可以访问那里的工作区。不知何故,桌面应用程序从未反射(reflect)过这种变化,也没有表现出知道工作空间的迹象
我是 Slack 的新手,刚刚创建了我的第一个 channel 。在设置 channel 时,我添加了一个目的,但是一旦创建,还有一个添加主题的地方。 channel 的目的和主题有什么区别?它们看起
线程将是 Slack 的一个非常好的特性,只是它们太窄了,我找不到任何方法来配置它们的宽度。 有谁知道如何扩展线程以使其占用所有屏幕空间(在 Slack 内)?放大不会改变窄度。它只会使字体更大。我想
我是一名优秀的程序员,十分优秀!