Go: panic: ข้อผิดพลาดรันไทม์: ที่อยู่หน่วยความจำไม่ถูกต้องหรือไม่มีการอ้างอิงศูนย์


99

เมื่อเรียกใช้โปรแกรม Go ของฉันมันจะตื่นตระหนกและส่งคืนสิ่งต่อไปนี้:

panic: runtime error: invalid memory address or nil pointer dereference
[signal 0xb code=0x1 addr=0x38 pc=0x26df]

goroutine 1 [running]:
main.getBody(0x1cdcd4, 0xf800000004, 0x1f2b44, 0x23, 0xf84005c800, ...)
        /Users/matt/Dropbox/code/go/scripts/cron/fido.go:65 +0x2bb
main.getToken(0xf84005c7e0, 0x10)
        /Users/matt/Dropbox/code/go/scripts/cron/fido.go:140 +0x156
main.main()
        /Users/matt/Dropbox/code/go/scripts/cron/fido.go:178 +0x61

goroutine 2 [syscall]:
created by runtime.main
        /usr/local/Cellar/go/1.0.3/src/pkg/runtime/proc.c:221

goroutine 3 [syscall]:
syscall.Syscall6()
        /usr/local/Cellar/go/1.0.3/src/pkg/syscall/asm_darwin_amd64.s:38 +0x5
syscall.kevent(0x6, 0x0, 0x0, 0xf840085188, 0xa, ...)
        /usr/local/Cellar/go/1.0.3/src/pkg/syscall/zsyscall_darwin_amd64.go:199 +0x88
syscall.Kevent(0xf800000006, 0x0, 0x0, 0xf840085188, 0xa0000000a, ...)
        /usr/local/Cellar/go/1.0.3/src/pkg/syscall/syscall_bsd.go:546 +0xa4
net.(*pollster).WaitFD(0xf840085180, 0xf840059040, 0x0, 0x0, 0x0, ...)
        /usr/local/Cellar/go/1.0.3/src/pkg/net/fd_darwin.go:96 +0x185
net.(*pollServer).Run(0xf840059040, 0x0)
        /usr/local/Cellar/go/1.0.3/src/pkg/net/fd.go:236 +0xe4
created by net.newPollServer
        /usr/local/Cellar/go/1.0.3/src/pkg/net/newpollserver.go:35 +0x382

ฉันดูคำตอบที่คนอื่นมีข้อยกเว้นเหมือนกัน แต่ไม่เห็นอะไรง่ายๆ (เช่นข้อผิดพลาดที่ไม่สามารถจัดการได้)

ฉันใช้งานบนเครื่องที่ไม่มีสิทธิ์เข้าถึงเซิร์ฟเวอร์ API ที่ระบุไว้ในโค้ด แต่ฉันหวังว่ามันจะส่งคืนข้อผิดพลาดที่เหมาะสม (เนื่องจากฉันพยายามตรวจจับข้อผิดพลาดประเภทนั้น)

package main

/*
Fido fetches the list of public images from the Glance server, captures the IDs of images with 'status': 'active' and then queues the images for pre-fetching with the Glance CLI utility `glance-cache-manage`. Once the images are added to the queue, `glance-cache-prefetcher` is called to actively fetch the queued images into the local compute nodes' image cache.

See http://docs.openstack.org/developer/glance/cache.html for further details on the Glance image cache.
*/

import (
    "bytes"
    "encoding/json"
    "fmt"
    "io/ioutil"
    /*
        "log"
        "log/syslog"
    */
    "net/http"
    "os"
    "os/exec"
)

func prefetchImages() error {

    cmd := exec.Command("glance-cache-prefetcher")
    err := cmd.Run()

    if err != nil {
        return fmt.Errorf("glance-cache-prefetcher failed to execute properly: %v", err)
    }

    return nil
}

func queueImages(hostname string, imageList []string) error {

    for _, image := range imageList {
        cmd := exec.Command("glance-cache-manage", "--host=", hostname, "queue-image", image)
        err := cmd.Run()

        if err != nil {
            return fmt.Errorf("glance-cache-manage failed to execute properly: %v", err)
        } else {
            fmt.Printf("Image %s queued", image)
        }
    }

    return nil
}

func getBody(method string, url string, headers map[string]string, body []byte) ([]byte, error) {

    client := &http.Client{}
    req, err := http.NewRequest(method, url, bytes.NewReader(body))

    if err != nil {
        return nil, err
    }

    for key, value := range headers {
        req.Header.Add(key, value)
    }

    res, err := client.Do(req)
    defer res.Body.Close()

    if err != nil {
        return nil, err
    }

    var bodyBytes []byte

    if res.StatusCode == 200 {
        bodyBytes, err = ioutil.ReadAll(res.Body)
    } else if err != nil {
        return nil, err
    } else {
        return nil, fmt.Errorf("The remote end did not return a HTTP 200 (OK) response.")
    }

    return bodyBytes, nil

}

func getImages(authToken string) ([]string, error) {

    type GlanceDetailResponse struct {
        Images []struct {
            Name   string `json:"name"`
            Status string `json:"status"`
            ID     string `json:"id"`
        }
    }

    method := "GET"
    url := "http://192.168.1.2:9292/v1.1/images/detail"
    headers := map[string]string{"X-Auth-Token": authToken}

    bodyBytes, err := getBody(method, url, headers, nil)

    if err != nil {
        return nil, fmt.Errorf("unable to retrieve the response body from the Glance API server: %v", err)
    }

    var glance GlanceDetailResponse
    err = json.Unmarshal(bodyBytes, &glance)

    if err != nil {
        return nil, fmt.Errorf("unable to parse the JSON response:", err)
    }

    imageList := make([]string, 10)

    for _, image := range glance.Images {
        if image.Status == "active" {
            imageList = append(imageList, image.ID)
        }
    }

    return imageList, nil

}

func getToken() (string, error) {

    type TokenResponse struct {
        Auth []struct {
            Token struct {
                Expires string `json:"expires"`
                ID      string `json:"id"`
            }
        }
    }

    method := "POST"
    url := "http://192.168.1.2:5000/v2.0/tokens"
    headers := map[string]string{"Content-type": "application/json"}
    creds := []byte(`{"auth":{"passwordCredentials":{"username": "glance", "password":"<password>"}, "tenantId":"<tenantkeygoeshere>"}}`)

    bodyBytes, err := getBody(method, url, headers, creds)

    if err != nil {
        return "", err
    }

    var keystone TokenResponse
    err = json.Unmarshal(bodyBytes, &keystone)

    if err != nil {
        return "", err
    }

    authToken := string((keystone.Auth[0].Token.ID))

    return authToken, nil
}

func main() {

    /*
        slog, err := syslog.New(syslog.LOG_ERR, "[fido]")

        if err != nil {
            log.Fatalf("unable to connect to syslog: %v", err)
            os.Exit(1)
        } else {
            defer slog.Close()
        }
    */

    hostname, err := os.Hostname()

    if err != nil {
        // slog.Err("Hostname not captured")
        os.Exit(1)
    }

    authToken, err := getToken()

    if err != nil {
        // slog.Err("The authentication token from the Glance API server was not retrieved")
        os.Exit(1)
    }

    imageList, err := getImages(authToken)

    err = queueImages(hostname, imageList)

    if err != nil {
        // slog.Err("Could not queue the images for pre-fetching")
        os.Exit(1)
    }

    err = prefetchImages()

    if err != nil {
        // slog.Err("Could not queue the images for pre-fetching")
        os.Exit(1)
    }

    return
}

คำตอบ:


122

ตามเอกสารสำหรับfunc (*Client) Do:

"ข้อผิดพลาดจะถูกส่งกลับหากเกิดจากนโยบายไคลเอนต์ (เช่น CheckRedirect) หรือหากมีข้อผิดพลาดของโปรโตคอล HTTP การตอบสนองที่ไม่ใช่ 2xx ไม่ก่อให้เกิดข้อผิดพลาด

เมื่อความผิดพลาดเป็นศูนย์การตอบสนองจะมีการตอบสนองที่ไม่เป็นศูนย์เสมอร่างกาย "

จากนั้นดูรหัสนี้:

res, err := client.Do(req)
defer res.Body.Close()

if err != nil {
    return nil, err
}

ฉันคาดเดาว่าไม่err nilคุณกำลังเข้าถึง.Close()วิธีการres.Bodyก่อนที่คุณจะตรวจสอบไฟล์err.

deferdefers เพียงการเรียกใช้ฟังก์ชัน สามารถเข้าถึงฟิลด์และวิธีการได้ทันที


ดังนั้นให้ลองตรวจสอบข้อผิดพลาดทันทีแทน

res, err := client.Do(req)

if err != nil {
    return nil, err
}
defer res.Body.Close()

1
สมบูรณ์แบบ! การเลื่อนการเลื่อนหลังจากการตรวจสอบข้อผิดพลาดแก้ไขได้
Melvin

ถ้า err! = nil, res.Body = nil ทำไม res.Body.Close () สามารถโทรได้?
oohcode

2
@oohcode if err! = nil แล้ว res.Body.Close () จะไม่ถูกเรียกเนื่องจากมีคำสั่ง return ภายใน if-block
giraffe.guru

12

การกำหนดค่าศูนย์ของตัวชี้ศูนย์อยู่ในบรรทัดที่ 65 ซึ่งเป็นการเลื่อนเข้า

res, err := client.Do(req)
defer res.Body.Close()

if err != nil {
    return nil, err
}

ถ้า err! = nil ให้ res == nil และ res ความตื่นตระหนกของร่างกาย จัดการกับข้อผิดพลาดก่อนที่จะเปลี่ยน res.Body.Close ()


4

เนื่องจากฉันมาถึงที่นี่พร้อมกับปัญหาของฉันฉันจะเพิ่มคำตอบนี้แม้ว่าจะไม่ตรงกับคำถามเดิมก็ตาม เมื่อคุณใช้งานอินเทอร์เฟซอย่าลืมเพิ่มตัวชี้ประเภทในการประกาศฟังก์ชันสมาชิกของคุณ ตัวอย่าง:

type AnimalSounder interface {
    MakeNoise()
}

type Dog struct {
    Name string
    mean bool
    BarkStrength int
}

func (dog *Dog) MakeNoise() {
    //implementation
}

ฉันลืมส่วน* (dog Dog)ฉันไม่แนะนำ จากนั้นคุณจะประสบปัญหาที่น่าเกลียดเมื่อเรียก MakeNoice บนตัวแปรอินเทอร์เฟซ AnimalSounder ประเภท Dog


-16

สำหรับฉันทางออกหนึ่งสำหรับปัญหานี้คือการเพิ่มใน sql เปิด ... sslmode = ปิดใช้งาน

โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.