diff --git a/main-packr.go b/main-packr.go index 771f349..549aab9 100644 --- a/main-packr.go +++ b/main-packr.go @@ -9,9 +9,9 @@ import "github.com/gobuffalo/packr" // Go binary. You can use the "packr clean" command to clean up this, // and any other packr generated files. func init() { - packr.PackJSONBytes("./tmpl", "model.html", "\"e3tkZWZpbmUgIm1vZGVsIn19CnBhY2thZ2Uge3sgLlBhY2thZ2VOYW1lIH19CgovKi0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLSsKfCBDb2RlIGdlbmVyYXRlZCBieSBtb2RlbGdlbiB8CnwgICAgICAgIERPIE5PVCBFRElULiAgICAgICAgfAorLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tKi8KCi8vIElmIHlvdSB3YW50IHRvIGV4dGVuZCBiZWhhdmlvdXI6Ci8vIFBsZWFzZSBjcmVhdGUgYSBjdXN0b20gZmlsZSBpbiB0aGlzIGRpcmVjdG9yeQovLyBhbmQgY3JlYXRlIHRoZSBtZXRob2RzIHRoZXJlIHRvIGF2b2lkIG92ZXJ3cml0aW5nIHlvdXIgY29kZSB3aXRoIHRoZSBnZW5lcmF0ZWQgb25lLgoKaW1wb3J0ICgKImZtdCIKImRhdGFiYXNlL3NxbCIKICAgIHt7IHJhbmdlICRrLCAkdjo9IC5Nb2RlbC5JbXBvcnRzIH19CiAgICAie3ska319IgogICAge3stIGVuZCB9fQopCgovLyB7ey5Nb2RlbC5OYW1lfX0gcmVwcmVzZW50cyBhIHJvdyBpbiB0aGUge3suTW9kZWwuVGFibGVOYW1lfX0gdGFibGUKdHlwZSB7ey5Nb2RlbC5OYW1lfX0gc3RydWN0IHsKICAgIHt7IHJhbmdlICRrLCAkdjo9IC5Nb2RlbC5GaWVsZHMgfX0KICAgICAgICB7eyAkdi5OYW1lIH19IHt7ICR2LlR5cGUgfX0gYGpzb246Int7JHYuQ29sdW1uTmFtZX19ImAKICAgIHt7LSBlbmQgfX0Kb2Zmc2V0IGludApsaW1pdCBpbnQKfQoKLy8gSW5zZXJ0IGEgbmV3IHt7Lk1vZGVsLk5hbWV9fSByb3cgaW4gdGhlIHt7Lk1vZGVsLlRhYmxlTmFtZX19IHRhYmxlCmZ1bmMgKHt7LlJlY2VpdmVyfX0gKnt7Lk1vZGVsLk5hbWV9fSkgSW5zZXJ0KGRiICpzcWwuREIpIChsYXN0SW5zZXJ0SUQgaW50NjQsIGVyciBlcnJvcikgewogICAgY29uc3Qgc3RtdCA9ICJJTlNFUlQgSU5UTyB7ey5Nb2RlbC5UYWJsZU5hbWV9fSAoe3suTW9kZWwuRmllbGRzIHwgaW5zZXJ0X2ZpZWxkc319KSBWQUxVRVMgKHt7Lk1vZGVsLkZpZWxkcyB8IGluc2VydF92YWx1ZXN9fSkiCiAgICByZXMsIGVyciA6PSBkYi5FeGVjKHN0bXQsIHt7IC4gfCBpbnNlcnRfYXJncyB9fSkKICAgIGlmIGVyciAhPSBuaWwgewogICAgICAgIHJldHVybiAwLCBlcnIKICAgIH0KICAgIHJldHVybiByZXMuTGFzdEluc2VydElkKCkKfQoKLy8gVXBkYXRlIGFuIGV4aXN0aW5nIHt7Lk1vZGVsLk5hbWV9fSByb3cgaW4gdGhlIHt7Lk1vZGVsLlRhYmxlTmFtZX19IHRhYmxlLgpmdW5jICh7ey5SZWNlaXZlcn19ICp7ey5Nb2RlbC5OYW1lfX0pIFVwZGF0ZShkYiAqc3FsLkRCLCBpZCBpbnQ2NCkgKGludDY0LCBlcnJvcikgewogICAgY29uc3Qgc3RtdCA9ICJVUERBVEUge3suTW9kZWwuVGFibGVOYW1lfX0gU0VUIHt7IC4gfCB1cGRhdGVfdmFsdWVzIH19IFdIRVJFIGlkID0gPyIKICAgIHJlc3VsdCwgZXJyIDo9IGRiLkV4ZWMoc3RtdCwge3sgLiB8IHVwZGF0ZV9hcmdzIH19ICwgaWQpCiAgICBpZiBlcnIgIT0gbmlsIHsKICAgICAgICByZXR1cm4gMCwgZXJyCiAgICB9CiAgICByZXR1cm4gcmVzdWx0LlJvd3NBZmZlY3RlZCgpCn0KCi8vIFVwc2VydCBpbnNlcnRzIGEgbmV3IHt7Lk1vZGVsLk5hbWV9fSByb3cgaW4gdGhlIHt7Lk1vZGVsLlRhYmxlTmFtZX19IHRhYmxlCi8vIGlmIHRoZSB1bmlxdWUgY29uc3RyYWludHMgYXJlIG5vdCBmb3VuZCwgb3RoZXJ3aXNlIGl0IHVwZGF0ZXMgaXQuCmZ1bmMgKHt7LlJlY2VpdmVyfX0gKnt7Lk1vZGVsLk5hbWV9fSkgVXBzZXJ0KGRiICpzcWwuREIpIChsYXN0SW5zZXJ0SUQgaW50NjQsIGVyciBlcnJvcikgewogICAgY29uc3Qgc3RtdCA9ICJJTlNFUlQgSU5UTyB7ey5Nb2RlbC5UYWJsZU5hbWV9fSAoe3suTW9kZWwuRmllbGRzIHwgdXBzZXJ0X2ZpZWxkc319KSBWQUxVRVMgKHt7Lk1vZGVsLkZpZWxkcyB8IHVwc2VydF92YWx1ZXN9fSkgT04gRFVQTElDQVRFIEtFWSBVUERBVEUge3sgLiB8IHVwc2VydF9vbl9kdXBsaWNhdGUgfX0iCiAgICByZXMsIGVyciA6PSBkYi5FeGVjKHN0bXQsIHt7IC4gfCB1cHNlcnRfYXJncyB9fSkKICAgIGlmIGVyciAhPSBuaWwgewogICAgICAgIHJldHVybiAwLCBlcnIKICAgIH0KICAgIHJldHVybiByZXMuTGFzdEluc2VydElkKCkKfQoKLy8gRmluZCBhbiBleGlzdGluZyB7ey5Nb2RlbC5OYW1lfX0gcm93IGluIHRoZSB7ey5Nb2RlbC5UYWJsZU5hbWV9fSB0YWJsZQpmdW5jICh7ey5SZWNlaXZlcn19ICp7ey5Nb2RlbC5OYW1lfX0pIEZpbmQoZGIgKnNxbC5EQiwgaWQgaW50NjQpIGVycm9yIHsKICAgIGNvbnN0IHN0bXQgPSAiU0VMRUNUICogRlJPTSB7ey5Nb2RlbC5UYWJsZU5hbWV9fSBXSEVSRSBpZCA9ID8iCiAgICByb3cgOj0gZGIuUXVlcnlSb3coc3RtdCwgaWQpCiAgICByZXR1cm4gcm93LlNjYW4oe3sgLiB8IHNjYW5fZmllbGRzfX0pCn0KCi8vIExvYWQgYWxsLCBvciBhIHN1YnNldCBvZiB7ey5Nb2RlbC5OYW1lfX0gcm93cyBmcm9tIHRoZSB7ey5Nb2RlbC5UYWJsZU5hbWV9fSB0YWJsZQpmdW5jICh7ey5SZWNlaXZlcn19ICp7ey5Nb2RlbC5OYW1lfX0pIExvYWQoZGIgKnNxbC5EQikgKHNldCBbXXt7Lk1vZGVsLk5hbWV9fSwgZXJyIGVycm9yKSB7CiAgICBzdG10IDo9ICJTRUxFQ1QgKiBGUk9NIHt7Lk1vZGVsLlRhYmxlTmFtZX19IgoKICAgIGlmIHt7LlJlY2VpdmVyfX0ubGltaXQgPT0gMCAmJiB7ey5SZWNlaXZlcn19Lm9mZnNldCA+IDAgewogICAgICAgIHJldHVybiBzZXQsIGZtdC5FcnJvcmYoImNhbm5vdCBxdWVyeSB3aXRoIG9mZnNldCBidXQgbm8gbGltaXQiKQogICAgfQoKICAgIGlmIHt7LlJlY2VpdmVyfX0ubGltaXQgPiAwIHsKICAgICAgICBzdG10ICs9IGZtdC5TcHJpbnRmKCIgTElNSVQgJWQiLCB7ey5SZWNlaXZlcn19LmxpbWl0KQogICAgfQogICAgaWYge3suUmVjZWl2ZXJ9fS5vZmZzZXQgPiAwIHsKICAgICAgICBzdG10ICs9IGZtdC5TcHJpbnRmKCIgT0ZGU0VUICVkIiwge3suUmVjZWl2ZXJ9fS5vZmZzZXQpCiAgICB9CiAgICBkZWZlciBmdW5jKCkgewogICAgICAgIHt7LlJlY2VpdmVyfX0ubGltaXQgPSAwCiAgICAgICAge3suUmVjZWl2ZXJ9fS5vZmZzZXQgPSAwCiAgICB9KCkKICAgIHJvd3MsIGVyciA6PSBkYi5RdWVyeShzdG10KQogICAgaWYgZXJyICE9IG5pbCB7CiAgICAgICAgcmV0dXJuCiAgICB9CiAgICBkZWZlciByb3dzLkNsb3NlKCkKICAgICAgICBmb3Igcm93cy5OZXh0KCkgewogICAgICAgICAgICBpZiBlcnIgPSByb3dzLlNjYW4oe3suIHwgc2Nhbl9maWVsZHN9fSk7IGVyciAhPSBuaWwgewogICAgICAgICAgICByZXR1cm4KICAgICAgICB9CiAgICAgICAgc2V0ID0gYXBwZW5kKHNldCwgKnt7LlJlY2VpdmVyfX0pCiAgICB9CgogICAgcmV0dXJuCn0KCi8vIERlbGV0ZSBhbiBleGlzdGluZyB7ey5Nb2RlbC5OYW1lfX0gcm93IGZyb20gdGhlIHt7Lk1vZGVsLlRhYmxlTmFtZX19IHRhYmxlCmZ1bmMgKHt7LlJlY2VpdmVyfX0gKnt7Lk1vZGVsLk5hbWV9fSkgRGVsZXRlKGRiICpzcWwuREIsIGlkIGludDY0KSAocm93c0FmZmVjdGVkIGludDY0LCBlcnIgZXJyb3IpIHsKICAgIGNvbnN0IHN0bXQgPSAiREVMRVRFIEZST00ge3suTW9kZWwuVGFibGVOYW1lfX0gV0hFUkUgaWQgPSA/IgogICAgcmVzdWx0LCBlcnIgOj0gZGIuRXhlYyhzdG10LCBpZCkKCWlmIGVyciAhPSBuaWwgewoJCXJldHVybgoJfQoKCXJldHVybiByZXN1bHQuUm93c0FmZmVjdGVkKCkKfQoKLy8gQ291bnQgdGhlIG51bWJlciBvZiByb3dzIGZyb20gdGhlIHt7Lk1vZGVsLlRhYmxlTmFtZX19IHRhYmxlCmZ1bmMoe3suUmVjZWl2ZXJ9fSAqe3suTW9kZWwuTmFtZX19KSBDb3VudChkYiAqc3FsLkRCKSAoY291bnQgaW50NjQsIGVyciBlcnJvcikgewogICAgY29uc3Qgc3RtdCA9ICJTRUxFQ1QgQ09VTlQoKikgRlJPTSB7ey5Nb2RlbC5UYWJsZU5hbWV9fSIKICAgIHJvdyA6PSBkYi5RdWVyeVJvdyhzdG10KQogICAgaWYgZXJyID0gcm93LlNjYW4oJmNvdW50KTsgZXJyICE9IG5pbCB7CiAgICAgICAgcmV0dXJuCiAgICB9CiAgICByZXR1cm4KfQoKLy8gRXhpc3RzIGNoZWNrcyBmb3IgdGhlIGl0ZW1zIGV4aXN0ZW5jZSBpbiB0aGUgZGF0YWJhc2UsIGJhc2VkIG9uIGl0J3MgaWQuCi8vIEFuIGVycm9yIHdpbGwgb25seSBiZSByZXR1cm5lZCBpZiBhIFNRTCByZWxhdGVkIGZhaWx1cmUgaGFwcGVucy4KLy8gSW4gYWxsIG90aGVyIGNhc2VzLCBhIGJvb2wgYW5kIG5pbCB3aWxsIHJldHVybi4KZnVuYyh7ey5SZWNlaXZlcn19ICp7ey5Nb2RlbC5OYW1lfX0pIEV4aXN0cyhkYiAqc3FsLkRCLCBpZCBpbnQ2NCkgKGV4aXN0cyBib29sLCBlcnIgZXJyb3IpIHsKICAgIGNvbnN0IHN0bXQgPSAiU0VMRUNUIEVYSVNUUyhTRUxFQ1QgMSBGUk9NIHt7Lk1vZGVsLlRhYmxlTmFtZX19IFdIRVJFIGlkID0gPyBMSU1JVCAxKSBBUyBgZXhpc3RzYCIKICAgIHZhciBjb3VudCBpbnQKICAgIHJvdyA6PSBkYi5RdWVyeVJvdyhzdG10LCBpZCkKICAgIGlmIGVyciA9IHJvdy5TY2FuKCZjb3VudCk7IGVyciAhPSBuaWwgewogICAgICAgIHJldHVybgogICAgfQogICAgcmV0dXJuIGNvdW50ID4gMCwgbmlsCn0KCi8vIFRhYmxlTmFtZSByZXR1cm5zIHRoZSB0YWJsZSBuYW1lCmZ1bmMgKHt7LlJlY2VpdmVyfX0gKnt7Lk1vZGVsLk5hbWV9fSkgVGFibGVOYW1lKCkgc3RyaW5nIHsKcmV0dXJuICJ7ey5Nb2RlbC5UYWJsZU5hbWV9fSIKfQoKLy8gU2V0TGltaXQgc2V0cyB0aGUgcXVlcnkgbGltaXQKZnVuYyAoe3suUmVjZWl2ZXJ9fSAqe3suTW9kZWwuTmFtZX19KSBTZXRMaW1pdChsaW1pdCBpbnQpICp7ey5Nb2RlbC5OYW1lfX0gewp7ey5SZWNlaXZlcn19LmxpbWl0ID0gbGltaXQKcmV0dXJuIHt7LlJlY2VpdmVyfX0KfQoKLy8gU2V0T2Zmc2V0IHNldHMgdGhlIHF1ZXJ5IG9mZnNldApmdW5jICh7ey5SZWNlaXZlcn19ICp7ey5Nb2RlbC5OYW1lfX0pIFNldE9mZnNldChvZmZzZXQgaW50KSAqe3suTW9kZWwuTmFtZX19IHsKe3suUmVjZWl2ZXJ9fS5vZmZzZXQgPSBvZmZzZXQKcmV0dXJuIHt7LlJlY2VpdmVyfX0KfQp7e2VuZH19Cgo=\"") + packr.PackJSONBytes("./tmpl", "model.html", "\"e3tkZWZpbmUgIm1vZGVsIn19CnBhY2thZ2Uge3sgLlBhY2thZ2VOYW1lIH19CgovKi0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLSsKfCBDb2RlIGdlbmVyYXRlZCBieSBtb2RlbGdlbiB8CnwgICAgICAgIERPIE5PVCBFRElULiAgICAgICAgfAorLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tKi8KCi8vIElmIHlvdSB3YW50IHRvIGV4dGVuZCBiZWhhdmlvdXI6Ci8vIFBsZWFzZSBjcmVhdGUgYSBjdXN0b20gZmlsZSBpbiB0aGlzIGRpcmVjdG9yeQovLyBhbmQgY3JlYXRlIHRoZSBtZXRob2RzIHRoZXJlIHRvIGF2b2lkIG92ZXJ3cml0aW5nIHlvdXIgY29kZSB3aXRoIHRoZSBnZW5lcmF0ZWQgb25lLgoKaW1wb3J0ICgKImZtdCIKICAgIHt7IHJhbmdlICRrLCAkdjo9IC5Nb2RlbC5JbXBvcnRzIH19CiAgICAie3ska319IgogICAge3stIGVuZCB9fQopCgovLyB7ey5Nb2RlbC5OYW1lfX0gcmVwcmVzZW50cyBhIHJvdyBpbiB0aGUge3suTW9kZWwuVGFibGVOYW1lfX0gdGFibGUKdHlwZSB7ey5Nb2RlbC5OYW1lfX0gc3RydWN0IHsKICAgIHt7IHJhbmdlICRrLCAkdjo9IC5Nb2RlbC5GaWVsZHMgfX0KICAgICAgICB7eyAkdi5OYW1lIH19IHt7ICR2LlR5cGUgfX0gYGpzb246Int7JHYuQ29sdW1uTmFtZX19ImAKICAgIHt7LSBlbmQgfX0Kb2Zmc2V0IGludApsaW1pdCBpbnQKfQoKLy8gSW5zZXJ0IGEgbmV3IHt7Lk1vZGVsLk5hbWV9fSByb3cgaW4gdGhlIHt7Lk1vZGVsLlRhYmxlTmFtZX19IHRhYmxlCmZ1bmMgKHt7LlJlY2VpdmVyfX0gKnt7Lk1vZGVsLk5hbWV9fSkgSW5zZXJ0KHEgUXVlcnllcikgKGxhc3RJbnNlcnRJRCBpbnQ2NCwgZXJyIGVycm9yKSB7CiAgICBjb25zdCBzdG10ID0gIklOU0VSVCBJTlRPIHt7Lk1vZGVsLlRhYmxlTmFtZX19ICh7ey5Nb2RlbC5GaWVsZHMgfCBpbnNlcnRfZmllbGRzfX0pIFZBTFVFUyAoe3suTW9kZWwuRmllbGRzIHwgaW5zZXJ0X3ZhbHVlc319KSIKICAgIHJlcywgZXJyIDo9IHEuRXhlYyhzdG10LCB7eyAuIHwgaW5zZXJ0X2FyZ3MgfX0pCiAgICBpZiBlcnIgIT0gbmlsIHsKICAgICAgICByZXR1cm4gMCwgZXJyCiAgICB9CiAgICByZXR1cm4gcmVzLkxhc3RJbnNlcnRJZCgpCn0KCi8vIFVwZGF0ZSBhbiBleGlzdGluZyB7ey5Nb2RlbC5OYW1lfX0gcm93IGluIHRoZSB7ey5Nb2RlbC5UYWJsZU5hbWV9fSB0YWJsZS4KZnVuYyAoe3suUmVjZWl2ZXJ9fSAqe3suTW9kZWwuTmFtZX19KSBVcGRhdGUocSBRdWVyeWVyLCBpZCBpbnQ2NCkgKGludDY0LCBlcnJvcikgewogICAgY29uc3Qgc3RtdCA9ICJVUERBVEUge3suTW9kZWwuVGFibGVOYW1lfX0gU0VUIHt7IC4gfCB1cGRhdGVfdmFsdWVzIH19IFdIRVJFIGlkID0gPyIKICAgIHJlc3VsdCwgZXJyIDo9IHEuRXhlYyhzdG10LCB7eyAuIHwgdXBkYXRlX2FyZ3MgfX0gLCBpZCkKICAgIGlmIGVyciAhPSBuaWwgewogICAgICAgIHJldHVybiAwLCBlcnIKICAgIH0KICAgIHJldHVybiByZXN1bHQuUm93c0FmZmVjdGVkKCkKfQoKLy8gVXBzZXJ0IGluc2VydHMgYSBuZXcge3suTW9kZWwuTmFtZX19IHJvdyBpbiB0aGUge3suTW9kZWwuVGFibGVOYW1lfX0gdGFibGUKLy8gaWYgdGhlIHVuaXF1ZSBjb25zdHJhaW50cyBhcmUgbm90IGZvdW5kLCBvdGhlcndpc2UgaXQgdXBkYXRlcyBpdC4KZnVuYyAoe3suUmVjZWl2ZXJ9fSAqe3suTW9kZWwuTmFtZX19KSBVcHNlcnQocSBRdWVyeWVyKSAobGFzdEluc2VydElEIGludDY0LCBlcnIgZXJyb3IpIHsKICAgIGNvbnN0IHN0bXQgPSAiSU5TRVJUIElOVE8ge3suTW9kZWwuVGFibGVOYW1lfX0gKHt7Lk1vZGVsLkZpZWxkcyB8IHVwc2VydF9maWVsZHN9fSkgVkFMVUVTICh7ey5Nb2RlbC5GaWVsZHMgfCB1cHNlcnRfdmFsdWVzfX0pIE9OIERVUExJQ0FURSBLRVkgVVBEQVRFIHt7IC4gfCB1cHNlcnRfb25fZHVwbGljYXRlIH19IgogICAgcmVzLCBlcnIgOj0gcS5FeGVjKHN0bXQsIHt7IC4gfCB1cHNlcnRfYXJncyB9fSkKICAgIGlmIGVyciAhPSBuaWwgewogICAgICAgIHJldHVybiAwLCBlcnIKICAgIH0KICAgIHJldHVybiByZXMuTGFzdEluc2VydElkKCkKfQoKLy8gRmluZCBhbiBleGlzdGluZyB7ey5Nb2RlbC5OYW1lfX0gcm93IGluIHRoZSB7ey5Nb2RlbC5UYWJsZU5hbWV9fSB0YWJsZQpmdW5jICh7ey5SZWNlaXZlcn19ICp7ey5Nb2RlbC5OYW1lfX0pIEZpbmQocSBRdWVyeWVyLCBpZCBpbnQ2NCkgZXJyb3IgewogICAgY29uc3Qgc3RtdCA9ICJTRUxFQ1QgKiBGUk9NIHt7Lk1vZGVsLlRhYmxlTmFtZX19IFdIRVJFIGlkID0gPyIKICAgIHJvdyA6PSBxLlF1ZXJ5Um93KHN0bXQsIGlkKQogICAgcmV0dXJuIHJvdy5TY2FuKHt7IC4gfCBzY2FuX2ZpZWxkc319KQp9CgovLyBMb2FkIGFsbCwgb3IgYSBzdWJzZXQgb2Yge3suTW9kZWwuTmFtZX19IHJvd3MgZnJvbSB0aGUge3suTW9kZWwuVGFibGVOYW1lfX0gdGFibGUKZnVuYyAoe3suUmVjZWl2ZXJ9fSAqe3suTW9kZWwuTmFtZX19KSBMb2FkKHEgUXVlcnllcikgKHNldCBbXXt7Lk1vZGVsLk5hbWV9fSwgZXJyIGVycm9yKSB7CiAgICBzdG10IDo9ICJTRUxFQ1QgKiBGUk9NIHt7Lk1vZGVsLlRhYmxlTmFtZX19IgoKICAgIGlmIHt7LlJlY2VpdmVyfX0ubGltaXQgPT0gMCAmJiB7ey5SZWNlaXZlcn19Lm9mZnNldCA+IDAgewogICAgICAgIHJldHVybiBzZXQsIGZtdC5FcnJvcmYoImNhbm5vdCBxdWVyeSB3aXRoIG9mZnNldCBidXQgbm8gbGltaXQiKQogICAgfQoKICAgIGlmIHt7LlJlY2VpdmVyfX0ubGltaXQgPiAwIHsKICAgICAgICBzdG10ICs9IGZtdC5TcHJpbnRmKCIgTElNSVQgJWQiLCB7ey5SZWNlaXZlcn19LmxpbWl0KQogICAgfQogICAgaWYge3suUmVjZWl2ZXJ9fS5vZmZzZXQgPiAwIHsKICAgICAgICBzdG10ICs9IGZtdC5TcHJpbnRmKCIgT0ZGU0VUICVkIiwge3suUmVjZWl2ZXJ9fS5vZmZzZXQpCiAgICB9CiAgICBkZWZlciBmdW5jKCkgewogICAgICAgIHt7LlJlY2VpdmVyfX0ubGltaXQgPSAwCiAgICAgICAge3suUmVjZWl2ZXJ9fS5vZmZzZXQgPSAwCiAgICB9KCkKICAgIHJvd3MsIGVyciA6PSBxLlF1ZXJ5KHN0bXQpCiAgICBpZiBlcnIgIT0gbmlsIHsKICAgICAgICByZXR1cm4KICAgIH0KICAgIGRlZmVyIHJvd3MuQ2xvc2UoKQogICAgICAgIGZvciByb3dzLk5leHQoKSB7CiAgICAgICAgICAgIGlmIGVyciA9IHJvd3MuU2Nhbih7ey4gfCBzY2FuX2ZpZWxkc319KTsgZXJyICE9IG5pbCB7CiAgICAgICAgICAgIHJldHVybgogICAgICAgIH0KICAgICAgICBzZXQgPSBhcHBlbmQoc2V0LCAqe3suUmVjZWl2ZXJ9fSkKICAgIH0KCiAgICByZXR1cm4KfQoKLy8gRGVsZXRlIGFuIGV4aXN0aW5nIHt7Lk1vZGVsLk5hbWV9fSByb3cgZnJvbSB0aGUge3suTW9kZWwuVGFibGVOYW1lfX0gdGFibGUKZnVuYyAoe3suUmVjZWl2ZXJ9fSAqe3suTW9kZWwuTmFtZX19KSBEZWxldGUocSBRdWVyeWVyLCBpZCBpbnQ2NCkgKHJvd3NBZmZlY3RlZCBpbnQ2NCwgZXJyIGVycm9yKSB7CiAgICBjb25zdCBzdG10ID0gIkRFTEVURSBGUk9NIHt7Lk1vZGVsLlRhYmxlTmFtZX19IFdIRVJFIGlkID0gPyIKICAgIHJlc3VsdCwgZXJyIDo9IHEuRXhlYyhzdG10LCBpZCkKCWlmIGVyciAhPSBuaWwgewoJCXJldHVybgoJfQoKCXJldHVybiByZXN1bHQuUm93c0FmZmVjdGVkKCkKfQoKLy8gQ291bnQgdGhlIG51bWJlciBvZiByb3dzIGZyb20gdGhlIHt7Lk1vZGVsLlRhYmxlTmFtZX19IHRhYmxlCmZ1bmMoe3suUmVjZWl2ZXJ9fSAqe3suTW9kZWwuTmFtZX19KSBDb3VudChxIFF1ZXJ5ZXIpIChjb3VudCBpbnQ2NCwgZXJyIGVycm9yKSB7CiAgICBjb25zdCBzdG10ID0gIlNFTEVDVCBDT1VOVCgqKSBGUk9NIHt7Lk1vZGVsLlRhYmxlTmFtZX19IgogICAgcm93IDo9IHEuUXVlcnlSb3coc3RtdCkKICAgIGlmIGVyciA9IHJvdy5TY2FuKCZjb3VudCk7IGVyciAhPSBuaWwgewogICAgICAgIHJldHVybgogICAgfQogICAgcmV0dXJuCn0KCi8vIEV4aXN0cyBjaGVja3MgZm9yIHRoZSBpdGVtcyBleGlzdGVuY2UgaW4gdGhlIGRhdGFiYXNlLCBiYXNlZCBvbiBpdCdzIGlkLgovLyBBbiBlcnJvciB3aWxsIG9ubHkgYmUgcmV0dXJuZWQgaWYgYSBTUUwgcmVsYXRlZCBmYWlsdXJlIGhhcHBlbnMuCi8vIEluIGFsbCBvdGhlciBjYXNlcywgYSBib29sIGFuZCBuaWwgd2lsbCByZXR1cm4uCmZ1bmMoe3suUmVjZWl2ZXJ9fSAqe3suTW9kZWwuTmFtZX19KSBFeGlzdHMocSBRdWVyeWVyLCBpZCBpbnQ2NCkgKGV4aXN0cyBib29sLCBlcnIgZXJyb3IpIHsKICAgIGNvbnN0IHN0bXQgPSAiU0VMRUNUIEVYSVNUUyhTRUxFQ1QgMSBGUk9NIHt7Lk1vZGVsLlRhYmxlTmFtZX19IFdIRVJFIGlkID0gPyBMSU1JVCAxKSBBUyBgZXhpc3RzYCIKICAgIHZhciBjb3VudCBpbnQKICAgIHJvdyA6PSBxLlF1ZXJ5Um93KHN0bXQsIGlkKQogICAgaWYgZXJyID0gcm93LlNjYW4oJmNvdW50KTsgZXJyICE9IG5pbCB7CiAgICAgICAgcmV0dXJuCiAgICB9CiAgICByZXR1cm4gY291bnQgPiAwLCBuaWwKfQoKLy8gVGFibGVOYW1lIHJldHVybnMgdGhlIHRhYmxlIG5hbWUKZnVuYyAoe3suUmVjZWl2ZXJ9fSAqe3suTW9kZWwuTmFtZX19KSBUYWJsZU5hbWUoKSBzdHJpbmcgewpyZXR1cm4gInt7Lk1vZGVsLlRhYmxlTmFtZX19Igp9CgovLyBTZXRMaW1pdCBzZXRzIHRoZSBxdWVyeSBsaW1pdApmdW5jICh7ey5SZWNlaXZlcn19ICp7ey5Nb2RlbC5OYW1lfX0pIFNldExpbWl0KGxpbWl0IGludCkgKnt7Lk1vZGVsLk5hbWV9fSB7Cnt7LlJlY2VpdmVyfX0ubGltaXQgPSBsaW1pdApyZXR1cm4ge3suUmVjZWl2ZXJ9fQp9CgovLyBTZXRPZmZzZXQgc2V0cyB0aGUgcXVlcnkgb2Zmc2V0CmZ1bmMgKHt7LlJlY2VpdmVyfX0gKnt7Lk1vZGVsLk5hbWV9fSkgU2V0T2Zmc2V0KG9mZnNldCBpbnQpICp7ey5Nb2RlbC5OYW1lfX0gewp7ey5SZWNlaXZlcn19Lm9mZnNldCA9IG9mZnNldApyZXR1cm4ge3suUmVjZWl2ZXJ9fQp9Cnt7ZW5kfX0KCg==\"") packr.PackJSONBytes("./tmpl", "tmpl.go", "\"cGFja2FnZSB0bXBsCgppbXBvcnQgKAoJImZtdCIKCSJodG1sL3RlbXBsYXRlIgoJInN0cmluZ3MiCikKCnZhciBGdW5jTWFwID0gdGVtcGxhdGUuRnVuY01hcHsKCSJpbnNlcnRfZmllbGRzIjogR2V0SW5zZXJ0RmllbGRzLAoJImluc2VydF92YWx1ZXMiOiBHZXRJbnNlcnRWYWx1ZXMsCgkiaW5zZXJ0X2FyZ3MiOiAgIEdldEluc2VydEFyZ3MsCgkic2Nhbl9maWVsZHMiOiAgIEdldFNjYW5GaWVsZHMsCgkidXBkYXRlX2FyZ3MiOiAgIEdldFVwZGF0ZUFyZ3MsCgkidXBkYXRlX3ZhbHVlcyI6IEdldFVwZGF0ZVZhbHVlcywKCSJ1cHNlcnRfZmllbGRzIjogR2V0VXBzZXJ0RmllbGRzLAoJInVwc2VydF92YWx1ZXMiOiBHZXRVcHNlcnRWYWx1ZXMsCgkidXBzZXJ0X29uX2R1cGxpY2F0ZSI6IEdldFVwc2VydE9uRHVwbGljYXRlLAoJInVwc2VydF9hcmdzIjogR2V0VXBzZXJ0QXJncywKfQoKZnVuYyBHZXRJbnNlcnRGaWVsZHMoZmllbGRzIFtdVG1wbEZpZWxkKSBzdHJpbmcgewoJdmFyIHBhcnRzIFtdc3RyaW5nCglmb3IgXywgZmwgOj0gcmFuZ2UgZmllbGRzIHsKCQlpZiBmbC5Db2x1bW5OYW1lID09ICJpZCIgewoJCQljb250aW51ZQoJCX0KCQlwYXJ0cyA9IGFwcGVuZChwYXJ0cywgImAiK2ZsLkNvbHVtbk5hbWUrImAiKQoJfQoJcmV0dXJuIHN0cmluZ3MuSm9pbihwYXJ0cywgIiwgIikKfQoKZnVuYyBHZXRJbnNlcnRWYWx1ZXMoZmllbGRzIFtdVG1wbEZpZWxkKSBzdHJpbmcgewoJdmFyIHBhcnRzIFtdc3RyaW5nCglmb3IgXywgZmwgOj0gcmFuZ2UgZmllbGRzIHsKCQlzd2l0Y2ggZmwuQ29sdW1uTmFtZSB7CgkJY2FzZSAiaWQiOgoJCQljb250aW51ZQoJCWNhc2UgImNyZWF0ZWRfYXQiOgoJCQlwYXJ0cyA9IGFwcGVuZChwYXJ0cywgIk5PVygpIikKCQkJY29udGludWUKCQlkZWZhdWx0OgoJCQlwYXJ0cyA9IGFwcGVuZChwYXJ0cywgIj8iKQoJCX0KCX0KCXJldHVybiBzdHJpbmdzLkpvaW4ocGFydHMsICIsICIpCn0KCmZ1bmMgR2V0SW5zZXJ0QXJncyhtIFN0cnVjdFRtcGxEYXRhKSBzdHJpbmcgewoJdmFyIHBhcnRzIFtdc3RyaW5nCglmb3IgXywgZmwgOj0gcmFuZ2UgbS5Nb2RlbC5GaWVsZHMgewoJCXN3aXRjaCBmbC5OYW1lIHsKCQljYXNlICJJRCIsICJDcmVhdGVkQXQiOgoJCQljb250aW51ZQoJCX0KCQlwYXJ0cyA9IGFwcGVuZChwYXJ0cywgZm10LlNwcmludGYoIiVzLiVzIiwgbS5SZWNlaXZlciwgZmwuTmFtZSkpCgl9CglyZXR1cm4gc3RyaW5ncy5Kb2luKHBhcnRzLCAiLCAiKQp9CgpmdW5jIEdldFNjYW5GaWVsZHMobSBTdHJ1Y3RUbXBsRGF0YSkgdGVtcGxhdGUuSFRNTCB7Cgl2YXIgcGFydHMgW11zdHJpbmcKCWZvciBfLCBmbCA6PSByYW5nZSBtLk1vZGVsLkZpZWxkcyB7CgkJcGFydHMgPSBhcHBlbmQocGFydHMsIGZtdC5TcHJpbnRmKCImJXMuJXMiLCBtLlJlY2VpdmVyLCBmbC5OYW1lKSkKCX0KCXJldHVybiB0ZW1wbGF0ZS5IVE1MKHN0cmluZ3MuSm9pbihwYXJ0cywgIiwgIikpCn0KCmZ1bmMgR2V0VXBkYXRlQXJncyhtIFN0cnVjdFRtcGxEYXRhKSB0ZW1wbGF0ZS5IVE1MIHsKCXZhciBwYXJ0cyBbXXN0cmluZwoJZm9yIF8sIGZsIDo9IHJhbmdlIG0uTW9kZWwuRmllbGRzIHsKCQlzd2l0Y2ggZmwuTmFtZSB7CgkJY2FzZSAiSUQiLCAiQ3JlYXRlZEF0IiwgIlVwZGF0ZWRBdCI6CgkJCWNvbnRpbnVlCgkJfQoJCXBhcnRzID0gYXBwZW5kKHBhcnRzLCBmbXQuU3ByaW50ZigiJXMuJXMiLCBtLlJlY2VpdmVyLCBmbC5OYW1lKSkKCX0KCXJldHVybiB0ZW1wbGF0ZS5IVE1MKHN0cmluZ3MuSm9pbihwYXJ0cywgIiwgIikpCn0KCmZ1bmMgR2V0VXBkYXRlVmFsdWVzKG0gU3RydWN0VG1wbERhdGEpIHN0cmluZyB7Cgl2YXIgcGFydHMgW11zdHJpbmcKCWZvciBfLCBmbCA6PSByYW5nZSBtLk1vZGVsLkZpZWxkcyB7CgkJc3dpdGNoIGZsLk5hbWUgewoJCWNhc2UgIklEIiwgIkNyZWF0ZWRBdCI6CgkJCWNvbnRpbnVlCgkJY2FzZSAiVXBkYXRlZEF0IjoKCQkJcGFydHMgPSBhcHBlbmQocGFydHMsIGZtdC5TcHJpbnRmKCJgJXNgPVVUQ19USU1FU1RBTVAoKSIsIGZsLkNvbHVtbk5hbWUpKQoJCWRlZmF1bHQ6CgkJCXBhcnRzID0gYXBwZW5kKHBhcnRzLCBmbXQuU3ByaW50ZigiYCVzYD0/IiwgZmwuQ29sdW1uTmFtZSkpCgkJfQoJfQoJcmV0dXJuIHN0cmluZ3MuSm9pbihwYXJ0cywgIiwgIikKfQoKZnVuYyBHZXRVcHNlcnRGaWVsZHMoZmllbGRzIFtdVG1wbEZpZWxkKSBzdHJpbmcgewoJdmFyIHBhcnRzIFtdc3RyaW5nCglmb3IgXywgZmwgOj0gcmFuZ2UgZmllbGRzIHsKCQlwYXJ0cyA9IGFwcGVuZChwYXJ0cywgImAiK2ZsLkNvbHVtbk5hbWUrImAiKQoJfQoJcmV0dXJuIHN0cmluZ3MuSm9pbihwYXJ0cywgIiwgIikKfQoKZnVuYyBHZXRVcHNlcnRWYWx1ZXMoZmllbGRzIFtdVG1wbEZpZWxkKSBzdHJpbmcgewoJdmFyIHBhcnRzIFtdc3RyaW5nCglmb3IgXywgZmwgOj0gcmFuZ2UgZmllbGRzIHsKCQlzd2l0Y2ggZmwuQ29sdW1uTmFtZSB7CgkJY2FzZSAiY3JlYXRlZF9hdCI6CgkJCXBhcnRzID0gYXBwZW5kKHBhcnRzLCAiTk9XKCkiKQoJCQljb250aW51ZQoJCWRlZmF1bHQ6CgkJCXBhcnRzID0gYXBwZW5kKHBhcnRzLCAiPyIpCgkJfQoJfQoJcmV0dXJuIHN0cmluZ3MuSm9pbihwYXJ0cywgIiwgIikKfQoKZnVuYyBHZXRVcHNlcnRPbkR1cGxpY2F0ZShtIFN0cnVjdFRtcGxEYXRhKSBzdHJpbmcgewoJdmFyIHBhcnRzIFtdc3RyaW5nCglmb3IgXywgZmwgOj0gcmFuZ2UgbS5Nb2RlbC5GaWVsZHMgewoJCXN3aXRjaCBmbC5OYW1lIHsKCQljYXNlICJDcmVhdGVkQXQiOgoJCQljb250aW51ZQoJCWNhc2UgIklEIjoKCQkJcGFydHMgPSBhcHBlbmQocGFydHMsIGZtdC5TcHJpbnRmKCJgJXNgPUxBU1RfSU5TRVJUX0lEKGAlc2ApIiwgZmwuQ29sdW1uTmFtZSwgZmwuQ29sdW1uTmFtZSkpCgkJY2FzZSAiVXBkYXRlZEF0IjoKCQkJcGFydHMgPSBhcHBlbmQocGFydHMsIGZtdC5TcHJpbnRmKCJgJXNgPVVUQ19USU1FU1RBTVAoKSIsIGZsLkNvbHVtbk5hbWUpKQoJCWRlZmF1bHQ6CgkJCXBhcnRzID0gYXBwZW5kKHBhcnRzLCBmbXQuU3ByaW50ZigiYCVzYD1WQUxVRVMoYCVzYCkiLCBmbC5Db2x1bW5OYW1lLCBmbC5Db2x1bW5OYW1lKSkKCQl9Cgl9CglyZXR1cm4gc3RyaW5ncy5Kb2luKHBhcnRzLCAiLCAiKQp9CgpmdW5jIEdldFVwc2VydEFyZ3MobSBTdHJ1Y3RUbXBsRGF0YSkgc3RyaW5nIHsKCXZhciBwYXJ0cyBbXXN0cmluZwoJZm9yIF8sIGZsIDo9IHJhbmdlIG0uTW9kZWwuRmllbGRzIHsKCQlzd2l0Y2ggZmwuTmFtZSB7CgkJY2FzZSAiQ3JlYXRlZEF0IjoKCQkJY29udGludWUKCQl9CgkJcGFydHMgPSBhcHBlbmQocGFydHMsIGZtdC5TcHJpbnRmKCIlcy4lcyIsIG0uUmVjZWl2ZXIsIGZsLk5hbWUpKQoJfQoJcmV0dXJuIHN0cmluZ3MuSm9pbihwYXJ0cywgIiwgIikKfQ==\"") packr.PackJSONBytes("./tmpl", "types.go", "\"cGFja2FnZSB0bXBsCgovLyBUbXBsU3RydWN0cyBpcyBhIGNvbGxlY3Rpb24gb24gVG1wbFN0cnVjdAp0eXBlIFRtcGxTdHJ1Y3RzIFtdVG1wbFN0cnVjdAoKLy8gVG1wbFN0cnVjdCBkZWZpbmVzIHRoZSB0YWJsZSBkYXRhIHRvIHBhc3MgdG8gdGhlIG1vZGVscwp0eXBlIFRtcGxTdHJ1Y3Qgc3RydWN0IHsKCU5hbWUgICAgICBzdHJpbmcKCVRhYmxlTmFtZSBzdHJpbmcKCUZpZWxkcyAgICBbXVRtcGxGaWVsZAoJSW1wb3J0cyAgIG1hcFtzdHJpbmddc3RydWN0e30KfQoKLy8gVG1wbEZpZWxkIGRlZmluZXMgYSB0YWJsZSBmaWVsZCB0ZW1wbGF0ZQp0eXBlIFRtcGxGaWVsZCBzdHJ1Y3QgewoJTmFtZSAgICAgICBzdHJpbmcKCVR5cGUgICAgICAgc3RyaW5nCglDb2x1bW5OYW1lIHN0cmluZwoJTnVsbGFibGUgICBib29sCn0KCi8vIFN0cnVjdFRtcGxEYXRhIGRlZmluZXMgdGhlIHRvcCBsZXZlbCBzdHJ1Y3QgZGF0YSB0byBwYXNzIHRvIHRoZSBtb2RlbHMKdHlwZSBTdHJ1Y3RUbXBsRGF0YSBzdHJ1Y3QgewoJTW9kZWwgICAgICAgVG1wbFN0cnVjdAoJUmVjZWl2ZXIgICAgc3RyaW5nCglQYWNrYWdlTmFtZSBzdHJpbmcKfQo=\"") - packr.PackJSONBytes("./tmpl", "x_helpers.html", "\"{{define "helpers"}}

package {{ .PackageName }}

/*---------------------------+
| Code generated by modelgen |
|        DO NOT EDIT.        |
+---------------------------*/

import (
"database/sql"
"database/sql/driver"
"encoding/json"
"reflect"
"strings"
"time"

"github.com/go-sql-driver/mysql"
)

// StdTime provides default SQL TIME format
const StdTime = "15:04:05"

// emptyTime allows default times to be considered
// null for insertion into the database.
var emptyTime = time.Time{}

/*-------------+
| Type aliases |
+-------------*/

// NullFloat64 aliases sql.NullFloat64
type NullFloat64 sql.NullFloat64

// NullString aliases sql.NullString
type NullString sql.NullString

// NullBool aliases sql.NullBool
type NullBool sql.NullBool

// NullInt64 aliases sql.NullInt64
type NullInt64 sql.NullInt64

// NullTime aliases sql.NullTime
type NullTime mysql.NullTime

// RawJSON aliases json.RawMessage
type RawJSON json.RawMessage

/*---------------------------+
| NullString implementations |
+---------------------------*/

// MarshalJSON for NullString
func (n NullString) MarshalJSON() ([]byte, error) {
var a *string
if n.Valid {
a = &n.String
}
return json.Marshal(a)
}

// UnmarshalJSON for NullString
func (n *NullString) UnmarshalJSON(b []byte) error {
err := json.Unmarshal(b, &n.String)
n.Valid = err == nil
return err
}

// Value for NullString
func (n NullString) Value() (driver.Value, error) {
if !n.Valid {
return nil, nil
}
return n.String, nil
}

// Scan for NullString
func (n *NullString) Scan(src interface{}) error {
var a sql.NullString
if err := a.Scan(src); err != nil {
return err
}
n.String = a.String
if reflect.TypeOf(src) != nil {
n.Valid = true
}
return nil
}

/*----------------------------+
| NullFloat64 implementations |
+----------------------------*/

// MarshalJSON for NullFloat64
func (n NullFloat64) MarshalJSON() ([]byte, error) {
var a *float64
if n.Valid {
a = &n.Float64
}
return json.Marshal(a)
}

// Value for NullFloat64
func (n NullFloat64) Value() (driver.Value, error) {
if !n.Valid {
return nil, nil
}
return n.Float64, nil
}

// UnmarshalJSON for NullFloat64
func (n *NullFloat64) UnmarshalJSON(b []byte) error {
err := json.Unmarshal(b, &n.Float64)
n.Valid = err == nil
return err
}

// Scan for NullFloat64
func (n *NullFloat64) Scan(src interface{}) error {
var a sql.NullFloat64
if err := a.Scan(src); err != nil {
return err
}
n.Float64 = a.Float64
if reflect.TypeOf(src) != nil {
n.Valid = true
}
return nil
}

/*--------------------------+
| NullInt64 implementations |
+--------------------------*/

// MarshalJSON for NullInt64
func (n NullInt64) MarshalJSON() ([]byte, error) {
var a *int64
if n.Valid {
a = &n.Int64
}
return json.Marshal(a)
}

// Value for NullInt64
func (n NullInt64) Value() (driver.Value, error) {
if !n.Valid {
return nil, nil
}
return n.Int64, nil
}

// UnmarshalJSON for NullInt64
func (n *NullInt64) UnmarshalJSON(b []byte) error {
err := json.Unmarshal(b, &n.Int64)
n.Valid = err == nil
return err
}

// Scan for NullInt64
func (n *NullInt64) Scan(src interface{}) error {
var a sql.NullInt64
if err := a.Scan(src); err != nil {
return err
}
n.Int64 = a.Int64
if reflect.TypeOf(src) != nil {
n.Valid = true
}
return nil
}

/*-------------------------+
| NullBool implementations |
+-------------------------*/

// MarshalJSON for NullBool
func (n NullBool) MarshalJSON() ([]byte, error) {
var a *bool
if n.Valid {
a = &n.Bool
}
return json.Marshal(a)
}

// Value for NullBool
func (n NullBool) Value() (driver.Value, error) {
if !n.Valid {
return nil, nil
}
return n.Bool, nil
}

// UnmarshalJSON for NullBool
func (n *NullBool) UnmarshalJSON(b []byte) error {
var field *bool
err := json.Unmarshal(b, &field)
if field != nil {
n.Valid = true
n.Bool = *field
}
return err
}


// Scan for NullBool
func (n *NullBool) Scan(src interface{}) error {
var a sql.NullBool
if err := a.Scan(src); err != nil {
return err
}
n.Bool = a.Bool
if reflect.TypeOf(src) != nil {
n.Valid = true
}
return nil
}

/*-------------------------+
| NullTime implementations |
+-------------------------*/

// MarshalJSON for NullTime
func (n NullTime) MarshalJSON() ([]byte, error) {
var a *time.Time
if n.Valid {
a = &n.Time
}
return json.Marshal(a)
}

// Value for NullTime
func (n NullTime) Value() (driver.Value, error) {
if !n.Valid {
return nil, nil
}
return n.Time, nil
}

// UnmarshalJSON for NullTime
func (n *NullTime) UnmarshalJSON(b []byte) error {
s := string(b)
s = strings.Trim(s, `"`)
tim, err := time.Parse(time.RFC3339, s)
if err != nil {
n.Valid = false
return err
}

n.Time = tim
n.Valid = true
return nil
}

// Scan for NullTime
func (n *NullTime) Scan(src interface{}) error {
// Set initial state for subsequent scans.
n.Valid = false

var a mysql.NullTime
if err := a.Scan(src); err != nil {
return err
}
n.Time = a.Time
if reflect.TypeOf(src) != nil {
n.Valid = true
}
return nil
}

/*------------------------+
| RawJSON implementations |
+------------------------*/

// MarshalJSON for NullString
func (n RawJSON) MarshalJSON() ([]byte, error) {
a := json.RawMessage(n)
return a.MarshalJSON()
}

// Value for NullString
func (n RawJSON) Value() (driver.Value, error) {
return string(n), nil
}

// UnmarshalJSON for NullString
func (n *RawJSON) UnmarshalJSON(b []byte) error {
var a json.RawMessage
json.Unmarshal(b, &a)
c := RawJSON(a)
*n = c
return nil
}

// Scan for NullString
func (n *RawJSON) Scan(src interface{}) error {
var a sql.NullString
if err := a.Scan(src); err != nil {
return err
}
jsn := RawJSON([]byte(a.String))
*n = jsn
return nil
}

/*-----------------+
| Helper functions |
+-----------------*/

// ToNullString returns a new NullString
func ToNullString(s *string) NullString {
if s == nil {
return NullString(sql.NullString{Valid: false})
}
return NullString(sql.NullString{String: *s, Valid: true})
}

// ToNullInt64 returns a new NullInt64
func ToNullInt64(i *int64) NullInt64 {
if i == nil {
return NullInt64(sql.NullInt64{Valid: false})
}
return NullInt64(sql.NullInt64{Int64: *i, Valid: true})
}

// ToNullFloat64 returns a new NullFloat64
func ToNullFloat64(i *float64) NullFloat64 {
if i == nil {
return NullFloat64(sql.NullFloat64{Valid: false})
}
return NullFloat64(sql.NullFloat64{Float64: *i, Valid: true})
}

// ToNullBool creates a new NullBool
func ToNullBool(b *bool) NullBool {
if b == nil {
return NullBool(sql.NullBool{Valid: false})
}
return NullBool(sql.NullBool{Bool: *b, Valid: true})
}

// ToNullTime creates a new NullTime
func ToNullTime(t time.Time) NullTime {
if t == emptyTime {
return NullTime(mysql.NullTime{Valid: false})
}
return NullTime(mysql.NullTime{Time: t, Valid: true})
}
{{end}}
\"") + packr.PackJSONBytes("./tmpl", "x_helpers.html", "\"{{define "helpers"}}

package {{ .PackageName }}

/*---------------------------+
| Code generated by modelgen |
|        DO NOT EDIT.        |
+---------------------------*/

import (
"database/sql"
"database/sql/driver"
"encoding/json"
"log"
"reflect"
"strings"
"time"

"github.com/go-sql-driver/mysql"
)

// StdTime provides default SQL TIME format
const StdTime = "15:04:05"

// emptyTime allows default times to be considered
// null for insertion into the database.
var emptyTime = time.Time{}

/********
* Types *
********/

// Queryer allows sql.DB and sql.Tx to be used interchangeably, allowing you
// to use any of the model methods inside transactions or standalone calls.
type Queryer interface {
	Query(query string, args ...interface{}) (*sql.Rows, error)
	QueryRow(query string, args ...interface{}) *sql.Row
	Exec(query string, args ...interface{}) (sql.Result, error)
}

/*-------------+
| Type aliases |
+-------------*/

// NullFloat64 aliases sql.NullFloat64
type NullFloat64 sql.NullFloat64

// NullString aliases sql.NullString
type NullString sql.NullString

// NullBool aliases sql.NullBool
type NullBool sql.NullBool

// NullInt64 aliases sql.NullInt64
type NullInt64 sql.NullInt64

// NullTime aliases sql.NullTime
type NullTime mysql.NullTime

// RawJSON aliases json.RawMessage
type RawJSON json.RawMessage

/*---------------------------+
| NullString implementations |
+---------------------------*/

// MarshalJSON for NullString
func (n NullString) MarshalJSON() ([]byte, error) {
var a *string
if n.Valid {
a = &n.String
}
return json.Marshal(a)
}

// UnmarshalJSON for NullString
func (n *NullString) UnmarshalJSON(b []byte) error {
err := json.Unmarshal(b, &n.String)
n.Valid = err == nil
return err
}

// Value for NullString
func (n NullString) Value() (driver.Value, error) {
if !n.Valid {
return nil, nil
}
return n.String, nil
}

// Scan for NullString
func (n *NullString) Scan(src interface{}) error {
var a sql.NullString
if err := a.Scan(src); err != nil {
return err
}
n.String = a.String
if reflect.TypeOf(src) != nil {
n.Valid = true
}
return nil
}

/*----------------------------+
| NullFloat64 implementations |
+----------------------------*/

// MarshalJSON for NullFloat64
func (n NullFloat64) MarshalJSON() ([]byte, error) {
var a *float64
if n.Valid {
a = &n.Float64
}
return json.Marshal(a)
}

// Value for NullFloat64
func (n NullFloat64) Value() (driver.Value, error) {
if !n.Valid {
return nil, nil
}
return n.Float64, nil
}

// UnmarshalJSON for NullFloat64
func (n *NullFloat64) UnmarshalJSON(b []byte) error {
err := json.Unmarshal(b, &n.Float64)
n.Valid = err == nil
return err
}

// Scan for NullFloat64
func (n *NullFloat64) Scan(src interface{}) error {
var a sql.NullFloat64
if err := a.Scan(src); err != nil {
return err
}
n.Float64 = a.Float64
if reflect.TypeOf(src) != nil {
n.Valid = true
}
return nil
}

/*--------------------------+
| NullInt64 implementations |
+--------------------------*/

// MarshalJSON for NullInt64
func (n NullInt64) MarshalJSON() ([]byte, error) {
var a *int64
if n.Valid {
a = &n.Int64
}
return json.Marshal(a)
}

// Value for NullInt64
func (n NullInt64) Value() (driver.Value, error) {
if !n.Valid {
return nil, nil
}
return n.Int64, nil
}

// UnmarshalJSON for NullInt64
func (n *NullInt64) UnmarshalJSON(b []byte) error {
err := json.Unmarshal(b, &n.Int64)
n.Valid = err == nil
return err
}

// Scan for NullInt64
func (n *NullInt64) Scan(src interface{}) error {
var a sql.NullInt64
if err := a.Scan(src); err != nil {
return err
}
n.Int64 = a.Int64
if reflect.TypeOf(src) != nil {
n.Valid = true
}
return nil
}

/*-------------------------+
| NullBool implementations |
+-------------------------*/

// MarshalJSON for NullBool
func (n NullBool) MarshalJSON() ([]byte, error) {
var a *bool
if n.Valid {
a = &n.Bool
}
return json.Marshal(a)
}

// Value for NullBool
func (n NullBool) Value() (driver.Value, error) {
if !n.Valid {
return nil, nil
}
return n.Bool, nil
}

// UnmarshalJSON for NullBool
func (n *NullBool) UnmarshalJSON(b []byte) error {
var field *bool
err := json.Unmarshal(b, &field)
if field != nil {
n.Valid = true
n.Bool = *field
}
return err
}


// Scan for NullBool
func (n *NullBool) Scan(src interface{}) error {
var a sql.NullBool
if err := a.Scan(src); err != nil {
return err
}
n.Bool = a.Bool
if reflect.TypeOf(src) != nil {
n.Valid = true
}
return nil
}

/*-------------------------+
| NullTime implementations |
+-------------------------*/

// MarshalJSON for NullTime
func (n NullTime) MarshalJSON() ([]byte, error) {
var a *time.Time
if n.Valid {
a = &n.Time
}
return json.Marshal(a)
}

// Value for NullTime
func (n NullTime) Value() (driver.Value, error) {
if !n.Valid {
return nil, nil
}
return n.Time, nil
}

// UnmarshalJSON for NullTime
func (n *NullTime) UnmarshalJSON(b []byte) error {
s := string(b)
s = strings.Trim(s, `"`)
tim, err := time.Parse(time.RFC3339, s)
if err != nil {
n.Valid = false
return err
}

n.Time = tim
n.Valid = true
return nil
}

// Scan for NullTime
func (n *NullTime) Scan(src interface{}) error {
// Set initial state for subsequent scans.
n.Valid = false

var a mysql.NullTime
if err := a.Scan(src); err != nil {
return err
}
n.Time = a.Time
if reflect.TypeOf(src) != nil {
n.Valid = true
}
return nil
}

/*------------------------+
| RawJSON implementations |
+------------------------*/

// MarshalJSON for NullString
func (n RawJSON) MarshalJSON() ([]byte, error) {
a := json.RawMessage(n)
return a.MarshalJSON()
}

// Value for NullString
func (n RawJSON) Value() (driver.Value, error) {
return string(n), nil
}

// UnmarshalJSON for NullString
func (n *RawJSON) UnmarshalJSON(b []byte) error {
var a json.RawMessage
json.Unmarshal(b, &a)
c := RawJSON(a)
*n = c
return nil
}

// Scan for NullString
func (n *RawJSON) Scan(src interface{}) error {
var a sql.NullString
if err := a.Scan(src); err != nil {
return err
}
jsn := RawJSON([]byte(a.String))
*n = jsn
return nil
}

/*-----------------+
| Helper functions |
+-----------------*/

// ToNullString returns a new NullString
func ToNullString(s *string) NullString {
if s == nil {
return NullString(sql.NullString{Valid: false})
}
return NullString(sql.NullString{String: *s, Valid: true})
}

// ToNullInt64 returns a new NullInt64
func ToNullInt64(i *int64) NullInt64 {
if i == nil {
return NullInt64(sql.NullInt64{Valid: false})
}
return NullInt64(sql.NullInt64{Int64: *i, Valid: true})
}

// ToNullFloat64 returns a new NullFloat64
func ToNullFloat64(i *float64) NullFloat64 {
if i == nil {
return NullFloat64(sql.NullFloat64{Valid: false})
}
return NullFloat64(sql.NullFloat64{Float64: *i, Valid: true})
}

// ToNullBool creates a new NullBool
func ToNullBool(b *bool) NullBool {
if b == nil {
return NullBool(sql.NullBool{Valid: false})
}
return NullBool(sql.NullBool{Bool: *b, Valid: true})
}

// ToNullTime creates a new NullTime
func ToNullTime(t time.Time) NullTime {
if t == emptyTime {
return NullTime(mysql.NullTime{Valid: false})
}
return NullTime(mysql.NullTime{Time: t, Valid: true})
}

// ExecuteTransaction closes over a transaction and automatically commits
// or rollbacks depending on whether errors were encountered.
func ExecuteTransaction(db *sql.DB, f func(*sql.Tx) error) error {
	tx, err := db.Begin()
	if err != nil {
		return err
	}

	defer func() error {
		if r := recover(); r != nil {
			// Only need to log here because panic won't report whether
			// the rollback was successful or not.
			if err = tx.Rollback(); err != nil {
				log.Println("db rollback error:", err)
			}
			panic(r)
		} else if err != nil {
			return fmt.Errorf("db error: %v rollback error: %v", err, tx.Rollback())
		} else {
			if err = tx.Commit(); err != nil {
				return err
			}
		}
		return nil
	}()

	return f(tx)
}
{{end}}
\"") packr.PackJSONBytes("./tmpl", "x_helpers_test.html", "\"{{define "helperstest"}}
//+build !helpers

package {{ .PackageName }}
/*---------------------------+
| Code generated by modelgen |
|        DO NOT EDIT.        |
+---------------------------*/

import (
"database/sql/driver"
"encoding/json"
"reflect"
"testing"
"time"
)

func TestStructEmbedding(t *testing.T) {
tim := time.Date(2017, 1, 1, 0, 0, 0, 0, time.UTC)
expected := []byte(`{"a":123,"b":true,"c":123.123,"d":"string","e":"2017-01-01T00:00:00Z","f":[1,2,3]}`)
type embed struct {
A NullInt64   `json:"a,omitempty"`
B NullBool    `json:"b,omitempty"`
C NullFloat64 `json:"c,omitempty"`
D NullString  `json:"d,omitempty"`
E NullTime    `json:"e,omitempty"`
F RawJSON     `json:"f,omitempty"`
}
em := embed{
A: NullInt64{Valid: true, Int64: 123},
B: NullBool{Valid: true, Bool: true},
C: NullFloat64{Valid: true, Float64: 123.123},
D: NullString{Valid: true, String: "string"},
E: NullTime{Valid: true, Time: tim},
F: RawJSON(`[1,2,3]`),
}
b, err := json.Marshal(em)
if err != nil {
t.Fatal(err)
}
if !reflect.DeepEqual(expected, b) {
t.Fatal("not the same JSON!")
}
if !(string(b) == string(expected)) {
t.Fatal("not the same!")
}

var em2 embed
if err := json.Unmarshal(expected, &em2); err != nil {
t.Fatal(err)
}
if !reflect.DeepEqual(em2, em) {
t.Fatal("not correct")
}
}

func TestNullString_UnmarshalJSON(t *testing.T) {
tests := []struct {
name    string
n       NullString
source  []byte
wantErr bool
}{
{
name:    "valid",
source:  []byte(`"hello"`),
wantErr: false,
},
{
name:    "invalid",
source:  []byte(`{"key":"value"}`),
wantErr: true,
},
{
name:    "empty",
source:  []byte{},
wantErr: true,
},
}
for _, tt := range tests {
t.Run(tt.name, func(t *testing.T) {
if err := tt.n.UnmarshalJSON(tt.source); (err != nil) != tt.wantErr {
t.Errorf("NullString.UnmarshalJSON() error = %v, wantErr %v", err, tt.wantErr)
}
})
}
}

func TestNullString_Value(t *testing.T) {
tests := []struct {
name    string
n       NullString
want    driver.Value
wantErr bool
}{
{
name: "valid",
n: NullString{
Valid:  true,
String: "hello",
},
want:    driver.Value("hello"),
wantErr: false,
},
{
name: "invalid",
n: NullString{
Valid: false,
},
want:    nil,
wantErr: false,
},
}
for _, tt := range tests {
t.Run(tt.name, func(t *testing.T) {
got, err := tt.n.Value()
if (err != nil) != tt.wantErr {
t.Errorf("NullString.Value() error = %v, wantErr %v", err, tt.wantErr)
return
}
if !reflect.DeepEqual(got, tt.want) {
t.Errorf("NullString.Value() = %v, want %v", got, tt.want)
}
})
}
}

func TestNullString_Scan(t *testing.T) {
tests := []struct {
name    string
n       *NullString
wantErr bool
src     interface{}
}{
{
name: "valid",
n: &NullString{
String: "hello",
Valid:  true,
},
src:     "",
wantErr: false,
},
{
name: "nil value",
n: &NullString{
String: "hello",
Valid:  false,
},
src:     nil,
wantErr: false,
},
}
for _, tt := range tests {
t.Run(tt.name, func(t *testing.T) {
if err := tt.n.Scan(tt.src); (err != nil) != tt.wantErr {
t.Errorf("NullString.Scan() error = %v, wantErr %v", err, tt.wantErr)
}
if tt.n.Valid == false && tt.src != nil {
t.Errorf("should return null")
}
if tt.n.Valid && tt.src != tt.n.String {
t.Errorf("invalid value")
}
})
}
}

func TestNullString_MarshalJSON(t *testing.T) {
tests := []struct {
name    string
n       *NullString
want    []byte
wantErr bool
}{
{
name: "valid",
n: &NullString{
String: "hello",
Valid:  true,
},
want:    []byte(`"hello"`),
wantErr: false,
},
{
name: "valid null",
n: &NullString{
String: "",
Valid:  false,
},
want:    []byte(`null`),
wantErr: false,
},
{
name: "invalid",
n: &NullString{
Valid: true,
},
want:    []byte(`""`),
wantErr: false,
},
}
for _, tt := range tests {
t.Run(tt.name, func(t *testing.T) {
got, err := tt.n.MarshalJSON()
if (err != nil) != tt.wantErr {
t.Errorf("NullString.MarshalJSON() error = %v, wantErr %v", err, tt.wantErr)
return
}
if !reflect.DeepEqual(got, tt.want) {
t.Errorf("NullString.MarshalJSON() = %v, want %v", got, tt.want)
}
})
}
}

func TestNullBool_UnmarshalJSON(t *testing.T) {
tests := []struct {
name         string
n            NullBool
source       []byte
wantErr      bool
wantValidity bool
}{
{
name:         "valid",
source:       []byte(`false`),
wantErr:      false,
wantValidity: true,
},
{
name:         "invalid",
source:       []byte(`{"key":"value"}`),
wantErr:      true,
wantValidity: false,
},
{
name:         "empty",
source:       []byte{},
wantErr:      true,
wantValidity: false,
},
{
name:         "explicit null",
source:       []byte("null"),
wantErr:      false,
wantValidity: false,
},
}
for _, tt := range tests {
t.Run(tt.name, func(t *testing.T) {
if err := tt.n.UnmarshalJSON(tt.source); (err != nil) != tt.wantErr && tt.n.Valid == tt.wantValidity {
    t.Errorf("NullBool.UnmarshalJSON() error = %v, wantErr %v", err, tt.wantErr)
}
})
}
}

func TestNullBool_Value(t *testing.T) {
tests := []struct {
name    string
n       NullBool
want    driver.Value
wantErr bool
}{
{
name: "valid",
n: NullBool{
Valid: true,
Bool:  true,
},
want:    driver.Value(true),
wantErr: false,
},
{
name: "invalid",
n: NullBool{
Valid: false,
},
want:    nil,
wantErr: false,
},
}
for _, tt := range tests {
t.Run(tt.name, func(t *testing.T) {
got, err := tt.n.Value()
if (err != nil) != tt.wantErr {
t.Errorf("NullBool.Value() error = %v, wantErr %v", err, tt.wantErr)
return
}
if !reflect.DeepEqual(got, tt.want) {
t.Errorf("NullBool.Value() = %v, want %v", got, tt.want)
}
})
}
}

func TestNullBool_Scan(t *testing.T) {
tests := []struct {
name    string
n       *NullBool
wantErr bool
src     interface{}
}{
{
name: "valid",
n: &NullBool{
Bool:  true,
Valid: true,
},
src:     true,
wantErr: false,
},
{
name: "nil value",
n: &NullBool{
Bool:  true,
Valid: false,
},
src:     false,
wantErr: false,
},
}
for _, tt := range tests {
t.Run(tt.name, func(t *testing.T) {
if err := tt.n.Scan(tt.src); (err != nil) != tt.wantErr {
t.Errorf("NullBool.Scan() error = %v, wantErr %v", err, tt.wantErr)
}
if tt.n.Valid == false && tt.src != nil {
t.Errorf("should return null")
}
if tt.n.Valid && tt.src != tt.n.Bool {
t.Errorf("invalid value")
}
})
}
}

func TestNullBool_MarshalJSON(t *testing.T) {
tests := []struct {
name    string
n       *NullBool
want    []byte
wantErr bool
}{
{
name: "valid",
n: &NullBool{
Valid: true,
},
want:    []byte(`false`),
wantErr: false,
},
{
name: "valid null",
n: &NullBool{
Valid: false,
},
want:    []byte(`null`),
wantErr: false,
},
{
name: "invalid",
n: &NullBool{
Valid: true,
},
want:    []byte(`false`),
wantErr: false,
},
}
for _, tt := range tests {
t.Run(tt.name, func(t *testing.T) {
got, err := tt.n.MarshalJSON()
if (err != nil) != tt.wantErr {
t.Errorf("NullBool.MarshalJSON() error = %v, wantErr %v", err, tt.wantErr)
return
}
if !reflect.DeepEqual(got, tt.want) {
t.Errorf("NullBool.MarshalJSON() = %v, want %v", got, tt.want)
}
})
}
}

func TestNullTime_UnmarshalJSON(t *testing.T) {
tests := []struct {
name    string
n       NullTime
source  []byte
wantErr bool
}{
{
name:    "valid",
source:  []byte(`"2017-11-24T00:00:00Z"`),
wantErr: false,
},
{
name:    "invalid",
source:  []byte(`{"key":"value"}`),
wantErr: true,
},
{
name:    "empty",
source:  []byte{},
wantErr: true,
},
}
for _, tt := range tests {
t.Run(tt.name, func(t *testing.T) {
if err := tt.n.UnmarshalJSON(tt.source); (err != nil) != tt.wantErr {
t.Errorf("NullBool.UnmarshalJSON() error = %v, wantErr %v", err, tt.wantErr)
}
})
}
}

func TestNullTime_Value(t *testing.T) {
tim := time.Now()
tests := []struct {
name    string
n       NullTime
want    driver.Value
wantErr bool
}{
{
name: "valid",
n: NullTime{
Valid: true,
Time:  tim,
},
want:    driver.Value(tim),
wantErr: false,
},
{
name: "invalid",
n: NullTime{
Valid: false,
},
want:    nil,
wantErr: false,
},
}
for _, tt := range tests {
t.Run(tt.name, func(t *testing.T) {
got, err := tt.n.Value()
if (err != nil) != tt.wantErr {
t.Errorf("NullTime.Value() error = %v, wantErr %v", err, tt.wantErr)
return
}
if !reflect.DeepEqual(got, tt.want) {
t.Errorf("NullTime.Value() = %v, want %v", got, tt.want)
}
})
}
}

func TestNullTime_Scan(t *testing.T) {
tim := time.Now()
tests := []struct {
name    string
n       *NullTime
wantErr bool
src     interface{}
}{
{
name: "valid",
n: &NullTime{
Time:  tim,
Valid: true,
},
src:     tim,
wantErr: false,
},
{
name: "nil value",
n: &NullTime{
Time:  tim,
Valid: false,
},
src:     time.Now(),
wantErr: false,
},
}
for _, tt := range tests {
t.Run(tt.name, func(t *testing.T) {
if err := tt.n.Scan(tt.src); (err != nil) != tt.wantErr {
t.Errorf("NullTime.Scan() error = %v, wantErr %v", err, tt.wantErr)
}
if tt.n.Valid == false && tt.src != nil {
t.Errorf("should return null")
}
if tt.n.Valid && tt.src != tt.n.Time {
t.Errorf("invalid value")
}
})
}
}

func TestNullTime_MarshalJSON(t *testing.T) {
tests := []struct {
name    string
n       *NullTime
want    []byte
wantErr bool
}{
{
name: "valid",
n: &NullTime{
Time:  time.Date(2017, 11, 24, 0, 0, 0, 0, time.UTC),
Valid: true,
},
want:    []byte(`"2017-11-24T00:00:00Z"`),
wantErr: false,
},
{
name: "valid null",
n: &NullTime{
Valid: false,
},
want:    []byte(`null`),
wantErr: false,
},
{
name: "invalid",
n: &NullTime{
Valid: true,
},
want:    []byte(`"0001-01-01T00:00:00Z"`),
wantErr: false,
},
}
for _, tt := range tests {
t.Run(tt.name, func(t *testing.T) {
got, err := tt.n.MarshalJSON()
if (err != nil) != tt.wantErr {
t.Errorf("NullTime.MarshalJSON() error = %v, wantErr %v", err, tt.wantErr)
return
}
if !reflect.DeepEqual(got, tt.want) {
t.Errorf("NullTime.MarshalJSON() = %v, want %v", got, tt.want)
}
})
}
}

func TestNullInt64_UnmarshalJSON(t *testing.T) {
tests := []struct {
name    string
n       NullInt64
source  []byte
wantErr bool
}{
{
name:    "valid",
source:  []byte(`123`),
wantErr: false,
},
{
name:    "invalid",
source:  []byte(`{"key":"value"}`),
wantErr: true,
},
{
name:    "empty",
source:  []byte{},
wantErr: true,
},
}
for _, tt := range tests {
t.Run(tt.name, func(t *testing.T) {
if err := tt.n.UnmarshalJSON(tt.source); (err != nil) != tt.wantErr {
t.Errorf("NullInt64.UnmarshalJSON() error = %v, wantErr %v", err, tt.wantErr)
}
})
}
}

func TestNullInt64_Value(t *testing.T) {
tests := []struct {
name    string
n       NullInt64
want    driver.Value
wantErr bool
}{
{
name: "valid",
n: NullInt64{
Valid: true,
Int64: 123,
},
want:    driver.Value(int64(123)),
wantErr: false,
},
{
name: "invalid",
n: NullInt64{
Valid: false,
},
want:    nil,
wantErr: false,
},
}
for _, tt := range tests {
t.Run(tt.name, func(t *testing.T) {
got, err := tt.n.Value()
if (err != nil) != tt.wantErr {
t.Errorf("NullInt64.Value() error = %v, wantErr %v", err, tt.wantErr)
return
}
if !reflect.DeepEqual(got, tt.want) {
t.Errorf("NullInt64.Value() = %v, want %v", got, tt.want)
}
})
}
}

func TestNullInt64_Scan(t *testing.T) {
tests := []struct {
name    string
n       *NullInt64
wantErr bool
src     interface{}
}{
{
name: "valid",
n: &NullInt64{
Int64: 123,
Valid: true,
},
src:     int64(123),
wantErr: false,
},
{
name: "nil value",
n: &NullInt64{
Valid: false,
},
src:     int64(123),
wantErr: false,
},
}
for _, tt := range tests {
t.Run(tt.name, func(t *testing.T) {
if err := tt.n.Scan(tt.src); (err != nil) != tt.wantErr {
t.Errorf("NullInt64.Scan() error = %v, wantErr %v", err, tt.wantErr)
}
if tt.n.Valid == false && tt.src != nil {
t.Errorf("should return null")
}
if tt.n.Valid && tt.src != tt.n.Int64 {
t.Errorf("invalid value")
}
})
}
}

func TestNullInt64_MarshalJSON(t *testing.T) {
tests := []struct {
name    string
n       *NullInt64
want    []byte
wantErr bool
}{
{
name: "valid",
n: &NullInt64{
Int64: 123,
Valid: true,
},
want:    []byte(`123`),
wantErr: false,
},
{
name: "valid null",
n: &NullInt64{
Valid: false,
},
want:    []byte(`null`),
wantErr: false,
},
{
name: "invalid",
n: &NullInt64{
Valid: true,
},
want:    []byte(`0`),
wantErr: false,
},
}
for _, tt := range tests {
t.Run(tt.name, func(t *testing.T) {
got, err := tt.n.MarshalJSON()
if (err != nil) != tt.wantErr {
t.Errorf("NullInt64.MarshalJSON() error = %v, wantErr %v", err, tt.wantErr)
return
}
if !reflect.DeepEqual(got, tt.want) {
t.Errorf("NullInt64.MarshalJSON() = %v, want %v", got, tt.want)
}
})
}
}
func TestNullFloat64_UnmarshalJSON(t *testing.T) {
tests := []struct {
name    string
n       NullFloat64
source  []byte
wantErr bool
}{
{
name:    "valid",
source:  []byte(`123.123`),
wantErr: false,
},
{
name:    "invalid",
source:  []byte(`{"key":"value"}`),
wantErr: true,
},
{
name:    "empty",
source:  []byte{},
wantErr: true,
},
}
for _, tt := range tests {
t.Run(tt.name, func(t *testing.T) {
if err := tt.n.UnmarshalJSON(tt.source); (err != nil) != tt.wantErr {
t.Errorf("NullFloat64.UnmarshalJSON() error = %v, wantErr %v", err, tt.wantErr)
}
})
}
}

func TestNullFloat64_Value(t *testing.T) {
tests := []struct {
name    string
n       NullFloat64
want    driver.Value
wantErr bool
}{
{
name: "valid",
n: NullFloat64{
Valid:   true,
Float64: 123.123,
},
want:    driver.Value(float64(123.123)),
wantErr: false,
},
{
name: "invalid",
n: NullFloat64{
Valid: false,
},
want:    nil,
wantErr: false,
},
}
for _, tt := range tests {
t.Run(tt.name, func(t *testing.T) {
got, err := tt.n.Value()
if (err != nil) != tt.wantErr {
t.Errorf("NullFloat64.Value() error = %v, wantErr %v", err, tt.wantErr)
return
}
if !reflect.DeepEqual(got, tt.want) {
t.Errorf("NullFloat64.Value() = %v, want %v", got, tt.want)
}
})
}
}

func TestNullFloat64_Scan(t *testing.T) {
tests := []struct {
name    string
n       *NullFloat64
wantErr bool
src     interface{}
}{
{
name: "valid",
n: &NullFloat64{
Float64: 123.123,
Valid:   true,
},
src:     float64(123),
wantErr: false,
},
{
name: "nil value",
n: &NullFloat64{
Valid: false,
},
src:     float64(123.123),
wantErr: false,
},
}
for _, tt := range tests {
t.Run(tt.name, func(t *testing.T) {
if err := tt.n.Scan(tt.src); (err != nil) != tt.wantErr {
t.Errorf("NullFloat64.Scan() error = %v, wantErr %v", err, tt.wantErr)
}
if tt.n.Valid == false && tt.src != nil {
t.Errorf("should return null")
}
if tt.n.Valid && tt.src != tt.n.Float64 {
t.Errorf("invalid value")
}
})
}
}

func TestNullFloat64_MarshalJSON(t *testing.T) {
tests := []struct {
name    string
n       *NullFloat64
want    []byte
wantErr bool
}{
{
name: "valid",
n: &NullFloat64{
Float64: 123.123,
Valid:   true,
},
want:    []byte(`123.123`),
wantErr: false,
},
{
name: "valid null",
n: &NullFloat64{
Valid: false,
},
want:    []byte(`null`),
wantErr: false,
},
{
name: "invalid",
n: &NullFloat64{
Valid: true,
},
want:    []byte(`0`),
wantErr: false,
},
}
for _, tt := range tests {
t.Run(tt.name, func(t *testing.T) {
got, err := tt.n.MarshalJSON()
if (err != nil) != tt.wantErr {
t.Errorf("NullFloat64.MarshalJSON() error = %v, wantErr %v", err, tt.wantErr)
return
}
if !reflect.DeepEqual(got, tt.want) {
t.Errorf("NullFloat64.MarshalJSON() = %v, want %v", got, tt.want)
}
})
}
}

func TestToNullBool(t *testing.T) {
b := true
bb := ToNullBool(&b)
if !bb.Valid {
t.Errorf("expected valid, got %v", bb.Valid)
}
if !bb.Bool {
t.Errorf("expected true, got %v", bb.Bool)
}

var b2 *bool
bb2 := ToNullBool(b2)
if bb2.Valid {
t.Errorf("expected not valid, got %v", bb2.Valid)
}
if bb2.Bool {
t.Errorf("expected false, got %v", bb2.Bool)
}
}
func TestToNullInt64(t *testing.T) {
b := int64(123)
bb := ToNullInt64(&b)
if !bb.Valid {
t.Errorf("expected valid, got %v", bb.Valid)
}
if bb.Int64 != 123 {
t.Errorf("expected 123, got %v", bb.Int64)
}

var b2 *int64
bb2 := ToNullInt64(b2)
if bb2.Valid {
t.Errorf("expected not valid, got %v", bb2.Valid)
}
if bb2.Int64 != 0 {
t.Errorf("expected 0, got %v", bb2.Int64)
}
}

func TestToNullFloat64(t *testing.T) {
b := float64(123.123)
bb := ToNullFloat64(&b)
if !bb.Valid {
t.Errorf("expected valid, got %v", bb.Valid)
}
if bb.Float64 != 123.123 {
t.Errorf("expected 123.123, got %v", bb.Float64)
}

var b2 *float64
bb2 := ToNullFloat64(b2)
if bb2.Valid {
t.Errorf("expected not valid, got %v", bb2.Valid)
}
if bb2.Float64 != 0 {
t.Errorf("expected 0, got %v", bb2.Float64)
}
}
func TestToNullString(t *testing.T) {
b := "qwe"
bb := ToNullString(&b)
if !bb.Valid {
t.Errorf("expected valid, got %v", bb.Valid)
}
if bb.String != "qwe" {
t.Errorf("expected qwe, got %v", bb.String)
}

var b2 *string
bb2 := ToNullString(b2)
if bb2.Valid {
t.Errorf("expected not valid, got %v", bb2.Valid)
}
if bb2.String != "" {
t.Errorf("expected <empty string>, got %v", bb2.String)
    }
    }
    func TestToNullTime(t *testing.T) {
    tim := time.Now()
    bb := ToNullTime(tim)
    if !bb.Valid {
    t.Errorf("expected valid, got %v", bb.Valid)
    }
    if bb.Time != tim {
    t.Errorf("expected %v, got %v", tim, bb.Time)
    }

    tim = time.Time{}
    bb = ToNullTime(tim)
    if bb.Valid {
    t.Errorf("expected invalid, got %v", bb.Valid)
    }
    if bb.Time != tim {
    t.Errorf("expected %v, got %v", tim, bb.Time)
    }
    }
{{end}}
\"") } diff --git a/tmpl/model.html b/tmpl/model.html index 24c08d7..8684e9b 100644 --- a/tmpl/model.html +++ b/tmpl/model.html @@ -12,7 +12,6 @@ import ( "fmt" -"database/sql" {{ range $k, $v:= .Model.Imports }} "{{$k}}" {{- end }} @@ -28,9 +27,9 @@ } // Insert a new {{.Model.Name}} row in the {{.Model.TableName}} table -func ({{.Receiver}} *{{.Model.Name}}) Insert(db *sql.DB) (lastInsertID int64, err error) { +func ({{.Receiver}} *{{.Model.Name}}) Insert(q Queryer) (lastInsertID int64, err error) { const stmt = "INSERT INTO {{.Model.TableName}} ({{.Model.Fields | insert_fields}}) VALUES ({{.Model.Fields | insert_values}})" - res, err := db.Exec(stmt, {{ . | insert_args }}) + res, err := q.Exec(stmt, {{ . | insert_args }}) if err != nil { return 0, err } @@ -38,9 +37,9 @@ } // Update an existing {{.Model.Name}} row in the {{.Model.TableName}} table. -func ({{.Receiver}} *{{.Model.Name}}) Update(db *sql.DB, id int64) (int64, error) { +func ({{.Receiver}} *{{.Model.Name}}) Update(q Queryer, id int64) (int64, error) { const stmt = "UPDATE {{.Model.TableName}} SET {{ . | update_values }} WHERE id = ?" - result, err := db.Exec(stmt, {{ . | update_args }} , id) + result, err := q.Exec(stmt, {{ . | update_args }} , id) if err != nil { return 0, err } @@ -49,9 +48,9 @@ // Upsert inserts a new {{.Model.Name}} row in the {{.Model.TableName}} table // if the unique constraints are not found, otherwise it updates it. -func ({{.Receiver}} *{{.Model.Name}}) Upsert(db *sql.DB) (lastInsertID int64, err error) { +func ({{.Receiver}} *{{.Model.Name}}) Upsert(q Queryer) (lastInsertID int64, err error) { const stmt = "INSERT INTO {{.Model.TableName}} ({{.Model.Fields | upsert_fields}}) VALUES ({{.Model.Fields | upsert_values}}) ON DUPLICATE KEY UPDATE {{ . | upsert_on_duplicate }}" - res, err := db.Exec(stmt, {{ . | upsert_args }}) + res, err := q.Exec(stmt, {{ . | upsert_args }}) if err != nil { return 0, err } @@ -59,14 +58,14 @@ } // Find an existing {{.Model.Name}} row in the {{.Model.TableName}} table -func ({{.Receiver}} *{{.Model.Name}}) Find(db *sql.DB, id int64) error { +func ({{.Receiver}} *{{.Model.Name}}) Find(q Queryer, id int64) error { const stmt = "SELECT * FROM {{.Model.TableName}} WHERE id = ?" - row := db.QueryRow(stmt, id) + row := q.QueryRow(stmt, id) return row.Scan({{ . | scan_fields}}) } // Load all, or a subset of {{.Model.Name}} rows from the {{.Model.TableName}} table -func ({{.Receiver}} *{{.Model.Name}}) Load(db *sql.DB) (set []{{.Model.Name}}, err error) { +func ({{.Receiver}} *{{.Model.Name}}) Load(q Queryer) (set []{{.Model.Name}}, err error) { stmt := "SELECT * FROM {{.Model.TableName}}" if {{.Receiver}}.limit == 0 && {{.Receiver}}.offset > 0 { @@ -83,7 +82,7 @@ {{.Receiver}}.limit = 0 {{.Receiver}}.offset = 0 }() - rows, err := db.Query(stmt) + rows, err := q.Query(stmt) if err != nil { return } @@ -99,9 +98,9 @@ } // Delete an existing {{.Model.Name}} row from the {{.Model.TableName}} table -func ({{.Receiver}} *{{.Model.Name}}) Delete(db *sql.DB, id int64) (rowsAffected int64, err error) { +func ({{.Receiver}} *{{.Model.Name}}) Delete(q Queryer, id int64) (rowsAffected int64, err error) { const stmt = "DELETE FROM {{.Model.TableName}} WHERE id = ?" - result, err := db.Exec(stmt, id) + result, err := q.Exec(stmt, id) if err != nil { return } @@ -110,9 +109,9 @@ } // Count the number of rows from the {{.Model.TableName}} table -func({{.Receiver}} *{{.Model.Name}}) Count(db *sql.DB) (count int64, err error) { +func({{.Receiver}} *{{.Model.Name}}) Count(q Queryer) (count int64, err error) { const stmt = "SELECT COUNT(*) FROM {{.Model.TableName}}" - row := db.QueryRow(stmt) + row := q.QueryRow(stmt) if err = row.Scan(&count); err != nil { return } @@ -122,10 +121,10 @@ // Exists checks for the items existence in the database, based on it's id. // An error will only be returned if a SQL related failure happens. // In all other cases, a bool and nil will return. -func({{.Receiver}} *{{.Model.Name}}) Exists(db *sql.DB, id int64) (exists bool, err error) { +func({{.Receiver}} *{{.Model.Name}}) Exists(q Queryer, id int64) (exists bool, err error) { const stmt = "SELECT EXISTS(SELECT 1 FROM {{.Model.TableName}} WHERE id = ? LIMIT 1) AS `exists`" var count int - row := db.QueryRow(stmt, id) + row := q.QueryRow(stmt, id) if err = row.Scan(&count); err != nil { return } diff --git a/tmpl/x_helpers.html b/tmpl/x_helpers.html index 08fb670..d05bbf4 100644 --- a/tmpl/x_helpers.html +++ b/tmpl/x_helpers.html @@ -11,6 +11,7 @@ "database/sql" "database/sql/driver" "encoding/json" +"log" "reflect" "strings" "time" @@ -25,6 +26,18 @@ // null for insertion into the database. var emptyTime = time.Time{} +/******** +* Types * +********/ + +// Queryer allows sql.DB and sql.Tx to be used interchangeably, allowing you +// to use any of the model methods inside transactions or standalone calls. +type Queryer interface { + Query(query string, args ...interface{}) (*sql.Rows, error) + QueryRow(query string, args ...interface{}) *sql.Row + Exec(query string, args ...interface{}) (sql.Result, error) +} + /*-------------+ | Type aliases | +-------------*/ @@ -346,4 +359,33 @@ } return NullTime(mysql.NullTime{Time: t, Valid: true}) } + +// ExecuteTransaction closes over a transaction and automatically commits +// or rollbacks depending on whether errors were encountered. +func ExecuteTransaction(db *sql.DB, f func(*sql.Tx) error) error { + tx, err := db.Begin() + if err != nil { + return err + } + + defer func() error { + if r := recover(); r != nil { + // Only need to log here because panic won't report whether + // the rollback was successful or not. + if err = tx.Rollback(); err != nil { + log.Println("db rollback error:", err) + } + panic(r) + } else if err != nil { + return fmt.Errorf("db error: %v rollback error: %v", err, tx.Rollback()) + } else { + if err = tx.Commit(); err != nil { + return err + } + } + return nil + }() + + return f(tx) +} {{end}}