func GetBalance(ussdRequest string) (float64, error) { log.Println("GetBalance...") //re-set encoding here? //m.SendCommand("AT+CSCS=\"GSM\"\r", true) //TODO: Is it necessery to run AT+CMGF=0 ??? SendCommand("AT+CMGF=0\r", true) SendCommand("AT^USSDMODE=1\r", true) request := strings.ToUpper(fmt.Sprintf("%x", pdu.Encode7Bit(ussdRequest))) _, err = SendCommand(fmt.Sprintf("AT+CUSD=1,\"%s\",15\r", request), true) if err != nil { return 0.0, err } status, err := WaitForOutput(10, "15\r\n") regex := regexp.MustCompile(`\+CUSD: \d{1},\"([a-zA-Z0-9]*)\",\d*`) if regex.MatchString(status) { balanceRaw := regex.FindStringSubmatch(status)[1] bytesWritten, _ := hex.DecodeString(balanceRaw) log.Println("Before decode", bytesWritten) balanceRaw, _ = pdu.Decode7Bit(bytesWritten) log.Println("After decode", balanceRaw) balanceParsed := regexp.MustCompile(`\d+\.\d+`).FindString(balanceRaw) if balanceParsed == "" { return 0.0, fmt.Errorf("GetBalance: Failed to find balance string in \"%s\"", balanceRaw) } balance, err := strconv.ParseFloat(balanceParsed, 64) if err != nil { return 0.0, fmt.Errorf("GetBalance: Failed to convert to float64 \"%s\"", balanceRaw) } return balance, nil } if err != nil { return 0.0, err } return 0.0, errors.New("GetBalace: Failed to get balance.") }
// handleReport detects and parses a report from the notification port represented // as a string. The parsed values may change the inner state or be sent over out channels. func (d *Device) handleReport(str string) (err error) { report := Reports.Resolve(str) str = strings.TrimSpace(strings.TrimPrefix(str, report.ID)) switch report { case Reports.Message: var report messageReport if err = report.Parse(str); err != nil { return } var octets []byte octets, err = d.Commands.CMGR(report.Index) if err != nil { return } if err = d.Commands.CMGD(report.Index, DeleteOptions.Index); err != nil { return } var msg sms.Message if _, err = msg.ReadFrom(octets); err != nil { return } d.messages <- &msg case Reports.Ussd: var ussd ussdReport if err = ussd.Parse(str); err != nil { return } var text string if ussd.Enc == Encodings.UCS2 { text, err = pdu.DecodeUcs2(ussd.Octets) if err != nil { return } } else if ussd.Enc == Encodings.Gsm7Bit { text, err = pdu.Decode7Bit(ussd.Octets) if err != nil { return } } else { return ErrUnknownEncoding } d.ussd <- Ussd(text) case Reports.SignalStrength: var rssi signalStrengthReport if err = rssi.Parse(str); err != nil { return } if d.State.SignalStrength != int(rssi) { d.State.SignalStrength = int(rssi) d.updated <- struct{}{} } case Reports.Mode: var report modeReport if err = report.Parse(str); err != nil { return } var updated bool if d.State.SystemMode != report.Mode { d.State.SystemMode = report.Mode updated = true } if d.State.SystemSubmode != report.Submode { d.State.SystemSubmode = report.Submode updated = true } if updated { d.updated <- struct{}{} } case Reports.ServiceState: var report serviceStateReport if err = report.Parse(str); err != nil { return } if d.State.ServiceState != Opt(report) { d.State.ServiceState = Opt(report) d.updated <- struct{}{} } case Reports.SimState: var report simStateReport if err = report.Parse(str); err != nil { return } if d.State.SimState != Opt(report) { d.State.SimState = Opt(report) d.updated <- struct{}{} } case Reports.BootHandshake: var token bootHandshakeReport if err = token.Parse(str); err != nil { return } if err = d.Commands.BOOT(uint64(token)); err != nil { return } case Reports.Stin: // ignore. what is this btw? default: switch FinalResults.Resolve(str) { case FinalResults.Noop, FinalResults.NotSupported, FinalResults.Timeout: // ignore default: return errors.New("at: unknown report: " + str) } } return }
// ReadFrom constructs a message from the supplied PDU octets. Returns the number of bytes read. // Complies with 3GPP TS 23.040. func (s *Message) ReadFrom(octets []byte) (n int, err error) { *s = Message{} buf := bytes.NewReader(octets) scLen, err := buf.ReadByte() n++ if err != nil { return } if scLen > 12 { return 0, ErrIncorrectSize } addr := make([]byte, scLen) off, err := io.ReadFull(buf, addr) n += off if err != nil { return } s.ServiceCenterAddress.ReadFrom(addr) msgType, err := buf.ReadByte() n++ if err != nil { return } n-- buf.UnreadByte() s.Type = MessageType(msgType & 0x03) switch s.Type { case MessageTypes.Deliver: var sms smsDeliver off, err2 := sms.FromBytes(octets[1+scLen:]) n += off if err2 != nil { return n, err2 } s.MoreMessagesToSend = sms.MoreMessagesToSend s.LoopPrevention = sms.LoopPrevention s.ReplyPathExists = sms.ReplyPath s.UserDataStartsWithHeader = sms.UserDataHeaderIndicator s.StatusReportIndication = sms.StatusReportIndication s.Address.ReadFrom(sms.OriginatingAddress[1:]) s.Encoding = Encoding(sms.DataCodingScheme) s.ServiceCenterTime.ReadFrom(sms.ServiceCentreTimestamp) switch s.Encoding { case Encodings.Gsm7Bit: s.Text, err = pdu.Decode7Bit(sms.UserData) if err != nil { return } s.Text = cutStr(s.Text, int(sms.UserDataLength)) case Encodings.UCS2: s.Text, err = pdu.DecodeUcs2(sms.UserData) if err != nil { return } default: return 0, ErrUnknownEncoding } case MessageTypes.Submit: var sms smsSubmit off, err2 := sms.FromBytes(octets[1+scLen:]) n += off if err2 != nil { return n, err2 } s.RejectDuplicates = sms.RejectDuplicates switch s.VPFormat { case ValidityPeriodFormats.Absolute, ValidityPeriodFormats.Enhanced: return n, ErrNonRelative default: s.VPFormat = ValidityPeriodFormat(sms.ValidityPeriodFormat) } s.ReplyPathExists = sms.ReplyPath s.UserDataStartsWithHeader = sms.UserDataHeaderIndicator s.StatusReportRequest = sms.StatusReportRequest s.Address.ReadFrom(sms.DestinationAddress[1:]) s.Encoding = Encoding(sms.DataCodingScheme) if s.VPFormat != ValidityPeriodFormats.FieldNotPresent { s.VP.ReadFrom(sms.ValidityPeriod) } switch s.Encoding { case Encodings.Gsm7Bit: s.Text, err = pdu.Decode7Bit(sms.UserData) if err != nil { return } s.Text = cutStr(s.Text, int(sms.UserDataLength)) case Encodings.UCS2: s.Text, err = pdu.DecodeUcs2(sms.UserData) if err != nil { return } default: return 0, ErrUnknownEncoding } default: return n, ErrUnknownMessageType } return }