From 9c82f7c534632b0d5b04c7c979d82cbf53a0c5ff Mon Sep 17 00:00:00 2001 From: JeGoi <13801368+JeGoi@users.noreply.github.com> Date: Fri, 1 Oct 2021 16:29:16 -0400 Subject: [PATCH 01/18] Add debug --- certificate.go | 20 +- parse.go | 671 ++++++++++++++++++++++++++++--------------------- utils.go | 205 +++++++++++---- 3 files changed, 545 insertions(+), 351 deletions(-) diff --git a/certificate.go b/certificate.go index cc49893..7c771f1 100644 --- a/certificate.go +++ b/certificate.go @@ -23,7 +23,7 @@ func getCAFingerprint(caFileBinary string) (string, error) { caFile, err := os.Open(caFileBinary) if err != nil { walk.MsgBox(windowMsgBox, T("errorWindowTitle"), T("cannotOpenCAFile"), walk.MsgBoxOK) - log.Fatal("Failed opening CA file: ", err) + log.Print("Failed opening CA file: ", err) return "", err } // close file @@ -34,7 +34,7 @@ func getCAFingerprint(caFileBinary string) (string, error) { // copy hash to the file if _, err := io.Copy(hashSha1, caFile); err != nil { walk.MsgBox(windowMsgBox, T("errorWindowTitle"), T("cannotCopyCAFile"), walk.MsgBoxOK) - log.Fatal("Failed copying CA file: ", err) + log.Print("Failed copying CA file: ", err) return "", err } // returns sha1 checksum of the data @@ -100,17 +100,17 @@ func addCertToMachine(userCertDecode string, CERTUTIL_PROGRAM_PATH string) error walk.MsgBox(windowMsgBox, T("errorWindowTitle"), T("invalidCertificate"), walk.MsgBoxOK) os.Remove(userCertDecode) os.Remove("profile.xml") - log.Fatal("Invalid certificate: ", exitStatus) + log.Print("Invalid certificate: ", exitStatus) case int(ERROR_FILE_NOT_FOUND): walk.MsgBox(windowMsgBox, T("errorWindowTitle"), T("cannotFindCertificateFile"), walk.MsgBoxOK) os.Remove(userCertDecode) os.Remove("profile.xml") - log.Fatal("Certificate not found: ", exitStatus) + log.Print("Certificate not found: ", exitStatus) default: walk.MsgBox(windowMsgBox, T("errorWindowTitle"), T("cannotInstallCertificate"), walk.MsgBoxOK) os.Remove(userCertDecode) os.Remove("profile.xml") - log.Fatal("Cannot install certificate: ", exitStatus) + log.Print("Cannot install certificate: ", exitStatus) } } } @@ -148,19 +148,13 @@ func addCAToMachine(caFileBinary string, CERTUTIL_PROGRAM_PATH string) error { retryOrCancel := walk.MsgBox(windowMsgBox, T("errorWindowTitle"), T("caErrorCanceled"), walk.MsgBoxRetryCancel) if retryOrCancel == 4 { log.Print("Failed installing certificate: ", err) - os.Remove(caFileBinary) - os.Remove("profile.xml") runCommand = true } else { - log.Fatal("Failed installing certificate: ", err) - os.Remove(caFileBinary) - os.Remove("profile.xml") + log.Print("Failed installing certificate: ", err) } } else { walk.MsgBox(windowMsgBox, T("errorWindowTitle"), T("cannotInstallCA"), walk.MsgBoxOK) - os.Remove(caFileBinary) - os.Remove("profile.xml") - log.Fatal("Failed installing certificate: ", err) + log.Print("Failed installing certificate: ", err) } } } diff --git a/parse.go b/parse.go index 61ed87c..6113490 100644 --- a/parse.go +++ b/parse.go @@ -61,114 +61,182 @@ type Template struct { } type Handle uintptr +var debug = false +var mw1 *walk.MainWindow +var configButton, debugButton, closedButton *walk.PushButton +var imgView *walk.ImageView +var debugTxt *walk.TextEdit +var debugGrpBox *walk.GroupBox +var cafilePath string + +var tempPath string +var pngFileName string +var pngFilePath string +var userCertPath string +var profilePath string +var stableCurrentWorkingDirectory string + func main() { hideConsole() + prepareMainWindow() + if err := (MainWindow{ + AssignTo: &mw1, + Title: fmt.Sprintf("%s - %s", PROGRAM_NAME, VERSION), + MinSize: Size{500, 400}, + Layout: VBox{}, + Background: SolidColorBrush{Color: walk.RGB(4, 5, 3)}, + Children: []Widget{ + ImageView{ + AssignTo: &imgView, + Background: SolidColorBrush{Color: walk.RGB(4, 5, 3)}, + //Image: pngFileName, + Visible: true, + }, + GroupBox{ + Title: "Configuration Debug output", + AssignTo: &debugGrpBox, + Visible: false, + Layout: VBox{}, + Children: []Widget{ + ScrollView{ + HorizontalFixed: true, + Layout: VBox{}, + Children: []Widget{ + TextEdit{ + MinSize: Size{400, 10}, + AssignTo: &debugTxt, + ReadOnly: false, + Text: "", + }, + }, + }, + }, + }, + PushButton{ + AssignTo: &configButton, + Background: SolidColorBrush{Color: walk.RGB(4, 5, 3)}, + MinSize: Size{50, 50}, + Text: "Configure", + OnClicked: func() { + fetchPortalDomainName() + }, + }, + PushButton{ + AssignTo: &debugButton, + Background: SolidColorBrush{Color: walk.RGB(4, 5, 3)}, + MinSize: Size{10, 10}, + Text: "Debug", + OnClicked: func() { + viewDebug() + }, + }, + PushButton{ + AssignTo: &closedButton, + Background: SolidColorBrush{Color: walk.RGB(4, 5, 3)}, + MinSize: Size{10, 10}, + Text: "Close", + Visible: false, + OnClicked: func() { + cleanAndExit() + mw1.Close() + }, + }, + }, + OnSizeChanged: func() { + mw1size := Size{500, 400} + mw1.SetSize(walk.Size(mw1size)) + }, + }.Create()); err != nil { + log.Print("Failed opening main window: ", err) + viewErrorAndExit(T("errorMainWindow: " + err.Error())) + } + prepareEnv() + prepareBackgroundImage() - log.Println("==================== PacketFence Provisioning Agent ===================") + mw1.Run() +} +func prepareMainWindow() { currentWorkingDirectory, err := os.Executable() if err != nil { walk.MsgBox(windowMsgBox, "Error", "Unable to get current working directory, please contact your local support.", walk.MsgBoxOK) } - stableCurrentWorkingDirectory := filepath.Dir(currentWorkingDirectory) + stableCurrentWorkingDirectory = filepath.Dir(currentWorkingDirectory) + tempPath = os.Getenv("tmp") + // Access to tmp path + if tempPath == "" { + viewErrorAndExit(T("invalidTempPath")) + } + walk.Resources.SetRootDirPath(tempPath) +} + +func prepareEnv() { + debug = true + addNewLinesToDebug("============== PacketFence Provisioning Agent ==============") // Internationalization (i18n) localeInfo := win.GetThreadUILanguage() - log.Printf("User's locale is: %#x", localeInfo) + addNewLinesToDebug("User's locale uint16 is: " + fmt.Sprintf("%v", localeInfo)) switch localeInfo { case SUBLANG_FRENCH, SUBLANG_FRENCH_CANADIAN, SUBLANG_FRENCH_BELGIAN, SUBLANG_FRENCH_LUXEMBOURG, SUBLANG_FRENCH_MONACO, SUBLANG_FRENCH_SWISS: languageFileName := "fr.json" createLanguageFile(stableCurrentWorkingDirectory, FRENCH_TRANSLATION, languageFileName) i18n.MustLoadTranslationFile(languageFileName) T, _ = i18n.Tfunc("fr") + addNewLinesToDebug("Program will be translated in french.") default: languageFileName := "en.json" createLanguageFile(stableCurrentWorkingDirectory, ENGLISH_TRANSLATION, languageFileName) i18n.MustLoadTranslationFile(languageFileName) T, _ = i18n.Tfunc("en") + addNewLinesToDebug("Program will be translated in english.") } - // Main window - tempPath := os.Getenv("tmp") - walk.Resources.SetRootDirPath(tempPath) - _, pfBg := base64ToPng(BACKGROUND_IMAGE_PF, tempPath) - var mw1 *walk.MainWindow - if _, err := (MainWindow{ - AssignTo: &mw1, - Title: fmt.Sprintf("%s - %s", PROGRAM_NAME, VERSION), - MinSize: Size{400, 400}, - Layout: VBox{}, - Background: SolidColorBrush{Color: walk.RGB(4, 5, 3)}, - Children: []Widget{ - ImageView{ - Background: SolidColorBrush{Color: walk.RGB(4, 5, 3)}, - Image: pfBg, - }, - PushButton{ - Background: SolidColorBrush{Color: walk.RGB(4, 5, 3)}, - MinSize: Size{50, 50}, - Text: "Configure", - OnClicked: func() { - Configure() - mw1.Close() - }, - }, - }, - }.Run()); err != nil { - walk.MsgBox(windowMsgBox, T("errorWindowTitle"), T("errorMainWindow"), walk.MsgBoxOK) - log.Fatal("Failed opening main window: ", err) - os.Exit(1) - } - os.Remove(tempPath + "\\" + "pf_bg.png") - os.Exit(0) + pngFileName = "pf_bg.png" + pngFilePath = tempPath + "\\" + pngFileName + userCertPath = "" + profilePath = tempPath + "\\template-out.xml" + debug = false } -func Configure() { - var WLAN_ERROR_MESSAGE = T("wlanErrorMessage") - var WIRED_ERROR_MESSAGE = T("wiredErrorMessage") - var WIRED_SUCCESS_MESSAGE = T("wiredSuccessMessage") +func fetchPortalDomainName() { var xmlPlistProfile map[string]interface{} - var eapType uint64 - var userCertDecode string - var templateToFile string - var elementsToReplaceInTemplate Template - var wifiKey string - var caFileBinary string - var wifiIndex int - var wiredIndex int - tempPath := os.Getenv("tmp") - if tempPath == "" { - walk.MsgBox(windowMsgBox, T("errorWindowTitle"), T("invalidTempPath"), walk.MsgBoxOK) - os.Exit(1) - } - profileFile := tempPath + "\\template-out.xml" // Download mobileconfig file err := writeProfileToLocalFile("profile.xml", PROFILE_URL) if err != nil { - walk.MsgBox(windowMsgBox, T("errorWindowTitle"), T("cannotRetrieveProfileFile"), walk.MsgBoxOK) - log.Fatal("Failed loading profile: ", err) - os.Exit(1) - } - - // Read xml profile, convert to string - data, err := ioutil.ReadFile("profile.xml") - if err != nil { - walk.MsgBox(windowMsgBox, T("errorWindowTitle"), T("cannotReadProfileData"), walk.MsgBoxOK) - os.Remove("profile.xml") - log.Fatal("Failed reading profile: ", err) + addNewLinesToDebug("Failed loading profile: " + err.Error()) + viewErrorAndExit(T("cannotRetrieveProfileFile")) + } else { + // Read xml profile, convert to string + data, err := ioutil.ReadFile("profile.xml") + if err != nil { + addNewLinesToDebug("Failed reading profile: " + err.Error()) + viewErrorAndExit(T("cannotReadProfileData")) + } else { + // Decode converted xml profile + dataToString := string(data) + buffer := bytes.NewReader([]byte(dataToString)) + decoder := plist.NewDecoder(buffer) + err = decoder.Decode(&xmlPlistProfile) + if err != nil { + addNewLinesToDebug("Failed decoding profile: " + err.Error()) + viewErrorAndExit(T("cannotDecodeProfileFile")) + } else { + fetchXML() + } + } } +} - // Decode converted xml profile - dataToString := string(data) - buffer := bytes.NewReader([]byte(dataToString)) - decoder := plist.NewDecoder(buffer) - err = decoder.Decode(&xmlPlistProfile) - if err != nil { - walk.MsgBox(windowMsgBox, T("errorWindowTitle"), T("cannotDecodeProfileFile"), walk.MsgBoxOK) - os.Remove("profile.xml") - log.Fatal("Failed decoding profile: ", err) - } +func fetchXML() { + var xmlPlistProfile map[string]interface{} + var wifiIndex int + var wiredIndex int + var eapType uint64 + var userCertDecode string + var caFileBinary string + var err error // Get data from the mobileconfig file shouldConfigureWifi := false @@ -192,6 +260,10 @@ func Configure() { eapClientConfiguration, ok := payloadContent["EAPClientConfiguration"].(map[string]interface{}) if ok { eapType = eapClientConfiguration["AcceptEAPTypes"].([]interface{})[0].(uint64) + addNewLinesToDebug("Extract Wireless configuration profile.") + } else { + addNewLinesToDebug("Failed Extract Wirless configuration profile") + viewErrorAndExit("Failed Extract Wireless configuration profile") } // Wired configuration case "com.apple.firstactiveethernet.managed": @@ -200,218 +272,254 @@ func Configure() { eapClientConfiguration, ok := payloadContent["EAPClientConfiguration"].(map[string]interface{}) if ok { eapType = eapClientConfiguration["AcceptEAPTypes"].([]interface{})[0].(uint64) + addNewLinesToDebug("Extract Wired configuration profile.") + } else { + addNewLinesToDebug("Failed Extract Wired configuration profile") + viewErrorAndExit("Failed Extract Wired configuration profile") } // User certificate configuration case "com.apple.security.pkcs12": userCert := payloadContent["PayloadContent"].(string) - userAuth := "certificate" - fileExtension := ".p12" - alertMessage := T("cannotGenerateCertificateFile") - userCertDecode, err = createCertTempFile(tempPath, userCert, userAuth, fileExtension, alertMessage) - err = addCertToMachine(userCertDecode, CERTUTIL_PROGRAM_PATH) + userCertPath = tempPath + "\\" + "certificate.p12" + err = createCertTempFile(userCert, userCertPath) if err != nil { - log.Fatal("Failed creating profile: ", err) - os.Exit(1) + addNewLinesToDebug("Failed Generating User Certificate : " + err.Error()) + viewErrorAndExit(T("cannotGenerateCertificateFile")) + } else { + err = addCertToMachine(userCertDecode, CERTUTIL_PROGRAM_PATH) + if err != nil { + addNewLinesToDebug("Failed creating profile: " + err.Error()) + viewErrorAndExit(T("cannotDecodeProfileFile")) + } } // Certificate of Authority configuration case "com.apple.security.root": if eapType == EAPTYPE_TLS { caName := payloadContent["PayloadCertificateFileName"].(string) caCert := payloadContent["PayloadContent"].(string) - fileExtension := ".cer" - alertMessage := T("cannotGenerateCAFile") - caFileBinary, err = createCertTempFile(tempPath, caCert, caName, fileExtension, alertMessage) + cafilePath = tempPath + "\\" + caName + ".cer" + err = createCertTempFile(caCert, cafilePath) + if err != nil { + addNewLinesToDebug("Failed Generating CA Certificate : " + err.Error()) + viewErrorAndExit(T("cannotGenerateCAFile")) + } err = addCAToMachine(caFileBinary, CERTUTIL_PROGRAM_PATH) if err != nil { - log.Fatal("Failed creating profile: ", err) - os.Exit(1) + addNewLinesToDebug("Failed creating profile: " + err.Error()) + viewErrorAndExit(T("cannotDecodeProfileFile")) } } default: - walk.MsgBox(windowMsgBox, T("errorWindowTitle"), T("Unexpected PayloadType {{.PayloadType}} please contact your local support.", map[string]interface{}{ + addNewLinesToDebug(T("Unexpected PayloadType: ", map[string]interface{}{ "PayloadType": payloadType, - }), walk.MsgBoxOK) - log.Fatal("Unexpected PayloadType: ", payloadType) - os.Exit(1) + })) + viewErrorAndExit(T("Unexpected PayloadType.")) } sum += i } - if shouldConfigureWifi { - // Get SSID information - payloadContent := xmlPlistProfile["PayloadContent"].([]interface{})[wifiIndex].(map[string]interface{}) - ssidString := payloadContent["SSID_STR"].(string) - ssidStringToHex := hex.EncodeToString([]byte(ssidString)) - ssidBroadcast := payloadContent["HIDDEN_NETWORK"].(bool) - securityType := payloadContent["EncryptionType"].(string) - if securityType == "None" { - securityType = "open" - } + configureWifi(xmlPlistProfile, wifiIndex, eapType, caFileBinary) + } + if shouldConfigureWired { + configureWired(xmlPlistProfile, wiredIndex, eapType) + } +} - addWLANProfileCommand := exec.Command("netsh", "wlan", "add", "profile", "filename="+profileFile, "user=all") - wlanSuccessMessage := T("The wireless profile was successfully added to the machine. \nPlease select your newly added profile {{.SsidString}} in the WiFi networks.", map[string]interface{}{ - "SsidString": ssidString, - }) +// Configure wifi +func configureWifi(xmlPlistProfile map[string]interface{}, wifiIndex int, eapType uint64, caFileBinary string) { + var WLAN_ERROR_MESSAGE = T("wlanErrorMessage") + var templateToFile string + var elementsToReplaceInTemplate Template + var wifiKey string + var err error + // Get SSID information + payloadContent := xmlPlistProfile["PayloadContent"].([]interface{})[wifiIndex].(map[string]interface{}) + ssidString := payloadContent["SSID_STR"].(string) + ssidStringToHex := hex.EncodeToString([]byte(ssidString)) + ssidBroadcast := payloadContent["HIDDEN_NETWORK"].(bool) + securityType := payloadContent["EncryptionType"].(string) + if securityType == "None" { + securityType = "open" + } + + wlanCmd := exec.Command("netsh", "wlan", "add", "profile", "filename="+profilePath, "user=all") + wlanCmdOutput := &bytes.Buffer{} + wlanCmd.Stdout = wlanCmdOutput + wlanCmdErr := wlanCmd.Run() + if wlanCmdErr != nil { + // There is an issue with the command line + addNewLinesToDebug(T("==> Executing: %s\n", strings.Join(wlanCmd.Args, " "))) + addNewLinesToDebug(T("==> Error: %s\n", wlanCmdErr.Error())) + addNewLinesToDebug(T("==> Ouput: %s\n", string(wlanCmdOutput.Bytes()))) + } + + wlanSuccessMessage := T("The wireless profile was successfully added to the machine. \nPlease select your newly added profile " + ssidString + " in the WiFi networks.") - // Security of the SSID - eapClientConfiguration, ok := payloadContent["EAPClientConfiguration"].(map[string]interface{}) + // Security of the SSID + eapClientConfiguration, ok := payloadContent["EAPClientConfiguration"].(map[string]interface{}) + if ok { + eapType = eapClientConfiguration["AcceptEAPTypes"].([]interface{})[0].(uint64) + userAuth, ok := eapClientConfiguration["UserName"].(string) if ok { - eapType = eapClientConfiguration["AcceptEAPTypes"].([]interface{})[0].(uint64) - userAuth, ok := eapClientConfiguration["UserName"].(string) - if ok { - if userAuth == "" { - userAuth = "certificate" - } - } else { + if userAuth == "" { userAuth = "certificate" } - eapType = eapClientConfiguration["AcceptEAPTypes"].([]interface{})[0].(uint64) - if eapType == EAPTYPE_PEAP { - // Search specific fields in wintemplate and replace them - elementsToReplaceInTemplate = Template{ - ProfileName: ssidString, - SsidStringToHex: ssidStringToHex, - IsSSIDBroadcast: ssidBroadcast, - SecAuth: "WPA2", - Encryption: "AES", - } - // executes the template - templateToFile, err := executeTemplate(WIFI_PEAP_TEMPLATE_NAME, WIFI_PEAP_TEMPLATE, elementsToReplaceInTemplate) - if err != nil { - log.Fatal("Failed executing template: ", err) - os.Exit(1) - } - // creates profile file with the executed template - err = createProfileFile(templateToFile) - if err != nil { - log.Fatal("Failed creating profile file: ", err) - os.Exit(1) - } - // adds the new profile to Windows with netsh command - addProfileToMachine(profileFile, addWLANProfileCommand, WLAN_ERROR_MESSAGE, wlanSuccessMessage) + } else { + userAuth = "certificate" + } + eapType = eapClientConfiguration["AcceptEAPTypes"].([]interface{})[0].(uint64) + if eapType == EAPTYPE_PEAP { + // Search specific fields in wintemplate and replace them + elementsToReplaceInTemplate = Template{ + ProfileName: ssidString, + SsidStringToHex: ssidStringToHex, + IsSSIDBroadcast: ssidBroadcast, + SecAuth: "WPA2", + Encryption: "AES", } - if eapType == EAPTYPE_TLS { - caFingerprint, err := getCAFingerprint(caFileBinary) - if err != nil { - os.Remove(caFileBinary) - os.Remove("profile.xml") - log.Fatal("Unable to get CA fingerprint: ", err) - } - elementsToReplaceInTemplate = Template{ - ProfileName: ssidString, - SsidStringToHex: ssidStringToHex, - IsSSIDBroadcast: ssidBroadcast, - SecAuth: "WPA2", - Encryption: "AES", - CaToTrust: caFingerprint, - } - os.Remove(caFileBinary) - templateToFile, err = executeTemplate(WIFI_TLS_TEMPLATE_NAME, WIFI_TLS_TEMPLATE, elementsToReplaceInTemplate) - if err != nil { - log.Fatal("Failed executing template: ", err) - os.Exit(1) - } - err = createProfileFile(templateToFile) - if err != nil { - log.Fatal("Failed creating profile file: ", err) - os.Exit(1) - } - addProfileToMachine(profileFile, addWLANProfileCommand, WLAN_ERROR_MESSAGE, wlanSuccessMessage) + // executes the template + templateToFile, err := executeTemplate(WIFI_PEAP_TEMPLATE_NAME, WIFI_PEAP_TEMPLATE, elementsToReplaceInTemplate) + if err != nil { + addNewLinesToDebug("Failed executing template: " + err.Error()) + viewErrorAndExit(T("Unexpected Error when executing the template.")) } - if (eapType != EAPTYPE_TLS) && (eapType != EAPTYPE_PEAP) { - // error handling - walk.MsgBox(windowMsgBox, T("errorWindowTitle"), T("unexpectedEAPType"), walk.MsgBoxOK) - log.Fatal("Incorrect EAP type: ", eapType) - os.Exit(1) + // creates profile file with the executed template + err = createProfileFile(templateToFile) + if err != nil { + addNewLinesToDebug("Failed creating profile file: " + err.Error()) + viewErrorAndExit(T("Unexpected Error when creating profile file.")) } - } else { - wifiKey = payloadContent["Password"].(string) - log.Println("Security type: ", securityType) - switch securityType { - case "WEP": - elementsToReplaceInTemplate = Template{ - ProfileName: ssidString, - SsidStringToHex: ssidStringToHex, - IsSSIDBroadcast: ssidBroadcast, - SecAuth: "open", - OpenPasscode: "passPhrase", - WifiKey: wifiKey, - Encryption: "WEP", - } - case "WPA": - elementsToReplaceInTemplate = Template{ - ProfileName: ssidString, - SsidStringToHex: ssidStringToHex, - IsSSIDBroadcast: ssidBroadcast, - SecAuth: "WPA2PSK", - OpenPasscode: "passPhrase", - WifiKey: wifiKey, - Encryption: "AES", - } - default: - elementsToReplaceInTemplate = Template{ - ProfileName: ssidString, - SsidStringToHex: ssidStringToHex, - IsSSIDBroadcast: ssidBroadcast, - SecAuth: "open", - OpenPasscode: "passPhrase", - WifiKey: wifiKey, - Encryption: "none", - } + // adds the new profile to Windows with netsh command + addProfileToMachine(profilePath, wlanCmd, WLAN_ERROR_MESSAGE, wlanSuccessMessage) + } else if eapType == EAPTYPE_TLS { + caFingerprint, err := getCAFingerprint(caFileBinary) + if err != nil { + addNewLinesToDebug("Unable to get CA fingerprint: " + err.Error()) + viewErrorAndExit(T("Unable to get CA fingerprint.")) + } + elementsToReplaceInTemplate = Template{ + ProfileName: ssidString, + SsidStringToHex: ssidStringToHex, + IsSSIDBroadcast: ssidBroadcast, + SecAuth: "WPA2", + Encryption: "AES", + CaToTrust: caFingerprint, } - templateToFile, err = executeTemplate(WIFI_OPEN_TEMPLATE_NAME, WIFI_OPEN_TEMPLATE, elementsToReplaceInTemplate) + templateToFile, err = executeTemplate(WIFI_TLS_TEMPLATE_NAME, WIFI_TLS_TEMPLATE, elementsToReplaceInTemplate) if err != nil { - log.Fatal("Failed executing template: ", err) - os.Exit(1) + addNewLinesToDebug("Failed executing template: " + err.Error()) + viewErrorAndExit(T("Failed executing template.")) } err = createProfileFile(templateToFile) if err != nil { - log.Fatal("Failed creating profile file: ", err) - os.Exit(1) + addNewLinesToDebug("Failed creating profile file: " + err.Error()) + viewErrorAndExit(T("Failed creating profile file.")) } - addProfileToMachine(profileFile, addWLANProfileCommand, WLAN_ERROR_MESSAGE, wlanSuccessMessage) + addProfileToMachine(profilePath, wlanCmd, WLAN_ERROR_MESSAGE, wlanSuccessMessage) + } else { + // error handling + addNewLinesToDebug(T("unexpectedEAPType") + fmt.Sprintf("%v", eapType)) + viewErrorAndExit(T("unexpectedEAPType")) } - } - if shouldConfigureWired { - dot3svc := exec.Command("net", "start", "dot3svc") - dot3svc.Start() - if err := dot3svc.Wait(); err != nil { - if exitErr, ok := err.(*exec.ExitError); ok { - if status, ok := exitErr.Sys().(syscall.WaitStatus); ok { - exitStatus := status.ExitStatus() - if exitStatus != 2 { - walk.MsgBox(windowMsgBox, T("errorWindowTitle"), T("dot3svcFail"), walk.MsgBoxOK) - log.Print("The Wired Autoconfig service could not be started.", err) - } - } + addNewLinesToDebug("EAPType is " + fmt.Sprintf("%v", eapType)) + } else { + wifiKey = payloadContent["Password"].(string) + addNewLinesToDebug("Security type: " + securityType) + switch securityType { + case "WEP": + elementsToReplaceInTemplate = Template{ + ProfileName: ssidString, + SsidStringToHex: ssidStringToHex, + IsSSIDBroadcast: ssidBroadcast, + SecAuth: "open", + OpenPasscode: "passPhrase", + WifiKey: wifiKey, + Encryption: "WEP", + } + case "WPA": + elementsToReplaceInTemplate = Template{ + ProfileName: ssidString, + SsidStringToHex: ssidStringToHex, + IsSSIDBroadcast: ssidBroadcast, + SecAuth: "WPA2PSK", + OpenPasscode: "passPhrase", + WifiKey: wifiKey, + Encryption: "AES", + } + default: + elementsToReplaceInTemplate = Template{ + ProfileName: ssidString, + SsidStringToHex: ssidStringToHex, + IsSSIDBroadcast: ssidBroadcast, + SecAuth: "open", + OpenPasscode: "passPhrase", + WifiKey: wifiKey, + Encryption: "none", } } - wiredNetshCommand := exec.Command("netsh", "lan", "add", "profile", "filename="+profileFile) - payloadContent := xmlPlistProfile["PayloadContent"].([]interface{})[wiredIndex].(map[string]interface{}) - eapClientConfiguration := payloadContent["EAPClientConfiguration"].(map[string]interface{}) - eapType = eapClientConfiguration["AcceptEAPTypes"].([]interface{})[0].(uint64) - if eapType == EAPTYPE_PEAP { - err = createProfileFile(WIRED_PEAP_TEMPLATE) - addProfileToMachine(profileFile, wiredNetshCommand, WIRED_ERROR_MESSAGE, WIRED_SUCCESS_MESSAGE) + templateToFile, err = executeTemplate(WIFI_OPEN_TEMPLATE_NAME, WIFI_OPEN_TEMPLATE, elementsToReplaceInTemplate) + if err != nil { + addNewLinesToDebug("Failed executing template: " + err.Error()) + viewErrorAndExit("Failed executing template.") + } else { + err = createProfileFile(templateToFile) if err != nil { - log.Fatal("Failed creating profile file: ", err) - os.Exit(1) + addNewLinesToDebug("Failed creating template: " + err.Error()) + viewErrorAndExit("Failed creating template.") + } else { + addProfileToMachine(profilePath, wlanCmd, WLAN_ERROR_MESSAGE, wlanSuccessMessage) } } - if eapType == EAPTYPE_TLS { - err = createProfileFile(WIRED_TLS_TEMPLATE) - addProfileToMachine(profileFile, wiredNetshCommand, WIRED_ERROR_MESSAGE, WIRED_SUCCESS_MESSAGE) - if err != nil { - log.Fatal("Failed creating profile file: ", err) - os.Exit(1) + } +} + +// Configuration for wired +func configureWired(xmlPlistProfile map[string]interface{}, wiredIndex int, eapType uint64) { + var WIRED_ERROR_MESSAGE = T("wiredErrorMessage") + var WIRED_SUCCESS_MESSAGE = T("wiredSuccessMessage") + var err error + + dot3svc := exec.Command("net", "start", "dot3svc") + dot3svc.Start() + if err := dot3svc.Wait(); err != nil { + if exitErr, ok := err.(*exec.ExitError); ok { + if status, ok := exitErr.Sys().(syscall.WaitStatus); ok { + exitStatus := status.ExitStatus() + if exitStatus != 2 { + addNewLinesToDebug("The Wired Autoconfig service could not be started due to: " + err.Error()) + viewErrorAndExit(T("dot3svcFail")) + } else { + addNewLinesToDebug("The Wired Autoconfig service has been started") + } } } - if (eapType != EAPTYPE_TLS) && (eapType != EAPTYPE_PEAP) { - // error handling - walk.MsgBox(windowMsgBox, T("errorWindowTitle"), T("unexpectedEAPType"), walk.MsgBoxOK) - log.Fatal("Incorrect EAP type: ", eapType) - os.Exit(1) + } + wiredNetshCommand := exec.Command("netsh", "lan", "add", "profile", "filename="+profilePath) + payloadContent := xmlPlistProfile["PayloadContent"].([]interface{})[wiredIndex].(map[string]interface{}) + eapClientConfiguration := payloadContent["EAPClientConfiguration"].(map[string]interface{}) + eapType = eapClientConfiguration["AcceptEAPTypes"].([]interface{})[0].(uint64) + if eapType == EAPTYPE_PEAP { + err = createProfileFile(WIRED_PEAP_TEMPLATE) + addProfileToMachine(profilePath, wiredNetshCommand, WIRED_ERROR_MESSAGE, WIRED_SUCCESS_MESSAGE) + if err != nil { + addNewLinesToDebug("Failed creating profile file: " + err.Error()) + viewErrorAndExit("Failed creating profile file.") + } else { + addNewLinesToDebug("Success creating profile file: " + err.Error()) + } + } else if eapType == EAPTYPE_TLS { + err = createProfileFile(WIRED_TLS_TEMPLATE) + addProfileToMachine(profilePath, wiredNetshCommand, WIRED_ERROR_MESSAGE, WIRED_SUCCESS_MESSAGE) + if err != nil { + addNewLinesToDebug("Failed creating profile file: " + err.Error()) + viewErrorAndExit("Failed creating profile file.") + } else { + addNewLinesToDebug("Success creating profile file: " + err.Error()) } + } else { + // error handling + addNewLinesToDebug(T("unexpectedEAPType") + err.Error()) + viewErrorAndExit(T("unexpectedEAPType")) } } @@ -445,59 +553,45 @@ func writeProfileToLocalFile(filepath string, url string) error { // Create, parse and execute templates func executeTemplate(nameTemplate, constTemplate string, templateToApply Template) (string, error) { newTemplate := template.New(nameTemplate) + var templateBuffer bytes.Buffer // parses template newTemplate, err := newTemplate.Parse(constTemplate) if err != nil { - walk.MsgBox(windowMsgBox, T("errorWindowTitle"), T("cannotParseTemplate"), walk.MsgBoxOK) - os.Remove("profile.xml") - log.Fatal("Failed parsing: ", err) - return "", err - } - // executes the template into the open file - var templateBuffer bytes.Buffer - err = newTemplate.Execute(&templateBuffer, templateToApply) - if err != nil { - log.Println("Error: ", err) - walk.MsgBox(windowMsgBox, T("errorWindowTitle"), T("cannotExecuteTemplate"), walk.MsgBoxOK) - os.Remove("profile.xml") - log.Fatal("Failed executing: ", err) - return templateBuffer.String(), err - } - // handles error - if err != nil { - walk.MsgBox(windowMsgBox, T("errorWindowTitle"), T("cannotCreateWLANProfile"), walk.MsgBoxOK) - os.Remove("profile.xml") - log.Fatal("Failed creating WLANProfile: ", err) + addNewLinesToDebug(T("cannotParseTemplate") + err.Error()) + viewErrorAndExit(T("cannotParseTemplate")) + } else { + // executes the template into the open file + err = newTemplate.Execute(&templateBuffer, templateToApply) + if err != nil { + addNewLinesToDebug(T("cannotExecuteTemplate") + err.Error()) + viewErrorAndExit(T("cannotExecuteTemplate")) + return "", err + } return templateBuffer.String(), err } - return templateBuffer.String(), nil + return "", nil } // Create and write profile file into templateToFile folder func createProfileFile(templateToFile string) error { - tempPath := os.Getenv("tmp") // create and open file - profileFilePath := tempPath + "\\" + "template-out.xml" - profileFile, err := os.Create(profileFilePath) + profileFile, err := os.Create(profilePath) if err != nil { - walk.MsgBox(windowMsgBox, T("errorWindowTitle"), T("cannotCreateProfileFile"), walk.MsgBoxOK) - os.Remove("profile.xml") - log.Fatal("Failed creating profile file: ", err) - return err - } - // close file - defer profileFile.Close() - // write the template into the new file - _, err = io.Copy(profileFile, strings.NewReader(templateToFile)) - if err != nil { - walk.MsgBox(windowMsgBox, T("errorWindowTitle"), T("cannotWriteIntoProfileFile"), walk.MsgBoxOK) - os.Remove("profile.xml") - os.Remove(profileFilePath) - log.Fatal("Failed writing template to file: ", err) + addNewLinesToDebug(T("cannotCreateProfileFile") + err.Error()) + viewErrorAndExit(T("cannotCreateProfileFile")) return err + } else { + // close file + defer profileFile.Close() + // write the template into the new file + _, err = io.Copy(profileFile, strings.NewReader(templateToFile)) + if err != nil { + addNewLinesToDebug(T("cannotWriteIntoProfileFile") + err.Error()) + viewErrorAndExit(T("cannotWriteIntoProfileFile")) + return err + } } - os.Remove("profile.xml") - log.Println("Information:", T("profileCreationSuccess")) + addNewLinesToDebug(T("profileCreationSuccess") + err.Error()) return nil } @@ -505,14 +599,11 @@ func createProfileFile(templateToFile string) error { func addProfileToMachine(profileFile string, cmd *exec.Cmd, ErrorMessage, SuccessMessage string) error { output, err := cmd.CombinedOutput() if err != nil { - log.Printf("Failed adding profile: output: %s\n", output, err) - walk.MsgBox(windowMsgBox, T("errorWindowTitle"), ErrorMessage, walk.MsgBoxOK) - os.Remove(profileFile) - log.Fatal("Failed adding profile: ", err, output) + addNewLinesToDebug("Failed adding profile" + ErrorMessage + err.Error() + fmt.Sprintf("%v", output)) + viewErrorAndExit("Failed adding profile") return err } else { - walk.MsgBox(windowMsgBox, "Information:", SuccessMessage, walk.MsgBoxOK) - os.Remove(profileFile) + addNewLinesToDebug("Failed adding profile" + SuccessMessage) + return nil } - return nil } diff --git a/utils.go b/utils.go index 9a81add..0283b44 100644 --- a/utils.go +++ b/utils.go @@ -11,7 +11,6 @@ import ( "image/png" "github.com/lxn/walk" - "github.com/tink-ab/tempfile" ) // Create and write profile file into templateToFile folder @@ -20,7 +19,7 @@ func createLanguageFile(currentDir, translationLanguage, languageFileName string languageFile, err := os.Create(currentDir + "\\" + languageFileName) if err != nil { walk.MsgBox(windowMsgBox, "Error", "Unable to create the language file, please contact your local support.", walk.MsgBoxOK) - log.Fatal("Failed creating language file: ", err) + log.Print("Failed creating language file: ", err) return err } // close file @@ -29,37 +28,156 @@ func createLanguageFile(currentDir, translationLanguage, languageFileName string _, err = io.Copy(languageFile, strings.NewReader(translationLanguage)) if err != nil { walk.MsgBox(windowMsgBox, "Error", "Unable to write into language file, please contact your local support.", walk.MsgBoxOK) - log.Fatal("Failed writing language constant to file: ", err) + log.Print("Failed writing language constant to file: ", err) return err } log.Print("Language file successfully created.") return nil } +func addNewLinesToDebug(mytxt string) { + if debug { + log.Print(mytxt) + myTxtC := chunks(mytxt, 60) + myCt := debugTxt.Text() + debugTxt.SetText(myCt + myTxtC + "\r\n") + numlines := strings.Count(debugTxt.Text(), "\r\n") + debugHeight := (13 * numlines) + 13 + if debugHeight <= 240 { + debugHeight = 240 + } + debugTxtsize := walk.Size{400, debugHeight} + debugTxt.SetMinMaxSize(walk.Size(debugTxtsize), walk.Size(debugTxtsize)) + } else { + log.Print(mytxt) + } +} + +// Chunk large string at x charaters +// Source: https://stackoverflow.com/a/61469854 +func chunks(s string, chunkSize int) string { + if len(s) == 0 { + return "" + } + if chunkSize >= len(s) { + return s + } + var chunks []string = make([]string, 0, (len(s)-1)/chunkSize+1) + currentLen := 0 + currentStart := 0 + for i := range s { + if currentLen == chunkSize { + chunks = append(chunks, s[currentStart:i]) + currentLen = 0 + currentStart = i + } + currentLen++ + } + chunks = append(chunks, s[currentStart:]) + justString := strings.Join(chunks, "\r\n") + return justString +} + +// Enable the debug view +func viewDebug() { + if debug { + debug = false + // Change view + scb, _ := walk.NewSolidColorBrush(walk.RGB(4, 5, 3)) + mw1.SetBackground(scb) + debugGrpBox.SetVisible(false) + imgView.SetVisible(true) + configButton.SetText("Configure") + } else { + debug = true + // Change view + scb, _ := walk.NewSolidColorBrush(walk.RGB(255, 255, 255)) + mw1.SetBackground(scb) + debugGrpBox.SetVisible(true) + imgView.SetVisible(false) + viewClosedButton(true) + configButton.SetText("Configure with debug") + } +} + +// Enable Close button view +func viewClosedButton(b bool) { + closedButton.SetVisible(b) +} + +// View Error +func viewErrorAndExit(s string) bool { + if !debug { + walk.MsgBox(windowMsgBox, T("errorWindowTitle"), s+"\r\nPlease enable Debug Mode and contact your local support.", walk.MsgBoxOK) + cleanAndExit() + } + return true +} + +func cleanAndExit() { + cleanTmpFiles() + mw1.Close() + os.Exit(1) +} + +func cleanTmpFiles() { + os.Remove(pngFilePath) + os.Remove(tempPath + "\\" + "template-out.xml") + os.Remove(profilePath) + os.Remove(cafilePath) +} + +// Prepare Background image // Converts base 64 background image to pf_bg.png -func base64ToPng(BACKGROUND_IMAGE_PF, tempPath string) (error, string) { +func prepareBackgroundImage() { + debug = true reader := base64.NewDecoder(base64.StdEncoding, strings.NewReader(BACKGROUND_IMAGE_PF)) decodeBase64ToPng, _, err := image.Decode(reader) + addNewLinesToDebug("Welcome to PF debug") if err != nil { - log.Fatal("Unable to decode base 64 background image: ", err) + addNewLinesToDebug("Unable to decode base 64 background image: " + err.Error()) + viewErrorAndExit("Unable to decode base 64 background image.") + } else { + //Encode from image format to writer + backgroundFile, err := os.Create(pngFilePath) + if err != nil { + addNewLinesToDebug("Unable to open or create background image: " + err.Error()) + viewErrorAndExit("Unable to open or create background image.") + } else { + err = png.Encode(backgroundFile, decodeBase64ToPng) + if err != nil { + addNewLinesToDebug("Unable to encode background image: " + err.Error()) + viewErrorAndExit("Unable to encode background image.") + } else { + addNewLinesToDebug("PNG file " + pngFileName + " successfully created at " + pngFilePath) + backgroundFile.Close() + } + } } - //Encode from image format to writer - pngFilename := "pf_bg.png" - pngFilePath := tempPath + "\\" + pngFilename - backgroundFile, err := os.Create(pngFilePath) - if err != nil { - log.Fatal("Unable to open or create background image: ", err) - return err, pngFilename + var img walk.Image + img, err = walk.NewImageFromFile(pngFilePath) + if img != nil { + if err := imgView.SetImage(img.(walk.Image)); err != nil { + addNewLinesToDebug("Unable to attach the background image: " + err.Error()) + viewErrorAndExit("Unable to attach background image.") + } else { + addNewLinesToDebug("Been able to attach background image.") + } + } else { + addNewLinesToDebug("Unable to grab background image") } - err = png.Encode(backgroundFile, decodeBase64ToPng) + debug = false +} + +// Create a filee +func createFile(filepath string) (*os.File, error) { + f, err := os.Create(filepath) if err != nil { - log.Fatal(err) - os.Remove(pngFilePath) - return err, pngFilename + addNewLinesToDebug("Unable to create file " + filepath + ": " + err.Error()) + viewErrorAndExit("Unable to create file " + filepath + ".") } - log.Println("PNG file", pngFilename, "successfully created.") - backgroundFile.Close() - return nil, pngFilename + defer f.Close() + return f, err } // Decode base64 certificate to string @@ -72,41 +190,32 @@ func decodeCertificate(certificate string) ([]byte, error) { } // Create certificate templateToFile files -func createCertTempFile(tempPath, certificate, fileName, fileExtension, alertMessage string) (string, error) { +func createCertTempFile(certificate, filePath string) error { // creates and opens new templateToFile file in directory - file, err := tempfile.TempFile(tempPath, fileName, fileExtension) + myFile, err := createFile(filePath) if err != nil { - walk.MsgBox(windowMsgBox, T("errorWindowTitle"), T("cannotCreateCertTempFile"), walk.MsgBoxOK) - // clean up - os.Remove("profile.xml") - log.Fatal("Failed creating temp file: ", err) - return file.Name(), err + addNewLinesToDebug("Failed creating temp file: " + err.Error()) + return err } - certName := file.Name() // write certificate into file decodedCertificate, err := decodeCertificate(certificate) if err != nil { - // handle error, exit if needed - walk.MsgBox(windowMsgBox, T("errorWindowTitle"), T("cannotdecodeCertificateFile"), walk.MsgBoxOK) - os.Remove("profile.xml") - log.Fatal("Failed decoding certificate: ", err) - os.Exit(1) - return certName, err - } - // write into new file - if _, err := file.Write(decodedCertificate); err != nil { - walk.MsgBox(windowMsgBox, T("errorWindowTitle"), T("cannotWriteIntoTempFile"), walk.MsgBoxOK) - os.Remove("profile.xml") - log.Fatal("Failed writing decoded certificate into temp file: ", err) - return certName, err - } - if err := file.Close(); err != nil { - walk.MsgBox(windowMsgBox, T("errorWindowTitle"), alertMessage, walk.MsgBoxOK) - os.Remove("profile.xml") - log.Fatal("Failed closing certificate file: ", err) - return certName, err - } - return certName, nil + addNewLinesToDebug("Failed decoding certificate: " + err.Error()) + return err + } + return writeInFIle(myFile, "", decodedCertificate) +} + +func writeInFIle(myFile *os.File, st string, b []byte) error { + if _, err := myFile.Write(b); err != nil { + addNewLinesToDebug("Failed writing decoded certificate into temp file: " + err.Error()) + return err + } + if err := myFile.Close(); err != nil { + addNewLinesToDebug("Failed closing certificate file: " + err.Error()) + return err + } + return nil } const BACKGROUND_IMAGE_PF = `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` From ce962a626576c4ad18fef1242edd32fa5192e3ca Mon Sep 17 00:00:00 2001 From: JeGoi <13801368+JeGoi@users.noreply.github.com> Date: Fri, 1 Oct 2021 16:31:16 -0400 Subject: [PATCH 02/18] Add debug in certificate --- certificate.go | 82 ++++++++++++++++++++++++-------------------------- 1 file changed, 39 insertions(+), 43 deletions(-) diff --git a/certificate.go b/certificate.go index 7c771f1..f7bd6f0 100644 --- a/certificate.go +++ b/certificate.go @@ -2,7 +2,6 @@ package main import ( "io" - "log" "os" "strings" "syscall" @@ -22,30 +21,33 @@ func getCAFingerprint(caFileBinary string) (string, error) { // open certificate of authority binary file caFile, err := os.Open(caFileBinary) if err != nil { - walk.MsgBox(windowMsgBox, T("errorWindowTitle"), T("cannotOpenCAFile"), walk.MsgBoxOK) - log.Print("Failed opening CA file: ", err) + addNewLinesToDebug(T("cannotOpenCAFile") + err.Error()) + viewErrorAndExit(T("cannotOpenCAFile")) + defer caFile.Close() return "", err - } - // close file - defer caFile.Close() + } else { + // close file + defer caFile.Close() - // create new hash - hashSha1 := sha1.New() - // copy hash to the file - if _, err := io.Copy(hashSha1, caFile); err != nil { - walk.MsgBox(windowMsgBox, T("errorWindowTitle"), T("cannotCopyCAFile"), walk.MsgBoxOK) - log.Print("Failed copying CA file: ", err) - return "", err - } - // returns sha1 checksum of the data - caFingerprintBytes := hashSha1.Sum(nil) - // convert sha1 to hex (base16) to string - caFingerprint = strings.ToLower(hex.EncodeToString(caFingerprintBytes)) - // add spaces every two characters - for i := 2; i < len(caFingerprint); i += 3 { - caFingerprint = caFingerprint[:i] + " " + caFingerprint[i:] + // create new hash + hashSha1 := sha1.New() + // copy hash to the file + if _, err := io.Copy(hashSha1, caFile); err != nil { + addNewLinesToDebug(T("cannotCopyCAFile") + err.Error()) + viewErrorAndExit(T("cannotCopyCAFile")) + return "", err + } else { + // returns sha1 checksum of the data + caFingerprintBytes := hashSha1.Sum(nil) + // convert sha1 to hex (base16) to string + caFingerprint = strings.ToLower(hex.EncodeToString(caFingerprintBytes)) + // add spaces every two characters + for i := 2; i < len(caFingerprint); i += 3 { + caFingerprint = caFingerprint[:i] + " " + caFingerprint[i:] + } + return caFingerprint, nil + } } - return caFingerprint, nil } // Add cert to windows @@ -90,33 +92,27 @@ func addCertToMachine(userCertDecode string, CERTUTIL_PROGRAM_PATH string) error if exitErr, ok := err.(*exec.ExitError); ok { if status, ok := exitErr.Sys().(syscall.WaitStatus); ok { exitStatus := status.ExitStatus() - log.Print("Exit Status: ", exitStatus) + addNewLinesToDebug("Exit Status: " + string(exitStatus)) switch exitStatus { case int(ERROR_INVALID_PASSWORD): - walk.MsgBox(windowMsgBox, T("errorWindowTitle"), T("wrongPassword"), walk.MsgBoxOK) + addNewLinesToDebug(T("wrongPassword")) + viewErrorAndExit(T("wrongPassword")) badCertificatePassword = true mw.Close() case int(ERROR_INVALID_DATA): - walk.MsgBox(windowMsgBox, T("errorWindowTitle"), T("invalidCertificate"), walk.MsgBoxOK) - os.Remove(userCertDecode) - os.Remove("profile.xml") - log.Print("Invalid certificate: ", exitStatus) + addNewLinesToDebug(T("invalidCertificate")) + viewErrorAndExit(T("invalidCertificate")) case int(ERROR_FILE_NOT_FOUND): - walk.MsgBox(windowMsgBox, T("errorWindowTitle"), T("cannotFindCertificateFile"), walk.MsgBoxOK) - os.Remove(userCertDecode) - os.Remove("profile.xml") - log.Print("Certificate not found: ", exitStatus) + addNewLinesToDebug(T("cannotFindCertificateFile") + string(exitStatus)) + viewErrorAndExit(T("cannotFindCertificateFile")) default: - walk.MsgBox(windowMsgBox, T("errorWindowTitle"), T("cannotInstallCertificate"), walk.MsgBoxOK) - os.Remove(userCertDecode) - os.Remove("profile.xml") - log.Print("Cannot install certificate: ", exitStatus) + addNewLinesToDebug(T("cannotInstallCertificate") + string(exitStatus)) + viewErrorAndExit(T("cannotInstallCertificate")) } } } } else { - log.Println(T("successWindowTitle"), T("certificateInstallationSuccess")) - os.Remove(userCertDecode) + addNewLinesToDebug(T("certificateInstallationSuccess")) } mw.Close() }, @@ -147,19 +143,19 @@ func addCAToMachine(caFileBinary string, CERTUTIL_PROGRAM_PATH string) error { // reprompt user to add certificate to windows retryOrCancel := walk.MsgBox(windowMsgBox, T("errorWindowTitle"), T("caErrorCanceled"), walk.MsgBoxRetryCancel) if retryOrCancel == 4 { - log.Print("Failed installing certificate: ", err) + addNewLinesToDebug("Failed installing certificate: " + err.Error()) runCommand = true } else { - log.Print("Failed installing certificate: ", err) + addNewLinesToDebug("Failed installing certificate: " + err.Error()) } } else { - walk.MsgBox(windowMsgBox, T("errorWindowTitle"), T("cannotInstallCA"), walk.MsgBoxOK) - log.Print("Failed installing certificate: ", err) + addNewLinesToDebug(T("cannotInstallCA") + err.Error()) + viewErrorAndExit(T("cannotInstallCA")) } } } } else { - log.Println(T("successWindowTitle"), T("caInstallationSuccess")) + addNewLinesToDebug(T("caInstallationSuccess") + err.Error()) } } return err From 204a55f87b5ee5bb12c05052e06eeb77f8f4c174 Mon Sep 17 00:00:00 2001 From: JeGoi <13801368+JeGoi@users.noreply.github.com> Date: Fri, 1 Oct 2021 16:40:18 -0400 Subject: [PATCH 03/18] Add more debug in utils --- utils.go | 29 ++++++++++++++++------------- 1 file changed, 16 insertions(+), 13 deletions(-) diff --git a/utils.go b/utils.go index 0283b44..aefb9a5 100644 --- a/utils.go +++ b/utils.go @@ -18,21 +18,23 @@ func createLanguageFile(currentDir, translationLanguage, languageFileName string // create and open file languageFile, err := os.Create(currentDir + "\\" + languageFileName) if err != nil { - walk.MsgBox(windowMsgBox, "Error", "Unable to create the language file, please contact your local support.", walk.MsgBoxOK) - log.Print("Failed creating language file: ", err) - return err - } - // close file - defer languageFile.Close() - // write the template into the new file - _, err = io.Copy(languageFile, strings.NewReader(translationLanguage)) - if err != nil { - walk.MsgBox(windowMsgBox, "Error", "Unable to write into language file, please contact your local support.", walk.MsgBoxOK) - log.Print("Failed writing language constant to file: ", err) + addNewLinesToDebug("Unable to create the language file:" + err.Error()) + viewErrorAndExit("Unable to create the language file.") return err + } else { + // close file + defer languageFile.Close() + // write the template into the new file + _, err = io.Copy(languageFile, strings.NewReader(translationLanguage)) + if err != nil { + addNewLinesToDebug("Unable to write into language file:" + err.Error()) + viewErrorAndExit("Unable to write into language file.") + return err + } else { + addNewLinesToDebug("Language file successfully created.") + return nil + } } - log.Print("Language file successfully created.") - return nil } func addNewLinesToDebug(mytxt string) { @@ -125,6 +127,7 @@ func cleanTmpFiles() { os.Remove(tempPath + "\\" + "template-out.xml") os.Remove(profilePath) os.Remove(cafilePath) + os.Remove(userCertPath) } // Prepare Background image From 978f0ee32cd848419c85d12fccec9dc9161c71d7 Mon Sep 17 00:00:00 2001 From: JeGoi <13801368+JeGoi@users.noreply.github.com> Date: Mon, 4 Oct 2021 12:06:59 -0400 Subject: [PATCH 04/18] Fix warnings, Merge exit in simple function --- certificate.go | 29 +++++-------- parse.go | 114 +++++++++++++++++-------------------------------- utils.go | 35 +++++++-------- 3 files changed, 66 insertions(+), 112 deletions(-) diff --git a/certificate.go b/certificate.go index f7bd6f0..4be1976 100644 --- a/certificate.go +++ b/certificate.go @@ -1,6 +1,7 @@ package main import ( + "fmt" "io" "os" "strings" @@ -21,8 +22,7 @@ func getCAFingerprint(caFileBinary string) (string, error) { // open certificate of authority binary file caFile, err := os.Open(caFileBinary) if err != nil { - addNewLinesToDebug(T("cannotOpenCAFile") + err.Error()) - viewErrorAndExit(T("cannotOpenCAFile")) + viewErrorAndExit(T("cannotOpenCAFile"), err.Error()) defer caFile.Close() return "", err } else { @@ -33,8 +33,7 @@ func getCAFingerprint(caFileBinary string) (string, error) { hashSha1 := sha1.New() // copy hash to the file if _, err := io.Copy(hashSha1, caFile); err != nil { - addNewLinesToDebug(T("cannotCopyCAFile") + err.Error()) - viewErrorAndExit(T("cannotCopyCAFile")) + viewErrorAndExit(T("cannotCopyCAFile"), err.Error()) return "", err } else { // returns sha1 checksum of the data @@ -72,7 +71,7 @@ func addCertToMachine(userCertDecode string, CERTUTIL_PROGRAM_PATH string) error MainWindow{ AssignTo: &mw, Title: PROGRAM_NAME, - MinSize: Size{350, 100}, + MinSize: Size{Width: 350, Height: 100}, Layout: VBox{}, Children: []Widget{ Label{Text: T("enterCertificatePassword")}, @@ -92,22 +91,18 @@ func addCertToMachine(userCertDecode string, CERTUTIL_PROGRAM_PATH string) error if exitErr, ok := err.(*exec.ExitError); ok { if status, ok := exitErr.Sys().(syscall.WaitStatus); ok { exitStatus := status.ExitStatus() - addNewLinesToDebug("Exit Status: " + string(exitStatus)) + addNewLinesToDebug("Exit Status: " + fmt.Sprint(exitStatus)) switch exitStatus { case int(ERROR_INVALID_PASSWORD): - addNewLinesToDebug(T("wrongPassword")) - viewErrorAndExit(T("wrongPassword")) + viewErrorAndExit(T("wrongPassword"), "") badCertificatePassword = true mw.Close() case int(ERROR_INVALID_DATA): - addNewLinesToDebug(T("invalidCertificate")) - viewErrorAndExit(T("invalidCertificate")) + viewErrorAndExit(T("invalidCertificate"), "") case int(ERROR_FILE_NOT_FOUND): - addNewLinesToDebug(T("cannotFindCertificateFile") + string(exitStatus)) - viewErrorAndExit(T("cannotFindCertificateFile")) + viewErrorAndExit(T("cannotFindCertificateFile"), " => "+fmt.Sprint(exitStatus)) default: - addNewLinesToDebug(T("cannotInstallCertificate") + string(exitStatus)) - viewErrorAndExit(T("cannotInstallCertificate")) + viewErrorAndExit(T("cannotInstallCertificate"), " => "+fmt.Sprint(exitStatus)) } } } @@ -127,8 +122,7 @@ func addCertToMachine(userCertDecode string, CERTUTIL_PROGRAM_PATH string) error // Add CA to the machine func addCAToMachine(caFileBinary string, CERTUTIL_PROGRAM_PATH string) error { var err error - var ERROR_CANCELED int64 - ERROR_CANCELED = 2147943623 + ERROR_CANCELED := 2147943623 runCommand := true for runCommand { runCommand = false @@ -149,8 +143,7 @@ func addCAToMachine(caFileBinary string, CERTUTIL_PROGRAM_PATH string) error { addNewLinesToDebug("Failed installing certificate: " + err.Error()) } } else { - addNewLinesToDebug(T("cannotInstallCA") + err.Error()) - viewErrorAndExit(T("cannotInstallCA")) + viewErrorAndExit(T("cannotInstallCA"), err.Error()) } } } diff --git a/parse.go b/parse.go index 6113490..db1b7b3 100644 --- a/parse.go +++ b/parse.go @@ -82,7 +82,7 @@ func main() { if err := (MainWindow{ AssignTo: &mw1, Title: fmt.Sprintf("%s - %s", PROGRAM_NAME, VERSION), - MinSize: Size{500, 400}, + MinSize: Size{Width: 500, Height: 400}, Layout: VBox{}, Background: SolidColorBrush{Color: walk.RGB(4, 5, 3)}, Children: []Widget{ @@ -103,7 +103,7 @@ func main() { Layout: VBox{}, Children: []Widget{ TextEdit{ - MinSize: Size{400, 10}, + MinSize: Size{Width: 400, Height: 10}, AssignTo: &debugTxt, ReadOnly: false, Text: "", @@ -115,7 +115,7 @@ func main() { PushButton{ AssignTo: &configButton, Background: SolidColorBrush{Color: walk.RGB(4, 5, 3)}, - MinSize: Size{50, 50}, + MinSize: Size{Width: 50, Height: 50}, Text: "Configure", OnClicked: func() { fetchPortalDomainName() @@ -124,7 +124,7 @@ func main() { PushButton{ AssignTo: &debugButton, Background: SolidColorBrush{Color: walk.RGB(4, 5, 3)}, - MinSize: Size{10, 10}, + MinSize: Size{Width: 10, Height: 10}, Text: "Debug", OnClicked: func() { viewDebug() @@ -133,7 +133,7 @@ func main() { PushButton{ AssignTo: &closedButton, Background: SolidColorBrush{Color: walk.RGB(4, 5, 3)}, - MinSize: Size{10, 10}, + MinSize: Size{Width: 10, Height: 51}, Text: "Close", Visible: false, OnClicked: func() { @@ -143,12 +143,12 @@ func main() { }, }, OnSizeChanged: func() { - mw1size := Size{500, 400} + mw1size := Size{Width: 500, Height: 400} mw1.SetSize(walk.Size(mw1size)) }, }.Create()); err != nil { - log.Print("Failed opening main window: ", err) - viewErrorAndExit(T("errorMainWindow: " + err.Error())) + log.Print("Failed opening main window: ", err.Error()) + viewErrorAndExit(T("errorMainWindow: ")+""+err.Error(), "") } prepareEnv() prepareBackgroundImage() @@ -165,7 +165,7 @@ func prepareMainWindow() { tempPath = os.Getenv("tmp") // Access to tmp path if tempPath == "" { - viewErrorAndExit(T("invalidTempPath")) + viewErrorAndExit(T("invalidTempPath"), "") } walk.Resources.SetRootDirPath(tempPath) } @@ -205,14 +205,12 @@ func fetchPortalDomainName() { // Download mobileconfig file err := writeProfileToLocalFile("profile.xml", PROFILE_URL) if err != nil { - addNewLinesToDebug("Failed loading profile: " + err.Error()) - viewErrorAndExit(T("cannotRetrieveProfileFile")) + viewErrorAndExit(T("cannotRetrieveProfileFile"), err.Error()) } else { // Read xml profile, convert to string data, err := ioutil.ReadFile("profile.xml") if err != nil { - addNewLinesToDebug("Failed reading profile: " + err.Error()) - viewErrorAndExit(T("cannotReadProfileData")) + viewErrorAndExit(T("cannotReadProfileData"), err.Error()) } else { // Decode converted xml profile dataToString := string(data) @@ -220,8 +218,7 @@ func fetchPortalDomainName() { decoder := plist.NewDecoder(buffer) err = decoder.Decode(&xmlPlistProfile) if err != nil { - addNewLinesToDebug("Failed decoding profile: " + err.Error()) - viewErrorAndExit(T("cannotDecodeProfileFile")) + viewErrorAndExit(T("cannotDecodeProfileFile"), err.Error()) } else { fetchXML() } @@ -260,10 +257,9 @@ func fetchXML() { eapClientConfiguration, ok := payloadContent["EAPClientConfiguration"].(map[string]interface{}) if ok { eapType = eapClientConfiguration["AcceptEAPTypes"].([]interface{})[0].(uint64) - addNewLinesToDebug("Extract Wireless configuration profile.") + addNewLinesToDebug("Extract Wireless configuration profile: " + fmt.Sprint(eapType)) } else { - addNewLinesToDebug("Failed Extract Wirless configuration profile") - viewErrorAndExit("Failed Extract Wireless configuration profile") + viewErrorAndExit("Failed Extract Wireless configuration profile", fmt.Sprint(eapType)) } // Wired configuration case "com.apple.firstactiveethernet.managed": @@ -272,10 +268,9 @@ func fetchXML() { eapClientConfiguration, ok := payloadContent["EAPClientConfiguration"].(map[string]interface{}) if ok { eapType = eapClientConfiguration["AcceptEAPTypes"].([]interface{})[0].(uint64) - addNewLinesToDebug("Extract Wired configuration profile.") + addNewLinesToDebug("Extract Wired configuration profile: " + fmt.Sprint(eapType)) } else { - addNewLinesToDebug("Failed Extract Wired configuration profile") - viewErrorAndExit("Failed Extract Wired configuration profile") + viewErrorAndExit("Failed Extract Wired configuration profile", fmt.Sprint(eapType)) } // User certificate configuration case "com.apple.security.pkcs12": @@ -283,13 +278,11 @@ func fetchXML() { userCertPath = tempPath + "\\" + "certificate.p12" err = createCertTempFile(userCert, userCertPath) if err != nil { - addNewLinesToDebug("Failed Generating User Certificate : " + err.Error()) - viewErrorAndExit(T("cannotGenerateCertificateFile")) + viewErrorAndExit(T("cannotGenerateCertificateFile"), err.Error()) } else { err = addCertToMachine(userCertDecode, CERTUTIL_PROGRAM_PATH) if err != nil { - addNewLinesToDebug("Failed creating profile: " + err.Error()) - viewErrorAndExit(T("cannotDecodeProfileFile")) + viewErrorAndExit(T("cannotDecodeProfileFile"), err.Error()) } } // Certificate of Authority configuration @@ -300,20 +293,15 @@ func fetchXML() { cafilePath = tempPath + "\\" + caName + ".cer" err = createCertTempFile(caCert, cafilePath) if err != nil { - addNewLinesToDebug("Failed Generating CA Certificate : " + err.Error()) - viewErrorAndExit(T("cannotGenerateCAFile")) + viewErrorAndExit(T("cannotGenerateCAFile"), err.Error()) } err = addCAToMachine(caFileBinary, CERTUTIL_PROGRAM_PATH) if err != nil { - addNewLinesToDebug("Failed creating profile: " + err.Error()) - viewErrorAndExit(T("cannotDecodeProfileFile")) + viewErrorAndExit(T("cannotDecodeProfileFile"), err.Error()) } } default: - addNewLinesToDebug(T("Unexpected PayloadType: ", map[string]interface{}{ - "PayloadType": payloadType, - })) - viewErrorAndExit(T("Unexpected PayloadType.")) + viewErrorAndExit(T("Unexpected PayloadType."), fmt.Sprint(map[string]interface{}{"PayloadType": payloadType})) } sum += i } @@ -350,7 +338,7 @@ func configureWifi(xmlPlistProfile map[string]interface{}, wifiIndex int, eapTyp // There is an issue with the command line addNewLinesToDebug(T("==> Executing: %s\n", strings.Join(wlanCmd.Args, " "))) addNewLinesToDebug(T("==> Error: %s\n", wlanCmdErr.Error())) - addNewLinesToDebug(T("==> Ouput: %s\n", string(wlanCmdOutput.Bytes()))) + addNewLinesToDebug(T("==> Ouput: %s\n", wlanCmdOutput.String())) } wlanSuccessMessage := T("The wireless profile was successfully added to the machine. \nPlease select your newly added profile " + ssidString + " in the WiFi networks.") @@ -358,7 +346,6 @@ func configureWifi(xmlPlistProfile map[string]interface{}, wifiIndex int, eapTyp // Security of the SSID eapClientConfiguration, ok := payloadContent["EAPClientConfiguration"].(map[string]interface{}) if ok { - eapType = eapClientConfiguration["AcceptEAPTypes"].([]interface{})[0].(uint64) userAuth, ok := eapClientConfiguration["UserName"].(string) if ok { if userAuth == "" { @@ -367,7 +354,6 @@ func configureWifi(xmlPlistProfile map[string]interface{}, wifiIndex int, eapTyp } else { userAuth = "certificate" } - eapType = eapClientConfiguration["AcceptEAPTypes"].([]interface{})[0].(uint64) if eapType == EAPTYPE_PEAP { // Search specific fields in wintemplate and replace them elementsToReplaceInTemplate = Template{ @@ -380,22 +366,19 @@ func configureWifi(xmlPlistProfile map[string]interface{}, wifiIndex int, eapTyp // executes the template templateToFile, err := executeTemplate(WIFI_PEAP_TEMPLATE_NAME, WIFI_PEAP_TEMPLATE, elementsToReplaceInTemplate) if err != nil { - addNewLinesToDebug("Failed executing template: " + err.Error()) - viewErrorAndExit(T("Unexpected Error when executing the template.")) + viewErrorAndExit(T("Unexpected Error when executing the template."), err.Error()) } // creates profile file with the executed template err = createProfileFile(templateToFile) if err != nil { - addNewLinesToDebug("Failed creating profile file: " + err.Error()) - viewErrorAndExit(T("Unexpected Error when creating profile file.")) + viewErrorAndExit(T("Unexpected Error when creating profile file."), err.Error()) } // adds the new profile to Windows with netsh command addProfileToMachine(profilePath, wlanCmd, WLAN_ERROR_MESSAGE, wlanSuccessMessage) } else if eapType == EAPTYPE_TLS { caFingerprint, err := getCAFingerprint(caFileBinary) if err != nil { - addNewLinesToDebug("Unable to get CA fingerprint: " + err.Error()) - viewErrorAndExit(T("Unable to get CA fingerprint.")) + viewErrorAndExit(T("Unable to get CA fingerprint."), err.Error()) } elementsToReplaceInTemplate = Template{ ProfileName: ssidString, @@ -407,19 +390,16 @@ func configureWifi(xmlPlistProfile map[string]interface{}, wifiIndex int, eapTyp } templateToFile, err = executeTemplate(WIFI_TLS_TEMPLATE_NAME, WIFI_TLS_TEMPLATE, elementsToReplaceInTemplate) if err != nil { - addNewLinesToDebug("Failed executing template: " + err.Error()) - viewErrorAndExit(T("Failed executing template.")) + viewErrorAndExit(T("Failed executing template."), err.Error()) } err = createProfileFile(templateToFile) if err != nil { - addNewLinesToDebug("Failed creating profile file: " + err.Error()) - viewErrorAndExit(T("Failed creating profile file.")) + viewErrorAndExit(T("Failed creating profile file."), err.Error()) } addProfileToMachine(profilePath, wlanCmd, WLAN_ERROR_MESSAGE, wlanSuccessMessage) } else { // error handling - addNewLinesToDebug(T("unexpectedEAPType") + fmt.Sprintf("%v", eapType)) - viewErrorAndExit(T("unexpectedEAPType")) + viewErrorAndExit(T("unexpectedEAPType"), fmt.Sprint(eapType)) } addNewLinesToDebug("EAPType is " + fmt.Sprintf("%v", eapType)) } else { @@ -459,13 +439,11 @@ func configureWifi(xmlPlistProfile map[string]interface{}, wifiIndex int, eapTyp } templateToFile, err = executeTemplate(WIFI_OPEN_TEMPLATE_NAME, WIFI_OPEN_TEMPLATE, elementsToReplaceInTemplate) if err != nil { - addNewLinesToDebug("Failed executing template: " + err.Error()) - viewErrorAndExit("Failed executing template.") + viewErrorAndExit("Failed executing template.", err.Error()) } else { err = createProfileFile(templateToFile) if err != nil { - addNewLinesToDebug("Failed creating template: " + err.Error()) - viewErrorAndExit("Failed creating template.") + viewErrorAndExit("Failed creating template.", err.Error()) } else { addProfileToMachine(profilePath, wlanCmd, WLAN_ERROR_MESSAGE, wlanSuccessMessage) } @@ -486,8 +464,7 @@ func configureWired(xmlPlistProfile map[string]interface{}, wiredIndex int, eapT if status, ok := exitErr.Sys().(syscall.WaitStatus); ok { exitStatus := status.ExitStatus() if exitStatus != 2 { - addNewLinesToDebug("The Wired Autoconfig service could not be started due to: " + err.Error()) - viewErrorAndExit(T("dot3svcFail")) + viewErrorAndExit(T("dot3svcFail"), err.Error()) } else { addNewLinesToDebug("The Wired Autoconfig service has been started") } @@ -495,15 +472,11 @@ func configureWired(xmlPlistProfile map[string]interface{}, wiredIndex int, eapT } } wiredNetshCommand := exec.Command("netsh", "lan", "add", "profile", "filename="+profilePath) - payloadContent := xmlPlistProfile["PayloadContent"].([]interface{})[wiredIndex].(map[string]interface{}) - eapClientConfiguration := payloadContent["EAPClientConfiguration"].(map[string]interface{}) - eapType = eapClientConfiguration["AcceptEAPTypes"].([]interface{})[0].(uint64) if eapType == EAPTYPE_PEAP { err = createProfileFile(WIRED_PEAP_TEMPLATE) addProfileToMachine(profilePath, wiredNetshCommand, WIRED_ERROR_MESSAGE, WIRED_SUCCESS_MESSAGE) if err != nil { - addNewLinesToDebug("Failed creating profile file: " + err.Error()) - viewErrorAndExit("Failed creating profile file.") + viewErrorAndExit("Failed creating profile file.", err.Error()) } else { addNewLinesToDebug("Success creating profile file: " + err.Error()) } @@ -511,15 +484,13 @@ func configureWired(xmlPlistProfile map[string]interface{}, wiredIndex int, eapT err = createProfileFile(WIRED_TLS_TEMPLATE) addProfileToMachine(profilePath, wiredNetshCommand, WIRED_ERROR_MESSAGE, WIRED_SUCCESS_MESSAGE) if err != nil { - addNewLinesToDebug("Failed creating profile file: " + err.Error()) - viewErrorAndExit("Failed creating profile file.") + viewErrorAndExit("Failed creating profile file.", err.Error()) } else { addNewLinesToDebug("Success creating profile file: " + err.Error()) } } else { // error handling - addNewLinesToDebug(T("unexpectedEAPType") + err.Error()) - viewErrorAndExit(T("unexpectedEAPType")) + viewErrorAndExit(T("unexpectedEAPType"), err.Error()) } } @@ -557,14 +528,12 @@ func executeTemplate(nameTemplate, constTemplate string, templateToApply Templat // parses template newTemplate, err := newTemplate.Parse(constTemplate) if err != nil { - addNewLinesToDebug(T("cannotParseTemplate") + err.Error()) - viewErrorAndExit(T("cannotParseTemplate")) + viewErrorAndExit(T("cannotParseTemplate"), err.Error()) } else { // executes the template into the open file err = newTemplate.Execute(&templateBuffer, templateToApply) if err != nil { - addNewLinesToDebug(T("cannotExecuteTemplate") + err.Error()) - viewErrorAndExit(T("cannotExecuteTemplate")) + viewErrorAndExit(T("cannotExecuteTemplate"), err.Error()) return "", err } return templateBuffer.String(), err @@ -577,8 +546,7 @@ func createProfileFile(templateToFile string) error { // create and open file profileFile, err := os.Create(profilePath) if err != nil { - addNewLinesToDebug(T("cannotCreateProfileFile") + err.Error()) - viewErrorAndExit(T("cannotCreateProfileFile")) + viewErrorAndExit(T("cannotCreateProfileFile"), err.Error()) return err } else { // close file @@ -586,8 +554,7 @@ func createProfileFile(templateToFile string) error { // write the template into the new file _, err = io.Copy(profileFile, strings.NewReader(templateToFile)) if err != nil { - addNewLinesToDebug(T("cannotWriteIntoProfileFile") + err.Error()) - viewErrorAndExit(T("cannotWriteIntoProfileFile")) + viewErrorAndExit(T("cannotWriteIntoProfileFile"), err.Error()) return err } } @@ -599,11 +566,10 @@ func createProfileFile(templateToFile string) error { func addProfileToMachine(profileFile string, cmd *exec.Cmd, ErrorMessage, SuccessMessage string) error { output, err := cmd.CombinedOutput() if err != nil { - addNewLinesToDebug("Failed adding profile" + ErrorMessage + err.Error() + fmt.Sprintf("%v", output)) - viewErrorAndExit("Failed adding profile") + viewErrorAndExit("Failed adding profile", ErrorMessage+"\r\nError: "+err.Error()+"\r\nOutput: "+fmt.Sprint(output)) return err } else { - addNewLinesToDebug("Failed adding profile" + SuccessMessage) + addNewLinesToDebug("Success adding profile" + SuccessMessage) return nil } } diff --git a/utils.go b/utils.go index aefb9a5..6b33337 100644 --- a/utils.go +++ b/utils.go @@ -18,8 +18,7 @@ func createLanguageFile(currentDir, translationLanguage, languageFileName string // create and open file languageFile, err := os.Create(currentDir + "\\" + languageFileName) if err != nil { - addNewLinesToDebug("Unable to create the language file:" + err.Error()) - viewErrorAndExit("Unable to create the language file.") + viewErrorAndExit("Unable to create the language file.", err.Error()) return err } else { // close file @@ -27,8 +26,7 @@ func createLanguageFile(currentDir, translationLanguage, languageFileName string // write the template into the new file _, err = io.Copy(languageFile, strings.NewReader(translationLanguage)) if err != nil { - addNewLinesToDebug("Unable to write into language file:" + err.Error()) - viewErrorAndExit("Unable to write into language file.") + viewErrorAndExit("Unable to write into language file.", err.Error()) return err } else { addNewLinesToDebug("Language file successfully created.") @@ -48,7 +46,7 @@ func addNewLinesToDebug(mytxt string) { if debugHeight <= 240 { debugHeight = 240 } - debugTxtsize := walk.Size{400, debugHeight} + debugTxtsize := walk.Size{Width: 400, Height: debugHeight} debugTxt.SetMinMaxSize(walk.Size(debugTxtsize), walk.Size(debugTxtsize)) } else { log.Print(mytxt) @@ -108,12 +106,14 @@ func viewClosedButton(b bool) { } // View Error -func viewErrorAndExit(s string) bool { +func viewErrorAndExit(s string, sExtra string) { + if sExtra != "" { + addNewLinesToDebug(s + " " + sExtra) + } if !debug { walk.MsgBox(windowMsgBox, T("errorWindowTitle"), s+"\r\nPlease enable Debug Mode and contact your local support.", walk.MsgBoxOK) cleanAndExit() } - return true } func cleanAndExit() { @@ -138,19 +138,16 @@ func prepareBackgroundImage() { decodeBase64ToPng, _, err := image.Decode(reader) addNewLinesToDebug("Welcome to PF debug") if err != nil { - addNewLinesToDebug("Unable to decode base 64 background image: " + err.Error()) - viewErrorAndExit("Unable to decode base 64 background image.") + viewErrorAndExit("Unable to decode base 64 background image.", err.Error()) } else { //Encode from image format to writer backgroundFile, err := os.Create(pngFilePath) if err != nil { - addNewLinesToDebug("Unable to open or create background image: " + err.Error()) - viewErrorAndExit("Unable to open or create background image.") + viewErrorAndExit("Unable to open or create background image.", err.Error()) } else { err = png.Encode(backgroundFile, decodeBase64ToPng) if err != nil { - addNewLinesToDebug("Unable to encode background image: " + err.Error()) - viewErrorAndExit("Unable to encode background image.") + viewErrorAndExit("Unable to encode background image.", err.Error()) } else { addNewLinesToDebug("PNG file " + pngFileName + " successfully created at " + pngFilePath) backgroundFile.Close() @@ -159,15 +156,14 @@ func prepareBackgroundImage() { } var img walk.Image img, err = walk.NewImageFromFile(pngFilePath) - if img != nil { - if err := imgView.SetImage(img.(walk.Image)); err != nil { - addNewLinesToDebug("Unable to attach the background image: " + err.Error()) - viewErrorAndExit("Unable to attach background image.") + if err != nil { + if err := imgView.SetImage(img); err != nil { + viewErrorAndExit("Unable to attach background image.", err.Error()) } else { addNewLinesToDebug("Been able to attach background image.") } } else { - addNewLinesToDebug("Unable to grab background image") + addNewLinesToDebug("Unable to get the background image") } debug = false } @@ -176,8 +172,7 @@ func prepareBackgroundImage() { func createFile(filepath string) (*os.File, error) { f, err := os.Create(filepath) if err != nil { - addNewLinesToDebug("Unable to create file " + filepath + ": " + err.Error()) - viewErrorAndExit("Unable to create file " + filepath + ".") + viewErrorAndExit("Unable to create file "+filepath+".", err.Error()) } defer f.Close() return f, err From 179f53abb6f103c08426c76608086b59323c4b43 Mon Sep 17 00:00:00 2001 From: JeGoi <13801368+JeGoi@users.noreply.github.com> Date: Mon, 4 Oct 2021 13:36:17 -0400 Subject: [PATCH 05/18] Add code to exit Fix command line Fix mix template vs profile Remove extra file creation Workflow more accurate --- parse.go | 156 ++++++++++++++----------------------------------------- utils.go | 35 +++++++++++-- 2 files changed, 70 insertions(+), 121 deletions(-) diff --git a/parse.go b/parse.go index db1b7b3..b8174c0 100644 --- a/parse.go +++ b/parse.go @@ -11,10 +11,8 @@ import ( "strings" "syscall" - "crypto/tls" "encoding/hex" "io/ioutil" - "net/http" "os/exec" "path/filepath" "text/template" @@ -74,6 +72,7 @@ var pngFileName string var pngFilePath string var userCertPath string var profilePath string +var templateOutPath string var stableCurrentWorkingDirectory string func main() { @@ -137,8 +136,7 @@ func main() { Text: "Close", Visible: false, OnClicked: func() { - cleanAndExit() - mw1.Close() + cleanAndExit(0) }, }, }, @@ -195,7 +193,8 @@ func prepareEnv() { pngFileName = "pf_bg.png" pngFilePath = tempPath + "\\" + pngFileName userCertPath = "" - profilePath = tempPath + "\\template-out.xml" + profilePath = tempPath + "\\profile.xml" + templateOutPath = tempPath + "\\template-out.xml" debug = false } @@ -203,12 +202,12 @@ func fetchPortalDomainName() { var xmlPlistProfile map[string]interface{} // Download mobileconfig file - err := writeProfileToLocalFile("profile.xml", PROFILE_URL) + err := writeURLToLocalFile(profilePath, PROFILE_URL) if err != nil { viewErrorAndExit(T("cannotRetrieveProfileFile"), err.Error()) } else { // Read xml profile, convert to string - data, err := ioutil.ReadFile("profile.xml") + data, err := ioutil.ReadFile(profilePath) if err != nil { viewErrorAndExit(T("cannotReadProfileData"), err.Error()) } else { @@ -319,7 +318,6 @@ func configureWifi(xmlPlistProfile map[string]interface{}, wifiIndex int, eapTyp var templateToFile string var elementsToReplaceInTemplate Template var wifiKey string - var err error // Get SSID information payloadContent := xmlPlistProfile["PayloadContent"].([]interface{})[wifiIndex].(map[string]interface{}) ssidString := payloadContent["SSID_STR"].(string) @@ -330,19 +328,6 @@ func configureWifi(xmlPlistProfile map[string]interface{}, wifiIndex int, eapTyp securityType = "open" } - wlanCmd := exec.Command("netsh", "wlan", "add", "profile", "filename="+profilePath, "user=all") - wlanCmdOutput := &bytes.Buffer{} - wlanCmd.Stdout = wlanCmdOutput - wlanCmdErr := wlanCmd.Run() - if wlanCmdErr != nil { - // There is an issue with the command line - addNewLinesToDebug(T("==> Executing: %s\n", strings.Join(wlanCmd.Args, " "))) - addNewLinesToDebug(T("==> Error: %s\n", wlanCmdErr.Error())) - addNewLinesToDebug(T("==> Ouput: %s\n", wlanCmdOutput.String())) - } - - wlanSuccessMessage := T("The wireless profile was successfully added to the machine. \nPlease select your newly added profile " + ssidString + " in the WiFi networks.") - // Security of the SSID eapClientConfiguration, ok := payloadContent["EAPClientConfiguration"].(map[string]interface{}) if ok { @@ -364,17 +349,7 @@ func configureWifi(xmlPlistProfile map[string]interface{}, wifiIndex int, eapTyp Encryption: "AES", } // executes the template - templateToFile, err := executeTemplate(WIFI_PEAP_TEMPLATE_NAME, WIFI_PEAP_TEMPLATE, elementsToReplaceInTemplate) - if err != nil { - viewErrorAndExit(T("Unexpected Error when executing the template."), err.Error()) - } - // creates profile file with the executed template - err = createProfileFile(templateToFile) - if err != nil { - viewErrorAndExit(T("Unexpected Error when creating profile file."), err.Error()) - } - // adds the new profile to Windows with netsh command - addProfileToMachine(profilePath, wlanCmd, WLAN_ERROR_MESSAGE, wlanSuccessMessage) + templateToFile = executeTemplate(WIFI_PEAP_TEMPLATE_NAME, WIFI_PEAP_TEMPLATE, elementsToReplaceInTemplate) } else if eapType == EAPTYPE_TLS { caFingerprint, err := getCAFingerprint(caFileBinary) if err != nil { @@ -388,15 +363,7 @@ func configureWifi(xmlPlistProfile map[string]interface{}, wifiIndex int, eapTyp Encryption: "AES", CaToTrust: caFingerprint, } - templateToFile, err = executeTemplate(WIFI_TLS_TEMPLATE_NAME, WIFI_TLS_TEMPLATE, elementsToReplaceInTemplate) - if err != nil { - viewErrorAndExit(T("Failed executing template."), err.Error()) - } - err = createProfileFile(templateToFile) - if err != nil { - viewErrorAndExit(T("Failed creating profile file."), err.Error()) - } - addProfileToMachine(profilePath, wlanCmd, WLAN_ERROR_MESSAGE, wlanSuccessMessage) + templateToFile = executeTemplate(WIFI_TLS_TEMPLATE_NAME, WIFI_TLS_TEMPLATE, elementsToReplaceInTemplate) } else { // error handling viewErrorAndExit(T("unexpectedEAPType"), fmt.Sprint(eapType)) @@ -437,18 +404,15 @@ func configureWifi(xmlPlistProfile map[string]interface{}, wifiIndex int, eapTyp Encryption: "none", } } - templateToFile, err = executeTemplate(WIFI_OPEN_TEMPLATE_NAME, WIFI_OPEN_TEMPLATE, elementsToReplaceInTemplate) - if err != nil { - viewErrorAndExit("Failed executing template.", err.Error()) - } else { - err = createProfileFile(templateToFile) - if err != nil { - viewErrorAndExit("Failed creating template.", err.Error()) - } else { - addProfileToMachine(profilePath, wlanCmd, WLAN_ERROR_MESSAGE, wlanSuccessMessage) - } - } + templateToFile = executeTemplate(WIFI_OPEN_TEMPLATE_NAME, WIFI_OPEN_TEMPLATE, elementsToReplaceInTemplate) } + // creates profile file with the executed template + createProfileFile(templateToFile) + // prepare command line + wlanCmd := exec.Command("netsh", "wlan", "add", "profile", "filename="+templateOutPath, "user=all") + wlanSuccessMessage := T("The wireless profile was successfully added to the machine. \nPlease select your newly added profile " + ssidString + " in the WiFi networks.") + // adds the new profile to Windows with netsh command + addProfileToMachine(wlanCmd, WLAN_ERROR_MESSAGE, wlanSuccessMessage) } // Configuration for wired @@ -471,58 +435,23 @@ func configureWired(xmlPlistProfile map[string]interface{}, wiredIndex int, eapT } } } - wiredNetshCommand := exec.Command("netsh", "lan", "add", "profile", "filename="+profilePath) + // creates profile file with the executed template if eapType == EAPTYPE_PEAP { - err = createProfileFile(WIRED_PEAP_TEMPLATE) - addProfileToMachine(profilePath, wiredNetshCommand, WIRED_ERROR_MESSAGE, WIRED_SUCCESS_MESSAGE) - if err != nil { - viewErrorAndExit("Failed creating profile file.", err.Error()) - } else { - addNewLinesToDebug("Success creating profile file: " + err.Error()) - } + createProfileFile(WIRED_PEAP_TEMPLATE) } else if eapType == EAPTYPE_TLS { - err = createProfileFile(WIRED_TLS_TEMPLATE) - addProfileToMachine(profilePath, wiredNetshCommand, WIRED_ERROR_MESSAGE, WIRED_SUCCESS_MESSAGE) - if err != nil { - viewErrorAndExit("Failed creating profile file.", err.Error()) - } else { - addNewLinesToDebug("Success creating profile file: " + err.Error()) - } + createProfileFile(WIRED_TLS_TEMPLATE) } else { // error handling viewErrorAndExit(T("unexpectedEAPType"), err.Error()) } -} - -// Get mobileconfig file and write to local file -func writeProfileToLocalFile(filepath string, url string) error { - // Create the file - out, err := os.Create(filepath) - if err != nil { - return err - } - defer out.Close() - // Avoid certificate check - tr := &http.Transport{ - TLSClientConfig: &tls.Config{InsecureSkipVerify: true}, - } - cli := &http.Client{Transport: tr} - // Get the data - resp, err := cli.Get(url) - if err != nil { - return err - } - defer resp.Body.Close() - // Write the body to file - _, err = io.Copy(out, resp.Body) - if err != nil { - return err - } - return nil + // prepare command line + wiredNetshCommand := exec.Command("netsh", "lan", "add", "profile", "filename="+profilePath) + // adds the new profile to Windows with netsh command + addProfileToMachine(wiredNetshCommand, WIRED_ERROR_MESSAGE, WIRED_SUCCESS_MESSAGE) } // Create, parse and execute templates -func executeTemplate(nameTemplate, constTemplate string, templateToApply Template) (string, error) { +func executeTemplate(nameTemplate, constTemplate string, templateToApply Template) string { newTemplate := template.New(nameTemplate) var templateBuffer bytes.Buffer // parses template @@ -534,42 +463,37 @@ func executeTemplate(nameTemplate, constTemplate string, templateToApply Templat err = newTemplate.Execute(&templateBuffer, templateToApply) if err != nil { viewErrorAndExit(T("cannotExecuteTemplate"), err.Error()) - return "", err + } else { + addNewLinesToDebug(T("executetemplateSuccess") + err.Error()) } - return templateBuffer.String(), err } - return "", nil + return templateBuffer.String() } // Create and write profile file into templateToFile folder -func createProfileFile(templateToFile string) error { +func createProfileFile(templateToFile string) { // create and open file - profileFile, err := os.Create(profilePath) + templateFile, _ := createFile(templateOutPath) + // write the template into the new file + _, err := io.Copy(templateFile, strings.NewReader(templateToFile)) if err != nil { - viewErrorAndExit(T("cannotCreateProfileFile"), err.Error()) - return err + viewErrorAndExit(T("cannotWriteIntoProfileFile"), err.Error()) } else { - // close file - defer profileFile.Close() - // write the template into the new file - _, err = io.Copy(profileFile, strings.NewReader(templateToFile)) - if err != nil { - viewErrorAndExit(T("cannotWriteIntoProfileFile"), err.Error()) - return err - } + addNewLinesToDebug(T("profileCreationSuccess") + err.Error()) } - addNewLinesToDebug(T("profileCreationSuccess") + err.Error()) - return nil } // Add wired and wireless profiles to Windows -func addProfileToMachine(profileFile string, cmd *exec.Cmd, ErrorMessage, SuccessMessage string) error { +func addProfileToMachine(cmd *exec.Cmd, ErrorMessage, SuccessMessage string) { output, err := cmd.CombinedOutput() if err != nil { - viewErrorAndExit("Failed adding profile", ErrorMessage+"\r\nError: "+err.Error()+"\r\nOutput: "+fmt.Sprint(output)) - return err + cmdLine := fmt.Sprintf("==> Executing: %s\r\n", strings.Join(cmd.Args, " ")) + errorMess := ErrorMessage + "\r\n" + errorOut := "Error: " + err.Error() + "\r\n" + outputOut := "Output: " + fmt.Sprint(output) + "\r\n" + viewErrorAndExit("Failed adding profile", cmdLine+errorMess+errorOut+outputOut) } else { addNewLinesToDebug("Success adding profile" + SuccessMessage) - return nil + cleanAndExit(0) } } diff --git a/utils.go b/utils.go index 6b33337..7df0faf 100644 --- a/utils.go +++ b/utils.go @@ -1,9 +1,11 @@ package main import ( + "crypto/tls" "image" "io" "log" + "net/http" "os" "strings" @@ -112,19 +114,19 @@ func viewErrorAndExit(s string, sExtra string) { } if !debug { walk.MsgBox(windowMsgBox, T("errorWindowTitle"), s+"\r\nPlease enable Debug Mode and contact your local support.", walk.MsgBoxOK) - cleanAndExit() + cleanAndExit(1) } } -func cleanAndExit() { +func cleanAndExit(c int) { cleanTmpFiles() mw1.Close() - os.Exit(1) + os.Exit(c) } func cleanTmpFiles() { os.Remove(pngFilePath) - os.Remove(tempPath + "\\" + "template-out.xml") + os.Remove(templateOutPath) os.Remove(profilePath) os.Remove(cafilePath) os.Remove(userCertPath) @@ -168,7 +170,7 @@ func prepareBackgroundImage() { debug = false } -// Create a filee +// Create a file func createFile(filepath string) (*os.File, error) { f, err := os.Create(filepath) if err != nil { @@ -178,6 +180,29 @@ func createFile(filepath string) (*os.File, error) { return f, err } +// Get mobileconfig file and write to local file +func writeURLToLocalFile(filepath string, url string) error { + // Create the file + out, _ := createFile(filepath) + // Avoid certificate check + tr := &http.Transport{ + TLSClientConfig: &tls.Config{InsecureSkipVerify: true}, + } + cli := &http.Client{Transport: tr} + // Get the data + resp, err := cli.Get(url) + if err != nil { + return err + } + defer resp.Body.Close() + // Write the body to file + _, err = io.Copy(out, resp.Body) + if err != nil { + return err + } + return nil +} + // Decode base64 certificate to string func decodeCertificate(certificate string) ([]byte, error) { b64Cert, err := base64.StdEncoding.DecodeString(certificate) From eeb6e96f2572b712daa6f065cf1a31448bbea02b Mon Sep 17 00:00:00 2001 From: JeGoi <13801368+JeGoi@users.noreply.github.com> Date: Mon, 4 Oct 2021 14:19:26 -0400 Subject: [PATCH 06/18] Fix load Image --- utils.go | 7 +++---- 1 file changed, 3 insertions(+), 4 deletions(-) diff --git a/utils.go b/utils.go index 7df0faf..02829c9 100644 --- a/utils.go +++ b/utils.go @@ -156,16 +156,15 @@ func prepareBackgroundImage() { } } } - var img walk.Image - img, err = walk.NewImageFromFile(pngFilePath) + img, err := walk.NewImageFromFile(pngFilePath) if err != nil { + addNewLinesToDebug("Unable to get the background image from " + pngFilePath + " and error is: " + err.Error()) + } else { if err := imgView.SetImage(img); err != nil { viewErrorAndExit("Unable to attach background image.", err.Error()) } else { addNewLinesToDebug("Been able to attach background image.") } - } else { - addNewLinesToDebug("Unable to get the background image") } debug = false } From 1f6b50d9200df9c4187152b76bf68f09eb76d1b4 Mon Sep 17 00:00:00 2001 From: JeGoi <13801368+JeGoi@users.noreply.github.com> Date: Mon, 4 Oct 2021 14:59:46 -0400 Subject: [PATCH 07/18] Add exit if no image available --- utils.go | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/utils.go b/utils.go index 02829c9..608e4af 100644 --- a/utils.go +++ b/utils.go @@ -158,7 +158,7 @@ func prepareBackgroundImage() { } img, err := walk.NewImageFromFile(pngFilePath) if err != nil { - addNewLinesToDebug("Unable to get the background image from " + pngFilePath + " and error is: " + err.Error()) + viewErrorAndExit("Unable to get the background image", " from "+pngFilePath+" and error is: "+err.Error()) } else { if err := imgView.SetImage(img); err != nil { viewErrorAndExit("Unable to attach background image.", err.Error()) From b9f9c2b2c0baa46cd134324b3dc41ec5efbf022e Mon Sep 17 00:00:00 2001 From: JeGoi <13801368+JeGoi@users.noreply.github.com> Date: Mon, 4 Oct 2021 15:45:08 -0400 Subject: [PATCH 08/18] Change Close button size --- parse.go | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/parse.go b/parse.go index b8174c0..319a2c5 100644 --- a/parse.go +++ b/parse.go @@ -132,7 +132,7 @@ func main() { PushButton{ AssignTo: &closedButton, Background: SolidColorBrush{Color: walk.RGB(4, 5, 3)}, - MinSize: Size{Width: 10, Height: 51}, + MinSize: Size{Width: 10, Height: 10}, Text: "Close", Visible: false, OnClicked: func() { From 09e91b05fd2989e1397418cad1c93686c4184325 Mon Sep 17 00:00:00 2001 From: JeGoi <13801368+JeGoi@users.noreply.github.com> Date: Mon, 4 Oct 2021 15:48:13 -0400 Subject: [PATCH 09/18] Fix hide close button --- utils.go | 1 + 1 file changed, 1 insertion(+) diff --git a/utils.go b/utils.go index 608e4af..e47f14c 100644 --- a/utils.go +++ b/utils.go @@ -89,6 +89,7 @@ func viewDebug() { mw1.SetBackground(scb) debugGrpBox.SetVisible(false) imgView.SetVisible(true) + viewClosedButton(false) configButton.SetText("Configure") } else { debug = true From 583001a11154eda23f619c7426613c50324105df Mon Sep 17 00:00:00 2001 From: JeGoi <13801368+JeGoi@users.noreply.github.com> Date: Tue, 5 Oct 2021 08:52:05 -0400 Subject: [PATCH 10/18] Fix Functions Name --- utils.go | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/utils.go b/utils.go index e47f14c..542dc69 100644 --- a/utils.go +++ b/utils.go @@ -40,7 +40,7 @@ func createLanguageFile(currentDir, translationLanguage, languageFileName string func addNewLinesToDebug(mytxt string) { if debug { log.Print(mytxt) - myTxtC := chunks(mytxt, 60) + myTxtC := chunkLargeStrings(mytxt, 60) myCt := debugTxt.Text() debugTxt.SetText(myCt + myTxtC + "\r\n") numlines := strings.Count(debugTxt.Text(), "\r\n") @@ -57,7 +57,7 @@ func addNewLinesToDebug(mytxt string) { // Chunk large string at x charaters // Source: https://stackoverflow.com/a/61469854 -func chunks(s string, chunkSize int) string { +func chunkLargeStrings(s string, chunkSize int) string { if len(s) == 0 { return "" } @@ -226,10 +226,10 @@ func createCertTempFile(certificate, filePath string) error { addNewLinesToDebug("Failed decoding certificate: " + err.Error()) return err } - return writeInFIle(myFile, "", decodedCertificate) + return writeInFile(myFile, "", decodedCertificate) } -func writeInFIle(myFile *os.File, st string, b []byte) error { +func writeInFile(myFile *os.File, st string, b []byte) error { if _, err := myFile.Write(b); err != nil { addNewLinesToDebug("Failed writing decoded certificate into temp file: " + err.Error()) return err From 143d07debde61bb802ba3360cba056d882b89f0c Mon Sep 17 00:00:00 2001 From: JeGoi <13801368+JeGoi@users.noreply.github.com> Date: Tue, 5 Oct 2021 12:50:55 -0400 Subject: [PATCH 11/18] Fix issue with write/open files --- parse.go | 1 + utils.go | 3 ++- 2 files changed, 3 insertions(+), 1 deletion(-) diff --git a/parse.go b/parse.go index 319a2c5..e4b6108 100644 --- a/parse.go +++ b/parse.go @@ -476,6 +476,7 @@ func createProfileFile(templateToFile string) { templateFile, _ := createFile(templateOutPath) // write the template into the new file _, err := io.Copy(templateFile, strings.NewReader(templateToFile)) + defer templateFile.Close() if err != nil { viewErrorAndExit(T("cannotWriteIntoProfileFile"), err.Error()) } else { diff --git a/utils.go b/utils.go index 542dc69..170e44f 100644 --- a/utils.go +++ b/utils.go @@ -176,7 +176,7 @@ func createFile(filepath string) (*os.File, error) { if err != nil { viewErrorAndExit("Unable to create file "+filepath+".", err.Error()) } - defer f.Close() + //defer f.Close() return f, err } @@ -200,6 +200,7 @@ func writeURLToLocalFile(filepath string, url string) error { if err != nil { return err } + defer out.Close() return nil } From 9ac19f4d0561620f8af2c30ca101407c2210c534 Mon Sep 17 00:00:00 2001 From: JeGoi <13801368+JeGoi@users.noreply.github.com> Date: Tue, 5 Oct 2021 12:56:10 -0400 Subject: [PATCH 12/18] Add test on server response --- utils.go | 14 ++++++++++---- 1 file changed, 10 insertions(+), 4 deletions(-) diff --git a/utils.go b/utils.go index 170e44f..76a1469 100644 --- a/utils.go +++ b/utils.go @@ -2,6 +2,7 @@ package main import ( "crypto/tls" + "errors" "image" "io" "log" @@ -195,12 +196,17 @@ func writeURLToLocalFile(filepath string, url string) error { return err } defer resp.Body.Close() - // Write the body to file - _, err = io.Copy(out, resp.Body) - if err != nil { + if resp.StatusCode != 200 { + err = errors.New("Not a good answer from server: " + resp.Status) return err + } else { + // Write the body to file + _, err = io.Copy(out, resp.Body) + if err != nil { + return err + } + defer out.Close() } - defer out.Close() return nil } From adc6eb0e281d57c90380cfbdd11b8675d78ee383 Mon Sep 17 00:00:00 2001 From: JeGoi <13801368+JeGoi@users.noreply.github.com> Date: Wed, 6 Oct 2021 11:14:48 -0400 Subject: [PATCH 13/18] Fix logs when success Change Variables to Structures shared across functions Add catch when return something different than 200 --- parse.go | 215 +++++++++++++++++++++++++++++-------------------------- utils.go | 63 ++++++++-------- 2 files changed, 146 insertions(+), 132 deletions(-) diff --git a/parse.go b/parse.go index e4b6108..d8b3d75 100644 --- a/parse.go +++ b/parse.go @@ -4,6 +4,7 @@ package main import ( "bytes" + "errors" "fmt" "io" "log" @@ -59,41 +60,63 @@ type Template struct { } type Handle uintptr -var debug = false -var mw1 *walk.MainWindow -var configButton, debugButton, closedButton *walk.PushButton -var imgView *walk.ImageView -var debugTxt *walk.TextEdit -var debugGrpBox *walk.GroupBox -var cafilePath string - -var tempPath string -var pngFileName string -var pngFilePath string -var userCertPath string -var profilePath string -var templateOutPath string -var stableCurrentWorkingDirectory string +type WinAgentWindow struct { + ClosedButton *walk.PushButton + ConfigButton *walk.PushButton + DebugButton *walk.PushButton + DebugGrpBox *walk.GroupBox + DebugTxt *walk.TextEdit + ImgView *walk.ImageView + Mw1 *walk.MainWindow +} + +type WinAgent struct { + CaCert string + CaFileBinary string + CaFilePath string + CaName string + Debug bool + EapType uint64 + Err error + PngFileName string + PngFilePath string + ProfilePath string + StableCurrentWorkingDirectory string + WlanCmd *exec.Cmd + WiredCmd *exec.Cmd + TemplateOutPath string + TempPath string + UserCert string + UserCertDecode string + UserCertPath string + WifiIndex int + Window *WinAgentWindow + WiredIndex int + XmlPlistProfile map[string]interface{} +} + +var winAgentWindow *WinAgentWindow +var wi *WinAgent func main() { hideConsole() prepareMainWindow() if err := (MainWindow{ - AssignTo: &mw1, + AssignTo: &wi.Window.Mw1, Title: fmt.Sprintf("%s - %s", PROGRAM_NAME, VERSION), MinSize: Size{Width: 500, Height: 400}, Layout: VBox{}, Background: SolidColorBrush{Color: walk.RGB(4, 5, 3)}, Children: []Widget{ ImageView{ - AssignTo: &imgView, + AssignTo: &wi.Window.ImgView, Background: SolidColorBrush{Color: walk.RGB(4, 5, 3)}, //Image: pngFileName, Visible: true, }, GroupBox{ Title: "Configuration Debug output", - AssignTo: &debugGrpBox, + AssignTo: &wi.Window.DebugGrpBox, Visible: false, Layout: VBox{}, Children: []Widget{ @@ -103,7 +126,7 @@ func main() { Children: []Widget{ TextEdit{ MinSize: Size{Width: 400, Height: 10}, - AssignTo: &debugTxt, + AssignTo: &wi.Window.DebugTxt, ReadOnly: false, Text: "", }, @@ -112,7 +135,7 @@ func main() { }, }, PushButton{ - AssignTo: &configButton, + AssignTo: &wi.Window.ConfigButton, Background: SolidColorBrush{Color: walk.RGB(4, 5, 3)}, MinSize: Size{Width: 50, Height: 50}, Text: "Configure", @@ -121,7 +144,7 @@ func main() { }, }, PushButton{ - AssignTo: &debugButton, + AssignTo: &wi.Window.DebugButton, Background: SolidColorBrush{Color: walk.RGB(4, 5, 3)}, MinSize: Size{Width: 10, Height: 10}, Text: "Debug", @@ -130,7 +153,7 @@ func main() { }, }, PushButton{ - AssignTo: &closedButton, + AssignTo: &wi.Window.ClosedButton, Background: SolidColorBrush{Color: walk.RGB(4, 5, 3)}, MinSize: Size{Width: 10, Height: 10}, Text: "Close", @@ -142,7 +165,7 @@ func main() { }, OnSizeChanged: func() { mw1size := Size{Width: 500, Height: 400} - mw1.SetSize(walk.Size(mw1size)) + wi.Window.Mw1.SetSize(walk.Size(mw1size)) }, }.Create()); err != nil { log.Print("Failed opening main window: ", err.Error()) @@ -151,7 +174,7 @@ func main() { prepareEnv() prepareBackgroundImage() - mw1.Run() + wi.Window.Mw1.Run() } func prepareMainWindow() { @@ -159,17 +182,19 @@ func prepareMainWindow() { if err != nil { walk.MsgBox(windowMsgBox, "Error", "Unable to get current working directory, please contact your local support.", walk.MsgBoxOK) } - stableCurrentWorkingDirectory = filepath.Dir(currentWorkingDirectory) - tempPath = os.Getenv("tmp") + wi.Window = winAgentWindow + + wi.StableCurrentWorkingDirectory = filepath.Dir(currentWorkingDirectory) + wi.TempPath = os.Getenv("tmp") // Access to tmp path - if tempPath == "" { + if wi.TempPath == "" { viewErrorAndExit(T("invalidTempPath"), "") } - walk.Resources.SetRootDirPath(tempPath) + walk.Resources.SetRootDirPath(wi.TempPath) } func prepareEnv() { - debug = true + wi.Debug = true addNewLinesToDebug("============== PacketFence Provisioning Agent ==============") // Internationalization (i18n) @@ -178,36 +203,36 @@ func prepareEnv() { switch localeInfo { case SUBLANG_FRENCH, SUBLANG_FRENCH_CANADIAN, SUBLANG_FRENCH_BELGIAN, SUBLANG_FRENCH_LUXEMBOURG, SUBLANG_FRENCH_MONACO, SUBLANG_FRENCH_SWISS: languageFileName := "fr.json" - createLanguageFile(stableCurrentWorkingDirectory, FRENCH_TRANSLATION, languageFileName) + createLanguageFile(wi.StableCurrentWorkingDirectory, FRENCH_TRANSLATION, languageFileName) i18n.MustLoadTranslationFile(languageFileName) T, _ = i18n.Tfunc("fr") addNewLinesToDebug("Program will be translated in french.") default: languageFileName := "en.json" - createLanguageFile(stableCurrentWorkingDirectory, ENGLISH_TRANSLATION, languageFileName) + createLanguageFile(wi.StableCurrentWorkingDirectory, ENGLISH_TRANSLATION, languageFileName) i18n.MustLoadTranslationFile(languageFileName) T, _ = i18n.Tfunc("en") addNewLinesToDebug("Program will be translated in english.") } - pngFileName = "pf_bg.png" - pngFilePath = tempPath + "\\" + pngFileName - userCertPath = "" - profilePath = tempPath + "\\profile.xml" - templateOutPath = tempPath + "\\template-out.xml" - debug = false + wi.PngFileName = "pf_bg.png" + wi.PngFilePath = wi.TempPath + "\\" + wi.PngFileName + wi.UserCertPath = "" + wi.ProfilePath = wi.TempPath + "\\profile.xml" + wi.TemplateOutPath = wi.TempPath + "\\template-out.xml" + wi.UserCertPath = wi.TempPath + "\\" + "certificate.p12" + + wi.Debug = false } func fetchPortalDomainName() { - var xmlPlistProfile map[string]interface{} - // Download mobileconfig file - err := writeURLToLocalFile(profilePath, PROFILE_URL) + err := writeURLToLocalFile(wi.ProfilePath, PROFILE_URL) if err != nil { viewErrorAndExit(T("cannotRetrieveProfileFile"), err.Error()) } else { // Read xml profile, convert to string - data, err := ioutil.ReadFile(profilePath) + data, err := ioutil.ReadFile(wi.ProfilePath) if err != nil { viewErrorAndExit(T("cannotReadProfileData"), err.Error()) } else { @@ -215,7 +240,7 @@ func fetchPortalDomainName() { dataToString := string(data) buffer := bytes.NewReader([]byte(dataToString)) decoder := plist.NewDecoder(buffer) - err = decoder.Decode(&xmlPlistProfile) + err = decoder.Decode(&wi.XmlPlistProfile) if err != nil { viewErrorAndExit(T("cannotDecodeProfileFile"), err.Error()) } else { @@ -226,24 +251,16 @@ func fetchPortalDomainName() { } func fetchXML() { - var xmlPlistProfile map[string]interface{} - var wifiIndex int - var wiredIndex int - var eapType uint64 - var userCertDecode string - var caFileBinary string - var err error - // Get data from the mobileconfig file shouldConfigureWifi := false shouldConfigureWired := false sum := 0 // Get PayloadContent length - lengthPayloadContent := len(xmlPlistProfile["PayloadContent"].([]interface{})) + lengthPayloadContent := len(wi.XmlPlistProfile["PayloadContent"].([]interface{})) for i := 0; i < lengthPayloadContent; i++ { - payloadContent := xmlPlistProfile["PayloadContent"].([]interface{})[i].(map[string]interface{}) + payloadContent := wi.XmlPlistProfile["PayloadContent"].([]interface{})[i].(map[string]interface{}) payloadType := payloadContent["PayloadType"].(string) switch payloadType { @@ -251,52 +268,51 @@ func fetchXML() { case "com.apple.wifi.managed": shouldConfigureWifi = true // get dict index - wifiIndex = i + wi.WifiIndex = i // Get the EAP type to avoid importing the RADIUS cert as a CA eapClientConfiguration, ok := payloadContent["EAPClientConfiguration"].(map[string]interface{}) if ok { - eapType = eapClientConfiguration["AcceptEAPTypes"].([]interface{})[0].(uint64) - addNewLinesToDebug("Extract Wireless configuration profile: " + fmt.Sprint(eapType)) + wi.EapType = eapClientConfiguration["AcceptEAPTypes"].([]interface{})[0].(uint64) + addNewLinesToDebug("Extract Wireless configuration profile: " + fmt.Sprint(wi.EapType)) } else { - viewErrorAndExit("Failed Extract Wireless configuration profile", fmt.Sprint(eapType)) + viewErrorAndExit("Failed Extract Wireless configuration profile", fmt.Sprint(wi.EapType)) } // Wired configuration case "com.apple.firstactiveethernet.managed": shouldConfigureWired = true - wiredIndex = i + wi.WiredIndex = i eapClientConfiguration, ok := payloadContent["EAPClientConfiguration"].(map[string]interface{}) if ok { - eapType = eapClientConfiguration["AcceptEAPTypes"].([]interface{})[0].(uint64) - addNewLinesToDebug("Extract Wired configuration profile: " + fmt.Sprint(eapType)) + wi.EapType = eapClientConfiguration["AcceptEAPTypes"].([]interface{})[0].(uint64) + addNewLinesToDebug("Extract Wired configuration profile: " + fmt.Sprint(wi.EapType)) } else { - viewErrorAndExit("Failed Extract Wired configuration profile", fmt.Sprint(eapType)) + viewErrorAndExit("Failed Extract Wired configuration profile", fmt.Sprint(wi.EapType)) } // User certificate configuration case "com.apple.security.pkcs12": - userCert := payloadContent["PayloadContent"].(string) - userCertPath = tempPath + "\\" + "certificate.p12" - err = createCertTempFile(userCert, userCertPath) - if err != nil { - viewErrorAndExit(T("cannotGenerateCertificateFile"), err.Error()) + wi.UserCert = payloadContent["PayloadContent"].(string) + wi.Err = createCertTempFile(wi.UserCert, wi.UserCertPath) + if wi.Err != nil { + viewErrorAndExit(T("cannotGenerateCertificateFile"), wi.Err.Error()) } else { - err = addCertToMachine(userCertDecode, CERTUTIL_PROGRAM_PATH) - if err != nil { - viewErrorAndExit(T("cannotDecodeProfileFile"), err.Error()) + wi.Err = addCertToMachine(wi.UserCertDecode, CERTUTIL_PROGRAM_PATH) + if wi.Err != nil { + viewErrorAndExit(T("cannotDecodeProfileFile"), wi.Err.Error()) } } // Certificate of Authority configuration case "com.apple.security.root": - if eapType == EAPTYPE_TLS { - caName := payloadContent["PayloadCertificateFileName"].(string) - caCert := payloadContent["PayloadContent"].(string) - cafilePath = tempPath + "\\" + caName + ".cer" - err = createCertTempFile(caCert, cafilePath) - if err != nil { - viewErrorAndExit(T("cannotGenerateCAFile"), err.Error()) + if wi.EapType == EAPTYPE_TLS { + wi.CaName = payloadContent["PayloadCertificateFileName"].(string) + wi.CaCert = payloadContent["PayloadContent"].(string) + wi.CaFilePath = wi.TempPath + "\\" + wi.CaName + ".cer" + wi.Err = createCertTempFile(wi.CaCert, wi.CaFilePath) + if wi.Err != nil { + viewErrorAndExit(T("cannotGenerateCAFile"), wi.Err.Error()) } - err = addCAToMachine(caFileBinary, CERTUTIL_PROGRAM_PATH) - if err != nil { - viewErrorAndExit(T("cannotDecodeProfileFile"), err.Error()) + wi.Err = addCAToMachine(wi.CaFileBinary, CERTUTIL_PROGRAM_PATH) + if wi.Err != nil { + viewErrorAndExit(T("cannotDecodeProfileFile"), wi.Err.Error()) } } default: @@ -305,21 +321,21 @@ func fetchXML() { sum += i } if shouldConfigureWifi { - configureWifi(xmlPlistProfile, wifiIndex, eapType, caFileBinary) + configureWifi() } if shouldConfigureWired { - configureWired(xmlPlistProfile, wiredIndex, eapType) + configureWired() } } // Configure wifi -func configureWifi(xmlPlistProfile map[string]interface{}, wifiIndex int, eapType uint64, caFileBinary string) { +func configureWifi() { var WLAN_ERROR_MESSAGE = T("wlanErrorMessage") var templateToFile string var elementsToReplaceInTemplate Template var wifiKey string // Get SSID information - payloadContent := xmlPlistProfile["PayloadContent"].([]interface{})[wifiIndex].(map[string]interface{}) + payloadContent := wi.XmlPlistProfile["PayloadContent"].([]interface{})[wi.WifiIndex].(map[string]interface{}) ssidString := payloadContent["SSID_STR"].(string) ssidStringToHex := hex.EncodeToString([]byte(ssidString)) ssidBroadcast := payloadContent["HIDDEN_NETWORK"].(bool) @@ -339,7 +355,7 @@ func configureWifi(xmlPlistProfile map[string]interface{}, wifiIndex int, eapTyp } else { userAuth = "certificate" } - if eapType == EAPTYPE_PEAP { + if wi.EapType == EAPTYPE_PEAP { // Search specific fields in wintemplate and replace them elementsToReplaceInTemplate = Template{ ProfileName: ssidString, @@ -350,8 +366,8 @@ func configureWifi(xmlPlistProfile map[string]interface{}, wifiIndex int, eapTyp } // executes the template templateToFile = executeTemplate(WIFI_PEAP_TEMPLATE_NAME, WIFI_PEAP_TEMPLATE, elementsToReplaceInTemplate) - } else if eapType == EAPTYPE_TLS { - caFingerprint, err := getCAFingerprint(caFileBinary) + } else if wi.EapType == EAPTYPE_TLS { + caFingerprint, err := getCAFingerprint(wi.CaFileBinary) if err != nil { viewErrorAndExit(T("Unable to get CA fingerprint."), err.Error()) } @@ -366,9 +382,9 @@ func configureWifi(xmlPlistProfile map[string]interface{}, wifiIndex int, eapTyp templateToFile = executeTemplate(WIFI_TLS_TEMPLATE_NAME, WIFI_TLS_TEMPLATE, elementsToReplaceInTemplate) } else { // error handling - viewErrorAndExit(T("unexpectedEAPType"), fmt.Sprint(eapType)) + viewErrorAndExit(T("unexpectedEAPType"), fmt.Sprint(wi.EapType)) } - addNewLinesToDebug("EAPType is " + fmt.Sprintf("%v", eapType)) + addNewLinesToDebug("EAPType is " + fmt.Sprint(wi.EapType)) } else { wifiKey = payloadContent["Password"].(string) addNewLinesToDebug("Security type: " + securityType) @@ -409,17 +425,16 @@ func configureWifi(xmlPlistProfile map[string]interface{}, wifiIndex int, eapTyp // creates profile file with the executed template createProfileFile(templateToFile) // prepare command line - wlanCmd := exec.Command("netsh", "wlan", "add", "profile", "filename="+templateOutPath, "user=all") + wi.WlanCmd = exec.Command("netsh", "wlan", "add", "profile", "filename="+wi.TemplateOutPath, "user=all") wlanSuccessMessage := T("The wireless profile was successfully added to the machine. \nPlease select your newly added profile " + ssidString + " in the WiFi networks.") // adds the new profile to Windows with netsh command - addProfileToMachine(wlanCmd, WLAN_ERROR_MESSAGE, wlanSuccessMessage) + addProfileToMachine(wi.WlanCmd, WLAN_ERROR_MESSAGE, wlanSuccessMessage) } // Configuration for wired -func configureWired(xmlPlistProfile map[string]interface{}, wiredIndex int, eapType uint64) { +func configureWired() { var WIRED_ERROR_MESSAGE = T("wiredErrorMessage") var WIRED_SUCCESS_MESSAGE = T("wiredSuccessMessage") - var err error dot3svc := exec.Command("net", "start", "dot3svc") dot3svc.Start() @@ -436,18 +451,19 @@ func configureWired(xmlPlistProfile map[string]interface{}, wiredIndex int, eapT } } // creates profile file with the executed template - if eapType == EAPTYPE_PEAP { + if wi.EapType == EAPTYPE_PEAP { createProfileFile(WIRED_PEAP_TEMPLATE) - } else if eapType == EAPTYPE_TLS { + } else if wi.EapType == EAPTYPE_TLS { createProfileFile(WIRED_TLS_TEMPLATE) } else { // error handling - viewErrorAndExit(T("unexpectedEAPType"), err.Error()) + wi.Err = errors.New("Not good EAP type: " + fmt.Sprint(wi.EapType)) + viewErrorAndExit(T("unexpectedEAPType"), wi.Err.Error()) } // prepare command line - wiredNetshCommand := exec.Command("netsh", "lan", "add", "profile", "filename="+profilePath) + wi.WiredCmd = exec.Command("netsh", "lan", "add", "profile", "filename="+wi.ProfilePath) // adds the new profile to Windows with netsh command - addProfileToMachine(wiredNetshCommand, WIRED_ERROR_MESSAGE, WIRED_SUCCESS_MESSAGE) + addProfileToMachine(wi.WiredCmd, WIRED_ERROR_MESSAGE, WIRED_SUCCESS_MESSAGE) } // Create, parse and execute templates @@ -464,7 +480,7 @@ func executeTemplate(nameTemplate, constTemplate string, templateToApply Templat if err != nil { viewErrorAndExit(T("cannotExecuteTemplate"), err.Error()) } else { - addNewLinesToDebug(T("executetemplateSuccess") + err.Error()) + addNewLinesToDebug(T("executetemplateSuccess")) } } return templateBuffer.String() @@ -473,14 +489,13 @@ func executeTemplate(nameTemplate, constTemplate string, templateToApply Templat // Create and write profile file into templateToFile folder func createProfileFile(templateToFile string) { // create and open file - templateFile, _ := createFile(templateOutPath) + templateFile, _ := createFile(wi.TemplateOutPath) // write the template into the new file _, err := io.Copy(templateFile, strings.NewReader(templateToFile)) - defer templateFile.Close() if err != nil { viewErrorAndExit(T("cannotWriteIntoProfileFile"), err.Error()) } else { - addNewLinesToDebug(T("profileCreationSuccess") + err.Error()) + addNewLinesToDebug(T("profileCreationSuccess")) } } diff --git a/utils.go b/utils.go index 76a1469..bd0fb6f 100644 --- a/utils.go +++ b/utils.go @@ -39,18 +39,18 @@ func createLanguageFile(currentDir, translationLanguage, languageFileName string } func addNewLinesToDebug(mytxt string) { - if debug { + if wi.Debug { log.Print(mytxt) myTxtC := chunkLargeStrings(mytxt, 60) - myCt := debugTxt.Text() - debugTxt.SetText(myCt + myTxtC + "\r\n") - numlines := strings.Count(debugTxt.Text(), "\r\n") + myCt := wi.Window.DebugTxt.Text() + wi.Window.DebugTxt.SetText(myCt + myTxtC + "\r\n") + numlines := strings.Count(wi.Window.DebugTxt.Text(), "\r\n") debugHeight := (13 * numlines) + 13 if debugHeight <= 240 { debugHeight = 240 } debugTxtsize := walk.Size{Width: 400, Height: debugHeight} - debugTxt.SetMinMaxSize(walk.Size(debugTxtsize), walk.Size(debugTxtsize)) + wi.Window.DebugTxt.SetMinMaxSize(walk.Size(debugTxtsize), walk.Size(debugTxtsize)) } else { log.Print(mytxt) } @@ -83,30 +83,30 @@ func chunkLargeStrings(s string, chunkSize int) string { // Enable the debug view func viewDebug() { - if debug { - debug = false + if wi.Debug { + wi.Debug = false // Change view scb, _ := walk.NewSolidColorBrush(walk.RGB(4, 5, 3)) - mw1.SetBackground(scb) - debugGrpBox.SetVisible(false) - imgView.SetVisible(true) + wi.Window.Mw1.SetBackground(scb) + wi.Window.DebugGrpBox.SetVisible(false) + wi.Window.ImgView.SetVisible(true) viewClosedButton(false) - configButton.SetText("Configure") + wi.Window.ConfigButton.SetText("Configure") } else { - debug = true + wi.Debug = true // Change view scb, _ := walk.NewSolidColorBrush(walk.RGB(255, 255, 255)) - mw1.SetBackground(scb) - debugGrpBox.SetVisible(true) - imgView.SetVisible(false) + wi.Window.Mw1.SetBackground(scb) + wi.Window.DebugGrpBox.SetVisible(true) + wi.Window.ImgView.SetVisible(false) viewClosedButton(true) - configButton.SetText("Configure with debug") + wi.Window.ConfigButton.SetText("Configure with debug") } } // Enable Close button view func viewClosedButton(b bool) { - closedButton.SetVisible(b) + wi.Window.ClosedButton.SetVisible(b) } // View Error @@ -114,7 +114,7 @@ func viewErrorAndExit(s string, sExtra string) { if sExtra != "" { addNewLinesToDebug(s + " " + sExtra) } - if !debug { + if !wi.Debug { walk.MsgBox(windowMsgBox, T("errorWindowTitle"), s+"\r\nPlease enable Debug Mode and contact your local support.", walk.MsgBoxOK) cleanAndExit(1) } @@ -122,22 +122,22 @@ func viewErrorAndExit(s string, sExtra string) { func cleanAndExit(c int) { cleanTmpFiles() - mw1.Close() + wi.Window.Mw1.Close() os.Exit(c) } func cleanTmpFiles() { - os.Remove(pngFilePath) - os.Remove(templateOutPath) - os.Remove(profilePath) - os.Remove(cafilePath) - os.Remove(userCertPath) + os.Remove(wi.PngFilePath) + os.Remove(wi.TemplateOutPath) + os.Remove(wi.ProfilePath) + os.Remove(wi.CaFilePath) + os.Remove(wi.UserCertPath) } // Prepare Background image // Converts base 64 background image to pf_bg.png func prepareBackgroundImage() { - debug = true + wi.Debug = true reader := base64.NewDecoder(base64.StdEncoding, strings.NewReader(BACKGROUND_IMAGE_PF)) decodeBase64ToPng, _, err := image.Decode(reader) addNewLinesToDebug("Welcome to PF debug") @@ -145,7 +145,7 @@ func prepareBackgroundImage() { viewErrorAndExit("Unable to decode base 64 background image.", err.Error()) } else { //Encode from image format to writer - backgroundFile, err := os.Create(pngFilePath) + backgroundFile, err := os.Create(wi.PngFilePath) if err != nil { viewErrorAndExit("Unable to open or create background image.", err.Error()) } else { @@ -153,22 +153,22 @@ func prepareBackgroundImage() { if err != nil { viewErrorAndExit("Unable to encode background image.", err.Error()) } else { - addNewLinesToDebug("PNG file " + pngFileName + " successfully created at " + pngFilePath) + addNewLinesToDebug("PNG file " + wi.PngFileName + " successfully created at " + wi.PngFilePath) backgroundFile.Close() } } } - img, err := walk.NewImageFromFile(pngFilePath) + img, err := walk.NewImageFromFile(wi.PngFilePath) if err != nil { - viewErrorAndExit("Unable to get the background image", " from "+pngFilePath+" and error is: "+err.Error()) + viewErrorAndExit("Unable to get the background image", " from "+wi.PngFilePath+" and error is: "+err.Error()) } else { - if err := imgView.SetImage(img); err != nil { + if err := wi.Window.ImgView.SetImage(img); err != nil { viewErrorAndExit("Unable to attach background image.", err.Error()) } else { addNewLinesToDebug("Been able to attach background image.") } } - debug = false + wi.Debug = false } // Create a file @@ -177,7 +177,6 @@ func createFile(filepath string) (*os.File, error) { if err != nil { viewErrorAndExit("Unable to create file "+filepath+".", err.Error()) } - //defer f.Close() return f, err } From d5ddd9a45c2f3d65060ad8efdef30c34a9986863 Mon Sep 17 00:00:00 2001 From: JeGoi <13801368+JeGoi@users.noreply.github.com> Date: Wed, 6 Oct 2021 11:50:40 -0400 Subject: [PATCH 14/18] Remove extra Variable Instanciate WinAgent* Objects --- parse.go | 15 ++++++--------- 1 file changed, 6 insertions(+), 9 deletions(-) diff --git a/parse.go b/parse.go index d8b3d75..a9d7d0c 100644 --- a/parse.go +++ b/parse.go @@ -95,7 +95,6 @@ type WinAgent struct { XmlPlistProfile map[string]interface{} } -var winAgentWindow *WinAgentWindow var wi *WinAgent func main() { @@ -178,12 +177,14 @@ func main() { } func prepareMainWindow() { + winAgent := new(WinAgent) + winAgentWindow := new(WinAgentWindow) + wi = winAgent + wi.Window = winAgentWindow currentWorkingDirectory, err := os.Executable() if err != nil { walk.MsgBox(windowMsgBox, "Error", "Unable to get current working directory, please contact your local support.", walk.MsgBoxOK) } - wi.Window = winAgentWindow - wi.StableCurrentWorkingDirectory = filepath.Dir(currentWorkingDirectory) wi.TempPath = os.Getenv("tmp") // Access to tmp path @@ -330,7 +331,6 @@ func fetchXML() { // Configure wifi func configureWifi() { - var WLAN_ERROR_MESSAGE = T("wlanErrorMessage") var templateToFile string var elementsToReplaceInTemplate Template var wifiKey string @@ -428,14 +428,11 @@ func configureWifi() { wi.WlanCmd = exec.Command("netsh", "wlan", "add", "profile", "filename="+wi.TemplateOutPath, "user=all") wlanSuccessMessage := T("The wireless profile was successfully added to the machine. \nPlease select your newly added profile " + ssidString + " in the WiFi networks.") // adds the new profile to Windows with netsh command - addProfileToMachine(wi.WlanCmd, WLAN_ERROR_MESSAGE, wlanSuccessMessage) + addProfileToMachine(wi.WlanCmd, T("wlanErrorMessage"), wlanSuccessMessage) } // Configuration for wired func configureWired() { - var WIRED_ERROR_MESSAGE = T("wiredErrorMessage") - var WIRED_SUCCESS_MESSAGE = T("wiredSuccessMessage") - dot3svc := exec.Command("net", "start", "dot3svc") dot3svc.Start() if err := dot3svc.Wait(); err != nil { @@ -463,7 +460,7 @@ func configureWired() { // prepare command line wi.WiredCmd = exec.Command("netsh", "lan", "add", "profile", "filename="+wi.ProfilePath) // adds the new profile to Windows with netsh command - addProfileToMachine(wi.WiredCmd, WIRED_ERROR_MESSAGE, WIRED_SUCCESS_MESSAGE) + addProfileToMachine(wi.WiredCmd, T("wiredErrorMessage"), T("wiredSuccessMessage")) } // Create, parse and execute templates From 98f0863f690174709750bd12d930ff45e29ebdc9 Mon Sep 17 00:00:00 2001 From: JeGoi <13801368+JeGoi@users.noreply.github.com> Date: Wed, 6 Oct 2021 12:12:39 -0400 Subject: [PATCH 15/18] Add message when exit on success --- parse.go | 4 +++- 1 file changed, 3 insertions(+), 1 deletion(-) diff --git a/parse.go b/parse.go index a9d7d0c..c409385 100644 --- a/parse.go +++ b/parse.go @@ -505,8 +505,10 @@ func addProfileToMachine(cmd *exec.Cmd, ErrorMessage, SuccessMessage string) { errorOut := "Error: " + err.Error() + "\r\n" outputOut := "Output: " + fmt.Sprint(output) + "\r\n" viewErrorAndExit("Failed adding profile", cmdLine+errorMess+errorOut+outputOut) + } else if wi.Debug { + addNewLinesToDebug("Success adding profile " + SuccessMessage) } else { - addNewLinesToDebug("Success adding profile" + SuccessMessage) + walk.MsgBox(windowMsgBox, "Success adding profile", SuccessMessage, walk.MsgBoxOK) cleanAndExit(0) } } From b5697eb27b4819445041ed7e6a1695f8a86ea201 Mon Sep 17 00:00:00 2001 From: Julien Semaan Date: Mon, 25 Oct 2021 07:57:01 -0700 Subject: [PATCH 16/18] ERROR_CANCELED is 64 bit long --- certificate.go | 2 +- parse.go | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/certificate.go b/certificate.go index 4be1976..458fa19 100644 --- a/certificate.go +++ b/certificate.go @@ -122,7 +122,7 @@ func addCertToMachine(userCertDecode string, CERTUTIL_PROGRAM_PATH string) error // Add CA to the machine func addCAToMachine(caFileBinary string, CERTUTIL_PROGRAM_PATH string) error { var err error - ERROR_CANCELED := 2147943623 + ERROR_CANCELED := int64(2147943623) runCommand := true for runCommand { runCommand = false diff --git a/parse.go b/parse.go index c409385..3ac4dc8 100644 --- a/parse.go +++ b/parse.go @@ -26,7 +26,7 @@ import ( ) const PROGRAM_NAME = "PacketFence Provisioning Agent" -const VERSION = "1.0.1" +const VERSION = "1.1.0" const CERTUTIL_PROGRAM_PATH = "C:\\Windows\\System32\\certutil.exe" const WIFI_PEAP_TEMPLATE_NAME = "wireless PEAP template" From c597b050aaac9696ed9d5b64fda663c7f6d282e7 Mon Sep 17 00:00:00 2001 From: JeGoi <13801368+JeGoi@users.noreply.github.com> Date: Tue, 26 Oct 2021 16:50:37 -0400 Subject: [PATCH 17/18] Remove extra error when install ca --- certificate.go | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/certificate.go b/certificate.go index 458fa19..9134d8c 100644 --- a/certificate.go +++ b/certificate.go @@ -148,7 +148,7 @@ func addCAToMachine(caFileBinary string, CERTUTIL_PROGRAM_PATH string) error { } } } else { - addNewLinesToDebug(T("caInstallationSuccess") + err.Error()) + addNewLinesToDebug(T("caInstallationSuccess")) } } return err From 867de2789b477d959bc99489f950f0569f195b38 Mon Sep 17 00:00:00 2001 From: JeGoi <13801368+JeGoi@users.noreply.github.com> Date: Mon, 1 Nov 2021 13:44:24 -0400 Subject: [PATCH 18/18] Change few debug lines + Add windows version --- go.mod | 1 + go.sum | 4 +++- parse.go | 18 +++++++++--------- utils.go | 28 ++++++++++++++++++++++++++++ 4 files changed, 41 insertions(+), 10 deletions(-) diff --git a/go.mod b/go.mod index e21a155..98da8ca 100644 --- a/go.mod +++ b/go.mod @@ -15,6 +15,7 @@ require ( github.com/BurntSushi/toml v0.4.1 // indirect github.com/davecgh/go-spew v1.1.1 // indirect github.com/pelletier/go-toml v1.2.1-0.20180724185102-c2dbbc24a979 // indirect + golang.org/x/sys v0.0.0-20211031064116-611d5d643895 gopkg.in/Knetic/govaluate.v3 v3.0.0 // indirect gopkg.in/yaml.v2 v2.2.1 // indirect ) diff --git a/go.sum b/go.sum index 44d98cc..99842f9 100644 --- a/go.sum +++ b/go.sum @@ -15,6 +15,8 @@ github.com/pelletier/go-toml v1.2.1-0.20180724185102-c2dbbc24a979 h1:kNmPAP94Bj9 github.com/pelletier/go-toml v1.2.1-0.20180724185102-c2dbbc24a979/go.mod h1:5z9KED0ma1S8pY6P1sdut58dfprrGBbd/94hg7ilaic= github.com/tink-ab/tempfile v0.0.0-20180226111222-33beb0518f1a h1:Qhm/9UKGO1+AjEKIsq8G72uCq4SrYxSxS5wiD0F3IC4= github.com/tink-ab/tempfile v0.0.0-20180226111222-33beb0518f1a/go.mod h1:Wt5qSdcHgX6XkqZKAZTxnN+93jnqtx0jEgTQakpZ1CE= +golang.org/x/sys v0.0.0-20211031064116-611d5d643895 h1:iaNpwpnrgL5jzWS0vCNnfa8HqzxveCFpFx3uC/X4Tps= +golang.org/x/sys v0.0.0-20211031064116-611d5d643895/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= gopkg.in/Knetic/govaluate.v3 v3.0.0 h1:18mUyIt4ZlRlFZAAfVetz4/rzlJs9yhN+U02F4u1AOc= gopkg.in/Knetic/govaluate.v3 v3.0.0/go.mod h1:csKLBORsPbafmSCGTEh3U7Ozmsuq8ZSIlKk1bcqph0E= gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405 h1:yhCVgyC4o1eVCa2tZl7eS0r+SDo693bJlVdllGtEeKM= @@ -22,4 +24,4 @@ gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8 gopkg.in/yaml.v2 v2.2.1 h1:mUhvW9EsL+naU5Q3cakzfE91YhliOondGd6ZrsDBHQE= gopkg.in/yaml.v2 v2.2.1/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= howett.net/plist v0.0.0-20180609054337-500bd5b9081b h1:r4LwkXZhdblHVSgAvfRjsFpQBorl6S9pAH+AOHVs+28= -howett.net/plist v0.0.0-20180609054337-500bd5b9081b/go.mod h1:jInWmjR7JRkkon4jlLXDZGVEeY/wo3kOOJEWYhNE+9Y= +howett.net/plist v0.0.0-20180609054337-500bd5b9081b/go.mod h1:jInWmjR7JRkkon4jlLXDZGVEeY/wo3kOOJEWYhNE+9Y= \ No newline at end of file diff --git a/parse.go b/parse.go index 3ac4dc8..956902f 100644 --- a/parse.go +++ b/parse.go @@ -19,7 +19,7 @@ import ( "text/template" "github.com/lxn/walk" - . "github.com/lxn/walk/declarative" + ."github.com/lxn/walk/declarative" "github.com/lxn/win" "github.com/nicksnyder/go-i18n/i18n" "howett.net/plist" @@ -89,6 +89,7 @@ type WinAgent struct { UserCert string UserCertDecode string UserCertPath string + VersionAfter2004 bool WifiIndex int Window *WinAgentWindow WiredIndex int @@ -222,7 +223,7 @@ func prepareEnv() { wi.ProfilePath = wi.TempPath + "\\profile.xml" wi.TemplateOutPath = wi.TempPath + "\\template-out.xml" wi.UserCertPath = wi.TempPath + "\\" + "certificate.p12" - + wi.VersionAfter2004 = windowsVersionAfter2004() wi.Debug = false } @@ -348,13 +349,12 @@ func configureWifi() { eapClientConfiguration, ok := payloadContent["EAPClientConfiguration"].(map[string]interface{}) if ok { userAuth, ok := eapClientConfiguration["UserName"].(string) - if ok { - if userAuth == "" { - userAuth = "certificate" - } - } else { + if !ok { + userAuth = "certificate" + } else if ok && userAuth == "" { userAuth = "certificate" } + if wi.EapType == EAPTYPE_PEAP { // Search specific fields in wintemplate and replace them elementsToReplaceInTemplate = Template{ @@ -384,10 +384,10 @@ func configureWifi() { // error handling viewErrorAndExit(T("unexpectedEAPType"), fmt.Sprint(wi.EapType)) } - addNewLinesToDebug("EAPType is " + fmt.Sprint(wi.EapType)) + addNewLinesToDebug("EAP Type is: " + fmt.Sprint(wi.EapType)) } else { wifiKey = payloadContent["Password"].(string) - addNewLinesToDebug("Security type: " + securityType) + addNewLinesToDebug("Security type is: " + securityType) switch securityType { case "WEP": elementsToReplaceInTemplate = Template{ diff --git a/utils.go b/utils.go index bd0fb6f..c5d5c6b 100644 --- a/utils.go +++ b/utils.go @@ -3,17 +3,20 @@ package main import ( "crypto/tls" "errors" + "fmt" "image" "io" "log" "net/http" "os" + "strconv" "strings" "encoding/base64" "image/png" "github.com/lxn/walk" + "golang.org/x/sys/windows/registry" ) // Create and write profile file into templateToFile folder @@ -38,6 +41,31 @@ func createLanguageFile(currentDir, translationLanguage, languageFileName string } } +func getCurrentBuikdWindows() int { + k, err := registry.OpenKey(registry.LOCAL_MACHINE, `SOFTWARE\Microsoft\Windows NT\CurrentVersion`, registry.QUERY_VALUE) + if err != nil { + addNewLinesToDebug("Failed to get the current windows version: " + fmt.Sprint(err.Error())) + } + defer k.Close() + cb, _, err := k.GetStringValue("CurrentBuild") + if err != nil { + addNewLinesToDebug("Failed to get the current build windows: " + fmt.Sprint(err.Error())) + } + i, err := strconv.ParseInt(cb, 10, 32) + return int(i) +} + +func windowsVersionAfter2004() bool { + i := getCurrentBuikdWindows() + if i >= 19041 { + addNewLinesToDebug("The current build windows is after 2004 version: " + fmt.Sprint(i)) + return true + } + // need to be developped + addNewLinesToDebug("The current build windows is before 2004 version: " + fmt.Sprint(i) + "So, may be fail for wireless if not admin") + return false +} + func addNewLinesToDebug(mytxt string) { if wi.Debug { log.Print(mytxt)