doushi8186 2019-03-26 15:05
浏览 157
已采纳

使用exec.Command运行sqlite3 .backup

I'm attempting to back up a db from Go using the sqlite3 .backup command in the manner give by this SO answer https://stackoverflow.com/a/25684912/426853.

The following always works from the command line:

sqlite3 /home/pi/pgclogs/smartlog.db ".backup '/home/pi/pgcdata/smartlog.db.bak'"

I've coded it in Go as follows:

func DbBackup() (err error) {
    dbpath := "/home/pi/pgclogs/smartlog.db"
    bakpath := "/home/pi/pgcdata/smartlog.db.bak"
    cmd := exec.Command("sqlite3", dbpath, fmt.Sprintf("\".backup '%s'\"", bakpath))
    out, err := cmd.CombinedOutput()
    if err != nil {
        return fmt.Errorf("dbBackup failed : %s : %v", string(out), err)
    }
    return
} 

and I have a test file that invokes it like this:

func TestDbBackup(t *testing.T) {
    var err error
    err = DbBackup()
    if err != nil {
        t.Errorf("backup failed : %v", err)
    }
} 

The test reports a syntax error from sqlite3.

--- FAIL: TestDbBackup (0.01s)
    db_backup_test.go:22: backup failed : dbBackup failed : Error: near "".backup '/home/pi/pgcdata/smartlog.db.bak'"": syntax error
         : exit status 1

I suspect the problem is the way cmd.CombinedOutput is passing the arguments to the shell and that the problem is in the quoting in the fmt.Sprintf call. I've tried changing it to

fmt.Sprintf(`".backup '%s'"`, bakpath)

but the result is the same. I also tried putting echo in front of the command, i.e exec.Command("echo", "sqlite3", dbpath, ... and printing the output. The output looks absolutely correct

sqlite3 /home/pi/pgclogs/smartlog.db ".backup '/home/pi/pgcdata/smartlog.db.bak'"

and it runs correctly when pasted to the command line. I'm sure I'm overlooking something simple, but I've spent over an hour and am still not seeing it.

  • 写回答

1条回答 默认 最新

  • dtpngq3378499 2019-03-26 16:11
    关注

    You don't need the additional quotes at all, so this should do:

    exec.Command("sqlite3", dbpath, fmt.Sprintf(".backup '%s'", bakpath))
    

    The quotes are parsed by the shell so that .backup 'somefile' gets passed to sqlite3 as a single argument. When using exec.Command, there is no shell involved, that's also the reason why you are passing the command splitted into multiple parameters.

    本回答被题主选为最佳回答 , 对您是否有帮助呢?
    评论

报告相同问题?

悬赏问题

  • ¥100 set_link_state
  • ¥15 虚幻5 UE美术毛发渲染
  • ¥15 CVRP 图论 物流运输优化
  • ¥15 Tableau online 嵌入ppt失败
  • ¥100 支付宝网页转账系统不识别账号
  • ¥15 基于单片机的靶位控制系统
  • ¥15 真我手机蓝牙传输进度消息被关闭了,怎么打开?(关键词-消息通知)
  • ¥15 装 pytorch 的时候出了好多问题,遇到这种情况怎么处理?
  • ¥20 IOS游览器某宝手机网页版自动立即购买JavaScript脚本
  • ¥15 手机接入宽带网线,如何释放宽带全部速度