Как зашифровать строку в sha1 в base64 с помощью Swift?

Я хочу криптовать privateKey & publicKey к sha1 в base64 с помощью Swift, но вывод не тот, который я вижу в PHP urlencode base64_encode, который я пробовал в Codecademy: «https://www.codecademy.com/courses/web-beginner-en-StaFQ/0/3?curriculum_id=5124ef4c78d510dd89003eb8».

Просьба увидеть следующие коды в Swift и Codecademy:

Swift:

//pls see func dataFromHexadecimalString() details here "http://stackoverflow.com/questions/26501276/convert-string-to-hex-string-in-swift/26502285#26502285"
extension String {

func dataFromHexadecimalString() -> NSData? {
let trimmedString = self.stringByTrimmingCharactersInSet(NSCharacterSet(charactersInString: "<> ")).stringByReplacingOccurrencesOfString(" ", withString: "")

var error: NSError?
let regex = NSRegularExpression(pattern: "^[0-9a-f]*$", options: .CaseInsensitive, error: &error)
let found = regex?.firstMatchInString(trimmedString, options: nil, range: NSMakeRange(0, count(trimmedString)))
if found == nil || found?.range.location == NSNotFound || count(trimmedString) % 2 != 0 {
return nil
}

let data = NSMutableData(capacity: count(trimmedString) / 2)

for var index = trimmedString.startIndex; index < trimmedString.endIndex; index = index.successor().successor() {
let byteString = trimmedString.substringWithRange(Range<String.Index>(start: index, end: index.successor().successor()))
let num = UInt8(byteString.withCString { strtoul($0, nil, 16) })
data?.appendBytes([num] as [UInt8], length: 1)
}

return data
}
}

func URLEcodekey() -> String {
let appid="a1b2c34d5e"let privateKey="ef7d6s0d"let areaid="101020900"let time="201507191254"let publicKey="http://open.weather.com.cn/data/?areaid=\(areaid)&type=forecast_v&date=\(time)&appid=\(appid)"
let cPrivateKey=privateKey.dataUsingEncoding(NSUTF8StringEncoding)!
let cPublicKey=publicKey.dataUsingEncoding(NSUTF8StringEncoding)!
var cHMAC = [CUnsignedChar](count: Int(CC_SHA1_DIGEST_LENGTH), repeatedValue: 0)
CCHmac(CCHmacAlgorithm(kCCHmacAlgSHA1), cPublicKey.bytes, Int(cPublicKey.length), cPrivateKey.bytes, Int(cPrivateKey.length), &cHMAC)

let hexKeyString=NSMutableString(capacity: Int(CC_SHA1_DIGEST_LENGTH))
for byte in cHMAC{
hexKeyString.appendFormat("%02hhx", byte)
}
println("hexKeyString:\(encryptedKey)")

let binaryData = hexKeyString.dataFromHexadecimalString()
let base64String = binaryData?.base64EncodedStringWithOptions(nil)
println("base64String:\(base64String)")

var urlEncodeKey=base64String!.stringByAddingPercentEncodingWithAllowedCharacters(.URLHostAllowedCharacterSet())!
println("urlEncodeKey:\(urlEncodeKey)")
return urlEncodeMessage
}

выходы:

hexKeyString: d4433d42b1505c00a4aa80205171d0d04754d254

base64String: 1EM9QrFQXACkqoAgUXHQ0EdU0lQ =

urlEncodeKey: 1EM9QrFQXACkqoAgUXHQ0EdU0lQ =

PHP в Codecademy:

echo urlencode(base64_encode(hash_hmac('sha1', " http://open.weather.com.cn/data/?areaid=101020900&type=forecast_v&date=201507191254&appid=a1b2c34d5e", "ef7d6s0d", TRUE)));

выход:

A5O59Y% 2BFbGjhVwaI9JNB7DkcX% 2F4% 3D // выход очень похож на
пример в API, который, я думаю, может быть правильным.

Итак, как я могу получить право urlEncodeKey для моего частного ключа & publicKey как в PHP?

Заранее большое спасибо!

1

Решение

Вы должны прочитать больше о криптографии и хешировании. В вашем случае нет открытого ключа, закрытого ключа, … SHA означает Безопасный алгоритм хеширования и то, что вы пытаетесь получить, Код аутентификации на основе хэша. Проверьте статьи Википедии о HMAC, SHA-1, Открытый ключ, … Я настоятельно рекомендую прочитать больше об этом, иначе вы можете нанести больше урона, если неправильно поймете.

Вернемся к вашей проблеме. Это в одном персонаже:

  • Свифт код — let publicKey="http://open.weather.com.cn...
  • Код PHP — hash_hmac('sha1', " http://open.weather.com.cn...

Вы видите, где проблема? В вашем PHP-коде есть один пробел персонаж как раз перед http, Этого персонажа нет в вашем коде Swift.

Честно говоря, я не проверил весь ваш код, потому что не знаю, почему вы пытаетесь преобразовать его из шестнадцатеричной строки и т. Д. Использовали только некоторые части и переписали его с нуля. Вот рабочий пример:

func URLEncodedKey() -> String? {
let appid = "a1b2c34d5e"let time = "201507191254"let areaid = "101020900"
let key = "ef7d6s0d"let string = " http://open.weather.com.cn/data/?areaid=\(areaid)&type=forecast_v&date=\(time)&appid=\(appid)"//            ^  <- in your PHP example, there's space

guard let keyData = key.dataUsingEncoding(NSUTF8StringEncoding),
stringData = string.dataUsingEncoding(NSUTF8StringEncoding),
outputData = NSMutableData(length: Int(CC_SHA1_DIGEST_LENGTH)) else {
return nil
}
outputData.length = Int(CC_SHA1_DIGEST_LENGTH)

CCHmac(CCHmacAlgorithm(kCCHmacAlgSHA1),
keyData.bytes, keyData.length,
stringData.bytes, stringData.length,
outputData.mutableBytes)

return outputData
.base64EncodedStringWithOptions([])
.stringByAddingPercentEncodingWithAllowedCharacters(.URLQueryAllowedCharacterSet())
}

Возвращаемое значение:

`Optional("A5O59Y+FbGjhVwaI9JNB7DkcX/4=")`

Что вы получаете, когда декодируете свой вывод PHP.

Просто замени URLQueryAllowedCharacterSet с любым из следующих наборов символов:

class func URLUserAllowedCharacterSet() -> NSCharacterSet
class func URLPasswordAllowedCharacterSet() -> NSCharacterSet
class func URLHostAllowedCharacterSet() -> NSCharacterSet
class func URLPathAllowedCharacterSet() -> NSCharacterSet
class func URLQueryAllowedCharacterSet() -> NSCharacterSet
class func URLFragmentAllowedCharacterSet() -> NSCharacterSet

Зависит от вашего варианта использования. IOW, в какой части URL вы хотели бы использовать его.

6

Другие решения

Других решений пока нет …

По вопросам рекламы [email protected]