TUN-4922: Downgrade quic-go library to 0.20.0
This commit is contained in:
parent
5f6e867685
commit
1082ac1c36
7
go.mod
7
go.mod
|
@ -29,7 +29,7 @@ require (
|
||||||
github.com/json-iterator/go v1.1.10
|
github.com/json-iterator/go v1.1.10
|
||||||
github.com/kr/text v0.2.0 // indirect
|
github.com/kr/text v0.2.0 // indirect
|
||||||
github.com/kylelemons/godebug v1.1.0 // indirect
|
github.com/kylelemons/godebug v1.1.0 // indirect
|
||||||
github.com/lucas-clemente/quic-go v0.21.1
|
github.com/lucas-clemente/quic-go v0.20.0
|
||||||
github.com/mattn/go-colorable v0.1.8
|
github.com/mattn/go-colorable v0.1.8
|
||||||
github.com/miekg/dns v1.1.31
|
github.com/miekg/dns v1.1.31
|
||||||
github.com/mitchellh/go-homedir v1.1.0
|
github.com/mitchellh/go-homedir v1.1.0
|
||||||
|
@ -38,6 +38,7 @@ require (
|
||||||
github.com/pkg/errors v0.9.1
|
github.com/pkg/errors v0.9.1
|
||||||
github.com/pquerna/cachecontrol v0.0.0-20180517163645-1555304b9b35 // indirect
|
github.com/pquerna/cachecontrol v0.0.0-20180517163645-1555304b9b35 // indirect
|
||||||
github.com/prometheus/client_golang v1.7.1
|
github.com/prometheus/client_golang v1.7.1
|
||||||
|
github.com/prometheus/client_model v0.2.0
|
||||||
github.com/prometheus/common v0.13.0 // indirect
|
github.com/prometheus/common v0.13.0 // indirect
|
||||||
github.com/rivo/tview v0.0.0-20200712113419-c65badfc3d92
|
github.com/rivo/tview v0.0.0-20200712113419-c65badfc3d92
|
||||||
github.com/rs/zerolog v1.20.0
|
github.com/rs/zerolog v1.20.0
|
||||||
|
@ -46,9 +47,9 @@ require (
|
||||||
github.com/urfave/cli/v2 v2.2.0
|
github.com/urfave/cli/v2 v2.2.0
|
||||||
go.uber.org/automaxprocs v1.4.0
|
go.uber.org/automaxprocs v1.4.0
|
||||||
golang.org/x/crypto v0.0.0-20200820211705-5c72a883971a
|
golang.org/x/crypto v0.0.0-20200820211705-5c72a883971a
|
||||||
golang.org/x/net v0.0.0-20210405180319-a5a99cb37ef4
|
golang.org/x/net v0.0.0-20200904194848-62affa334b73
|
||||||
golang.org/x/oauth2 v0.0.0-20200902213428-5d25da1a8d43 // indirect
|
golang.org/x/oauth2 v0.0.0-20200902213428-5d25da1a8d43 // indirect
|
||||||
golang.org/x/sync v0.0.0-20210220032951-036812b2e83c
|
golang.org/x/sync v0.0.0-20200625203802-6e8e738ad208
|
||||||
golang.org/x/sys v0.0.0-20210510120138-977fb7262007
|
golang.org/x/sys v0.0.0-20210510120138-977fb7262007
|
||||||
golang.org/x/term v0.0.0-20201210144234-2321bbc49cbf
|
golang.org/x/term v0.0.0-20201210144234-2321bbc49cbf
|
||||||
google.golang.org/genproto v0.0.0-20200904004341-0bd0a958aa1d // indirect
|
google.golang.org/genproto v0.0.0-20200904004341-0bd0a958aa1d // indirect
|
||||||
|
|
20
go.sum
20
go.sum
|
@ -263,7 +263,8 @@ github.com/golang/mock v1.4.0/go.mod h1:UOMv5ysSaYNkG+OFQykRIcU/QvvxJf3p21QfJ2Bt
|
||||||
github.com/golang/mock v1.4.1/go.mod h1:UOMv5ysSaYNkG+OFQykRIcU/QvvxJf3p21QfJ2Bt3cw=
|
github.com/golang/mock v1.4.1/go.mod h1:UOMv5ysSaYNkG+OFQykRIcU/QvvxJf3p21QfJ2Bt3cw=
|
||||||
github.com/golang/mock v1.4.3/go.mod h1:UOMv5ysSaYNkG+OFQykRIcU/QvvxJf3p21QfJ2Bt3cw=
|
github.com/golang/mock v1.4.3/go.mod h1:UOMv5ysSaYNkG+OFQykRIcU/QvvxJf3p21QfJ2Bt3cw=
|
||||||
github.com/golang/mock v1.4.4/go.mod h1:l3mdAwkq5BuhzHwde/uurv3sEJeZMXNpwsxVWU71h+4=
|
github.com/golang/mock v1.4.4/go.mod h1:l3mdAwkq5BuhzHwde/uurv3sEJeZMXNpwsxVWU71h+4=
|
||||||
github.com/golang/mock v1.6.0/go.mod h1:p6yTPP+5HYm5mzsMV8JkE6ZKdX+/wYM6Hr+LicevLPs=
|
github.com/golang/mock v1.5.0 h1:jlYHihg//f7RRwuPfptm04yp4s7O6Kw8EZiVYIGcH0g=
|
||||||
|
github.com/golang/mock v1.5.0/go.mod h1:CWnOUgYIOo4TcNZ0wHX3YZCqsaM1I1Jvs6v3mP3KVu8=
|
||||||
github.com/golang/protobuf v0.0.0-20161109072736-4bd1920723d7/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U=
|
github.com/golang/protobuf v0.0.0-20161109072736-4bd1920723d7/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U=
|
||||||
github.com/golang/protobuf v1.2.0/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U=
|
github.com/golang/protobuf v1.2.0/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U=
|
||||||
github.com/golang/protobuf v1.3.0/go.mod h1:Qd/q+1AKNOZr9uGQzbzCmRO6sUih6GTPZv6a1/R87v0=
|
github.com/golang/protobuf v1.3.0/go.mod h1:Qd/q+1AKNOZr9uGQzbzCmRO6sUih6GTPZv6a1/R87v0=
|
||||||
|
@ -414,8 +415,8 @@ github.com/lightstep/lightstep-tracer-go v0.18.1/go.mod h1:jlF1pusYV4pidLvZ+XD0U
|
||||||
github.com/linode/linodego v0.10.0/go.mod h1:cziNP7pbvE3mXIPneHj0oRY8L1WtGEIKlZ8LANE4eXA=
|
github.com/linode/linodego v0.10.0/go.mod h1:cziNP7pbvE3mXIPneHj0oRY8L1WtGEIKlZ8LANE4eXA=
|
||||||
github.com/liquidweb/liquidweb-go v1.6.0/go.mod h1:UDcVnAMDkZxpw4Y7NOHkqoeiGacVLEIG/i5J9cyixzQ=
|
github.com/liquidweb/liquidweb-go v1.6.0/go.mod h1:UDcVnAMDkZxpw4Y7NOHkqoeiGacVLEIG/i5J9cyixzQ=
|
||||||
github.com/lucas-clemente/quic-go v0.13.1/go.mod h1:Vn3/Fb0/77b02SGhQk36KzOUmXgVpFfizUfW5WMaqyU=
|
github.com/lucas-clemente/quic-go v0.13.1/go.mod h1:Vn3/Fb0/77b02SGhQk36KzOUmXgVpFfizUfW5WMaqyU=
|
||||||
github.com/lucas-clemente/quic-go v0.21.1 h1:uuhCcu885TE9u/piPYMChI/yqA1lXfaLUEx8uCMxf8w=
|
github.com/lucas-clemente/quic-go v0.20.0 h1:FSU3YN5VnLafHR27Ejs1r1CYMS7XMyIVDzRewkDLNBw=
|
||||||
github.com/lucas-clemente/quic-go v0.21.1/go.mod h1:U9kFi5LKbNIlU30dkuM9vxmTxWq4Bvzee/MjBI+07UA=
|
github.com/lucas-clemente/quic-go v0.20.0/go.mod h1:fZq/HUDIM+mW6X6wtzORjC0E/WDBMKe5Hf9bgjISwLk=
|
||||||
github.com/lucasb-eyer/go-colorful v1.0.2/go.mod h1:0MS4r+7BZKSJ5mw4/S5MPN+qHFF1fYclkSPilDOKW0s=
|
github.com/lucasb-eyer/go-colorful v1.0.2/go.mod h1:0MS4r+7BZKSJ5mw4/S5MPN+qHFF1fYclkSPilDOKW0s=
|
||||||
github.com/lucasb-eyer/go-colorful v1.0.3 h1:QIbQXiugsb+q10B+MI+7DI1oQLdmnep86tWFlaaUAac=
|
github.com/lucasb-eyer/go-colorful v1.0.3 h1:QIbQXiugsb+q10B+MI+7DI1oQLdmnep86tWFlaaUAac=
|
||||||
github.com/lucasb-eyer/go-colorful v1.0.3/go.mod h1:R4dSotOR9KMtayYi1e77YzuveK+i7ruzyGqttikkLy0=
|
github.com/lucasb-eyer/go-colorful v1.0.3/go.mod h1:R4dSotOR9KMtayYi1e77YzuveK+i7ruzyGqttikkLy0=
|
||||||
|
@ -431,8 +432,6 @@ github.com/marten-seemann/qtls-go1-15 v0.1.4 h1:RehYMOyRW8hPVEja1KBVsFVNSm35Jj9M
|
||||||
github.com/marten-seemann/qtls-go1-15 v0.1.4/go.mod h1:GyFwywLKkRt+6mfU99csTEY1joMZz5vmB1WNZH3P81I=
|
github.com/marten-seemann/qtls-go1-15 v0.1.4/go.mod h1:GyFwywLKkRt+6mfU99csTEY1joMZz5vmB1WNZH3P81I=
|
||||||
github.com/marten-seemann/qtls-go1-16 v0.1.3 h1:XEZ1xGorVy9u+lJq+WXNE+hiqRYLNvJGYmwfwKQN2gU=
|
github.com/marten-seemann/qtls-go1-16 v0.1.3 h1:XEZ1xGorVy9u+lJq+WXNE+hiqRYLNvJGYmwfwKQN2gU=
|
||||||
github.com/marten-seemann/qtls-go1-16 v0.1.3/go.mod h1:gNpI2Ol+lRS3WwSOtIUUtRwZEQMXjYK+dQSBFbethAk=
|
github.com/marten-seemann/qtls-go1-16 v0.1.3/go.mod h1:gNpI2Ol+lRS3WwSOtIUUtRwZEQMXjYK+dQSBFbethAk=
|
||||||
github.com/marten-seemann/qtls-go1-17 v0.1.0-beta.1.2 h1:SficYjyOthSrliKI+EaFuXS6HqSsX3dkY9AqxAAjBjw=
|
|
||||||
github.com/marten-seemann/qtls-go1-17 v0.1.0-beta.1.2/go.mod h1:fz4HIxByo+LlWcreM4CZOYNuz3taBQ8rN2X6FqvaWo8=
|
|
||||||
github.com/mattn/go-colorable v0.0.9/go.mod h1:9vuHe8Xs5qXnSaW/c/ABM9alt+Vo+STaOChaDxuIBZU=
|
github.com/mattn/go-colorable v0.0.9/go.mod h1:9vuHe8Xs5qXnSaW/c/ABM9alt+Vo+STaOChaDxuIBZU=
|
||||||
github.com/mattn/go-colorable v0.1.8 h1:c1ghPdyEDarC70ftn0y+A/Ee++9zz8ljHG1b13eJ0s8=
|
github.com/mattn/go-colorable v0.1.8 h1:c1ghPdyEDarC70ftn0y+A/Ee++9zz8ljHG1b13eJ0s8=
|
||||||
github.com/mattn/go-colorable v0.1.8/go.mod h1:u6P/XSegPjTcexA+o6vUJrdnUu04hMope9wVRipJSqc=
|
github.com/mattn/go-colorable v0.1.8/go.mod h1:u6P/XSegPjTcexA+o6vUJrdnUu04hMope9wVRipJSqc=
|
||||||
|
@ -492,6 +491,7 @@ github.com/niemeyer/pretty v0.0.0-20200227124842-a10e7caefd8e/go.mod h1:zD1mROLA
|
||||||
github.com/nrdcg/auroradns v1.0.0/go.mod h1:6JPXKzIRzZzMqtTDgueIhTi6rFf1QvYE/HzqidhOhjw=
|
github.com/nrdcg/auroradns v1.0.0/go.mod h1:6JPXKzIRzZzMqtTDgueIhTi6rFf1QvYE/HzqidhOhjw=
|
||||||
github.com/nrdcg/goinwx v0.6.1/go.mod h1:XPiut7enlbEdntAqalBIqcYcTEVhpv/dKWgDCX2SwKQ=
|
github.com/nrdcg/goinwx v0.6.1/go.mod h1:XPiut7enlbEdntAqalBIqcYcTEVhpv/dKWgDCX2SwKQ=
|
||||||
github.com/nrdcg/namesilo v0.2.1/go.mod h1:lwMvfQTyYq+BbjJd30ylEG4GPSS6PII0Tia4rRpRiyw=
|
github.com/nrdcg/namesilo v0.2.1/go.mod h1:lwMvfQTyYq+BbjJd30ylEG4GPSS6PII0Tia4rRpRiyw=
|
||||||
|
github.com/nxadm/tail v1.4.4 h1:DQuhQpB1tVlglWS2hLQ5OV6B5r8aGxSrPc5Qo6uTN78=
|
||||||
github.com/nxadm/tail v1.4.4/go.mod h1:kenIhsEOeOJmVchQTgglprH7qJGnHDVpk1VPCcaMI8A=
|
github.com/nxadm/tail v1.4.4/go.mod h1:kenIhsEOeOJmVchQTgglprH7qJGnHDVpk1VPCcaMI8A=
|
||||||
github.com/oklog/oklog v0.3.2/go.mod h1:FCV+B7mhrz4o+ueLpx+KqkyXRGMWOYEvfiXtdGtbWGs=
|
github.com/oklog/oklog v0.3.2/go.mod h1:FCV+B7mhrz4o+ueLpx+KqkyXRGMWOYEvfiXtdGtbWGs=
|
||||||
github.com/oklog/run v1.0.0/go.mod h1:dlhp/R75TPv97u0XWUtDeV/lRKWPKSdTuV0TZvrmrQA=
|
github.com/oklog/run v1.0.0/go.mod h1:dlhp/R75TPv97u0XWUtDeV/lRKWPKSdTuV0TZvrmrQA=
|
||||||
|
@ -502,11 +502,13 @@ github.com/onsi/ginkgo v1.6.0/go.mod h1:lLunBs/Ym6LB5Z9jYTR76FiuTmxDTDusOGeTQH+W
|
||||||
github.com/onsi/ginkgo v1.7.0/go.mod h1:lLunBs/Ym6LB5Z9jYTR76FiuTmxDTDusOGeTQH+WWjE=
|
github.com/onsi/ginkgo v1.7.0/go.mod h1:lLunBs/Ym6LB5Z9jYTR76FiuTmxDTDusOGeTQH+WWjE=
|
||||||
github.com/onsi/ginkgo v1.11.0/go.mod h1:lLunBs/Ym6LB5Z9jYTR76FiuTmxDTDusOGeTQH+WWjE=
|
github.com/onsi/ginkgo v1.11.0/go.mod h1:lLunBs/Ym6LB5Z9jYTR76FiuTmxDTDusOGeTQH+WWjE=
|
||||||
github.com/onsi/ginkgo v1.12.1/go.mod h1:zj2OWP4+oCPe1qIXoGWkgMRwljMUYCdkwsT2108oapk=
|
github.com/onsi/ginkgo v1.12.1/go.mod h1:zj2OWP4+oCPe1qIXoGWkgMRwljMUYCdkwsT2108oapk=
|
||||||
|
github.com/onsi/ginkgo v1.14.0 h1:2mOpI4JVVPBN+WQRa0WKH2eXR+Ey+uK4n7Zj0aYpIQA=
|
||||||
github.com/onsi/ginkgo v1.14.0/go.mod h1:iSB4RoI2tjJc9BBv4NKIKWKya62Rps+oPG/Lv9klQyY=
|
github.com/onsi/ginkgo v1.14.0/go.mod h1:iSB4RoI2tjJc9BBv4NKIKWKya62Rps+oPG/Lv9klQyY=
|
||||||
github.com/onsi/gomega v0.0.0-20170829124025-dcabb60a477c/go.mod h1:C1qb7wdrVGGVU+Z6iS04AVkA3Q65CEZX59MT0QO5uiA=
|
github.com/onsi/gomega v0.0.0-20170829124025-dcabb60a477c/go.mod h1:C1qb7wdrVGGVU+Z6iS04AVkA3Q65CEZX59MT0QO5uiA=
|
||||||
github.com/onsi/gomega v1.4.3/go.mod h1:ex+gbHU/CVuBBDIJjb2X0qEXbFg53c61hWP/1CpauHY=
|
github.com/onsi/gomega v1.4.3/go.mod h1:ex+gbHU/CVuBBDIJjb2X0qEXbFg53c61hWP/1CpauHY=
|
||||||
github.com/onsi/gomega v1.7.0/go.mod h1:ex+gbHU/CVuBBDIJjb2X0qEXbFg53c61hWP/1CpauHY=
|
github.com/onsi/gomega v1.7.0/go.mod h1:ex+gbHU/CVuBBDIJjb2X0qEXbFg53c61hWP/1CpauHY=
|
||||||
github.com/onsi/gomega v1.7.1/go.mod h1:XdKZgCCFLUoM/7CFJVPcG8C1xQ1AJ0vpAezJrB7JYyY=
|
github.com/onsi/gomega v1.7.1/go.mod h1:XdKZgCCFLUoM/7CFJVPcG8C1xQ1AJ0vpAezJrB7JYyY=
|
||||||
|
github.com/onsi/gomega v1.10.1 h1:o0+MgICZLuZ7xjH7Vx6zS/zcu93/BEp1VwkIW1mEXCE=
|
||||||
github.com/onsi/gomega v1.10.1/go.mod h1:iN09h71vgCQne3DLsj+A5owkum+a2tYe+TOCB1ybHNo=
|
github.com/onsi/gomega v1.10.1/go.mod h1:iN09h71vgCQne3DLsj+A5owkum+a2tYe+TOCB1ybHNo=
|
||||||
github.com/op/go-logging v0.0.0-20160315200505-970db520ece7/go.mod h1:HzydrMdWErDVzsI23lYNej1Htcns9BCg93Dk0bBINWk=
|
github.com/op/go-logging v0.0.0-20160315200505-970db520ece7/go.mod h1:HzydrMdWErDVzsI23lYNej1Htcns9BCg93Dk0bBINWk=
|
||||||
github.com/opentracing-contrib/go-observer v0.0.0-20170622124052-a52f23424492/go.mod h1:Ngi6UdF0k5OKD5t5wlmGhe/EDKPoUM3BXZSSfIuJbis=
|
github.com/opentracing-contrib/go-observer v0.0.0-20170622124052-a52f23424492/go.mod h1:Ngi6UdF0k5OKD5t5wlmGhe/EDKPoUM3BXZSSfIuJbis=
|
||||||
|
@ -588,7 +590,6 @@ github.com/rogpeppe/go-internal v1.3.0/go.mod h1:M8bDsm7K2OlrFYOpmOWEs/qY81heoFR
|
||||||
github.com/rs/xid v1.2.1/go.mod h1:+uKXf+4Djp6Md1KODXJxgGQPKngRmWyn10oCKFzNHOQ=
|
github.com/rs/xid v1.2.1/go.mod h1:+uKXf+4Djp6Md1KODXJxgGQPKngRmWyn10oCKFzNHOQ=
|
||||||
github.com/rs/zerolog v1.20.0 h1:38k9hgtUBdxFwE34yS8rTHmHBa4eN16E4DJlv177LNs=
|
github.com/rs/zerolog v1.20.0 h1:38k9hgtUBdxFwE34yS8rTHmHBa4eN16E4DJlv177LNs=
|
||||||
github.com/rs/zerolog v1.20.0/go.mod h1:IzD0RJ65iWH0w97OQQebJEvTZYvsCUm9WVLWBQrJRjo=
|
github.com/rs/zerolog v1.20.0/go.mod h1:IzD0RJ65iWH0w97OQQebJEvTZYvsCUm9WVLWBQrJRjo=
|
||||||
github.com/russross/blackfriday v0.0.0-20170610170232-067529f716f4 h1:S9YlS71UNJIyS61OqGAmLXv3w5zclSidN+qwr80XxKs=
|
|
||||||
github.com/russross/blackfriday v0.0.0-20170610170232-067529f716f4/go.mod h1:JO/DiYxRf+HjHt06OyowR9PTA263kcR/rfWxYHBV53g=
|
github.com/russross/blackfriday v0.0.0-20170610170232-067529f716f4/go.mod h1:JO/DiYxRf+HjHt06OyowR9PTA263kcR/rfWxYHBV53g=
|
||||||
github.com/russross/blackfriday v1.5.2 h1:HyvC0ARfnZBqnXwABFeSZHpKvJHJJfPz81GNueLj0oo=
|
github.com/russross/blackfriday v1.5.2 h1:HyvC0ARfnZBqnXwABFeSZHpKvJHJJfPz81GNueLj0oo=
|
||||||
github.com/russross/blackfriday v1.5.2/go.mod h1:JO/DiYxRf+HjHt06OyowR9PTA263kcR/rfWxYHBV53g=
|
github.com/russross/blackfriday v1.5.2/go.mod h1:JO/DiYxRf+HjHt06OyowR9PTA263kcR/rfWxYHBV53g=
|
||||||
|
@ -671,7 +672,6 @@ github.com/xiang90/probing v0.0.0-20190116061207-43a291ad63a2/go.mod h1:UETIi67q
|
||||||
github.com/yuin/goldmark v1.1.25/go.mod h1:3hX8gzYuyVAZsxl0MRgGTJEmQBFcNTphYh9decYSb74=
|
github.com/yuin/goldmark v1.1.25/go.mod h1:3hX8gzYuyVAZsxl0MRgGTJEmQBFcNTphYh9decYSb74=
|
||||||
github.com/yuin/goldmark v1.1.27/go.mod h1:3hX8gzYuyVAZsxl0MRgGTJEmQBFcNTphYh9decYSb74=
|
github.com/yuin/goldmark v1.1.27/go.mod h1:3hX8gzYuyVAZsxl0MRgGTJEmQBFcNTphYh9decYSb74=
|
||||||
github.com/yuin/goldmark v1.1.32/go.mod h1:3hX8gzYuyVAZsxl0MRgGTJEmQBFcNTphYh9decYSb74=
|
github.com/yuin/goldmark v1.1.32/go.mod h1:3hX8gzYuyVAZsxl0MRgGTJEmQBFcNTphYh9decYSb74=
|
||||||
github.com/yuin/goldmark v1.3.5/go.mod h1:mwnBkeHKe2W/ZEtQ+71ViKU8L12m81fl3OWwC1Zlc8k=
|
|
||||||
go.etcd.io/bbolt v1.3.3/go.mod h1:IbVyRI1SCnLcuJnV2u8VeU0CEYM7e686BmAb1XKL+uU=
|
go.etcd.io/bbolt v1.3.3/go.mod h1:IbVyRI1SCnLcuJnV2u8VeU0CEYM7e686BmAb1XKL+uU=
|
||||||
go.etcd.io/etcd v0.0.0-20191023171146-3cf2f69b5738/go.mod h1:dnLIgRNXwCJa5e+c6mIZCrds/GIG4ncV9HhK5PX7jPg=
|
go.etcd.io/etcd v0.0.0-20191023171146-3cf2f69b5738/go.mod h1:dnLIgRNXwCJa5e+c6mIZCrds/GIG4ncV9HhK5PX7jPg=
|
||||||
go.etcd.io/etcd v0.5.0-alpha.5.0.20200306183522-221f0cc107cb/go.mod h1:VZB9Yx4s43MHItytoe8jcvaEFEgF2QzHDZGfQ/XQjvQ=
|
go.etcd.io/etcd v0.5.0-alpha.5.0.20200306183522-221f0cc107cb/go.mod h1:VZB9Yx4s43MHItytoe8jcvaEFEgF2QzHDZGfQ/XQjvQ=
|
||||||
|
@ -752,7 +752,6 @@ golang.org/x/mod v0.1.1-0.20191105210325-c90efee705ee/go.mod h1:QqPTAvyqsEbceGzB
|
||||||
golang.org/x/mod v0.1.1-0.20191107180719-034126e5016b/go.mod h1:QqPTAvyqsEbceGzBzNggFXnrqF1CaUcvgkdR5Ot7KZg=
|
golang.org/x/mod v0.1.1-0.20191107180719-034126e5016b/go.mod h1:QqPTAvyqsEbceGzBzNggFXnrqF1CaUcvgkdR5Ot7KZg=
|
||||||
golang.org/x/mod v0.2.0/go.mod h1:s0Qsj1ACt9ePp/hMypM3fl4fZqREWJwdYDEqhRiZZUA=
|
golang.org/x/mod v0.2.0/go.mod h1:s0Qsj1ACt9ePp/hMypM3fl4fZqREWJwdYDEqhRiZZUA=
|
||||||
golang.org/x/mod v0.3.0/go.mod h1:s0Qsj1ACt9ePp/hMypM3fl4fZqREWJwdYDEqhRiZZUA=
|
golang.org/x/mod v0.3.0/go.mod h1:s0Qsj1ACt9ePp/hMypM3fl4fZqREWJwdYDEqhRiZZUA=
|
||||||
golang.org/x/mod v0.4.2/go.mod h1:s0Qsj1ACt9ePp/hMypM3fl4fZqREWJwdYDEqhRiZZUA=
|
|
||||||
golang.org/x/net v0.0.0-20170114055629-f2499483f923/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
|
golang.org/x/net v0.0.0-20170114055629-f2499483f923/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
|
||||||
golang.org/x/net v0.0.0-20180611182652-db08ff08e862/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
|
golang.org/x/net v0.0.0-20180611182652-db08ff08e862/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
|
||||||
golang.org/x/net v0.0.0-20180724234803-3673e40ba225/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
|
golang.org/x/net v0.0.0-20180724234803-3673e40ba225/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
|
||||||
|
@ -881,8 +880,7 @@ golang.org/x/sys v0.0.0-20200615200032-f1bc736245b1/go.mod h1:h1NjWce9XRLGQEsW7w
|
||||||
golang.org/x/sys v0.0.0-20200625212154-ddb9806d33ae/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
golang.org/x/sys v0.0.0-20200625212154-ddb9806d33ae/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
||||||
golang.org/x/sys v0.0.0-20200803210538-64077c9b5642/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
golang.org/x/sys v0.0.0-20200803210538-64077c9b5642/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
||||||
golang.org/x/sys v0.0.0-20201119102817-f84b799fce68/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
golang.org/x/sys v0.0.0-20201119102817-f84b799fce68/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
||||||
golang.org/x/sys v0.0.0-20210119212857-b64e53b001e4 h1:myAQVi0cGEoqQVR5POX+8RR2mrocKqNN1hmeMqhX27k=
|
golang.org/x/sys v0.0.0-20201231184435-2d18734c6014/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
||||||
golang.org/x/sys v0.0.0-20210119212857-b64e53b001e4/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
|
||||||
golang.org/x/sys v0.0.0-20210330210617-4fbd30eecc44/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
golang.org/x/sys v0.0.0-20210330210617-4fbd30eecc44/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
||||||
golang.org/x/sys v0.0.0-20210510120138-977fb7262007 h1:gG67DSER+11cZvqIMb8S8bt0vZtiN6xWYARwirrOSfE=
|
golang.org/x/sys v0.0.0-20210510120138-977fb7262007 h1:gG67DSER+11cZvqIMb8S8bt0vZtiN6xWYARwirrOSfE=
|
||||||
golang.org/x/sys v0.0.0-20210510120138-977fb7262007/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
|
golang.org/x/sys v0.0.0-20210510120138-977fb7262007/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
|
||||||
|
@ -953,7 +951,6 @@ golang.org/x/tools v0.0.0-20200618134242-20370b0cb4b2/go.mod h1:EkVYQZoAsY45+roY
|
||||||
golang.org/x/tools v0.0.0-20200729194436-6467de6f59a7/go.mod h1:njjCfa9FT2d7l9Bc6FUM5FLjQPp3cFF28FI3qnDFljA=
|
golang.org/x/tools v0.0.0-20200729194436-6467de6f59a7/go.mod h1:njjCfa9FT2d7l9Bc6FUM5FLjQPp3cFF28FI3qnDFljA=
|
||||||
golang.org/x/tools v0.0.0-20200804011535-6c149bb5ef0d/go.mod h1:njjCfa9FT2d7l9Bc6FUM5FLjQPp3cFF28FI3qnDFljA=
|
golang.org/x/tools v0.0.0-20200804011535-6c149bb5ef0d/go.mod h1:njjCfa9FT2d7l9Bc6FUM5FLjQPp3cFF28FI3qnDFljA=
|
||||||
golang.org/x/tools v0.0.0-20200825202427-b303f430e36d/go.mod h1:njjCfa9FT2d7l9Bc6FUM5FLjQPp3cFF28FI3qnDFljA=
|
golang.org/x/tools v0.0.0-20200825202427-b303f430e36d/go.mod h1:njjCfa9FT2d7l9Bc6FUM5FLjQPp3cFF28FI3qnDFljA=
|
||||||
golang.org/x/tools v0.1.1/go.mod h1:o0xws9oXOQQZyjljx8fwUC0k7L1pTE6eaCbjGeHmOkk=
|
|
||||||
golang.org/x/xerrors v0.0.0-20190717185122-a985d3407aa7/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0=
|
golang.org/x/xerrors v0.0.0-20190717185122-a985d3407aa7/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0=
|
||||||
golang.org/x/xerrors v0.0.0-20191011141410-1b5146add898/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0=
|
golang.org/x/xerrors v0.0.0-20191011141410-1b5146add898/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0=
|
||||||
golang.org/x/xerrors v0.0.0-20191204190536-9bdfabe68543/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0=
|
golang.org/x/xerrors v0.0.0-20191204190536-9bdfabe68543/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0=
|
||||||
|
@ -1084,6 +1081,7 @@ gopkg.in/resty.v1 v1.12.0/go.mod h1:mDo4pnntr5jdWRML875a/NmxYqAlA73dVijT2AXvQQo=
|
||||||
gopkg.in/square/go-jose.v2 v2.3.1/go.mod h1:M9dMgbHiYLoDGQrXy7OpJDJWiKiU//h+vD76mk0e1AI=
|
gopkg.in/square/go-jose.v2 v2.3.1/go.mod h1:M9dMgbHiYLoDGQrXy7OpJDJWiKiU//h+vD76mk0e1AI=
|
||||||
gopkg.in/square/go-jose.v2 v2.4.0 h1:0kXPskUMGAXXWJlP05ktEMOV0vmzFQUWw6d+aZJQU8A=
|
gopkg.in/square/go-jose.v2 v2.4.0 h1:0kXPskUMGAXXWJlP05ktEMOV0vmzFQUWw6d+aZJQU8A=
|
||||||
gopkg.in/square/go-jose.v2 v2.4.0/go.mod h1:M9dMgbHiYLoDGQrXy7OpJDJWiKiU//h+vD76mk0e1AI=
|
gopkg.in/square/go-jose.v2 v2.4.0/go.mod h1:M9dMgbHiYLoDGQrXy7OpJDJWiKiU//h+vD76mk0e1AI=
|
||||||
|
gopkg.in/tomb.v1 v1.0.0-20141024135613-dd632973f1e7 h1:uRGJdciOHaEIrze2W8Q3AKkepLTh2hOroT7a+7czfdQ=
|
||||||
gopkg.in/tomb.v1 v1.0.0-20141024135613-dd632973f1e7/go.mod h1:dt/ZhP58zS4L8KSrWDmTeBkI65Dw0HsyUHuEVlX15mw=
|
gopkg.in/tomb.v1 v1.0.0-20141024135613-dd632973f1e7/go.mod h1:dt/ZhP58zS4L8KSrWDmTeBkI65Dw0HsyUHuEVlX15mw=
|
||||||
gopkg.in/warnings.v0 v0.1.2/go.mod h1:jksf8JmL6Qr/oQM2OXTHunEvvTAsrWBLb6OOjuVWRNI=
|
gopkg.in/warnings.v0 v0.1.2/go.mod h1:jksf8JmL6Qr/oQM2OXTHunEvvTAsrWBLb6OOjuVWRNI=
|
||||||
gopkg.in/yaml.v2 v2.0.0-20170812160011-eb3733d160e7/go.mod h1:JAlM8MvJe8wmxCU4Bli9HhUf9+ttbYbLASfIpnQbh74=
|
gopkg.in/yaml.v2 v2.0.0-20170812160011-eb3733d160e7/go.mod h1:JAlM8MvJe8wmxCU4Bli9HhUf9+ttbYbLASfIpnQbh74=
|
||||||
|
|
|
@ -0,0 +1,5 @@
|
||||||
|
root = true
|
||||||
|
|
||||||
|
[*]
|
||||||
|
indent_style = tab
|
||||||
|
indent_size = 2
|
|
@ -8,12 +8,17 @@
|
||||||
[![Windows Build Status](https://img.shields.io/appveyor/ci/lucas-clemente/quic-go/master.svg?style=flat-square&label=windows+build)](https://ci.appveyor.com/project/lucas-clemente/quic-go/branch/master)
|
[![Windows Build Status](https://img.shields.io/appveyor/ci/lucas-clemente/quic-go/master.svg?style=flat-square&label=windows+build)](https://ci.appveyor.com/project/lucas-clemente/quic-go/branch/master)
|
||||||
[![Code Coverage](https://img.shields.io/codecov/c/github/lucas-clemente/quic-go/master.svg?style=flat-square)](https://codecov.io/gh/lucas-clemente/quic-go/)
|
[![Code Coverage](https://img.shields.io/codecov/c/github/lucas-clemente/quic-go/master.svg?style=flat-square)](https://codecov.io/gh/lucas-clemente/quic-go/)
|
||||||
|
|
||||||
quic-go is an implementation of the [QUIC protocol, RFC 9000](https://datatracker.ietf.org/doc/html/rfc9000) protocol in Go.
|
quic-go is an implementation of the [QUIC](https://en.wikipedia.org/wiki/QUIC) protocol in Go. It implements the [IETF QUIC draft-29](https://tools.ietf.org/html/draft-ietf-quic-transport-29), [draft-32](https://tools.ietf.org/html/draft-ietf-quic-transport-32) and [draft-34](https://tools.ietf.org/html/draft-ietf-quic-transport-34).
|
||||||
In addition to RFC 9000, it currently implements the [IETF QUIC draft-29](https://tools.ietf.org/html/draft-ietf-quic-transport-29), [draft-32](https://tools.ietf.org/html/draft-ietf-quic-transport-32) and [draft-34](https://tools.ietf.org/html/draft-ietf-quic-transport-34). Support for draft versions will be eventually be dropped, as these are phased out of the ecosystem.
|
|
||||||
|
## Version compatibility
|
||||||
|
|
||||||
|
Since quic-go is under active development, there's no guarantee that two builds of different commits are interoperable. The QUIC version used in the *master* branch is just a placeholder, and should not be considered stable.
|
||||||
|
|
||||||
|
When using quic-go as a library, please always use a [tagged release](https://github.com/lucas-clemente/quic-go/releases). Only these releases use the official draft version numbers.
|
||||||
|
|
||||||
## Guides
|
## Guides
|
||||||
|
|
||||||
*We currently support Go 1.15.x, Go 1.16.x and Go 1.17 Beta 1, with [Go modules](https://github.com/golang/go/wiki/Modules) support enabled.*
|
*We currently support Go 1.15+, with [Go modules](https://github.com/golang/go/wiki/Modules) support enabled.*
|
||||||
|
|
||||||
Running tests:
|
Running tests:
|
||||||
|
|
||||||
|
|
|
@ -38,7 +38,6 @@ type client struct {
|
||||||
session quicSession
|
session quicSession
|
||||||
|
|
||||||
tracer logging.ConnectionTracer
|
tracer logging.ConnectionTracer
|
||||||
tracingID uint64
|
|
||||||
logger utils.Logger
|
logger utils.Logger
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -203,16 +202,8 @@ func dialContext(
|
||||||
}
|
}
|
||||||
c.packetHandlers = packetHandlers
|
c.packetHandlers = packetHandlers
|
||||||
|
|
||||||
c.tracingID = nextSessionTracingID()
|
|
||||||
if c.config.Tracer != nil {
|
if c.config.Tracer != nil {
|
||||||
c.tracer = c.config.Tracer.TracerForConnection(
|
c.tracer = c.config.Tracer.TracerForConnection(protocol.PerspectiveClient, c.destConnID)
|
||||||
context.WithValue(ctx, SessionTracingKey, c.tracingID),
|
|
||||||
protocol.PerspectiveClient,
|
|
||||||
c.destConnID,
|
|
||||||
)
|
|
||||||
}
|
|
||||||
if c.tracer != nil {
|
|
||||||
c.tracer.StartedConnection(c.conn.LocalAddr(), c.conn.RemoteAddr(), c.srcConnID, c.destConnID)
|
|
||||||
}
|
}
|
||||||
if err := c.dial(ctx); err != nil {
|
if err := c.dial(ctx); err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
|
@ -279,6 +270,9 @@ func newClient(
|
||||||
|
|
||||||
func (c *client) dial(ctx context.Context) error {
|
func (c *client) dial(ctx context.Context) error {
|
||||||
c.logger.Infof("Starting new connection to %s (%s -> %s), source connection ID %s, destination connection ID %s, version %s", c.tlsConf.ServerName, c.conn.LocalAddr(), c.conn.RemoteAddr(), c.srcConnID, c.destConnID, c.version)
|
c.logger.Infof("Starting new connection to %s (%s -> %s), source connection ID %s, destination connection ID %s, version %s", c.tlsConf.ServerName, c.conn.LocalAddr(), c.conn.RemoteAddr(), c.srcConnID, c.destConnID, c.version)
|
||||||
|
if c.tracer != nil {
|
||||||
|
c.tracer.StartedConnection(c.conn.LocalAddr(), c.conn.RemoteAddr(), c.version, c.srcConnID, c.destConnID)
|
||||||
|
}
|
||||||
|
|
||||||
c.session = newClientSession(
|
c.session = newClientSession(
|
||||||
c.conn,
|
c.conn,
|
||||||
|
@ -291,7 +285,6 @@ func (c *client) dial(ctx context.Context) error {
|
||||||
c.use0RTT,
|
c.use0RTT,
|
||||||
c.hasNegotiatedVersion,
|
c.hasNegotiatedVersion,
|
||||||
c.tracer,
|
c.tracer,
|
||||||
c.tracingID,
|
|
||||||
c.logger,
|
c.logger,
|
||||||
c.version,
|
c.version,
|
||||||
)
|
)
|
||||||
|
@ -300,7 +293,7 @@ func (c *client) dial(ctx context.Context) error {
|
||||||
errorChan := make(chan error, 1)
|
errorChan := make(chan error, 1)
|
||||||
go func() {
|
go func() {
|
||||||
err := c.session.run() // returns as soon as the session is closed
|
err := c.session.run() // returns as soon as the session is closed
|
||||||
if !errors.Is(err, &errCloseForRecreating{}) && c.createdPacketConn {
|
if !errors.Is(err, errCloseForRecreating{}) && c.createdPacketConn {
|
||||||
c.packetHandlers.Destroy()
|
c.packetHandlers.Destroy()
|
||||||
}
|
}
|
||||||
errorChan <- err
|
errorChan <- err
|
||||||
|
|
|
@ -73,10 +73,7 @@ func (m *connIDGenerator) SetMaxActiveConnIDs(limit uint64) error {
|
||||||
|
|
||||||
func (m *connIDGenerator) Retire(seq uint64, sentWithDestConnID protocol.ConnectionID) error {
|
func (m *connIDGenerator) Retire(seq uint64, sentWithDestConnID protocol.ConnectionID) error {
|
||||||
if seq > m.highestSeq {
|
if seq > m.highestSeq {
|
||||||
return &qerr.TransportError{
|
return qerr.NewError(qerr.ProtocolViolation, fmt.Sprintf("tried to retire connection ID %d. Highest issued: %d", seq, m.highestSeq))
|
||||||
ErrorCode: qerr.ProtocolViolation,
|
|
||||||
ErrorMessage: fmt.Sprintf("retired connection ID %d (highest issued: %d)", seq, m.highestSeq),
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
connID, ok := m.activeSrcConnIDs[seq]
|
connID, ok := m.activeSrcConnIDs[seq]
|
||||||
// We might already have deleted this connection ID, if this is a duplicate frame.
|
// We might already have deleted this connection ID, if this is a duplicate frame.
|
||||||
|
@ -84,10 +81,7 @@ func (m *connIDGenerator) Retire(seq uint64, sentWithDestConnID protocol.Connect
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
if connID.Equal(sentWithDestConnID) && !protocol.UseRetireBugBackwardsCompatibilityMode(RetireBugBackwardsCompatibilityMode, m.version) {
|
if connID.Equal(sentWithDestConnID) && !protocol.UseRetireBugBackwardsCompatibilityMode(RetireBugBackwardsCompatibilityMode, m.version) {
|
||||||
return &qerr.TransportError{
|
return qerr.NewError(qerr.ProtocolViolation, fmt.Sprintf("tried to retire connection ID %d (%s), which was used as the Destination Connection ID on this packet", seq, connID))
|
||||||
ErrorCode: qerr.ProtocolViolation,
|
|
||||||
ErrorMessage: fmt.Sprintf("retired connection ID %d (%s), which was used as the Destination Connection ID on this packet", seq, connID),
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
m.retireConnectionID(connID)
|
m.retireConnectionID(connID)
|
||||||
delete(m.activeSrcConnIDs, seq)
|
delete(m.activeSrcConnIDs, seq)
|
||||||
|
|
|
@ -53,7 +53,7 @@ func (h *connIDManager) Add(f *wire.NewConnectionIDFrame) error {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
if h.queue.Len() >= protocol.MaxActiveConnectionIDs {
|
if h.queue.Len() >= protocol.MaxActiveConnectionIDs {
|
||||||
return &qerr.TransportError{ErrorCode: qerr.ConnectionIDLimitError}
|
return qerr.ConnectionIDLimitError
|
||||||
}
|
}
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
|
@ -11,21 +11,7 @@ import (
|
||||||
"golang.org/x/sys/windows"
|
"golang.org/x/sys/windows"
|
||||||
)
|
)
|
||||||
|
|
||||||
const IP_DONTFRAGMENT = 14
|
func newConn(c net.PacketConn) (connection, error) {
|
||||||
|
|
||||||
func newConn(c OOBCapablePacketConn) (connection, error) {
|
|
||||||
rawConn, err := c.SyscallConn()
|
|
||||||
if err != nil {
|
|
||||||
return nil, fmt.Errorf("couldn't get syscall.RawConn: %w", err)
|
|
||||||
}
|
|
||||||
if err := rawConn.Control(func(fd uintptr) {
|
|
||||||
// This should succeed if the connection is a IPv4 or a dual-stack connection.
|
|
||||||
// It will fail for IPv6 connections.
|
|
||||||
// TODO: properly handle error.
|
|
||||||
_ = windows.SetsockoptInt(windows.Handle(fd), windows.IPPROTO_IP, IP_DONTFRAGMENT, 1)
|
|
||||||
}); err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
return &basicConn{PacketConn: c}, nil
|
return &basicConn{PacketConn: c}, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -39,18 +39,12 @@ func newCryptoStream() cryptoStream {
|
||||||
func (s *cryptoStreamImpl) HandleCryptoFrame(f *wire.CryptoFrame) error {
|
func (s *cryptoStreamImpl) HandleCryptoFrame(f *wire.CryptoFrame) error {
|
||||||
highestOffset := f.Offset + protocol.ByteCount(len(f.Data))
|
highestOffset := f.Offset + protocol.ByteCount(len(f.Data))
|
||||||
if maxOffset := highestOffset; maxOffset > protocol.MaxCryptoStreamOffset {
|
if maxOffset := highestOffset; maxOffset > protocol.MaxCryptoStreamOffset {
|
||||||
return &qerr.TransportError{
|
return qerr.NewError(qerr.CryptoBufferExceeded, fmt.Sprintf("received invalid offset %d on crypto stream, maximum allowed %d", maxOffset, protocol.MaxCryptoStreamOffset))
|
||||||
ErrorCode: qerr.CryptoBufferExceeded,
|
|
||||||
ErrorMessage: fmt.Sprintf("received invalid offset %d on crypto stream, maximum allowed %d", maxOffset, protocol.MaxCryptoStreamOffset),
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
if s.finished {
|
if s.finished {
|
||||||
if highestOffset > s.highestOffset {
|
if highestOffset > s.highestOffset {
|
||||||
// reject crypto data received after this stream was already finished
|
// reject crypto data received after this stream was already finished
|
||||||
return &qerr.TransportError{
|
return qerr.NewError(qerr.ProtocolViolation, "received crypto data after change of encryption level")
|
||||||
ErrorCode: qerr.ProtocolViolation,
|
|
||||||
ErrorMessage: "received crypto data after change of encryption level",
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
// ignore data with a smaller offset than the highest received
|
// ignore data with a smaller offset than the highest received
|
||||||
// could e.g. be a retransmission
|
// could e.g. be a retransmission
|
||||||
|
@ -86,10 +80,7 @@ func (s *cryptoStreamImpl) GetCryptoData() []byte {
|
||||||
|
|
||||||
func (s *cryptoStreamImpl) Finish() error {
|
func (s *cryptoStreamImpl) Finish() error {
|
||||||
if s.queue.HasMoreData() {
|
if s.queue.HasMoreData() {
|
||||||
return &qerr.TransportError{
|
return qerr.NewError(qerr.ProtocolViolation, "encryption level changed, but crypto stream has more data to read")
|
||||||
ErrorCode: qerr.ProtocolViolation,
|
|
||||||
ErrorMessage: "encryption level changed, but crypto stream has more data to read",
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
s.finished = true
|
s.finished = true
|
||||||
return nil
|
return nil
|
||||||
|
|
|
@ -1,58 +0,0 @@
|
||||||
package quic
|
|
||||||
|
|
||||||
import (
|
|
||||||
"fmt"
|
|
||||||
|
|
||||||
"github.com/lucas-clemente/quic-go/internal/qerr"
|
|
||||||
)
|
|
||||||
|
|
||||||
type (
|
|
||||||
TransportError = qerr.TransportError
|
|
||||||
ApplicationError = qerr.ApplicationError
|
|
||||||
VersionNegotiationError = qerr.VersionNegotiationError
|
|
||||||
StatelessResetError = qerr.StatelessResetError
|
|
||||||
IdleTimeoutError = qerr.IdleTimeoutError
|
|
||||||
HandshakeTimeoutError = qerr.HandshakeTimeoutError
|
|
||||||
)
|
|
||||||
|
|
||||||
type (
|
|
||||||
TransportErrorCode = qerr.TransportErrorCode
|
|
||||||
ApplicationErrorCode = qerr.ApplicationErrorCode
|
|
||||||
StreamErrorCode = qerr.StreamErrorCode
|
|
||||||
)
|
|
||||||
|
|
||||||
const (
|
|
||||||
NoError = qerr.NoError
|
|
||||||
InternalError = qerr.InternalError
|
|
||||||
ConnectionRefused = qerr.ConnectionRefused
|
|
||||||
FlowControlError = qerr.FlowControlError
|
|
||||||
StreamLimitError = qerr.StreamLimitError
|
|
||||||
StreamStateError = qerr.StreamStateError
|
|
||||||
FinalSizeError = qerr.FinalSizeError
|
|
||||||
FrameEncodingError = qerr.FrameEncodingError
|
|
||||||
TransportParameterError = qerr.TransportParameterError
|
|
||||||
ConnectionIDLimitError = qerr.ConnectionIDLimitError
|
|
||||||
ProtocolViolation = qerr.ProtocolViolation
|
|
||||||
InvalidToken = qerr.InvalidToken
|
|
||||||
ApplicationErrorErrorCode = qerr.ApplicationErrorErrorCode
|
|
||||||
CryptoBufferExceeded = qerr.CryptoBufferExceeded
|
|
||||||
KeyUpdateError = qerr.KeyUpdateError
|
|
||||||
AEADLimitReached = qerr.AEADLimitReached
|
|
||||||
NoViablePathError = qerr.NoViablePathError
|
|
||||||
)
|
|
||||||
|
|
||||||
// A StreamError is used for Stream.CancelRead and Stream.CancelWrite.
|
|
||||||
// It is also returned from Stream.Read and Stream.Write if the peer canceled reading or writing.
|
|
||||||
type StreamError struct {
|
|
||||||
StreamID StreamID
|
|
||||||
ErrorCode StreamErrorCode
|
|
||||||
}
|
|
||||||
|
|
||||||
func (e *StreamError) Is(target error) bool {
|
|
||||||
_, ok := target.(*StreamError)
|
|
||||||
return ok
|
|
||||||
}
|
|
||||||
|
|
||||||
func (e *StreamError) Error() string {
|
|
||||||
return fmt.Sprintf("stream %d canceled with error code %d", e.StreamID, e.ErrorCode)
|
|
||||||
}
|
|
|
@ -1,20 +1,19 @@
|
||||||
module github.com/lucas-clemente/quic-go
|
module github.com/lucas-clemente/quic-go
|
||||||
|
|
||||||
go 1.15
|
go 1.14
|
||||||
|
|
||||||
require (
|
require (
|
||||||
github.com/cheekybits/genny v1.0.0
|
github.com/cheekybits/genny v1.0.0
|
||||||
github.com/francoispqt/gojay v1.2.13
|
github.com/francoispqt/gojay v1.2.13
|
||||||
github.com/golang/mock v1.6.0
|
github.com/golang/mock v1.5.0
|
||||||
github.com/marten-seemann/qpack v0.2.1
|
github.com/marten-seemann/qpack v0.2.1
|
||||||
github.com/marten-seemann/qtls-go1-15 v0.1.4
|
github.com/marten-seemann/qtls-go1-15 v0.1.4
|
||||||
github.com/marten-seemann/qtls-go1-16 v0.1.3
|
github.com/marten-seemann/qtls-go1-16 v0.1.3
|
||||||
github.com/marten-seemann/qtls-go1-17 v0.1.0-beta.1.2
|
|
||||||
github.com/onsi/ginkgo v1.14.0
|
github.com/onsi/ginkgo v1.14.0
|
||||||
github.com/onsi/gomega v1.10.1
|
github.com/onsi/gomega v1.10.1
|
||||||
golang.org/x/crypto v0.0.0-20200622213623-75b288015ac9
|
golang.org/x/crypto v0.0.0-20200622213623-75b288015ac9
|
||||||
golang.org/x/net v0.0.0-20210405180319-a5a99cb37ef4
|
golang.org/x/net v0.0.0-20200707034311-ab3426394381
|
||||||
golang.org/x/sync v0.0.0-20210220032951-036812b2e83c
|
golang.org/x/sync v0.0.0-20190911185100-cd5d95a43a6e
|
||||||
golang.org/x/sys v0.0.0-20210510120138-977fb7262007
|
golang.org/x/sys v0.0.0-20201231184435-2d18734c6014
|
||||||
gopkg.in/check.v1 v1.0.0-20180628173108-788fd7840127 // indirect
|
gopkg.in/check.v1 v1.0.0-20180628173108-788fd7840127 // indirect
|
||||||
)
|
)
|
||||||
|
|
|
@ -34,8 +34,8 @@ github.com/golang/lint v0.0.0-20180702182130-06c8688daad7/go.mod h1:tluoj9z5200j
|
||||||
github.com/golang/mock v1.1.1/go.mod h1:oTYuIxOrZwtPieC+H1uAHpcLFnEyAGVDL/k47Jfbm0A=
|
github.com/golang/mock v1.1.1/go.mod h1:oTYuIxOrZwtPieC+H1uAHpcLFnEyAGVDL/k47Jfbm0A=
|
||||||
github.com/golang/mock v1.2.0/go.mod h1:oTYuIxOrZwtPieC+H1uAHpcLFnEyAGVDL/k47Jfbm0A=
|
github.com/golang/mock v1.2.0/go.mod h1:oTYuIxOrZwtPieC+H1uAHpcLFnEyAGVDL/k47Jfbm0A=
|
||||||
github.com/golang/mock v1.4.4/go.mod h1:l3mdAwkq5BuhzHwde/uurv3sEJeZMXNpwsxVWU71h+4=
|
github.com/golang/mock v1.4.4/go.mod h1:l3mdAwkq5BuhzHwde/uurv3sEJeZMXNpwsxVWU71h+4=
|
||||||
github.com/golang/mock v1.6.0 h1:ErTB+efbowRARo13NNdxyJji2egdxLGQhRaY+DUumQc=
|
github.com/golang/mock v1.5.0 h1:jlYHihg//f7RRwuPfptm04yp4s7O6Kw8EZiVYIGcH0g=
|
||||||
github.com/golang/mock v1.6.0/go.mod h1:p6yTPP+5HYm5mzsMV8JkE6ZKdX+/wYM6Hr+LicevLPs=
|
github.com/golang/mock v1.5.0/go.mod h1:CWnOUgYIOo4TcNZ0wHX3YZCqsaM1I1Jvs6v3mP3KVu8=
|
||||||
github.com/golang/protobuf v1.2.0/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U=
|
github.com/golang/protobuf v1.2.0/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U=
|
||||||
github.com/golang/protobuf v1.3.1/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U=
|
github.com/golang/protobuf v1.3.1/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U=
|
||||||
github.com/golang/protobuf v1.4.0-rc.1/go.mod h1:ceaxUfeHdC40wWswd/P6IGgMaK3YpKi5j83Wpe3EHw8=
|
github.com/golang/protobuf v1.4.0-rc.1/go.mod h1:ceaxUfeHdC40wWswd/P6IGgMaK3YpKi5j83Wpe3EHw8=
|
||||||
|
@ -79,8 +79,6 @@ github.com/marten-seemann/qtls-go1-15 v0.1.4 h1:RehYMOyRW8hPVEja1KBVsFVNSm35Jj9M
|
||||||
github.com/marten-seemann/qtls-go1-15 v0.1.4/go.mod h1:GyFwywLKkRt+6mfU99csTEY1joMZz5vmB1WNZH3P81I=
|
github.com/marten-seemann/qtls-go1-15 v0.1.4/go.mod h1:GyFwywLKkRt+6mfU99csTEY1joMZz5vmB1WNZH3P81I=
|
||||||
github.com/marten-seemann/qtls-go1-16 v0.1.3 h1:XEZ1xGorVy9u+lJq+WXNE+hiqRYLNvJGYmwfwKQN2gU=
|
github.com/marten-seemann/qtls-go1-16 v0.1.3 h1:XEZ1xGorVy9u+lJq+WXNE+hiqRYLNvJGYmwfwKQN2gU=
|
||||||
github.com/marten-seemann/qtls-go1-16 v0.1.3/go.mod h1:gNpI2Ol+lRS3WwSOtIUUtRwZEQMXjYK+dQSBFbethAk=
|
github.com/marten-seemann/qtls-go1-16 v0.1.3/go.mod h1:gNpI2Ol+lRS3WwSOtIUUtRwZEQMXjYK+dQSBFbethAk=
|
||||||
github.com/marten-seemann/qtls-go1-17 v0.1.0-beta.1.2 h1:SficYjyOthSrliKI+EaFuXS6HqSsX3dkY9AqxAAjBjw=
|
|
||||||
github.com/marten-seemann/qtls-go1-17 v0.1.0-beta.1.2/go.mod h1:fz4HIxByo+LlWcreM4CZOYNuz3taBQ8rN2X6FqvaWo8=
|
|
||||||
github.com/matttproud/golang_protobuf_extensions v1.0.1/go.mod h1:D8He9yQNgCq6Z5Ld7szi9bcBfOoFv/3dc6xSMkL2PC0=
|
github.com/matttproud/golang_protobuf_extensions v1.0.1/go.mod h1:D8He9yQNgCq6Z5Ld7szi9bcBfOoFv/3dc6xSMkL2PC0=
|
||||||
github.com/microcosm-cc/bluemonday v1.0.1/go.mod h1:hsXNsILzKxV+sX77C5b8FSuKF00vh2OMYv+xgHpAMF4=
|
github.com/microcosm-cc/bluemonday v1.0.1/go.mod h1:hsXNsILzKxV+sX77C5b8FSuKF00vh2OMYv+xgHpAMF4=
|
||||||
github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd/go.mod h1:6dJC0mAP4ikYIbvyc7fijjWJddQyLn8Ig3JB5CqoB9Q=
|
github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd/go.mod h1:6dJC0mAP4ikYIbvyc7fijjWJddQyLn8Ig3JB5CqoB9Q=
|
||||||
|
@ -135,7 +133,6 @@ github.com/stretchr/testify v1.2.2/go.mod h1:a8OnRcib4nhh0OaRAV+Yts87kKdq0PP7pXf
|
||||||
github.com/tarm/serial v0.0.0-20180830185346-98f6abe2eb07/go.mod h1:kDXzergiv9cbyO7IOYJZWg1U88JhDg3PB6klq9Hg2pA=
|
github.com/tarm/serial v0.0.0-20180830185346-98f6abe2eb07/go.mod h1:kDXzergiv9cbyO7IOYJZWg1U88JhDg3PB6klq9Hg2pA=
|
||||||
github.com/viant/assertly v0.4.8/go.mod h1:aGifi++jvCrUaklKEKT0BU95igDNaqkvz+49uaYMPRU=
|
github.com/viant/assertly v0.4.8/go.mod h1:aGifi++jvCrUaklKEKT0BU95igDNaqkvz+49uaYMPRU=
|
||||||
github.com/viant/toolbox v0.24.0/go.mod h1:OxMCG57V0PXuIP2HNQrtJf2CjqdmbrOx5EkMILuUhzM=
|
github.com/viant/toolbox v0.24.0/go.mod h1:OxMCG57V0PXuIP2HNQrtJf2CjqdmbrOx5EkMILuUhzM=
|
||||||
github.com/yuin/goldmark v1.3.5/go.mod h1:mwnBkeHKe2W/ZEtQ+71ViKU8L12m81fl3OWwC1Zlc8k=
|
|
||||||
go.opencensus.io v0.18.0/go.mod h1:vKdFvxhtzZ9onBp9VKHK8z/sRpBMnKAsufL7wlDrCOA=
|
go.opencensus.io v0.18.0/go.mod h1:vKdFvxhtzZ9onBp9VKHK8z/sRpBMnKAsufL7wlDrCOA=
|
||||||
go4.org v0.0.0-20180809161055-417644f6feb5/go.mod h1:MkTOUMDaeVYJUOUsaDXIhWPZYa1yOyC1qaOBpL57BhE=
|
go4.org v0.0.0-20180809161055-417644f6feb5/go.mod h1:MkTOUMDaeVYJUOUsaDXIhWPZYa1yOyC1qaOBpL57BhE=
|
||||||
golang.org/x/build v0.0.0-20190111050920-041ab4dc3f9d/go.mod h1:OWs+y06UdEOHN4y+MfF/py+xQ/tYqIWW03b70/CG9Rw=
|
golang.org/x/build v0.0.0-20190111050920-041ab4dc3f9d/go.mod h1:OWs+y06UdEOHN4y+MfF/py+xQ/tYqIWW03b70/CG9Rw=
|
||||||
|
@ -150,7 +147,7 @@ golang.org/x/exp v0.0.0-20190121172915-509febef88a4/go.mod h1:CJ0aWSM057203Lf6IL
|
||||||
golang.org/x/lint v0.0.0-20180702182130-06c8688daad7/go.mod h1:UVdnD1Gm6xHRNCYTkRU2/jEulfH38KcIWyp/GAMgvoE=
|
golang.org/x/lint v0.0.0-20180702182130-06c8688daad7/go.mod h1:UVdnD1Gm6xHRNCYTkRU2/jEulfH38KcIWyp/GAMgvoE=
|
||||||
golang.org/x/lint v0.0.0-20181026193005-c67002cb31c3/go.mod h1:UVdnD1Gm6xHRNCYTkRU2/jEulfH38KcIWyp/GAMgvoE=
|
golang.org/x/lint v0.0.0-20181026193005-c67002cb31c3/go.mod h1:UVdnD1Gm6xHRNCYTkRU2/jEulfH38KcIWyp/GAMgvoE=
|
||||||
golang.org/x/lint v0.0.0-20190227174305-5b3e6a55c961/go.mod h1:wehouNa3lNwaWXcvxsM5YxQ5yQlVC4a0KAMCusXpPoU=
|
golang.org/x/lint v0.0.0-20190227174305-5b3e6a55c961/go.mod h1:wehouNa3lNwaWXcvxsM5YxQ5yQlVC4a0KAMCusXpPoU=
|
||||||
golang.org/x/mod v0.4.2/go.mod h1:s0Qsj1ACt9ePp/hMypM3fl4fZqREWJwdYDEqhRiZZUA=
|
golang.org/x/mod v0.3.0/go.mod h1:s0Qsj1ACt9ePp/hMypM3fl4fZqREWJwdYDEqhRiZZUA=
|
||||||
golang.org/x/net v0.0.0-20180724234803-3673e40ba225/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
|
golang.org/x/net v0.0.0-20180724234803-3673e40ba225/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
|
||||||
golang.org/x/net v0.0.0-20180826012351-8a410e7b638d/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
|
golang.org/x/net v0.0.0-20180826012351-8a410e7b638d/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
|
||||||
golang.org/x/net v0.0.0-20180906233101-161cd47e91fd/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
|
golang.org/x/net v0.0.0-20180906233101-161cd47e91fd/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
|
||||||
|
@ -163,9 +160,8 @@ golang.org/x/net v0.0.0-20190313220215-9f648a60d977/go.mod h1:t9HGtf8HONx5eT2rtn
|
||||||
golang.org/x/net v0.0.0-20190404232315-eb5bcb51f2a3/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg=
|
golang.org/x/net v0.0.0-20190404232315-eb5bcb51f2a3/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg=
|
||||||
golang.org/x/net v0.0.0-20190620200207-3b0461eec859/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s=
|
golang.org/x/net v0.0.0-20190620200207-3b0461eec859/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s=
|
||||||
golang.org/x/net v0.0.0-20200520004742-59133d7f0dd7/go.mod h1:qpuaurCH72eLCgpAm/N6yyVIVM9cpaDIP3A8BGJEC5A=
|
golang.org/x/net v0.0.0-20200520004742-59133d7f0dd7/go.mod h1:qpuaurCH72eLCgpAm/N6yyVIVM9cpaDIP3A8BGJEC5A=
|
||||||
|
golang.org/x/net v0.0.0-20200707034311-ab3426394381 h1:VXak5I6aEWmAXeQjA+QSZzlgNrpq9mjcfDemuexIKsU=
|
||||||
golang.org/x/net v0.0.0-20200707034311-ab3426394381/go.mod h1:/O7V0waA8r7cgGh81Ro3o1hOxt32SMVPicZroKQ2sZA=
|
golang.org/x/net v0.0.0-20200707034311-ab3426394381/go.mod h1:/O7V0waA8r7cgGh81Ro3o1hOxt32SMVPicZroKQ2sZA=
|
||||||
golang.org/x/net v0.0.0-20210405180319-a5a99cb37ef4 h1:4nGaVu0QrbjT/AK2PRLuQfQuh6DJve+pELhqTdAj3x0=
|
|
||||||
golang.org/x/net v0.0.0-20210405180319-a5a99cb37ef4/go.mod h1:p54w0d4576C0XHj96bSt6lcn1PtDYWL6XObtHCRCNQM=
|
|
||||||
golang.org/x/oauth2 v0.0.0-20180821212333-d2e6202438be/go.mod h1:N/0e6XlmueqKjAGxoOufVs8QHGRruUQn6yWY3a++T0U=
|
golang.org/x/oauth2 v0.0.0-20180821212333-d2e6202438be/go.mod h1:N/0e6XlmueqKjAGxoOufVs8QHGRruUQn6yWY3a++T0U=
|
||||||
golang.org/x/oauth2 v0.0.0-20181017192945-9dcd33a902f4/go.mod h1:N/0e6XlmueqKjAGxoOufVs8QHGRruUQn6yWY3a++T0U=
|
golang.org/x/oauth2 v0.0.0-20181017192945-9dcd33a902f4/go.mod h1:N/0e6XlmueqKjAGxoOufVs8QHGRruUQn6yWY3a++T0U=
|
||||||
golang.org/x/oauth2 v0.0.0-20181203162652-d668ce993890/go.mod h1:N/0e6XlmueqKjAGxoOufVs8QHGRruUQn6yWY3a++T0U=
|
golang.org/x/oauth2 v0.0.0-20181203162652-d668ce993890/go.mod h1:N/0e6XlmueqKjAGxoOufVs8QHGRruUQn6yWY3a++T0U=
|
||||||
|
@ -176,8 +172,8 @@ golang.org/x/sync v0.0.0-20181108010431-42b317875d0f/go.mod h1:RxMgew5VJxzue5/jJ
|
||||||
golang.org/x/sync v0.0.0-20181221193216-37e7f081c4d4/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
|
golang.org/x/sync v0.0.0-20181221193216-37e7f081c4d4/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
|
||||||
golang.org/x/sync v0.0.0-20190227155943-e225da77a7e6/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
|
golang.org/x/sync v0.0.0-20190227155943-e225da77a7e6/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
|
||||||
golang.org/x/sync v0.0.0-20190423024810-112230192c58/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
|
golang.org/x/sync v0.0.0-20190423024810-112230192c58/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
|
||||||
golang.org/x/sync v0.0.0-20210220032951-036812b2e83c h1:5KslGYwFpkhGh+Q16bwMP3cOontH8FOep7tGV86Y7SQ=
|
golang.org/x/sync v0.0.0-20190911185100-cd5d95a43a6e h1:vcxGaoTs7kV8m5Np9uUNQin4BrLOthgV7252N8V+FwY=
|
||||||
golang.org/x/sync v0.0.0-20210220032951-036812b2e83c/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
|
golang.org/x/sync v0.0.0-20190911185100-cd5d95a43a6e/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
|
||||||
golang.org/x/sys v0.0.0-20180830151530-49385e6e1522/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
|
golang.org/x/sys v0.0.0-20180830151530-49385e6e1522/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
|
||||||
golang.org/x/sys v0.0.0-20180909124046-d0be0721c37e/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
|
golang.org/x/sys v0.0.0-20180909124046-d0be0721c37e/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
|
||||||
golang.org/x/sys v0.0.0-20181029174526-d69651ed3497/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
|
golang.org/x/sys v0.0.0-20181029174526-d69651ed3497/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
|
||||||
|
@ -190,16 +186,12 @@ golang.org/x/sys v0.0.0-20191120155948-bd437916bb0e/go.mod h1:h1NjWce9XRLGQEsW7w
|
||||||
golang.org/x/sys v0.0.0-20200223170610-d5e6a3e2c0ae/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
golang.org/x/sys v0.0.0-20200223170610-d5e6a3e2c0ae/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
||||||
golang.org/x/sys v0.0.0-20200323222414-85ca7c5b95cd/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
golang.org/x/sys v0.0.0-20200323222414-85ca7c5b95cd/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
||||||
golang.org/x/sys v0.0.0-20200519105757-fe76b779f299/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
golang.org/x/sys v0.0.0-20200519105757-fe76b779f299/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
||||||
golang.org/x/sys v0.0.0-20201119102817-f84b799fce68/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
golang.org/x/sys v0.0.0-20201231184435-2d18734c6014 h1:joucsQqXmyBVxViHCPFjG3hx8JzIFSaym3l3MM/Jsdg=
|
||||||
golang.org/x/sys v0.0.0-20210330210617-4fbd30eecc44/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
golang.org/x/sys v0.0.0-20201231184435-2d18734c6014/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
||||||
golang.org/x/sys v0.0.0-20210510120138-977fb7262007 h1:gG67DSER+11cZvqIMb8S8bt0vZtiN6xWYARwirrOSfE=
|
|
||||||
golang.org/x/sys v0.0.0-20210510120138-977fb7262007/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
|
|
||||||
golang.org/x/term v0.0.0-20201126162022-7de9c90e9dd1/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo=
|
|
||||||
golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ=
|
golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ=
|
||||||
golang.org/x/text v0.3.1-0.20180807135948-17ff2d5776d2/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ=
|
golang.org/x/text v0.3.1-0.20180807135948-17ff2d5776d2/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ=
|
||||||
|
golang.org/x/text v0.3.2 h1:tW2bmiBqwgJj/UpqtC8EpXEZVYOwU0yG4iWbprSVAcs=
|
||||||
golang.org/x/text v0.3.2/go.mod h1:bEr9sfX3Q8Zfm5fL9x+3itogRgK3+ptLWKqgva+5dAk=
|
golang.org/x/text v0.3.2/go.mod h1:bEr9sfX3Q8Zfm5fL9x+3itogRgK3+ptLWKqgva+5dAk=
|
||||||
golang.org/x/text v0.3.3 h1:cokOdA+Jmi5PJGXLlLllQSgYigAEfHXJAERHVMaCc2k=
|
|
||||||
golang.org/x/text v0.3.3/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ=
|
|
||||||
golang.org/x/time v0.0.0-20180412165947-fbb02b2291d2/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ=
|
golang.org/x/time v0.0.0-20180412165947-fbb02b2291d2/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ=
|
||||||
golang.org/x/time v0.0.0-20181108054448-85acf8d2951c/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ=
|
golang.org/x/time v0.0.0-20181108054448-85acf8d2951c/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ=
|
||||||
golang.org/x/tools v0.0.0-20180828015842-6cd1fcedba52/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ=
|
golang.org/x/tools v0.0.0-20180828015842-6cd1fcedba52/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ=
|
||||||
|
@ -209,12 +201,10 @@ golang.org/x/tools v0.0.0-20190114222345-bf090417da8b/go.mod h1:n7NCudcB/nEzxVGm
|
||||||
golang.org/x/tools v0.0.0-20190226205152-f727befe758c/go.mod h1:9Yl7xja0Znq3iFh3HoIrodX9oNMXvdceNzlUR8zjMvY=
|
golang.org/x/tools v0.0.0-20190226205152-f727befe758c/go.mod h1:9Yl7xja0Znq3iFh3HoIrodX9oNMXvdceNzlUR8zjMvY=
|
||||||
golang.org/x/tools v0.0.0-20190425150028-36563e24a262/go.mod h1:RgjU9mgBXZiqYHBnxXauZ1Gv1EHHAz9KjViQ78xBX0Q=
|
golang.org/x/tools v0.0.0-20190425150028-36563e24a262/go.mod h1:RgjU9mgBXZiqYHBnxXauZ1Gv1EHHAz9KjViQ78xBX0Q=
|
||||||
golang.org/x/tools v0.0.0-20191119224855-298f0cb1881e/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo=
|
golang.org/x/tools v0.0.0-20191119224855-298f0cb1881e/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo=
|
||||||
golang.org/x/tools v0.1.1/go.mod h1:o0xws9oXOQQZyjljx8fwUC0k7L1pTE6eaCbjGeHmOkk=
|
|
||||||
golang.org/x/xerrors v0.0.0-20190717185122-a985d3407aa7/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0=
|
golang.org/x/xerrors v0.0.0-20190717185122-a985d3407aa7/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0=
|
||||||
golang.org/x/xerrors v0.0.0-20191011141410-1b5146add898/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0=
|
golang.org/x/xerrors v0.0.0-20191011141410-1b5146add898/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0=
|
||||||
|
golang.org/x/xerrors v0.0.0-20191204190536-9bdfabe68543 h1:E7g+9GITq07hpfrRu66IVDexMakfv52eLZ2CXBWiKr4=
|
||||||
golang.org/x/xerrors v0.0.0-20191204190536-9bdfabe68543/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0=
|
golang.org/x/xerrors v0.0.0-20191204190536-9bdfabe68543/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0=
|
||||||
golang.org/x/xerrors v0.0.0-20200804184101-5ec99f83aff1 h1:go1bK/D/BFZV2I8cIQd1NKEZ+0owSTG1fDTci4IqFcE=
|
|
||||||
golang.org/x/xerrors v0.0.0-20200804184101-5ec99f83aff1/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0=
|
|
||||||
google.golang.org/api v0.0.0-20180910000450-7ca32eb868bf/go.mod h1:4mhQ8q/RsB7i+udVvVy5NUi08OU8ZlA0gRVgrF7VFY0=
|
google.golang.org/api v0.0.0-20180910000450-7ca32eb868bf/go.mod h1:4mhQ8q/RsB7i+udVvVy5NUi08OU8ZlA0gRVgrF7VFY0=
|
||||||
google.golang.org/api v0.0.0-20181030000543-1d582fd0359e/go.mod h1:4mhQ8q/RsB7i+udVvVy5NUi08OU8ZlA0gRVgrF7VFY0=
|
google.golang.org/api v0.0.0-20181030000543-1d582fd0359e/go.mod h1:4mhQ8q/RsB7i+udVvVy5NUi08OU8ZlA0gRVgrF7VFY0=
|
||||||
google.golang.org/api v0.1.0/go.mod h1:UGEZY7KEX120AnNLIHFMKIo4obdJhkp2tPbaPlQx13Y=
|
google.golang.org/api v0.1.0/go.mod h1:UGEZY7KEX120AnNLIHFMKIo4obdJhkp2tPbaPlQx13Y=
|
||||||
|
|
|
@ -33,10 +33,6 @@ const (
|
||||||
VersionDraft29 = protocol.VersionDraft29
|
VersionDraft29 = protocol.VersionDraft29
|
||||||
// VersionDraft32 is IETF QUIC draft-32
|
// VersionDraft32 is IETF QUIC draft-32
|
||||||
VersionDraft32 = protocol.VersionDraft32
|
VersionDraft32 = protocol.VersionDraft32
|
||||||
// VersionDraft34 is IETF QUIC draft-34
|
|
||||||
VersionDraft34 = protocol.VersionDraft34
|
|
||||||
// Version1 is RFC 9000
|
|
||||||
Version1 = protocol.Version1
|
|
||||||
)
|
)
|
||||||
|
|
||||||
// A Token can be used to verify the ownership of the client address.
|
// A Token can be used to verify the ownership of the client address.
|
||||||
|
@ -66,6 +62,10 @@ type TokenStore interface {
|
||||||
Put(key string, token *ClientToken)
|
Put(key string, token *ClientToken)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// An ErrorCode is an application-defined error code.
|
||||||
|
// Valid values range between 0 and MAX_UINT62.
|
||||||
|
type ErrorCode = protocol.ApplicationErrorCode
|
||||||
|
|
||||||
// Err0RTTRejected is the returned from:
|
// Err0RTTRejected is the returned from:
|
||||||
// * Open{Uni}Stream{Sync}
|
// * Open{Uni}Stream{Sync}
|
||||||
// * Accept{Uni}Stream
|
// * Accept{Uni}Stream
|
||||||
|
@ -73,16 +73,7 @@ type TokenStore interface {
|
||||||
// when the server rejects a 0-RTT connection attempt.
|
// when the server rejects a 0-RTT connection attempt.
|
||||||
var Err0RTTRejected = errors.New("0-RTT rejected")
|
var Err0RTTRejected = errors.New("0-RTT rejected")
|
||||||
|
|
||||||
// SessionTracingKey can be used to associate a ConnectionTracer with a Session.
|
|
||||||
// It is set on the Session.Context() context,
|
|
||||||
// as well as on the context passed to logging.Tracer.NewConnectionTracer.
|
|
||||||
var SessionTracingKey = sessionTracingCtxKey{}
|
|
||||||
|
|
||||||
type sessionTracingCtxKey struct{}
|
|
||||||
|
|
||||||
// Stream is the interface implemented by QUIC streams
|
// Stream is the interface implemented by QUIC streams
|
||||||
// In addition to the errors listed on the Session,
|
|
||||||
// calls to stream functions can return a StreamError if the stream is canceled.
|
|
||||||
type Stream interface {
|
type Stream interface {
|
||||||
ReceiveStream
|
ReceiveStream
|
||||||
SendStream
|
SendStream
|
||||||
|
@ -108,7 +99,7 @@ type ReceiveStream interface {
|
||||||
// It will ask the peer to stop transmitting stream data.
|
// It will ask the peer to stop transmitting stream data.
|
||||||
// Read will unblock immediately, and future Read calls will fail.
|
// Read will unblock immediately, and future Read calls will fail.
|
||||||
// When called multiple times or after reading the io.EOF it is a no-op.
|
// When called multiple times or after reading the io.EOF it is a no-op.
|
||||||
CancelRead(StreamErrorCode)
|
CancelRead(ErrorCode)
|
||||||
// SetReadDeadline sets the deadline for future Read calls and
|
// SetReadDeadline sets the deadline for future Read calls and
|
||||||
// any currently-blocked Read call.
|
// any currently-blocked Read call.
|
||||||
// A zero value for t means Read will not time out.
|
// A zero value for t means Read will not time out.
|
||||||
|
@ -137,7 +128,7 @@ type SendStream interface {
|
||||||
// Data already written, but not yet delivered to the peer is not guaranteed to be delivered reliably.
|
// Data already written, but not yet delivered to the peer is not guaranteed to be delivered reliably.
|
||||||
// Write will unblock immediately, and future calls to Write will fail.
|
// Write will unblock immediately, and future calls to Write will fail.
|
||||||
// When called multiple times or after closing the stream it is a no-op.
|
// When called multiple times or after closing the stream it is a no-op.
|
||||||
CancelWrite(StreamErrorCode)
|
CancelWrite(ErrorCode)
|
||||||
// The context is canceled as soon as the write-side of the stream is closed.
|
// The context is canceled as soon as the write-side of the stream is closed.
|
||||||
// This happens when Close() or CancelWrite() is called, or when the peer
|
// This happens when Close() or CancelWrite() is called, or when the peer
|
||||||
// cancels the read-side of their stream.
|
// cancels the read-side of their stream.
|
||||||
|
@ -151,14 +142,14 @@ type SendStream interface {
|
||||||
SetWriteDeadline(t time.Time) error
|
SetWriteDeadline(t time.Time) error
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// StreamError is returned by Read and Write when the peer cancels the stream.
|
||||||
|
type StreamError interface {
|
||||||
|
error
|
||||||
|
Canceled() bool
|
||||||
|
ErrorCode() ErrorCode
|
||||||
|
}
|
||||||
|
|
||||||
// A Session is a QUIC connection between two peers.
|
// A Session is a QUIC connection between two peers.
|
||||||
// Calls to the session (and to streams) can return the following types of errors:
|
|
||||||
// * ApplicationError: for errors triggered by the application running on top of QUIC
|
|
||||||
// * TransportError: for errors triggered by the QUIC transport (in many cases a misbehaving peer)
|
|
||||||
// * IdleTimeoutError: when the peer goes away unexpectedly (this is a net.Error timeout error)
|
|
||||||
// * HandshakeTimeoutError: when the cryptographic handshake takes too long (this is a net.Error timeout error)
|
|
||||||
// * StatelessResetError: when we receive a stateless reset (this is a net.Error temporary error)
|
|
||||||
// * VersionNegotiationError: returned by the client, when there's no version overlap between the peers
|
|
||||||
type Session interface {
|
type Session interface {
|
||||||
// AcceptStream returns the next stream opened by the peer, blocking until one is available.
|
// AcceptStream returns the next stream opened by the peer, blocking until one is available.
|
||||||
// If the session was closed due to a timeout, the error satisfies
|
// If the session was closed due to a timeout, the error satisfies
|
||||||
|
@ -194,9 +185,9 @@ type Session interface {
|
||||||
LocalAddr() net.Addr
|
LocalAddr() net.Addr
|
||||||
// RemoteAddr returns the address of the peer.
|
// RemoteAddr returns the address of the peer.
|
||||||
RemoteAddr() net.Addr
|
RemoteAddr() net.Addr
|
||||||
// CloseWithError closes the connection with an error.
|
// Close the connection with an error.
|
||||||
// The error string will be sent to the peer.
|
// The error string will be sent to the peer.
|
||||||
CloseWithError(ApplicationErrorCode, string) error
|
CloseWithError(ErrorCode, string) error
|
||||||
// The context is cancelled when the session is closed.
|
// The context is cancelled when the session is closed.
|
||||||
// Warning: This API should not be considered stable and might change soon.
|
// Warning: This API should not be considered stable and might change soon.
|
||||||
Context() context.Context
|
Context() context.Context
|
||||||
|
|
|
@ -9,13 +9,12 @@ import (
|
||||||
// NewAckHandler creates a new SentPacketHandler and a new ReceivedPacketHandler
|
// NewAckHandler creates a new SentPacketHandler and a new ReceivedPacketHandler
|
||||||
func NewAckHandler(
|
func NewAckHandler(
|
||||||
initialPacketNumber protocol.PacketNumber,
|
initialPacketNumber protocol.PacketNumber,
|
||||||
initialMaxDatagramSize protocol.ByteCount,
|
|
||||||
rttStats *utils.RTTStats,
|
rttStats *utils.RTTStats,
|
||||||
pers protocol.Perspective,
|
pers protocol.Perspective,
|
||||||
tracer logging.ConnectionTracer,
|
tracer logging.ConnectionTracer,
|
||||||
logger utils.Logger,
|
logger utils.Logger,
|
||||||
version protocol.VersionNumber,
|
version protocol.VersionNumber,
|
||||||
) (SentPacketHandler, ReceivedPacketHandler) {
|
) (SentPacketHandler, ReceivedPacketHandler) {
|
||||||
sph := newSentPacketHandler(initialPacketNumber, initialMaxDatagramSize, rttStats, pers, tracer, logger)
|
sph := newSentPacketHandler(initialPacketNumber, rttStats, pers, tracer, logger)
|
||||||
return sph, newReceivedPacketHandler(sph, rttStats, logger, version)
|
return sph, newReceivedPacketHandler(sph, rttStats, logger, version)
|
||||||
}
|
}
|
||||||
|
|
|
@ -27,7 +27,7 @@ type Packet struct {
|
||||||
type SentPacketHandler interface {
|
type SentPacketHandler interface {
|
||||||
// SentPacket may modify the packet
|
// SentPacket may modify the packet
|
||||||
SentPacket(packet *Packet)
|
SentPacket(packet *Packet)
|
||||||
ReceivedAck(ackFrame *wire.AckFrame, encLevel protocol.EncryptionLevel, recvTime time.Time) (bool /* 1-RTT packet acked */, error)
|
ReceivedAck(ackFrame *wire.AckFrame, encLevel protocol.EncryptionLevel, recvTime time.Time) error
|
||||||
ReceivedBytes(protocol.ByteCount)
|
ReceivedBytes(protocol.ByteCount)
|
||||||
DropPackets(protocol.EncryptionLevel)
|
DropPackets(protocol.EncryptionLevel)
|
||||||
ResetForRetry() error
|
ResetForRetry() error
|
||||||
|
|
157
vendor/github.com/lucas-clemente/quic-go/internal/ackhandler/sent_packet_handler.go
generated
vendored
157
vendor/github.com/lucas-clemente/quic-go/internal/ackhandler/sent_packet_handler.go
generated
vendored
|
@ -21,8 +21,6 @@ const (
|
||||||
packetThreshold = 3
|
packetThreshold = 3
|
||||||
// Before validating the client's address, the server won't send more than 3x bytes than it received.
|
// Before validating the client's address, the server won't send more than 3x bytes than it received.
|
||||||
amplificationFactor = 3
|
amplificationFactor = 3
|
||||||
// We use Retry packets to derive an RTT estimate. Make sure we don't set the RTT to a super low value yet.
|
|
||||||
minRTTAfterRetry = 5 * time.Millisecond
|
|
||||||
)
|
)
|
||||||
|
|
||||||
type packetNumberSpace struct {
|
type packetNumberSpace struct {
|
||||||
|
@ -103,7 +101,6 @@ var (
|
||||||
|
|
||||||
func newSentPacketHandler(
|
func newSentPacketHandler(
|
||||||
initialPN protocol.PacketNumber,
|
initialPN protocol.PacketNumber,
|
||||||
initialMaxDatagramSize protocol.ByteCount,
|
|
||||||
rttStats *utils.RTTStats,
|
rttStats *utils.RTTStats,
|
||||||
pers protocol.Perspective,
|
pers protocol.Perspective,
|
||||||
tracer logging.ConnectionTracer,
|
tracer logging.ConnectionTracer,
|
||||||
|
@ -112,7 +109,6 @@ func newSentPacketHandler(
|
||||||
congestion := congestion.NewCubicSender(
|
congestion := congestion.NewCubicSender(
|
||||||
congestion.DefaultClock{},
|
congestion.DefaultClock{},
|
||||||
rttStats,
|
rttStats,
|
||||||
initialMaxDatagramSize,
|
|
||||||
true, // use Reno
|
true, // use Reno
|
||||||
tracer,
|
tracer,
|
||||||
)
|
)
|
||||||
|
@ -198,17 +194,12 @@ func (h *sentPacketHandler) dropPackets(encLevel protocol.EncryptionLevel) {
|
||||||
}
|
}
|
||||||
|
|
||||||
func (h *sentPacketHandler) ReceivedBytes(n protocol.ByteCount) {
|
func (h *sentPacketHandler) ReceivedBytes(n protocol.ByteCount) {
|
||||||
wasAmplificationLimit := h.isAmplificationLimited()
|
|
||||||
h.bytesReceived += n
|
h.bytesReceived += n
|
||||||
if wasAmplificationLimit && !h.isAmplificationLimited() {
|
|
||||||
h.setLossDetectionTimer()
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func (h *sentPacketHandler) ReceivedPacket(l protocol.EncryptionLevel) {
|
func (h *sentPacketHandler) ReceivedPacket(encLevel protocol.EncryptionLevel) {
|
||||||
if h.perspective == protocol.PerspectiveServer && l == protocol.EncryptionHandshake && !h.peerAddressValidated {
|
if h.perspective == protocol.PerspectiveServer && encLevel == protocol.EncryptionHandshake {
|
||||||
h.peerAddressValidated = true
|
h.peerAddressValidated = true
|
||||||
h.setLossDetectionTimer()
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -277,15 +268,12 @@ func (h *sentPacketHandler) sentPacketImpl(packet *Packet) bool /* is ack-elicit
|
||||||
return isAckEliciting
|
return isAckEliciting
|
||||||
}
|
}
|
||||||
|
|
||||||
func (h *sentPacketHandler) ReceivedAck(ack *wire.AckFrame, encLevel protocol.EncryptionLevel, rcvTime time.Time) (bool /* contained 1-RTT packet */, error) {
|
func (h *sentPacketHandler) ReceivedAck(ack *wire.AckFrame, encLevel protocol.EncryptionLevel, rcvTime time.Time) error {
|
||||||
pnSpace := h.getPacketNumberSpace(encLevel)
|
pnSpace := h.getPacketNumberSpace(encLevel)
|
||||||
|
|
||||||
largestAcked := ack.LargestAcked()
|
largestAcked := ack.LargestAcked()
|
||||||
if largestAcked > pnSpace.largestSent {
|
if largestAcked > pnSpace.largestSent {
|
||||||
return false, &qerr.TransportError{
|
return qerr.NewError(qerr.ProtocolViolation, "Received ACK for an unsent packet")
|
||||||
ErrorCode: qerr.ProtocolViolation,
|
|
||||||
ErrorMessage: "received ACK for an unsent packet",
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
pnSpace.largestAcked = utils.MaxPacketNumber(pnSpace.largestAcked, largestAcked)
|
pnSpace.largestAcked = utils.MaxPacketNumber(pnSpace.largestAcked, largestAcked)
|
||||||
|
@ -302,7 +290,7 @@ func (h *sentPacketHandler) ReceivedAck(ack *wire.AckFrame, encLevel protocol.En
|
||||||
priorInFlight := h.bytesInFlight
|
priorInFlight := h.bytesInFlight
|
||||||
ackedPackets, err := h.detectAndRemoveAckedPackets(ack, encLevel)
|
ackedPackets, err := h.detectAndRemoveAckedPackets(ack, encLevel)
|
||||||
if err != nil || len(ackedPackets) == 0 {
|
if err != nil || len(ackedPackets) == 0 {
|
||||||
return false, err
|
return err
|
||||||
}
|
}
|
||||||
// update the RTT, if the largest acked is newly acknowledged
|
// update the RTT, if the largest acked is newly acknowledged
|
||||||
if len(ackedPackets) > 0 {
|
if len(ackedPackets) > 0 {
|
||||||
|
@ -320,16 +308,15 @@ func (h *sentPacketHandler) ReceivedAck(ack *wire.AckFrame, encLevel protocol.En
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if err := h.detectLostPackets(rcvTime, encLevel); err != nil {
|
if err := h.detectLostPackets(rcvTime, encLevel); err != nil {
|
||||||
return false, err
|
return err
|
||||||
}
|
}
|
||||||
var acked1RTTPacket bool
|
|
||||||
for _, p := range ackedPackets {
|
for _, p := range ackedPackets {
|
||||||
|
if p.skippedPacket {
|
||||||
|
return fmt.Errorf("received an ACK for skipped packet number: %d (%s)", p.PacketNumber, encLevel)
|
||||||
|
}
|
||||||
if p.includedInBytesInFlight && !p.declaredLost {
|
if p.includedInBytesInFlight && !p.declaredLost {
|
||||||
h.congestion.OnPacketAcked(p.PacketNumber, p.Length, priorInFlight, rcvTime)
|
h.congestion.OnPacketAcked(p.PacketNumber, p.Length, priorInFlight, rcvTime)
|
||||||
}
|
}
|
||||||
if p.EncryptionLevel == protocol.Encryption1RTT {
|
|
||||||
acked1RTTPacket = true
|
|
||||||
}
|
|
||||||
h.removeFromBytesInFlight(p)
|
h.removeFromBytesInFlight(p)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -348,7 +335,7 @@ func (h *sentPacketHandler) ReceivedAck(ack *wire.AckFrame, encLevel protocol.En
|
||||||
|
|
||||||
pnSpace.history.DeleteOldPackets(rcvTime)
|
pnSpace.history.DeleteOldPackets(rcvTime)
|
||||||
h.setLossDetectionTimer()
|
h.setLossDetectionTimer()
|
||||||
return acked1RTTPacket, nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (h *sentPacketHandler) GetLowestPacketNotConfirmedAcked() protocol.PacketNumber {
|
func (h *sentPacketHandler) GetLowestPacketNotConfirmedAcked() protocol.PacketNumber {
|
||||||
|
@ -380,20 +367,15 @@ func (h *sentPacketHandler) detectAndRemoveAckedPackets(ack *wire.AckFrame, encL
|
||||||
ackRange = ack.AckRanges[len(ack.AckRanges)-1-ackRangeIndex]
|
ackRange = ack.AckRanges[len(ack.AckRanges)-1-ackRangeIndex]
|
||||||
}
|
}
|
||||||
|
|
||||||
if p.PacketNumber < ackRange.Smallest { // packet not contained in ACK range
|
if p.PacketNumber >= ackRange.Smallest { // packet i contained in ACK range
|
||||||
return true, nil
|
|
||||||
}
|
|
||||||
if p.PacketNumber > ackRange.Largest {
|
if p.PacketNumber > ackRange.Largest {
|
||||||
return false, fmt.Errorf("BUG: ackhandler would have acked wrong packet %d, while evaluating range %d -> %d", p.PacketNumber, ackRange.Smallest, ackRange.Largest)
|
return false, fmt.Errorf("BUG: ackhandler would have acked wrong packet %d, while evaluating range %d -> %d", p.PacketNumber, ackRange.Smallest, ackRange.Largest)
|
||||||
}
|
}
|
||||||
}
|
|
||||||
if p.skippedPacket {
|
|
||||||
return false, &qerr.TransportError{
|
|
||||||
ErrorCode: qerr.ProtocolViolation,
|
|
||||||
ErrorMessage: fmt.Sprintf("received an ACK for skipped packet number: %d (%s)", p.PacketNumber, encLevel),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
h.ackedPackets = append(h.ackedPackets, p)
|
h.ackedPackets = append(h.ackedPackets, p)
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
h.ackedPackets = append(h.ackedPackets, p)
|
||||||
|
}
|
||||||
return true, nil
|
return true, nil
|
||||||
})
|
})
|
||||||
if h.logger.Debug() && len(h.ackedPackets) > 0 {
|
if h.logger.Debug() && len(h.ackedPackets) > 0 {
|
||||||
|
@ -417,9 +399,6 @@ func (h *sentPacketHandler) detectAndRemoveAckedPackets(ack *wire.AckFrame, encL
|
||||||
if err := pnSpace.history.Remove(p.PacketNumber); err != nil {
|
if err := pnSpace.history.Remove(p.PacketNumber); err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
if h.tracer != nil {
|
|
||||||
h.tracer.AcknowledgedPacket(encLevel, p.PacketNumber)
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return h.ackedPackets, err
|
return h.ackedPackets, err
|
||||||
|
@ -445,20 +424,20 @@ func (h *sentPacketHandler) getLossTimeAndSpace() (time.Time, protocol.Encryptio
|
||||||
}
|
}
|
||||||
|
|
||||||
// same logic as getLossTimeAndSpace, but for lastAckElicitingPacketTime instead of lossTime
|
// same logic as getLossTimeAndSpace, but for lastAckElicitingPacketTime instead of lossTime
|
||||||
func (h *sentPacketHandler) getPTOTimeAndSpace() (pto time.Time, encLevel protocol.EncryptionLevel, ok bool) {
|
func (h *sentPacketHandler) getPTOTimeAndSpace() (time.Time, protocol.EncryptionLevel) {
|
||||||
// We only send application data probe packets once the handshake is confirmed,
|
if !h.hasOutstandingPackets() {
|
||||||
// because before that, we don't have the keys to decrypt ACKs sent in 1-RTT packets.
|
|
||||||
if !h.handshakeConfirmed && !h.hasOutstandingCryptoPackets() {
|
|
||||||
if h.peerCompletedAddressValidation {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
t := time.Now().Add(h.rttStats.PTO(false) << h.ptoCount)
|
t := time.Now().Add(h.rttStats.PTO(false) << h.ptoCount)
|
||||||
if h.initialPackets != nil {
|
if h.initialPackets != nil {
|
||||||
return t, protocol.EncryptionInitial, true
|
return t, protocol.EncryptionInitial
|
||||||
}
|
}
|
||||||
return t, protocol.EncryptionHandshake, true
|
return t, protocol.EncryptionHandshake
|
||||||
}
|
}
|
||||||
|
|
||||||
|
var (
|
||||||
|
encLevel protocol.EncryptionLevel
|
||||||
|
pto time.Time
|
||||||
|
)
|
||||||
|
|
||||||
if h.initialPackets != nil {
|
if h.initialPackets != nil {
|
||||||
encLevel = protocol.EncryptionInitial
|
encLevel = protocol.EncryptionInitial
|
||||||
if t := h.initialPackets.lastAckElicitingPacketTime; !t.IsZero() {
|
if t := h.initialPackets.lastAckElicitingPacketTime; !t.IsZero() {
|
||||||
|
@ -479,7 +458,7 @@ func (h *sentPacketHandler) getPTOTimeAndSpace() (pto time.Time, encLevel protoc
|
||||||
encLevel = protocol.Encryption1RTT
|
encLevel = protocol.Encryption1RTT
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return pto, encLevel, true
|
return pto, encLevel
|
||||||
}
|
}
|
||||||
|
|
||||||
func (h *sentPacketHandler) hasOutstandingCryptoPackets() bool {
|
func (h *sentPacketHandler) hasOutstandingCryptoPackets() bool {
|
||||||
|
@ -494,13 +473,15 @@ func (h *sentPacketHandler) hasOutstandingCryptoPackets() bool {
|
||||||
}
|
}
|
||||||
|
|
||||||
func (h *sentPacketHandler) hasOutstandingPackets() bool {
|
func (h *sentPacketHandler) hasOutstandingPackets() bool {
|
||||||
return h.appDataPackets.history.HasOutstandingPackets() || h.hasOutstandingCryptoPackets()
|
// We only send application data probe packets once the handshake completes,
|
||||||
|
// because before that, we don't have the keys to decrypt ACKs sent in 1-RTT packets.
|
||||||
|
return (h.handshakeConfirmed && h.appDataPackets.history.HasOutstandingPackets()) ||
|
||||||
|
h.hasOutstandingCryptoPackets()
|
||||||
}
|
}
|
||||||
|
|
||||||
func (h *sentPacketHandler) setLossDetectionTimer() {
|
func (h *sentPacketHandler) setLossDetectionTimer() {
|
||||||
oldAlarm := h.alarm // only needed in case tracing is enabled
|
oldAlarm := h.alarm // only needed in case tracing is enabled
|
||||||
lossTime, encLevel := h.getLossTimeAndSpace()
|
if lossTime, encLevel := h.getLossTimeAndSpace(); !lossTime.IsZero() {
|
||||||
if !lossTime.IsZero() {
|
|
||||||
// Early retransmit timer or time loss detection.
|
// Early retransmit timer or time loss detection.
|
||||||
h.alarm = lossTime
|
h.alarm = lossTime
|
||||||
if h.tracer != nil && h.alarm != oldAlarm {
|
if h.tracer != nil && h.alarm != oldAlarm {
|
||||||
|
@ -509,35 +490,18 @@ func (h *sentPacketHandler) setLossDetectionTimer() {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
// Cancel the alarm if amplification limited.
|
|
||||||
if h.isAmplificationLimited() {
|
|
||||||
h.alarm = time.Time{}
|
|
||||||
if !oldAlarm.IsZero() {
|
|
||||||
h.logger.Debugf("Canceling loss detection timer. Amplification limited.")
|
|
||||||
if h.tracer != nil {
|
|
||||||
h.tracer.LossTimerCanceled()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
// Cancel the alarm if no packets are outstanding
|
// Cancel the alarm if no packets are outstanding
|
||||||
if !h.hasOutstandingPackets() && h.peerCompletedAddressValidation {
|
if !h.hasOutstandingPackets() && h.peerCompletedAddressValidation {
|
||||||
h.alarm = time.Time{}
|
h.alarm = time.Time{}
|
||||||
if !oldAlarm.IsZero() {
|
|
||||||
h.logger.Debugf("Canceling loss detection timer. No packets in flight.")
|
h.logger.Debugf("Canceling loss detection timer. No packets in flight.")
|
||||||
if h.tracer != nil {
|
if h.tracer != nil && !oldAlarm.IsZero() {
|
||||||
h.tracer.LossTimerCanceled()
|
h.tracer.LossTimerCanceled()
|
||||||
}
|
}
|
||||||
}
|
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
// PTO alarm
|
// PTO alarm
|
||||||
ptoTime, encLevel, ok := h.getPTOTimeAndSpace()
|
ptoTime, encLevel := h.getPTOTimeAndSpace()
|
||||||
if !ok {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
h.alarm = ptoTime
|
h.alarm = ptoTime
|
||||||
if h.tracer != nil && h.alarm != oldAlarm {
|
if h.tracer != nil && h.alarm != oldAlarm {
|
||||||
h.tracer.SetLossTimer(logging.TimerTypePTO, encLevel, h.alarm)
|
h.tracer.SetLossTimer(logging.TimerTypePTO, encLevel, h.alarm)
|
||||||
|
@ -605,7 +569,20 @@ func (h *sentPacketHandler) detectLostPackets(now time.Time, encLevel protocol.E
|
||||||
}
|
}
|
||||||
|
|
||||||
func (h *sentPacketHandler) OnLossDetectionTimeout() error {
|
func (h *sentPacketHandler) OnLossDetectionTimeout() error {
|
||||||
defer h.setLossDetectionTimer()
|
// When all outstanding are acknowledged, the alarm is canceled in
|
||||||
|
// setLossDetectionTimer. This doesn't reset the timer in the session though.
|
||||||
|
// When OnAlarm is called, we therefore need to make sure that there are
|
||||||
|
// actually packets outstanding.
|
||||||
|
if h.hasOutstandingPackets() || !h.peerCompletedAddressValidation {
|
||||||
|
if err := h.onVerifiedLossDetectionTimeout(); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
h.setLossDetectionTimer()
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (h *sentPacketHandler) onVerifiedLossDetectionTimeout() error {
|
||||||
earliestLossTime, encLevel := h.getLossTimeAndSpace()
|
earliestLossTime, encLevel := h.getLossTimeAndSpace()
|
||||||
if !earliestLossTime.IsZero() {
|
if !earliestLossTime.IsZero() {
|
||||||
if h.logger.Debug() {
|
if h.logger.Debug() {
|
||||||
|
@ -619,31 +596,9 @@ func (h *sentPacketHandler) OnLossDetectionTimeout() error {
|
||||||
}
|
}
|
||||||
|
|
||||||
// PTO
|
// PTO
|
||||||
// When all outstanding are acknowledged, the alarm is canceled in
|
|
||||||
// setLossDetectionTimer. This doesn't reset the timer in the session though.
|
|
||||||
// When OnAlarm is called, we therefore need to make sure that there are
|
|
||||||
// actually packets outstanding.
|
|
||||||
if h.bytesInFlight == 0 && !h.peerCompletedAddressValidation {
|
|
||||||
h.ptoCount++
|
|
||||||
h.numProbesToSend++
|
|
||||||
if h.initialPackets != nil {
|
|
||||||
h.ptoMode = SendPTOInitial
|
|
||||||
} else if h.handshakePackets != nil {
|
|
||||||
h.ptoMode = SendPTOHandshake
|
|
||||||
} else {
|
|
||||||
return errors.New("sentPacketHandler BUG: PTO fired, but bytes_in_flight is 0 and Initial and Handshake already dropped")
|
|
||||||
}
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
_, encLevel, ok := h.getPTOTimeAndSpace()
|
|
||||||
if !ok {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
if ps := h.getPacketNumberSpace(encLevel); !ps.history.HasOutstandingPackets() && !h.peerCompletedAddressValidation {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
h.ptoCount++
|
h.ptoCount++
|
||||||
|
if h.bytesInFlight > 0 {
|
||||||
|
_, encLevel = h.getPTOTimeAndSpace()
|
||||||
if h.logger.Debug() {
|
if h.logger.Debug() {
|
||||||
h.logger.Debugf("Loss detection alarm for %s fired in PTO mode. PTO count: %d", encLevel, h.ptoCount)
|
h.logger.Debugf("Loss detection alarm for %s fired in PTO mode. PTO count: %d", encLevel, h.ptoCount)
|
||||||
}
|
}
|
||||||
|
@ -665,6 +620,19 @@ func (h *sentPacketHandler) OnLossDetectionTimeout() error {
|
||||||
default:
|
default:
|
||||||
return fmt.Errorf("PTO timer in unexpected encryption level: %s", encLevel)
|
return fmt.Errorf("PTO timer in unexpected encryption level: %s", encLevel)
|
||||||
}
|
}
|
||||||
|
} else {
|
||||||
|
if h.perspective == protocol.PerspectiveServer {
|
||||||
|
return errors.New("sentPacketHandler BUG: PTO fired, but bytes_in_flight is 0")
|
||||||
|
}
|
||||||
|
h.numProbesToSend++
|
||||||
|
if h.initialPackets != nil {
|
||||||
|
h.ptoMode = SendPTOInitial
|
||||||
|
} else if h.handshakePackets != nil {
|
||||||
|
h.ptoMode = SendPTOHandshake
|
||||||
|
} else {
|
||||||
|
return errors.New("sentPacketHandler BUG: PTO fired, but bytes_in_flight is 0 and Initial and Handshake already dropped")
|
||||||
|
}
|
||||||
|
}
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -800,9 +768,8 @@ func (h *sentPacketHandler) ResetForRetry() error {
|
||||||
// Only use the Retry to estimate the RTT if we didn't send any retransmission for the Initial.
|
// Only use the Retry to estimate the RTT if we didn't send any retransmission for the Initial.
|
||||||
// Otherwise, we don't know which Initial the Retry was sent in response to.
|
// Otherwise, we don't know which Initial the Retry was sent in response to.
|
||||||
if h.ptoCount == 0 {
|
if h.ptoCount == 0 {
|
||||||
// Don't set the RTT to a value lower than 5ms here.
|
|
||||||
now := time.Now()
|
now := time.Now()
|
||||||
h.rttStats.UpdateRTT(utils.MaxDuration(minRTTAfterRetry, now.Sub(firstPacketSendTime)), 0, now)
|
h.rttStats.UpdateRTT(now.Sub(firstPacketSendTime), 0, now)
|
||||||
if h.logger.Debug() {
|
if h.logger.Debug() {
|
||||||
h.logger.Debugf("\tupdated RTT: %s (σ: %s)", h.rttStats.SmoothedRTT(), h.rttStats.MeanDeviation())
|
h.logger.Debugf("\tupdated RTT: %s (σ: %s)", h.rttStats.SmoothedRTT(), h.rttStats.MeanDeviation())
|
||||||
}
|
}
|
||||||
|
|
5
vendor/github.com/lucas-clemente/quic-go/internal/ackhandler/sent_packet_history.go
generated
vendored
5
vendor/github.com/lucas-clemente/quic-go/internal/ackhandler/sent_packet_history.go
generated
vendored
|
@ -64,9 +64,8 @@ func (h *sentPacketHistory) Iterate(cb func(*Packet) (cont bool, err error)) err
|
||||||
// FirstOutStanding returns the first outstanding packet.
|
// FirstOutStanding returns the first outstanding packet.
|
||||||
func (h *sentPacketHistory) FirstOutstanding() *Packet {
|
func (h *sentPacketHistory) FirstOutstanding() *Packet {
|
||||||
for el := h.packetList.Front(); el != nil; el = el.Next() {
|
for el := h.packetList.Front(); el != nil; el = el.Next() {
|
||||||
p := &el.Value
|
if !el.Value.declaredLost && !el.Value.skippedPacket {
|
||||||
if !p.declaredLost && !p.skippedPacket && !p.IsPathMTUProbePacket {
|
return &el.Value
|
||||||
return p
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return nil
|
return nil
|
||||||
|
|
|
@ -1,7 +1,6 @@
|
||||||
package congestion
|
package congestion
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"fmt"
|
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
"github.com/lucas-clemente/quic-go/internal/protocol"
|
"github.com/lucas-clemente/quic-go/internal/protocol"
|
||||||
|
@ -15,8 +14,9 @@ const (
|
||||||
initialMaxDatagramSize = protocol.ByteCount(protocol.InitialPacketSizeIPv4)
|
initialMaxDatagramSize = protocol.ByteCount(protocol.InitialPacketSizeIPv4)
|
||||||
maxBurstPackets = 3
|
maxBurstPackets = 3
|
||||||
renoBeta = 0.7 // Reno backoff factor.
|
renoBeta = 0.7 // Reno backoff factor.
|
||||||
|
initialMaxCongestionWindow = protocol.MaxCongestionWindowPackets * initialMaxDatagramSize
|
||||||
minCongestionWindowPackets = 2
|
minCongestionWindowPackets = 2
|
||||||
initialCongestionWindow = 32
|
initialCongestionWindow = 32 * initialMaxDatagramSize
|
||||||
)
|
)
|
||||||
|
|
||||||
type cubicSender struct {
|
type cubicSender struct {
|
||||||
|
@ -65,33 +65,11 @@ var (
|
||||||
)
|
)
|
||||||
|
|
||||||
// NewCubicSender makes a new cubic sender
|
// NewCubicSender makes a new cubic sender
|
||||||
func NewCubicSender(
|
func NewCubicSender(clock Clock, rttStats *utils.RTTStats, reno bool, tracer logging.ConnectionTracer) *cubicSender {
|
||||||
clock Clock,
|
return newCubicSender(clock, rttStats, reno, initialCongestionWindow, initialMaxCongestionWindow, tracer)
|
||||||
rttStats *utils.RTTStats,
|
|
||||||
initialMaxDatagramSize protocol.ByteCount,
|
|
||||||
reno bool,
|
|
||||||
tracer logging.ConnectionTracer,
|
|
||||||
) *cubicSender {
|
|
||||||
return newCubicSender(
|
|
||||||
clock,
|
|
||||||
rttStats,
|
|
||||||
reno,
|
|
||||||
initialMaxDatagramSize,
|
|
||||||
initialCongestionWindow*initialMaxDatagramSize,
|
|
||||||
protocol.MaxCongestionWindowPackets*initialMaxDatagramSize,
|
|
||||||
tracer,
|
|
||||||
)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func newCubicSender(
|
func newCubicSender(clock Clock, rttStats *utils.RTTStats, reno bool, initialCongestionWindow, initialMaxCongestionWindow protocol.ByteCount, tracer logging.ConnectionTracer) *cubicSender {
|
||||||
clock Clock,
|
|
||||||
rttStats *utils.RTTStats,
|
|
||||||
reno bool,
|
|
||||||
initialMaxDatagramSize,
|
|
||||||
initialCongestionWindow,
|
|
||||||
initialMaxCongestionWindow protocol.ByteCount,
|
|
||||||
tracer logging.ConnectionTracer,
|
|
||||||
) *cubicSender {
|
|
||||||
c := &cubicSender{
|
c := &cubicSender{
|
||||||
rttStats: rttStats,
|
rttStats: rttStats,
|
||||||
largestSentPacketNumber: protocol.InvalidPacketNumber,
|
largestSentPacketNumber: protocol.InvalidPacketNumber,
|
||||||
|
@ -305,7 +283,7 @@ func (c *cubicSender) maybeTraceStateChange(new logging.CongestionState) {
|
||||||
|
|
||||||
func (c *cubicSender) SetMaxDatagramSize(s protocol.ByteCount) {
|
func (c *cubicSender) SetMaxDatagramSize(s protocol.ByteCount) {
|
||||||
if s < c.maxDatagramSize {
|
if s < c.maxDatagramSize {
|
||||||
panic(fmt.Sprintf("congestion BUG: decreased max datagram size from %d to %d", c.maxDatagramSize, s))
|
panic("congestion BUG: decreased max datagram size")
|
||||||
}
|
}
|
||||||
cwndIsMinCwnd := c.congestionWindow == c.minCongestionWindow()
|
cwndIsMinCwnd := c.congestionWindow == c.minCongestionWindow()
|
||||||
c.maxDatagramSize = s
|
c.maxDatagramSize = s
|
||||||
|
|
|
@ -50,10 +50,7 @@ func (c *connectionFlowController) IncrementHighestReceived(increment protocol.B
|
||||||
|
|
||||||
c.highestReceived += increment
|
c.highestReceived += increment
|
||||||
if c.checkFlowControlViolation() {
|
if c.checkFlowControlViolation() {
|
||||||
return &qerr.TransportError{
|
return qerr.NewError(qerr.FlowControlError, fmt.Sprintf("Received %d bytes for the connection, allowed %d bytes", c.highestReceived, c.receiveWindow))
|
||||||
ErrorCode: qerr.FlowControlError,
|
|
||||||
ErrorMessage: fmt.Sprintf("received %d bytes for the connection, allowed %d bytes", c.highestReceived, c.receiveWindow),
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
20
vendor/github.com/lucas-clemente/quic-go/internal/flowcontrol/stream_flow_controller.go
generated
vendored
20
vendor/github.com/lucas-clemente/quic-go/internal/flowcontrol/stream_flow_controller.go
generated
vendored
|
@ -54,17 +54,11 @@ func (c *streamFlowController) UpdateHighestReceived(offset protocol.ByteCount,
|
||||||
if c.receivedFinalOffset {
|
if c.receivedFinalOffset {
|
||||||
// If we receive another final offset, check that it's the same.
|
// If we receive another final offset, check that it's the same.
|
||||||
if final && offset != c.highestReceived {
|
if final && offset != c.highestReceived {
|
||||||
return &qerr.TransportError{
|
return qerr.NewError(qerr.FinalSizeError, fmt.Sprintf("Received inconsistent final offset for stream %d (old: %d, new: %d bytes)", c.streamID, c.highestReceived, offset))
|
||||||
ErrorCode: qerr.FinalSizeError,
|
|
||||||
ErrorMessage: fmt.Sprintf("received inconsistent final offset for stream %d (old: %d, new: %d bytes)", c.streamID, c.highestReceived, offset),
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
// Check that the offset is below the final offset.
|
// Check that the offset is below the final offset.
|
||||||
if offset > c.highestReceived {
|
if offset > c.highestReceived {
|
||||||
return &qerr.TransportError{
|
return qerr.NewError(qerr.FinalSizeError, fmt.Sprintf("Received offset %d for stream %d. Final offset was already received at %d", offset, c.streamID, c.highestReceived))
|
||||||
ErrorCode: qerr.FinalSizeError,
|
|
||||||
ErrorMessage: fmt.Sprintf("received offset %d for stream %d, but final offset was already received at %d", offset, c.streamID, c.highestReceived),
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -78,10 +72,7 @@ func (c *streamFlowController) UpdateHighestReceived(offset protocol.ByteCount,
|
||||||
// This can happen due to reordering.
|
// This can happen due to reordering.
|
||||||
if offset <= c.highestReceived {
|
if offset <= c.highestReceived {
|
||||||
if final {
|
if final {
|
||||||
return &qerr.TransportError{
|
return qerr.NewError(qerr.FinalSizeError, fmt.Sprintf("Received final offset %d for stream %d, but already received offset %d before", offset, c.streamID, c.highestReceived))
|
||||||
ErrorCode: qerr.FinalSizeError,
|
|
||||||
ErrorMessage: fmt.Sprintf("received final offset %d for stream %d, but already received offset %d before", offset, c.streamID, c.highestReceived),
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
@ -89,10 +80,7 @@ func (c *streamFlowController) UpdateHighestReceived(offset protocol.ByteCount,
|
||||||
increment := offset - c.highestReceived
|
increment := offset - c.highestReceived
|
||||||
c.highestReceived = offset
|
c.highestReceived = offset
|
||||||
if c.checkFlowControlViolation() {
|
if c.checkFlowControlViolation() {
|
||||||
return &qerr.TransportError{
|
return qerr.NewError(qerr.FlowControlError, fmt.Sprintf("Received %d bytes on stream %d, allowed %d bytes", offset, c.streamID, c.receiveWindow))
|
||||||
ErrorCode: qerr.FlowControlError,
|
|
||||||
ErrorMessage: fmt.Sprintf("received %d bytes on stream %d, allowed %d bytes", offset, c.streamID, c.receiveWindow),
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
return c.connection.IncrementHighestReceived(increment)
|
return c.connection.IncrementHighestReceived(increment)
|
||||||
}
|
}
|
||||||
|
|
|
@ -403,10 +403,7 @@ func (h *cryptoSetup) checkEncryptionLevel(msgType messageType, encLevel protoco
|
||||||
func (h *cryptoSetup) handleTransportParameters(data []byte) {
|
func (h *cryptoSetup) handleTransportParameters(data []byte) {
|
||||||
var tp wire.TransportParameters
|
var tp wire.TransportParameters
|
||||||
if err := tp.Unmarshal(data, h.perspective.Opposite()); err != nil {
|
if err := tp.Unmarshal(data, h.perspective.Opposite()); err != nil {
|
||||||
h.runner.OnError(&qerr.TransportError{
|
h.runner.OnError(qerr.NewError(qerr.TransportParameterError, err.Error()))
|
||||||
ErrorCode: qerr.TransportParameterError,
|
|
||||||
ErrorMessage: err.Error(),
|
|
||||||
})
|
|
||||||
}
|
}
|
||||||
h.peerParams = &tp
|
h.peerParams = &tp
|
||||||
h.runner.OnReceivedParams(h.peerParams)
|
h.runner.OnReceivedParams(h.peerParams)
|
||||||
|
@ -558,7 +555,7 @@ func (h *cryptoSetup) SetReadKey(encLevel qtls.EncryptionLevel, suite *qtls.Ciph
|
||||||
newHeaderProtector(suite, trafficSecret, true),
|
newHeaderProtector(suite, trafficSecret, true),
|
||||||
)
|
)
|
||||||
h.mutex.Unlock()
|
h.mutex.Unlock()
|
||||||
h.logger.Debugf("Installed 0-RTT Read keys (using %s)", tls.CipherSuiteName(suite.ID))
|
h.logger.Debugf("Installed 0-RTT Read keys (using %s)", qtls.CipherSuiteName(suite.ID))
|
||||||
if h.tracer != nil {
|
if h.tracer != nil {
|
||||||
h.tracer.UpdatedKeyFromTLS(protocol.Encryption0RTT, h.perspective.Opposite())
|
h.tracer.UpdatedKeyFromTLS(protocol.Encryption0RTT, h.perspective.Opposite())
|
||||||
}
|
}
|
||||||
|
@ -571,12 +568,12 @@ func (h *cryptoSetup) SetReadKey(encLevel qtls.EncryptionLevel, suite *qtls.Ciph
|
||||||
h.dropInitialKeys,
|
h.dropInitialKeys,
|
||||||
h.perspective,
|
h.perspective,
|
||||||
)
|
)
|
||||||
h.logger.Debugf("Installed Handshake Read keys (using %s)", tls.CipherSuiteName(suite.ID))
|
h.logger.Debugf("Installed Handshake Read keys (using %s)", qtls.CipherSuiteName(suite.ID))
|
||||||
case qtls.EncryptionApplication:
|
case qtls.EncryptionApplication:
|
||||||
h.readEncLevel = protocol.Encryption1RTT
|
h.readEncLevel = protocol.Encryption1RTT
|
||||||
h.aead.SetReadKey(suite, trafficSecret)
|
h.aead.SetReadKey(suite, trafficSecret)
|
||||||
h.has1RTTOpener = true
|
h.has1RTTOpener = true
|
||||||
h.logger.Debugf("Installed 1-RTT Read keys (using %s)", tls.CipherSuiteName(suite.ID))
|
h.logger.Debugf("Installed 1-RTT Read keys (using %s)", qtls.CipherSuiteName(suite.ID))
|
||||||
default:
|
default:
|
||||||
panic("unexpected read encryption level")
|
panic("unexpected read encryption level")
|
||||||
}
|
}
|
||||||
|
@ -598,7 +595,7 @@ func (h *cryptoSetup) SetWriteKey(encLevel qtls.EncryptionLevel, suite *qtls.Cip
|
||||||
newHeaderProtector(suite, trafficSecret, true),
|
newHeaderProtector(suite, trafficSecret, true),
|
||||||
)
|
)
|
||||||
h.mutex.Unlock()
|
h.mutex.Unlock()
|
||||||
h.logger.Debugf("Installed 0-RTT Write keys (using %s)", tls.CipherSuiteName(suite.ID))
|
h.logger.Debugf("Installed 0-RTT Write keys (using %s)", qtls.CipherSuiteName(suite.ID))
|
||||||
if h.tracer != nil {
|
if h.tracer != nil {
|
||||||
h.tracer.UpdatedKeyFromTLS(protocol.Encryption0RTT, h.perspective)
|
h.tracer.UpdatedKeyFromTLS(protocol.Encryption0RTT, h.perspective)
|
||||||
}
|
}
|
||||||
|
@ -611,12 +608,12 @@ func (h *cryptoSetup) SetWriteKey(encLevel qtls.EncryptionLevel, suite *qtls.Cip
|
||||||
h.dropInitialKeys,
|
h.dropInitialKeys,
|
||||||
h.perspective,
|
h.perspective,
|
||||||
)
|
)
|
||||||
h.logger.Debugf("Installed Handshake Write keys (using %s)", tls.CipherSuiteName(suite.ID))
|
h.logger.Debugf("Installed Handshake Write keys (using %s)", qtls.CipherSuiteName(suite.ID))
|
||||||
case qtls.EncryptionApplication:
|
case qtls.EncryptionApplication:
|
||||||
h.writeEncLevel = protocol.Encryption1RTT
|
h.writeEncLevel = protocol.Encryption1RTT
|
||||||
h.aead.SetWriteKey(suite, trafficSecret)
|
h.aead.SetWriteKey(suite, trafficSecret)
|
||||||
h.has1RTTSealer = true
|
h.has1RTTSealer = true
|
||||||
h.logger.Debugf("Installed 1-RTT Write keys (using %s)", tls.CipherSuiteName(suite.ID))
|
h.logger.Debugf("Installed 1-RTT Write keys (using %s)", qtls.CipherSuiteName(suite.ID))
|
||||||
if h.zeroRTTSealer != nil {
|
if h.zeroRTTSealer != nil {
|
||||||
h.zeroRTTSealer = nil
|
h.zeroRTTSealer = nil
|
||||||
h.logger.Debugf("Dropping 0-RTT keys.")
|
h.logger.Debugf("Dropping 0-RTT keys.")
|
||||||
|
|
|
@ -4,8 +4,6 @@ import (
|
||||||
"crypto"
|
"crypto"
|
||||||
"crypto/tls"
|
"crypto/tls"
|
||||||
|
|
||||||
"golang.org/x/crypto/hkdf"
|
|
||||||
|
|
||||||
"github.com/lucas-clemente/quic-go/internal/protocol"
|
"github.com/lucas-clemente/quic-go/internal/protocol"
|
||||||
"github.com/lucas-clemente/quic-go/internal/qtls"
|
"github.com/lucas-clemente/quic-go/internal/qtls"
|
||||||
)
|
)
|
||||||
|
@ -16,7 +14,7 @@ var (
|
||||||
)
|
)
|
||||||
|
|
||||||
func getSalt(v protocol.VersionNumber) []byte {
|
func getSalt(v protocol.VersionNumber) []byte {
|
||||||
if v == protocol.VersionDraft34 || v == protocol.Version1 {
|
if v == protocol.VersionDraft34 {
|
||||||
return quicSaltDraft34
|
return quicSaltDraft34
|
||||||
}
|
}
|
||||||
return quicSaltOld
|
return quicSaltOld
|
||||||
|
@ -51,7 +49,7 @@ func NewInitialAEAD(connID protocol.ConnectionID, pers protocol.Perspective, v p
|
||||||
}
|
}
|
||||||
|
|
||||||
func computeSecrets(connID protocol.ConnectionID, v protocol.VersionNumber) (clientSecret, serverSecret []byte) {
|
func computeSecrets(connID protocol.ConnectionID, v protocol.VersionNumber) (clientSecret, serverSecret []byte) {
|
||||||
initialSecret := hkdf.Extract(crypto.SHA256.New, connID, getSalt(v))
|
initialSecret := qtls.HkdfExtract(crypto.SHA256, connID, getSalt(v))
|
||||||
clientSecret = hkdfExpandLabel(crypto.SHA256, initialSecret, []byte{}, "client in", crypto.SHA256.Size())
|
clientSecret = hkdfExpandLabel(crypto.SHA256, initialSecret, []byte{}, "client in", crypto.SHA256.Size())
|
||||||
serverSecret = hkdfExpandLabel(crypto.SHA256, initialSecret, []byte{}, "server in", crypto.SHA256.Size())
|
serverSecret = hkdfExpandLabel(crypto.SHA256, initialSecret, []byte{}, "server in", crypto.SHA256.Size())
|
||||||
return
|
return
|
||||||
|
|
|
@ -48,7 +48,7 @@ func GetRetryIntegrityTag(retry []byte, origDestConnID protocol.ConnectionID, ve
|
||||||
|
|
||||||
var tag [16]byte
|
var tag [16]byte
|
||||||
var sealed []byte
|
var sealed []byte
|
||||||
if version != protocol.VersionDraft34 && version != protocol.Version1 {
|
if version != protocol.VersionDraft34 {
|
||||||
sealed = oldRetryAEAD.Seal(tag[:0], oldRetryNonce[:], nil, retryBuf.Bytes())
|
sealed = oldRetryAEAD.Seal(tag[:0], oldRetryNonce[:], nil, retryBuf.Bytes())
|
||||||
} else {
|
} else {
|
||||||
sealed = retryAEAD.Seal(tag[:0], retryNonce[:], nil, retryBuf.Bytes())
|
sealed = retryAEAD.Seal(tag[:0], retryNonce[:], nil, retryBuf.Bytes())
|
||||||
|
|
2
vendor/github.com/lucas-clemente/quic-go/internal/handshake/tls_extension_handler.go
generated
vendored
2
vendor/github.com/lucas-clemente/quic-go/internal/handshake/tls_extension_handler.go
generated
vendored
|
@ -24,7 +24,7 @@ var _ tlsExtensionHandler = &extensionHandler{}
|
||||||
// newExtensionHandler creates a new extension handler
|
// newExtensionHandler creates a new extension handler
|
||||||
func newExtensionHandler(params []byte, pers protocol.Perspective, v protocol.VersionNumber) tlsExtensionHandler {
|
func newExtensionHandler(params []byte, pers protocol.Perspective, v protocol.VersionNumber) tlsExtensionHandler {
|
||||||
et := uint16(quicTLSExtensionType)
|
et := uint16(quicTLSExtensionType)
|
||||||
if v != protocol.VersionDraft34 && v != protocol.Version1 {
|
if v != protocol.VersionDraft34 {
|
||||||
et = quicTLSExtensionTypeOldDrafts
|
et = quicTLSExtensionTypeOldDrafts
|
||||||
}
|
}
|
||||||
return &extensionHandler{
|
return &extensionHandler{
|
||||||
|
|
|
@ -163,7 +163,7 @@ func (a *updatableAEAD) Open(dst, src []byte, rcvTime time.Time, pn protocol.Pac
|
||||||
if err == ErrDecryptionFailed {
|
if err == ErrDecryptionFailed {
|
||||||
a.invalidPacketCount++
|
a.invalidPacketCount++
|
||||||
if a.invalidPacketCount >= a.invalidPacketLimit {
|
if a.invalidPacketCount >= a.invalidPacketLimit {
|
||||||
return nil, &qerr.TransportError{ErrorCode: qerr.AEADLimitReached}
|
return nil, qerr.AEADLimitReached
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if err == nil {
|
if err == nil {
|
||||||
|
@ -201,10 +201,7 @@ func (a *updatableAEAD) open(dst, src []byte, rcvTime time.Time, pn protocol.Pac
|
||||||
}
|
}
|
||||||
// Opening succeeded. Check if the peer was allowed to update.
|
// Opening succeeded. Check if the peer was allowed to update.
|
||||||
if a.keyPhase > 0 && a.firstSentWithCurrentKey == protocol.InvalidPacketNumber {
|
if a.keyPhase > 0 && a.firstSentWithCurrentKey == protocol.InvalidPacketNumber {
|
||||||
return nil, &qerr.TransportError{
|
return nil, qerr.NewError(qerr.KeyUpdateError, "keys updated too quickly")
|
||||||
ErrorCode: qerr.KeyUpdateError,
|
|
||||||
ErrorMessage: "keys updated too quickly",
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
a.rollKeys()
|
a.rollKeys()
|
||||||
a.logger.Debugf("Peer updated keys to %d", a.keyPhase)
|
a.logger.Debugf("Peer updated keys to %d", a.keyPhase)
|
||||||
|
@ -253,10 +250,7 @@ func (a *updatableAEAD) Seal(dst, src []byte, pn protocol.PacketNumber, ad []byt
|
||||||
func (a *updatableAEAD) SetLargestAcked(pn protocol.PacketNumber) error {
|
func (a *updatableAEAD) SetLargestAcked(pn protocol.PacketNumber) error {
|
||||||
if a.firstSentWithCurrentKey != protocol.InvalidPacketNumber &&
|
if a.firstSentWithCurrentKey != protocol.InvalidPacketNumber &&
|
||||||
pn >= a.firstSentWithCurrentKey && a.numRcvdWithCurrentKey == 0 {
|
pn >= a.firstSentWithCurrentKey && a.numRcvdWithCurrentKey == 0 {
|
||||||
return &qerr.TransportError{
|
return qerr.NewError(qerr.KeyUpdateError, fmt.Sprintf("received ACK for key phase %d, but peer didn't update keys", a.keyPhase))
|
||||||
ErrorCode: qerr.KeyUpdateError,
|
|
||||||
ErrorMessage: fmt.Sprintf("received ACK for key phase %d, but peer didn't update keys", a.keyPhase),
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
a.largestAcked = pn
|
a.largestAcked = pn
|
||||||
return nil
|
return nil
|
||||||
|
|
|
@ -52,6 +52,9 @@ const MaxByteCount = ByteCount(1<<62 - 1)
|
||||||
// InvalidByteCount is an invalid byte count
|
// InvalidByteCount is an invalid byte count
|
||||||
const InvalidByteCount ByteCount = -1
|
const InvalidByteCount ByteCount = -1
|
||||||
|
|
||||||
|
// An ApplicationErrorCode is an application-defined error code.
|
||||||
|
type ApplicationErrorCode uint64
|
||||||
|
|
||||||
// A StatelessResetToken is a stateless reset token.
|
// A StatelessResetToken is a stateless reset token.
|
||||||
type StatelessResetToken [16]byte
|
type StatelessResetToken [16]byte
|
||||||
|
|
||||||
|
|
|
@ -18,18 +18,17 @@ const (
|
||||||
|
|
||||||
// The version numbers, making grepping easier
|
// The version numbers, making grepping easier
|
||||||
const (
|
const (
|
||||||
VersionTLS VersionNumber = 0x1
|
VersionTLS VersionNumber = 0xff00001d // draft-29
|
||||||
VersionWhatever VersionNumber = math.MaxUint32 - 1 // for when the version doesn't matter
|
VersionWhatever VersionNumber = 1 // for when the version doesn't matter
|
||||||
VersionUnknown VersionNumber = math.MaxUint32
|
VersionUnknown VersionNumber = math.MaxUint32
|
||||||
VersionDraft29 VersionNumber = 0xff00001d
|
VersionDraft29 VersionNumber = 0xff00001d
|
||||||
VersionDraft32 VersionNumber = 0xff000020
|
VersionDraft32 VersionNumber = 0xff000020
|
||||||
VersionDraft34 VersionNumber = 0xff000022
|
VersionDraft34 VersionNumber = 0xff000022 // If everything goes according to plan at the IETF, this will one day be QUIC v1.
|
||||||
Version1 VersionNumber = 0x1
|
|
||||||
)
|
)
|
||||||
|
|
||||||
// SupportedVersions lists the versions that the server supports
|
// SupportedVersions lists the versions that the server supports
|
||||||
// must be in sorted descending order
|
// must be in sorted descending order
|
||||||
var SupportedVersions = []VersionNumber{Version1, VersionDraft34, VersionDraft32, VersionDraft29}
|
var SupportedVersions = []VersionNumber{VersionDraft29, VersionDraft34, VersionDraft32}
|
||||||
|
|
||||||
// IsValidVersion says if the version is known to quic-go
|
// IsValidVersion says if the version is known to quic-go
|
||||||
func IsValidVersion(v VersionNumber) bool {
|
func IsValidVersion(v VersionNumber) bool {
|
||||||
|
@ -39,7 +38,7 @@ func IsValidVersion(v VersionNumber) bool {
|
||||||
func (vn VersionNumber) String() string {
|
func (vn VersionNumber) String() string {
|
||||||
// For releases, VersionTLS will be set to a draft version.
|
// For releases, VersionTLS will be set to a draft version.
|
||||||
// A switch statement can't contain duplicate cases.
|
// A switch statement can't contain duplicate cases.
|
||||||
if vn == VersionTLS && VersionTLS != VersionDraft29 && VersionTLS != VersionDraft32 && VersionTLS != Version1 {
|
if vn == VersionTLS && VersionTLS != VersionDraft29 && VersionTLS != VersionDraft32 {
|
||||||
return "TLS dev version (WIP)"
|
return "TLS dev version (WIP)"
|
||||||
}
|
}
|
||||||
//nolint:exhaustive
|
//nolint:exhaustive
|
||||||
|
@ -54,8 +53,6 @@ func (vn VersionNumber) String() string {
|
||||||
return "draft-32"
|
return "draft-32"
|
||||||
case VersionDraft34:
|
case VersionDraft34:
|
||||||
return "draft-34"
|
return "draft-34"
|
||||||
case Version1:
|
|
||||||
return "v1"
|
|
||||||
default:
|
default:
|
||||||
if vn.isGQUIC() {
|
if vn.isGQUIC() {
|
||||||
return fmt.Sprintf("gQUIC %d", vn.toGQUICVersion())
|
return fmt.Sprintf("gQUIC %d", vn.toGQUICVersion())
|
||||||
|
|
|
@ -6,44 +6,51 @@ import (
|
||||||
"github.com/lucas-clemente/quic-go/internal/qtls"
|
"github.com/lucas-clemente/quic-go/internal/qtls"
|
||||||
)
|
)
|
||||||
|
|
||||||
// TransportErrorCode is a QUIC transport error.
|
// ErrorCode can be used as a normal error without reason.
|
||||||
type TransportErrorCode uint64
|
type ErrorCode uint64
|
||||||
|
|
||||||
// The error codes defined by QUIC
|
// The error codes defined by QUIC
|
||||||
const (
|
const (
|
||||||
NoError TransportErrorCode = 0x0
|
NoError ErrorCode = 0x0
|
||||||
InternalError TransportErrorCode = 0x1
|
InternalError ErrorCode = 0x1
|
||||||
ConnectionRefused TransportErrorCode = 0x2
|
ConnectionRefused ErrorCode = 0x2
|
||||||
FlowControlError TransportErrorCode = 0x3
|
FlowControlError ErrorCode = 0x3
|
||||||
StreamLimitError TransportErrorCode = 0x4
|
StreamLimitError ErrorCode = 0x4
|
||||||
StreamStateError TransportErrorCode = 0x5
|
StreamStateError ErrorCode = 0x5
|
||||||
FinalSizeError TransportErrorCode = 0x6
|
FinalSizeError ErrorCode = 0x6
|
||||||
FrameEncodingError TransportErrorCode = 0x7
|
FrameEncodingError ErrorCode = 0x7
|
||||||
TransportParameterError TransportErrorCode = 0x8
|
TransportParameterError ErrorCode = 0x8
|
||||||
ConnectionIDLimitError TransportErrorCode = 0x9
|
ConnectionIDLimitError ErrorCode = 0x9
|
||||||
ProtocolViolation TransportErrorCode = 0xa
|
ProtocolViolation ErrorCode = 0xa
|
||||||
InvalidToken TransportErrorCode = 0xb
|
InvalidToken ErrorCode = 0xb
|
||||||
ApplicationErrorErrorCode TransportErrorCode = 0xc
|
ApplicationError ErrorCode = 0xc
|
||||||
CryptoBufferExceeded TransportErrorCode = 0xd
|
CryptoBufferExceeded ErrorCode = 0xd
|
||||||
KeyUpdateError TransportErrorCode = 0xe
|
KeyUpdateError ErrorCode = 0xe
|
||||||
AEADLimitReached TransportErrorCode = 0xf
|
AEADLimitReached ErrorCode = 0xf
|
||||||
NoViablePathError TransportErrorCode = 0x10
|
NoViablePathError ErrorCode = 0x10
|
||||||
)
|
)
|
||||||
|
|
||||||
func (e TransportErrorCode) IsCryptoError() bool {
|
func (e ErrorCode) isCryptoError() bool {
|
||||||
return e >= 0x100 && e < 0x200
|
return e >= 0x100 && e < 0x200
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (e ErrorCode) Error() string {
|
||||||
|
if e.isCryptoError() {
|
||||||
|
return fmt.Sprintf("%s: %s", e.String(), e.Message())
|
||||||
|
}
|
||||||
|
return e.String()
|
||||||
|
}
|
||||||
|
|
||||||
// Message is a description of the error.
|
// Message is a description of the error.
|
||||||
// It only returns a non-empty string for crypto errors.
|
// It only returns a non-empty string for crypto errors.
|
||||||
func (e TransportErrorCode) Message() string {
|
func (e ErrorCode) Message() string {
|
||||||
if !e.IsCryptoError() {
|
if !e.isCryptoError() {
|
||||||
return ""
|
return ""
|
||||||
}
|
}
|
||||||
return qtls.Alert(e - 0x100).Error()
|
return qtls.Alert(e - 0x100).Error()
|
||||||
}
|
}
|
||||||
|
|
||||||
func (e TransportErrorCode) String() string {
|
func (e ErrorCode) String() string {
|
||||||
switch e {
|
switch e {
|
||||||
case NoError:
|
case NoError:
|
||||||
return "NO_ERROR"
|
return "NO_ERROR"
|
||||||
|
@ -69,7 +76,7 @@ func (e TransportErrorCode) String() string {
|
||||||
return "PROTOCOL_VIOLATION"
|
return "PROTOCOL_VIOLATION"
|
||||||
case InvalidToken:
|
case InvalidToken:
|
||||||
return "INVALID_TOKEN"
|
return "INVALID_TOKEN"
|
||||||
case ApplicationErrorErrorCode:
|
case ApplicationError:
|
||||||
return "APPLICATION_ERROR"
|
return "APPLICATION_ERROR"
|
||||||
case CryptoBufferExceeded:
|
case CryptoBufferExceeded:
|
||||||
return "CRYPTO_BUFFER_EXCEEDED"
|
return "CRYPTO_BUFFER_EXCEEDED"
|
||||||
|
@ -80,7 +87,7 @@ func (e TransportErrorCode) String() string {
|
||||||
case NoViablePathError:
|
case NoViablePathError:
|
||||||
return "NO_VIABLE_PATH"
|
return "NO_VIABLE_PATH"
|
||||||
default:
|
default:
|
||||||
if e.IsCryptoError() {
|
if e.isCryptoError() {
|
||||||
return fmt.Sprintf("CRYPTO_ERROR (%#x)", uint16(e))
|
return fmt.Sprintf("CRYPTO_ERROR (%#x)", uint16(e))
|
||||||
}
|
}
|
||||||
return fmt.Sprintf("unknown error code: %#x", uint16(e))
|
return fmt.Sprintf("unknown error code: %#x", uint16(e))
|
||||||
|
|
|
@ -1,106 +0,0 @@
|
||||||
package qerr
|
|
||||||
|
|
||||||
import (
|
|
||||||
"fmt"
|
|
||||||
"net"
|
|
||||||
|
|
||||||
"github.com/lucas-clemente/quic-go/internal/protocol"
|
|
||||||
)
|
|
||||||
|
|
||||||
var (
|
|
||||||
ErrHandshakeTimeout = &HandshakeTimeoutError{}
|
|
||||||
ErrIdleTimeout = &IdleTimeoutError{}
|
|
||||||
)
|
|
||||||
|
|
||||||
type TransportError struct {
|
|
||||||
Remote bool
|
|
||||||
FrameType uint64
|
|
||||||
ErrorCode TransportErrorCode
|
|
||||||
ErrorMessage string
|
|
||||||
}
|
|
||||||
|
|
||||||
var _ error = &TransportError{}
|
|
||||||
|
|
||||||
// NewCryptoError create a new TransportError instance for a crypto error
|
|
||||||
func NewCryptoError(tlsAlert uint8, errorMessage string) *TransportError {
|
|
||||||
return &TransportError{
|
|
||||||
ErrorCode: 0x100 + TransportErrorCode(tlsAlert),
|
|
||||||
ErrorMessage: errorMessage,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (e *TransportError) Error() string {
|
|
||||||
str := e.ErrorCode.String()
|
|
||||||
if e.FrameType != 0 {
|
|
||||||
str += fmt.Sprintf(" (frame type: %#x)", e.FrameType)
|
|
||||||
}
|
|
||||||
msg := e.ErrorMessage
|
|
||||||
if len(msg) == 0 {
|
|
||||||
msg = e.ErrorCode.Message()
|
|
||||||
}
|
|
||||||
if len(msg) == 0 {
|
|
||||||
return str
|
|
||||||
}
|
|
||||||
return str + ": " + msg
|
|
||||||
}
|
|
||||||
|
|
||||||
// An ApplicationErrorCode is an application-defined error code.
|
|
||||||
type ApplicationErrorCode uint64
|
|
||||||
|
|
||||||
// A StreamErrorCode is an error code used to cancel streams.
|
|
||||||
type StreamErrorCode uint64
|
|
||||||
|
|
||||||
type ApplicationError struct {
|
|
||||||
Remote bool
|
|
||||||
ErrorCode ApplicationErrorCode
|
|
||||||
ErrorMessage string
|
|
||||||
}
|
|
||||||
|
|
||||||
var _ error = &ApplicationError{}
|
|
||||||
|
|
||||||
func (e *ApplicationError) Error() string {
|
|
||||||
if len(e.ErrorMessage) == 0 {
|
|
||||||
return fmt.Sprintf("Application error %#x", e.ErrorCode)
|
|
||||||
}
|
|
||||||
return fmt.Sprintf("Application error %#x: %s", e.ErrorCode, e.ErrorMessage)
|
|
||||||
}
|
|
||||||
|
|
||||||
type IdleTimeoutError struct{}
|
|
||||||
|
|
||||||
var _ error = &IdleTimeoutError{}
|
|
||||||
|
|
||||||
func (e *IdleTimeoutError) Timeout() bool { return true }
|
|
||||||
func (e *IdleTimeoutError) Temporary() bool { return false }
|
|
||||||
func (e *IdleTimeoutError) Error() string { return "timeout: no recent network activity" }
|
|
||||||
|
|
||||||
type HandshakeTimeoutError struct{}
|
|
||||||
|
|
||||||
var _ error = &HandshakeTimeoutError{}
|
|
||||||
|
|
||||||
func (e *HandshakeTimeoutError) Timeout() bool { return true }
|
|
||||||
func (e *HandshakeTimeoutError) Temporary() bool { return false }
|
|
||||||
func (e *HandshakeTimeoutError) Error() string { return "timeout: handshake did not complete in time" }
|
|
||||||
|
|
||||||
// A VersionNegotiationError occurs when the client and the server can't agree on a QUIC version.
|
|
||||||
type VersionNegotiationError struct {
|
|
||||||
Ours []protocol.VersionNumber
|
|
||||||
Theirs []protocol.VersionNumber
|
|
||||||
}
|
|
||||||
|
|
||||||
func (e *VersionNegotiationError) Error() string {
|
|
||||||
return fmt.Sprintf("no compatible QUIC version found (we support %s, server offered %s)", e.Ours, e.Theirs)
|
|
||||||
}
|
|
||||||
|
|
||||||
// A StatelessResetError occurs when we receive a stateless reset.
|
|
||||||
type StatelessResetError struct {
|
|
||||||
Token protocol.StatelessResetToken
|
|
||||||
}
|
|
||||||
|
|
||||||
var _ net.Error = &StatelessResetError{}
|
|
||||||
|
|
||||||
func (e *StatelessResetError) Error() string {
|
|
||||||
return fmt.Sprintf("received a stateless reset with token %x", e.Token)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (e *StatelessResetError) Timeout() bool { return false }
|
|
||||||
func (e *StatelessResetError) Temporary() bool { return true }
|
|
|
@ -1,55 +0,0 @@
|
||||||
// +build go1.16
|
|
||||||
|
|
||||||
package qerr
|
|
||||||
|
|
||||||
import (
|
|
||||||
"net"
|
|
||||||
)
|
|
||||||
|
|
||||||
func (e *TransportError) Is(target error) bool {
|
|
||||||
_, ok := target.(*TransportError)
|
|
||||||
if ok {
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
return target == net.ErrClosed
|
|
||||||
}
|
|
||||||
|
|
||||||
func (e *ApplicationError) Is(target error) bool {
|
|
||||||
_, ok := target.(*ApplicationError)
|
|
||||||
if ok {
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
return target == net.ErrClosed
|
|
||||||
}
|
|
||||||
|
|
||||||
func (e *IdleTimeoutError) Is(target error) bool {
|
|
||||||
_, ok := target.(*IdleTimeoutError)
|
|
||||||
if ok {
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
return target == net.ErrClosed
|
|
||||||
}
|
|
||||||
|
|
||||||
func (e *HandshakeTimeoutError) Is(target error) bool {
|
|
||||||
_, ok := target.(*HandshakeTimeoutError)
|
|
||||||
if ok {
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
return target == net.ErrClosed
|
|
||||||
}
|
|
||||||
|
|
||||||
func (e *VersionNegotiationError) Is(target error) bool {
|
|
||||||
_, ok := target.(*VersionNegotiationError)
|
|
||||||
if ok {
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
return target == net.ErrClosed
|
|
||||||
}
|
|
||||||
|
|
||||||
func (e *StatelessResetError) Is(target error) bool {
|
|
||||||
_, ok := target.(*StatelessResetError)
|
|
||||||
if ok {
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
return target == net.ErrClosed
|
|
||||||
}
|
|
|
@ -1,33 +0,0 @@
|
||||||
// +build !go1.16
|
|
||||||
|
|
||||||
package qerr
|
|
||||||
|
|
||||||
func (e *TransportError) Is(target error) bool {
|
|
||||||
_, ok := target.(*TransportError)
|
|
||||||
return ok
|
|
||||||
}
|
|
||||||
|
|
||||||
func (e *ApplicationError) Is(target error) bool {
|
|
||||||
_, ok := target.(*ApplicationError)
|
|
||||||
return ok
|
|
||||||
}
|
|
||||||
|
|
||||||
func (e *IdleTimeoutError) Is(target error) bool {
|
|
||||||
_, ok := target.(*IdleTimeoutError)
|
|
||||||
return ok
|
|
||||||
}
|
|
||||||
|
|
||||||
func (e *HandshakeTimeoutError) Is(target error) bool {
|
|
||||||
_, ok := target.(*HandshakeTimeoutError)
|
|
||||||
return ok
|
|
||||||
}
|
|
||||||
|
|
||||||
func (e *VersionNegotiationError) Is(target error) bool {
|
|
||||||
_, ok := target.(*VersionNegotiationError)
|
|
||||||
return ok
|
|
||||||
}
|
|
||||||
|
|
||||||
func (e *StatelessResetError) Is(target error) bool {
|
|
||||||
_, ok := target.(*StatelessResetError)
|
|
||||||
return ok
|
|
||||||
}
|
|
112
vendor/github.com/lucas-clemente/quic-go/internal/qerr/quic_error.go
generated
vendored
Normal file
112
vendor/github.com/lucas-clemente/quic-go/internal/qerr/quic_error.go
generated
vendored
Normal file
|
@ -0,0 +1,112 @@
|
||||||
|
package qerr
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"net"
|
||||||
|
)
|
||||||
|
|
||||||
|
// A QuicError consists of an error code plus a error reason
|
||||||
|
type QuicError struct {
|
||||||
|
ErrorCode ErrorCode
|
||||||
|
FrameType uint64 // only valid if this not an application error
|
||||||
|
ErrorMessage string
|
||||||
|
isTimeout bool
|
||||||
|
isApplicationError bool
|
||||||
|
}
|
||||||
|
|
||||||
|
var _ net.Error = &QuicError{}
|
||||||
|
|
||||||
|
// NewError creates a new QuicError instance
|
||||||
|
func NewError(errorCode ErrorCode, errorMessage string) *QuicError {
|
||||||
|
return &QuicError{
|
||||||
|
ErrorCode: errorCode,
|
||||||
|
ErrorMessage: errorMessage,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewErrorWithFrameType creates a new QuicError instance for a specific frame type
|
||||||
|
func NewErrorWithFrameType(errorCode ErrorCode, frameType uint64, errorMessage string) *QuicError {
|
||||||
|
return &QuicError{
|
||||||
|
ErrorCode: errorCode,
|
||||||
|
FrameType: frameType,
|
||||||
|
ErrorMessage: errorMessage,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewTimeoutError creates a new QuicError instance for a timeout error
|
||||||
|
func NewTimeoutError(errorMessage string) *QuicError {
|
||||||
|
return &QuicError{
|
||||||
|
ErrorMessage: errorMessage,
|
||||||
|
isTimeout: true,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewCryptoError create a new QuicError instance for a crypto error
|
||||||
|
func NewCryptoError(tlsAlert uint8, errorMessage string) *QuicError {
|
||||||
|
return &QuicError{
|
||||||
|
ErrorCode: 0x100 + ErrorCode(tlsAlert),
|
||||||
|
ErrorMessage: errorMessage,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewApplicationError creates a new QuicError instance for an application error
|
||||||
|
func NewApplicationError(errorCode ErrorCode, errorMessage string) *QuicError {
|
||||||
|
return &QuicError{
|
||||||
|
ErrorCode: errorCode,
|
||||||
|
ErrorMessage: errorMessage,
|
||||||
|
isApplicationError: true,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (e *QuicError) Error() string {
|
||||||
|
if e.isApplicationError {
|
||||||
|
if len(e.ErrorMessage) == 0 {
|
||||||
|
return fmt.Sprintf("Application error %#x", uint64(e.ErrorCode))
|
||||||
|
}
|
||||||
|
return fmt.Sprintf("Application error %#x: %s", uint64(e.ErrorCode), e.ErrorMessage)
|
||||||
|
}
|
||||||
|
str := e.ErrorCode.String()
|
||||||
|
if e.FrameType != 0 {
|
||||||
|
str += fmt.Sprintf(" (frame type: %#x)", e.FrameType)
|
||||||
|
}
|
||||||
|
msg := e.ErrorMessage
|
||||||
|
if len(msg) == 0 {
|
||||||
|
msg = e.ErrorCode.Message()
|
||||||
|
}
|
||||||
|
if len(msg) == 0 {
|
||||||
|
return str
|
||||||
|
}
|
||||||
|
return str + ": " + msg
|
||||||
|
}
|
||||||
|
|
||||||
|
// IsCryptoError says if this error is a crypto error
|
||||||
|
func (e *QuicError) IsCryptoError() bool {
|
||||||
|
return e.ErrorCode.isCryptoError()
|
||||||
|
}
|
||||||
|
|
||||||
|
// IsApplicationError says if this error is an application error
|
||||||
|
func (e *QuicError) IsApplicationError() bool {
|
||||||
|
return e.isApplicationError
|
||||||
|
}
|
||||||
|
|
||||||
|
// Temporary says if the error is temporary.
|
||||||
|
func (e *QuicError) Temporary() bool {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
// Timeout says if this error is a timeout.
|
||||||
|
func (e *QuicError) Timeout() bool {
|
||||||
|
return e.isTimeout
|
||||||
|
}
|
||||||
|
|
||||||
|
// ToQuicError converts an arbitrary error to a QuicError. It leaves QuicErrors
|
||||||
|
// unchanged, and properly handles `ErrorCode`s.
|
||||||
|
func ToQuicError(err error) *QuicError {
|
||||||
|
switch e := err.(type) {
|
||||||
|
case *QuicError:
|
||||||
|
return e
|
||||||
|
case ErrorCode:
|
||||||
|
return NewError(e, "")
|
||||||
|
}
|
||||||
|
return NewError(InternalError, err.Error())
|
||||||
|
}
|
|
@ -10,7 +10,7 @@ import (
|
||||||
"net"
|
"net"
|
||||||
"unsafe"
|
"unsafe"
|
||||||
|
|
||||||
"github.com/marten-seemann/qtls-go1-15"
|
qtls "github.com/marten-seemann/qtls-go1-15"
|
||||||
)
|
)
|
||||||
|
|
||||||
type (
|
type (
|
||||||
|
@ -53,6 +53,21 @@ const (
|
||||||
EncryptionApplication = qtls.EncryptionApplication
|
EncryptionApplication = qtls.EncryptionApplication
|
||||||
)
|
)
|
||||||
|
|
||||||
|
// CipherSuiteName gets the name of a cipher suite.
|
||||||
|
func CipherSuiteName(id uint16) string {
|
||||||
|
return qtls.CipherSuiteName(id)
|
||||||
|
}
|
||||||
|
|
||||||
|
// HkdfExtract generates a pseudorandom key for use with Expand from an input secret and an optional independent salt.
|
||||||
|
func HkdfExtract(hash crypto.Hash, newSecret, currentSecret []byte) []byte {
|
||||||
|
return qtls.HkdfExtract(hash, newSecret, currentSecret)
|
||||||
|
}
|
||||||
|
|
||||||
|
// HkdfExpandLabel HKDF expands a label
|
||||||
|
func HkdfExpandLabel(hash crypto.Hash, secret, hashValue []byte, label string, L int) []byte {
|
||||||
|
return qtls.HkdfExpandLabel(hash, secret, hashValue, label, L)
|
||||||
|
}
|
||||||
|
|
||||||
// AEADAESGCMTLS13 creates a new AES-GCM AEAD for TLS 1.3
|
// AEADAESGCMTLS13 creates a new AES-GCM AEAD for TLS 1.3
|
||||||
func AEADAESGCMTLS13(key, fixedNonce []byte) cipher.AEAD {
|
func AEADAESGCMTLS13(key, fixedNonce []byte) cipher.AEAD {
|
||||||
return qtls.AEADAESGCMTLS13(key, fixedNonce)
|
return qtls.AEADAESGCMTLS13(key, fixedNonce)
|
||||||
|
|
|
@ -1,5 +1,4 @@
|
||||||
// +build go1.16
|
// +build go1.16
|
||||||
// +build !go1.17
|
|
||||||
|
|
||||||
package qtls
|
package qtls
|
||||||
|
|
||||||
|
@ -10,7 +9,7 @@ import (
|
||||||
"net"
|
"net"
|
||||||
"unsafe"
|
"unsafe"
|
||||||
|
|
||||||
"github.com/marten-seemann/qtls-go1-16"
|
qtls "github.com/marten-seemann/qtls-go1-16"
|
||||||
)
|
)
|
||||||
|
|
||||||
type (
|
type (
|
||||||
|
@ -53,6 +52,21 @@ const (
|
||||||
EncryptionApplication = qtls.EncryptionApplication
|
EncryptionApplication = qtls.EncryptionApplication
|
||||||
)
|
)
|
||||||
|
|
||||||
|
// CipherSuiteName gets the name of a cipher suite.
|
||||||
|
func CipherSuiteName(id uint16) string {
|
||||||
|
return qtls.CipherSuiteName(id)
|
||||||
|
}
|
||||||
|
|
||||||
|
// HkdfExtract generates a pseudorandom key for use with Expand from an input secret and an optional independent salt.
|
||||||
|
func HkdfExtract(hash crypto.Hash, newSecret, currentSecret []byte) []byte {
|
||||||
|
return qtls.HkdfExtract(hash, newSecret, currentSecret)
|
||||||
|
}
|
||||||
|
|
||||||
|
// HkdfExpandLabel HKDF expands a label
|
||||||
|
func HkdfExpandLabel(hash crypto.Hash, secret, hashValue []byte, label string, L int) []byte {
|
||||||
|
return qtls.HkdfExpandLabel(hash, secret, hashValue, label, L)
|
||||||
|
}
|
||||||
|
|
||||||
// AEADAESGCMTLS13 creates a new AES-GCM AEAD for TLS 1.3
|
// AEADAESGCMTLS13 creates a new AES-GCM AEAD for TLS 1.3
|
||||||
func AEADAESGCMTLS13(key, fixedNonce []byte) cipher.AEAD {
|
func AEADAESGCMTLS13(key, fixedNonce []byte) cipher.AEAD {
|
||||||
return qtls.AEADAESGCMTLS13(key, fixedNonce)
|
return qtls.AEADAESGCMTLS13(key, fixedNonce)
|
||||||
|
|
|
@ -1,99 +0,0 @@
|
||||||
// +build go1.17
|
|
||||||
|
|
||||||
package qtls
|
|
||||||
|
|
||||||
import (
|
|
||||||
"crypto"
|
|
||||||
"crypto/cipher"
|
|
||||||
"crypto/tls"
|
|
||||||
"net"
|
|
||||||
"unsafe"
|
|
||||||
|
|
||||||
"github.com/marten-seemann/qtls-go1-17"
|
|
||||||
)
|
|
||||||
|
|
||||||
type (
|
|
||||||
// Alert is a TLS alert
|
|
||||||
Alert = qtls.Alert
|
|
||||||
// A Certificate is qtls.Certificate.
|
|
||||||
Certificate = qtls.Certificate
|
|
||||||
// CertificateRequestInfo contains inforamtion about a certificate request.
|
|
||||||
CertificateRequestInfo = qtls.CertificateRequestInfo
|
|
||||||
// A CipherSuiteTLS13 is a cipher suite for TLS 1.3
|
|
||||||
CipherSuiteTLS13 = qtls.CipherSuiteTLS13
|
|
||||||
// ClientHelloInfo contains information about a ClientHello.
|
|
||||||
ClientHelloInfo = qtls.ClientHelloInfo
|
|
||||||
// ClientSessionCache is a cache used for session resumption.
|
|
||||||
ClientSessionCache = qtls.ClientSessionCache
|
|
||||||
// ClientSessionState is a state needed for session resumption.
|
|
||||||
ClientSessionState = qtls.ClientSessionState
|
|
||||||
// A Config is a qtls.Config.
|
|
||||||
Config = qtls.Config
|
|
||||||
// A Conn is a qtls.Conn.
|
|
||||||
Conn = qtls.Conn
|
|
||||||
// ConnectionState contains information about the state of the connection.
|
|
||||||
ConnectionState = qtls.ConnectionStateWith0RTT
|
|
||||||
// EncryptionLevel is the encryption level of a message.
|
|
||||||
EncryptionLevel = qtls.EncryptionLevel
|
|
||||||
// Extension is a TLS extension
|
|
||||||
Extension = qtls.Extension
|
|
||||||
// ExtraConfig is the qtls.ExtraConfig
|
|
||||||
ExtraConfig = qtls.ExtraConfig
|
|
||||||
// RecordLayer is a qtls RecordLayer.
|
|
||||||
RecordLayer = qtls.RecordLayer
|
|
||||||
)
|
|
||||||
|
|
||||||
const (
|
|
||||||
// EncryptionHandshake is the Handshake encryption level
|
|
||||||
EncryptionHandshake = qtls.EncryptionHandshake
|
|
||||||
// Encryption0RTT is the 0-RTT encryption level
|
|
||||||
Encryption0RTT = qtls.Encryption0RTT
|
|
||||||
// EncryptionApplication is the application data encryption level
|
|
||||||
EncryptionApplication = qtls.EncryptionApplication
|
|
||||||
)
|
|
||||||
|
|
||||||
// AEADAESGCMTLS13 creates a new AES-GCM AEAD for TLS 1.3
|
|
||||||
func AEADAESGCMTLS13(key, fixedNonce []byte) cipher.AEAD {
|
|
||||||
return qtls.AEADAESGCMTLS13(key, fixedNonce)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Client returns a new TLS client side connection.
|
|
||||||
func Client(conn net.Conn, config *Config, extraConfig *ExtraConfig) *Conn {
|
|
||||||
return qtls.Client(conn, config, extraConfig)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Server returns a new TLS server side connection.
|
|
||||||
func Server(conn net.Conn, config *Config, extraConfig *ExtraConfig) *Conn {
|
|
||||||
return qtls.Server(conn, config, extraConfig)
|
|
||||||
}
|
|
||||||
|
|
||||||
func GetConnectionState(conn *Conn) ConnectionState {
|
|
||||||
return conn.ConnectionStateWith0RTT()
|
|
||||||
}
|
|
||||||
|
|
||||||
// ToTLSConnectionState extracts the tls.ConnectionState
|
|
||||||
func ToTLSConnectionState(cs ConnectionState) tls.ConnectionState {
|
|
||||||
return cs.ConnectionState
|
|
||||||
}
|
|
||||||
|
|
||||||
type cipherSuiteTLS13 struct {
|
|
||||||
ID uint16
|
|
||||||
KeyLen int
|
|
||||||
AEAD func(key, fixedNonce []byte) cipher.AEAD
|
|
||||||
Hash crypto.Hash
|
|
||||||
}
|
|
||||||
|
|
||||||
//go:linkname cipherSuiteTLS13ByID github.com/marten-seemann/qtls-go1-17.cipherSuiteTLS13ByID
|
|
||||||
func cipherSuiteTLS13ByID(id uint16) *cipherSuiteTLS13
|
|
||||||
|
|
||||||
// CipherSuiteTLS13ByID gets a TLS 1.3 cipher suite.
|
|
||||||
func CipherSuiteTLS13ByID(id uint16) *CipherSuiteTLS13 {
|
|
||||||
val := cipherSuiteTLS13ByID(id)
|
|
||||||
cs := (*cipherSuiteTLS13)(unsafe.Pointer(val))
|
|
||||||
return &qtls.CipherSuiteTLS13{
|
|
||||||
ID: cs.ID,
|
|
||||||
KeyLen: cs.KeyLen,
|
|
||||||
AEAD: cs.AEAD,
|
|
||||||
Hash: cs.Hash,
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -1,5 +0,0 @@
|
||||||
// +build go1.18
|
|
||||||
|
|
||||||
package qtls
|
|
||||||
|
|
||||||
var _ int = "quic-go doesn't build on Go 1.18 yet."
|
|
11
vendor/github.com/lucas-clemente/quic-go/internal/wire/connection_close_frame.go
generated
vendored
11
vendor/github.com/lucas-clemente/quic-go/internal/wire/connection_close_frame.go
generated
vendored
|
@ -5,13 +5,14 @@ import (
|
||||||
"io"
|
"io"
|
||||||
|
|
||||||
"github.com/lucas-clemente/quic-go/internal/protocol"
|
"github.com/lucas-clemente/quic-go/internal/protocol"
|
||||||
|
"github.com/lucas-clemente/quic-go/internal/qerr"
|
||||||
"github.com/lucas-clemente/quic-go/quicvarint"
|
"github.com/lucas-clemente/quic-go/quicvarint"
|
||||||
)
|
)
|
||||||
|
|
||||||
// A ConnectionCloseFrame is a CONNECTION_CLOSE frame
|
// A ConnectionCloseFrame is a CONNECTION_CLOSE frame
|
||||||
type ConnectionCloseFrame struct {
|
type ConnectionCloseFrame struct {
|
||||||
IsApplicationError bool
|
IsApplicationError bool
|
||||||
ErrorCode uint64
|
ErrorCode qerr.ErrorCode
|
||||||
FrameType uint64
|
FrameType uint64
|
||||||
ReasonPhrase string
|
ReasonPhrase string
|
||||||
}
|
}
|
||||||
|
@ -27,7 +28,7 @@ func parseConnectionCloseFrame(r *bytes.Reader, _ protocol.VersionNumber) (*Conn
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
f.ErrorCode = ec
|
f.ErrorCode = qerr.ErrorCode(ec)
|
||||||
// read the Frame Type, if this is not an application error
|
// read the Frame Type, if this is not an application error
|
||||||
if !f.IsApplicationError {
|
if !f.IsApplicationError {
|
||||||
ft, err := quicvarint.Read(r)
|
ft, err := quicvarint.Read(r)
|
||||||
|
@ -58,8 +59,8 @@ func parseConnectionCloseFrame(r *bytes.Reader, _ protocol.VersionNumber) (*Conn
|
||||||
}
|
}
|
||||||
|
|
||||||
// Length of a written frame
|
// Length of a written frame
|
||||||
func (f *ConnectionCloseFrame) Length(protocol.VersionNumber) protocol.ByteCount {
|
func (f *ConnectionCloseFrame) Length(version protocol.VersionNumber) protocol.ByteCount {
|
||||||
length := 1 + quicvarint.Len(f.ErrorCode) + quicvarint.Len(uint64(len(f.ReasonPhrase))) + protocol.ByteCount(len(f.ReasonPhrase))
|
length := 1 + quicvarint.Len(uint64(f.ErrorCode)) + quicvarint.Len(uint64(len(f.ReasonPhrase))) + protocol.ByteCount(len(f.ReasonPhrase))
|
||||||
if !f.IsApplicationError {
|
if !f.IsApplicationError {
|
||||||
length += quicvarint.Len(f.FrameType) // for the frame type
|
length += quicvarint.Len(f.FrameType) // for the frame type
|
||||||
}
|
}
|
||||||
|
@ -73,7 +74,7 @@ func (f *ConnectionCloseFrame) Write(b *bytes.Buffer, version protocol.VersionNu
|
||||||
b.WriteByte(0x1c)
|
b.WriteByte(0x1c)
|
||||||
}
|
}
|
||||||
|
|
||||||
quicvarint.Write(b, f.ErrorCode)
|
quicvarint.Write(b, uint64(f.ErrorCode))
|
||||||
if !f.IsApplicationError {
|
if !f.IsApplicationError {
|
||||||
quicvarint.Write(b, f.FrameType)
|
quicvarint.Write(b, f.FrameType)
|
||||||
}
|
}
|
||||||
|
|
|
@ -26,7 +26,7 @@ func NewFrameParser(supportsDatagrams bool, v protocol.VersionNumber) FrameParse
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// ParseNext parses the next frame.
|
// ParseNextFrame parses the next frame
|
||||||
// It skips PADDING frames.
|
// It skips PADDING frames.
|
||||||
func (p *frameParser) ParseNext(r *bytes.Reader, encLevel protocol.EncryptionLevel) (Frame, error) {
|
func (p *frameParser) ParseNext(r *bytes.Reader, encLevel protocol.EncryptionLevel) (Frame, error) {
|
||||||
for r.Len() != 0 {
|
for r.Len() != 0 {
|
||||||
|
@ -38,11 +38,7 @@ func (p *frameParser) ParseNext(r *bytes.Reader, encLevel protocol.EncryptionLev
|
||||||
|
|
||||||
f, err := p.parseFrame(r, typeByte, encLevel)
|
f, err := p.parseFrame(r, typeByte, encLevel)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, &qerr.TransportError{
|
return nil, qerr.NewErrorWithFrameType(qerr.FrameEncodingError, uint64(typeByte), err.Error())
|
||||||
FrameType: uint64(typeByte),
|
|
||||||
ErrorCode: qerr.FrameEncodingError,
|
|
||||||
ErrorMessage: err.Error(),
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
return f, nil
|
return f, nil
|
||||||
}
|
}
|
||||||
|
|
|
@ -4,14 +4,13 @@ import (
|
||||||
"bytes"
|
"bytes"
|
||||||
|
|
||||||
"github.com/lucas-clemente/quic-go/internal/protocol"
|
"github.com/lucas-clemente/quic-go/internal/protocol"
|
||||||
"github.com/lucas-clemente/quic-go/internal/qerr"
|
|
||||||
"github.com/lucas-clemente/quic-go/quicvarint"
|
"github.com/lucas-clemente/quic-go/quicvarint"
|
||||||
)
|
)
|
||||||
|
|
||||||
// A ResetStreamFrame is a RESET_STREAM frame in QUIC
|
// A ResetStreamFrame is a RESET_STREAM frame in QUIC
|
||||||
type ResetStreamFrame struct {
|
type ResetStreamFrame struct {
|
||||||
StreamID protocol.StreamID
|
StreamID protocol.StreamID
|
||||||
ErrorCode qerr.StreamErrorCode
|
ErrorCode protocol.ApplicationErrorCode
|
||||||
FinalSize protocol.ByteCount
|
FinalSize protocol.ByteCount
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -39,7 +38,7 @@ func parseResetStreamFrame(r *bytes.Reader, _ protocol.VersionNumber) (*ResetStr
|
||||||
|
|
||||||
return &ResetStreamFrame{
|
return &ResetStreamFrame{
|
||||||
StreamID: streamID,
|
StreamID: streamID,
|
||||||
ErrorCode: qerr.StreamErrorCode(errorCode),
|
ErrorCode: protocol.ApplicationErrorCode(errorCode),
|
||||||
FinalSize: byteOffset,
|
FinalSize: byteOffset,
|
||||||
}, nil
|
}, nil
|
||||||
}
|
}
|
||||||
|
|
|
@ -4,14 +4,13 @@ import (
|
||||||
"bytes"
|
"bytes"
|
||||||
|
|
||||||
"github.com/lucas-clemente/quic-go/internal/protocol"
|
"github.com/lucas-clemente/quic-go/internal/protocol"
|
||||||
"github.com/lucas-clemente/quic-go/internal/qerr"
|
|
||||||
"github.com/lucas-clemente/quic-go/quicvarint"
|
"github.com/lucas-clemente/quic-go/quicvarint"
|
||||||
)
|
)
|
||||||
|
|
||||||
// A StopSendingFrame is a STOP_SENDING frame
|
// A StopSendingFrame is a STOP_SENDING frame
|
||||||
type StopSendingFrame struct {
|
type StopSendingFrame struct {
|
||||||
StreamID protocol.StreamID
|
StreamID protocol.StreamID
|
||||||
ErrorCode qerr.StreamErrorCode
|
ErrorCode protocol.ApplicationErrorCode
|
||||||
}
|
}
|
||||||
|
|
||||||
// parseStopSendingFrame parses a STOP_SENDING frame
|
// parseStopSendingFrame parses a STOP_SENDING frame
|
||||||
|
@ -31,7 +30,7 @@ func parseStopSendingFrame(r *bytes.Reader, _ protocol.VersionNumber) (*StopSend
|
||||||
|
|
||||||
return &StopSendingFrame{
|
return &StopSendingFrame{
|
||||||
StreamID: protocol.StreamID(streamID),
|
StreamID: protocol.StreamID(streamID),
|
||||||
ErrorCode: qerr.StreamErrorCode(errorCode),
|
ErrorCode: protocol.ApplicationErrorCode(errorCode),
|
||||||
}, nil
|
}, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -6,6 +6,7 @@ import (
|
||||||
"io"
|
"io"
|
||||||
|
|
||||||
"github.com/lucas-clemente/quic-go/internal/protocol"
|
"github.com/lucas-clemente/quic-go/internal/protocol"
|
||||||
|
"github.com/lucas-clemente/quic-go/internal/qerr"
|
||||||
"github.com/lucas-clemente/quic-go/quicvarint"
|
"github.com/lucas-clemente/quic-go/quicvarint"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@ -78,7 +79,7 @@ func parseStreamFrame(r *bytes.Reader, _ protocol.VersionNumber) (*StreamFrame,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if frame.Offset+frame.DataLen() > protocol.MaxByteCount {
|
if frame.Offset+frame.DataLen() > protocol.MaxByteCount {
|
||||||
return nil, errors.New("stream data overflows maximum offset")
|
return nil, qerr.NewError(qerr.FrameEncodingError, "stream data overflows maximum offset")
|
||||||
}
|
}
|
||||||
return frame, nil
|
return frame, nil
|
||||||
}
|
}
|
||||||
|
|
|
@ -90,10 +90,7 @@ type TransportParameters struct {
|
||||||
// Unmarshal the transport parameters
|
// Unmarshal the transport parameters
|
||||||
func (p *TransportParameters) Unmarshal(data []byte, sentBy protocol.Perspective) error {
|
func (p *TransportParameters) Unmarshal(data []byte, sentBy protocol.Perspective) error {
|
||||||
if err := p.unmarshal(bytes.NewReader(data), sentBy, false); err != nil {
|
if err := p.unmarshal(bytes.NewReader(data), sentBy, false); err != nil {
|
||||||
return &qerr.TransportError{
|
return qerr.NewError(qerr.TransportParameterError, err.Error())
|
||||||
ErrorCode: qerr.TransportParameterError,
|
|
||||||
ErrorMessage: err.Error(),
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
@ -262,7 +259,7 @@ func (p *TransportParameters) readNumericTransportParameter(
|
||||||
return fmt.Errorf("error while reading transport parameter %d: %s", paramID, err)
|
return fmt.Errorf("error while reading transport parameter %d: %s", paramID, err)
|
||||||
}
|
}
|
||||||
if remainingLen-r.Len() != expectedLen {
|
if remainingLen-r.Len() != expectedLen {
|
||||||
return fmt.Errorf("inconsistent transport parameter length for transport parameter %#x", paramID)
|
return fmt.Errorf("inconsistent transport parameter length for %d", paramID)
|
||||||
}
|
}
|
||||||
//nolint:exhaustive // This only covers the numeric transport parameters.
|
//nolint:exhaustive // This only covers the numeric transport parameters.
|
||||||
switch paramID {
|
switch paramID {
|
||||||
|
|
|
@ -0,0 +1,78 @@
|
||||||
|
package logging
|
||||||
|
|
||||||
|
// A CloseReason is the reason why a QUIC connection is closed.
|
||||||
|
// It falls in one of 4 categories:
|
||||||
|
// 1. The application closed the connection (with an application-specific error code).
|
||||||
|
// 2. The transport closed the connection with a transport-error code.
|
||||||
|
// 3. The connection timed out, either during the handshake, or due to an idle timeout.
|
||||||
|
// 4. A stateless reset was received.
|
||||||
|
type CloseReason struct {
|
||||||
|
remote bool
|
||||||
|
applicationError *ApplicationError
|
||||||
|
transportError *TransportError
|
||||||
|
|
||||||
|
timeout *TimeoutReason
|
||||||
|
statelessResetToken *StatelessResetToken
|
||||||
|
versions []VersionNumber
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewApplicationCloseReason creates a new CloseReason for an application error.
|
||||||
|
func NewApplicationCloseReason(errorCode ApplicationError, remote bool) CloseReason {
|
||||||
|
return CloseReason{remote: remote, applicationError: &errorCode}
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewTransportCloseReason creates a new CloseReason for a transport error.
|
||||||
|
func NewTransportCloseReason(errorCode TransportError, remote bool) CloseReason {
|
||||||
|
return CloseReason{remote: remote, transportError: &errorCode}
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewTimeoutCloseReason creates a new CloseReason for a connection timeout.
|
||||||
|
func NewTimeoutCloseReason(r TimeoutReason) CloseReason {
|
||||||
|
return CloseReason{timeout: &r}
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewStatelessResetCloseReason creates a new CloseReason for a stateless reset.
|
||||||
|
func NewStatelessResetCloseReason(token StatelessResetToken) CloseReason {
|
||||||
|
return CloseReason{statelessResetToken: &token}
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewVersionNegotiationError creates a new CloseReason for a version negotiation error.
|
||||||
|
func NewVersionNegotiationError(versions []VersionNumber) CloseReason {
|
||||||
|
return CloseReason{versions: versions}
|
||||||
|
}
|
||||||
|
|
||||||
|
// ApplicationError gets the application error.
|
||||||
|
func (r *CloseReason) ApplicationError() (errorCode ApplicationError, remote bool, ok bool) {
|
||||||
|
if r.applicationError == nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
return *r.applicationError, r.remote, true
|
||||||
|
}
|
||||||
|
|
||||||
|
// TransportError gets the transport error.
|
||||||
|
func (r *CloseReason) TransportError() (errorCode TransportError, remote bool, ok bool) {
|
||||||
|
if r.transportError == nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
return *r.transportError, r.remote, true
|
||||||
|
}
|
||||||
|
|
||||||
|
// Timeout gets the timeout error.
|
||||||
|
func (r *CloseReason) Timeout() (reason TimeoutReason, ok bool) {
|
||||||
|
if r.timeout == nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
return *r.timeout, true
|
||||||
|
}
|
||||||
|
|
||||||
|
// StatelessReset gets the stateless reset token.
|
||||||
|
func (r *CloseReason) StatelessReset() (token StatelessResetToken, ok bool) {
|
||||||
|
if r.statelessResetToken == nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
return *r.statelessResetToken, true
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *CloseReason) VersionNegotiation() (versions []VersionNumber, ok bool) {
|
||||||
|
return r.versions, len(r.versions) > 0
|
||||||
|
}
|
|
@ -3,7 +3,6 @@
|
||||||
package logging
|
package logging
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"context"
|
|
||||||
"net"
|
"net"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
|
@ -50,9 +49,9 @@ type (
|
||||||
PreferredAddress = wire.PreferredAddress
|
PreferredAddress = wire.PreferredAddress
|
||||||
|
|
||||||
// A TransportError is a transport-level error code.
|
// A TransportError is a transport-level error code.
|
||||||
TransportError = qerr.TransportErrorCode
|
TransportError = qerr.ErrorCode
|
||||||
// An ApplicationError is an application-defined error code.
|
// An ApplicationError is an application-defined error code.
|
||||||
ApplicationError = qerr.TransportErrorCode
|
ApplicationError = qerr.ErrorCode
|
||||||
|
|
||||||
// The RTTStats contain statistics used by the congestion controller.
|
// The RTTStats contain statistics used by the congestion controller.
|
||||||
RTTStats = utils.RTTStats
|
RTTStats = utils.RTTStats
|
||||||
|
@ -92,11 +91,11 @@ const (
|
||||||
|
|
||||||
// A Tracer traces events.
|
// A Tracer traces events.
|
||||||
type Tracer interface {
|
type Tracer interface {
|
||||||
// TracerForConnection requests a new tracer for a connection.
|
// ConnectionTracer requests a new tracer for a connection.
|
||||||
// The ODCID is the original destination connection ID:
|
// The ODCID is the original destination connection ID:
|
||||||
// The destination connection ID that the client used on the first Initial packet it sent on this connection.
|
// The destination connection ID that the client used on the first Initial packet it sent on this connection.
|
||||||
// If nil is returned, tracing will be disabled for this connection.
|
// If nil is returned, tracing will be disabled for this connection.
|
||||||
TracerForConnection(ctx context.Context, p Perspective, odcid ConnectionID) ConnectionTracer
|
TracerForConnection(p Perspective, odcid ConnectionID) ConnectionTracer
|
||||||
|
|
||||||
SentPacket(net.Addr, *Header, ByteCount, []Frame)
|
SentPacket(net.Addr, *Header, ByteCount, []Frame)
|
||||||
DroppedPacket(net.Addr, PacketType, ByteCount, PacketDropReason)
|
DroppedPacket(net.Addr, PacketType, ByteCount, PacketDropReason)
|
||||||
|
@ -104,9 +103,8 @@ type Tracer interface {
|
||||||
|
|
||||||
// A ConnectionTracer records events.
|
// A ConnectionTracer records events.
|
||||||
type ConnectionTracer interface {
|
type ConnectionTracer interface {
|
||||||
StartedConnection(local, remote net.Addr, srcConnID, destConnID ConnectionID)
|
StartedConnection(local, remote net.Addr, version VersionNumber, srcConnID, destConnID ConnectionID)
|
||||||
NegotiatedVersion(chosen VersionNumber, clientVersions, serverVersions []VersionNumber)
|
ClosedConnection(CloseReason)
|
||||||
ClosedConnection(error)
|
|
||||||
SentTransportParameters(*TransportParameters)
|
SentTransportParameters(*TransportParameters)
|
||||||
ReceivedTransportParameters(*TransportParameters)
|
ReceivedTransportParameters(*TransportParameters)
|
||||||
RestoredTransportParameters(parameters *TransportParameters) // for 0-RTT
|
RestoredTransportParameters(parameters *TransportParameters) // for 0-RTT
|
||||||
|
@ -117,7 +115,6 @@ type ConnectionTracer interface {
|
||||||
BufferedPacket(PacketType)
|
BufferedPacket(PacketType)
|
||||||
DroppedPacket(PacketType, ByteCount, PacketDropReason)
|
DroppedPacket(PacketType, ByteCount, PacketDropReason)
|
||||||
UpdatedMetrics(rttStats *RTTStats, cwnd, bytesInFlight ByteCount, packetsInFlight int)
|
UpdatedMetrics(rttStats *RTTStats, cwnd, bytesInFlight ByteCount, packetsInFlight int)
|
||||||
AcknowledgedPacket(EncryptionLevel, PacketNumber)
|
|
||||||
LostPacket(EncryptionLevel, PacketNumber, PacketLossReason)
|
LostPacket(EncryptionLevel, PacketNumber, PacketLossReason)
|
||||||
UpdatedCongestionState(CongestionState)
|
UpdatedCongestionState(CongestionState)
|
||||||
UpdatedPTOCount(value uint32)
|
UpdatedPTOCount(value uint32)
|
||||||
|
|
|
@ -1,7 +1,6 @@
|
||||||
package logging
|
package logging
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"context"
|
|
||||||
"net"
|
"net"
|
||||||
"time"
|
"time"
|
||||||
)
|
)
|
||||||
|
@ -23,10 +22,10 @@ func NewMultiplexedTracer(tracers ...Tracer) Tracer {
|
||||||
return &tracerMultiplexer{tracers}
|
return &tracerMultiplexer{tracers}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (m *tracerMultiplexer) TracerForConnection(ctx context.Context, p Perspective, odcid ConnectionID) ConnectionTracer {
|
func (m *tracerMultiplexer) TracerForConnection(p Perspective, odcid ConnectionID) ConnectionTracer {
|
||||||
var connTracers []ConnectionTracer
|
var connTracers []ConnectionTracer
|
||||||
for _, t := range m.tracers {
|
for _, t := range m.tracers {
|
||||||
if ct := t.TracerForConnection(ctx, p, odcid); ct != nil {
|
if ct := t.TracerForConnection(p, odcid); ct != nil {
|
||||||
connTracers = append(connTracers, ct)
|
connTracers = append(connTracers, ct)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -62,21 +61,15 @@ func NewMultiplexedConnectionTracer(tracers ...ConnectionTracer) ConnectionTrace
|
||||||
return &connTracerMultiplexer{tracers: tracers}
|
return &connTracerMultiplexer{tracers: tracers}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (m *connTracerMultiplexer) StartedConnection(local, remote net.Addr, srcConnID, destConnID ConnectionID) {
|
func (m *connTracerMultiplexer) StartedConnection(local, remote net.Addr, version VersionNumber, srcConnID, destConnID ConnectionID) {
|
||||||
for _, t := range m.tracers {
|
for _, t := range m.tracers {
|
||||||
t.StartedConnection(local, remote, srcConnID, destConnID)
|
t.StartedConnection(local, remote, version, srcConnID, destConnID)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (m *connTracerMultiplexer) NegotiatedVersion(chosen VersionNumber, clientVersions, serverVersions []VersionNumber) {
|
func (m *connTracerMultiplexer) ClosedConnection(reason CloseReason) {
|
||||||
for _, t := range m.tracers {
|
for _, t := range m.tracers {
|
||||||
t.NegotiatedVersion(chosen, clientVersions, serverVersions)
|
t.ClosedConnection(reason)
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (m *connTracerMultiplexer) ClosedConnection(e error) {
|
|
||||||
for _, t := range m.tracers {
|
|
||||||
t.ClosedConnection(e)
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -146,12 +139,6 @@ func (m *connTracerMultiplexer) UpdatedMetrics(rttStats *RTTStats, cwnd, bytesIn
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (m *connTracerMultiplexer) AcknowledgedPacket(encLevel EncryptionLevel, pn PacketNumber) {
|
|
||||||
for _, t := range m.tracers {
|
|
||||||
t.AcknowledgedPacket(encLevel, pn)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (m *connTracerMultiplexer) LostPacket(encLevel EncryptionLevel, pn PacketNumber, reason PacketLossReason) {
|
func (m *connTracerMultiplexer) LostPacket(encLevel EncryptionLevel, pn PacketNumber, reason PacketLossReason) {
|
||||||
for _, t := range m.tracers {
|
for _, t := range m.tracers {
|
||||||
t.LostPacket(encLevel, pn, reason)
|
t.LostPacket(encLevel, pn, reason)
|
||||||
|
|
|
@ -1,7 +1,9 @@
|
||||||
package logging
|
package logging
|
||||||
|
|
||||||
|
import "github.com/lucas-clemente/quic-go/internal/protocol"
|
||||||
|
|
||||||
// PacketType is the packet type of a QUIC packet
|
// PacketType is the packet type of a QUIC packet
|
||||||
type PacketType uint8
|
type PacketType = protocol.PacketType
|
||||||
|
|
||||||
const (
|
const (
|
||||||
// PacketTypeInitial is the packet type of an Initial packet
|
// PacketTypeInitial is the packet type of an Initial packet
|
||||||
|
|
|
@ -18,6 +18,14 @@ import (
|
||||||
"github.com/lucas-clemente/quic-go/logging"
|
"github.com/lucas-clemente/quic-go/logging"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
type statelessResetErr struct {
|
||||||
|
token protocol.StatelessResetToken
|
||||||
|
}
|
||||||
|
|
||||||
|
func (e statelessResetErr) Error() string {
|
||||||
|
return fmt.Sprintf("received a stateless reset with token %x", e.token)
|
||||||
|
}
|
||||||
|
|
||||||
type zeroRTTQueue struct {
|
type zeroRTTQueue struct {
|
||||||
queue []*receivedPacket
|
queue []*receivedPacket
|
||||||
retireTimer *time.Timer
|
retireTimer *time.Timer
|
||||||
|
@ -422,7 +430,7 @@ func (h *packetHandlerMap) maybeHandleStatelessReset(data []byte) bool {
|
||||||
copy(token[:], data[len(data)-16:])
|
copy(token[:], data[len(data)-16:])
|
||||||
if sess, ok := h.resetTokens[token]; ok {
|
if sess, ok := h.resetTokens[token]; ok {
|
||||||
h.logger.Debugf("Received a stateless reset with token %#x. Closing session.", token)
|
h.logger.Debugf("Received a stateless reset with token %#x. Closing session.", token)
|
||||||
go sess.destroy(&StatelessResetError{Token: token})
|
go sess.destroy(statelessResetErr{token: token})
|
||||||
return true
|
return true
|
||||||
}
|
}
|
||||||
return false
|
return false
|
||||||
|
|
|
@ -20,8 +20,7 @@ type packer interface {
|
||||||
PackPacket() (*packedPacket, error)
|
PackPacket() (*packedPacket, error)
|
||||||
MaybePackProbePacket(protocol.EncryptionLevel) (*packedPacket, error)
|
MaybePackProbePacket(protocol.EncryptionLevel) (*packedPacket, error)
|
||||||
MaybePackAckPacket(handshakeConfirmed bool) (*packedPacket, error)
|
MaybePackAckPacket(handshakeConfirmed bool) (*packedPacket, error)
|
||||||
PackConnectionClose(*qerr.TransportError) (*coalescedPacket, error)
|
PackConnectionClose(*qerr.QuicError) (*coalescedPacket, error)
|
||||||
PackApplicationClose(*qerr.ApplicationError) (*coalescedPacket, error)
|
|
||||||
|
|
||||||
SetMaxPacketSize(protocol.ByteCount)
|
SetMaxPacketSize(protocol.ByteCount)
|
||||||
PackMTUProbePacket(ping ackhandler.Frame, size protocol.ByteCount) (*packedPacket, error)
|
PackMTUProbePacket(ping ackhandler.Frame, size protocol.ByteCount) (*packedPacket, error)
|
||||||
|
@ -204,27 +203,14 @@ func newPacketPacker(
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// PackConnectionClose packs a packet that closes the connection with a transport error.
|
// PackConnectionClose packs a packet that ONLY contains a ConnectionCloseFrame
|
||||||
func (p *packetPacker) PackConnectionClose(e *qerr.TransportError) (*coalescedPacket, error) {
|
func (p *packetPacker) PackConnectionClose(quicErr *qerr.QuicError) (*coalescedPacket, error) {
|
||||||
var reason string
|
var reason string
|
||||||
// don't send details of crypto errors
|
// don't send details of crypto errors
|
||||||
if !e.ErrorCode.IsCryptoError() {
|
if !quicErr.IsCryptoError() {
|
||||||
reason = e.ErrorMessage
|
reason = quicErr.ErrorMessage
|
||||||
}
|
}
|
||||||
return p.packConnectionClose(false, uint64(e.ErrorCode), e.FrameType, reason)
|
|
||||||
}
|
|
||||||
|
|
||||||
// PackApplicationClose packs a packet that closes the connection with an application error.
|
|
||||||
func (p *packetPacker) PackApplicationClose(e *qerr.ApplicationError) (*coalescedPacket, error) {
|
|
||||||
return p.packConnectionClose(true, uint64(e.ErrorCode), 0, e.ErrorMessage)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (p *packetPacker) packConnectionClose(
|
|
||||||
isApplicationError bool,
|
|
||||||
errorCode uint64,
|
|
||||||
frameType uint64,
|
|
||||||
reason string,
|
|
||||||
) (*coalescedPacket, error) {
|
|
||||||
var sealers [4]sealer
|
var sealers [4]sealer
|
||||||
var hdrs [4]*wire.ExtendedHeader
|
var hdrs [4]*wire.ExtendedHeader
|
||||||
var payloads [4]*payload
|
var payloads [4]*payload
|
||||||
|
@ -235,17 +221,20 @@ func (p *packetPacker) packConnectionClose(
|
||||||
if p.perspective == protocol.PerspectiveServer && encLevel == protocol.Encryption0RTT {
|
if p.perspective == protocol.PerspectiveServer && encLevel == protocol.Encryption0RTT {
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
ccf := &wire.ConnectionCloseFrame{
|
quicErrToSend := quicErr
|
||||||
IsApplicationError: isApplicationError,
|
reasonPhrase := reason
|
||||||
ErrorCode: errorCode,
|
if encLevel == protocol.EncryptionInitial || encLevel == protocol.EncryptionHandshake {
|
||||||
FrameType: frameType,
|
|
||||||
ReasonPhrase: reason,
|
|
||||||
}
|
|
||||||
// don't send application errors in Initial or Handshake packets
|
// don't send application errors in Initial or Handshake packets
|
||||||
if isApplicationError && (encLevel == protocol.EncryptionInitial || encLevel == protocol.EncryptionHandshake) {
|
if quicErr.IsApplicationError() {
|
||||||
ccf.IsApplicationError = false
|
quicErrToSend = qerr.NewError(qerr.ApplicationError, "")
|
||||||
ccf.ErrorCode = uint64(qerr.ApplicationErrorErrorCode)
|
reasonPhrase = ""
|
||||||
ccf.ReasonPhrase = ""
|
}
|
||||||
|
}
|
||||||
|
ccf := &wire.ConnectionCloseFrame{
|
||||||
|
IsApplicationError: quicErrToSend.IsApplicationError(),
|
||||||
|
ErrorCode: quicErrToSend.ErrorCode,
|
||||||
|
FrameType: quicErrToSend.FrameType,
|
||||||
|
ReasonPhrase: reasonPhrase,
|
||||||
}
|
}
|
||||||
payload := &payload{
|
payload := &payload{
|
||||||
frames: []ackhandler.Frame{{Frame: ccf}},
|
frames: []ackhandler.Frame{{Frame: ccf}},
|
||||||
|
|
|
@ -8,7 +8,6 @@ import (
|
||||||
|
|
||||||
"github.com/lucas-clemente/quic-go/internal/flowcontrol"
|
"github.com/lucas-clemente/quic-go/internal/flowcontrol"
|
||||||
"github.com/lucas-clemente/quic-go/internal/protocol"
|
"github.com/lucas-clemente/quic-go/internal/protocol"
|
||||||
"github.com/lucas-clemente/quic-go/internal/qerr"
|
|
||||||
"github.com/lucas-clemente/quic-go/internal/utils"
|
"github.com/lucas-clemente/quic-go/internal/utils"
|
||||||
"github.com/lucas-clemente/quic-go/internal/wire"
|
"github.com/lucas-clemente/quic-go/internal/wire"
|
||||||
)
|
)
|
||||||
|
@ -39,7 +38,7 @@ type receiveStream struct {
|
||||||
|
|
||||||
closeForShutdownErr error
|
closeForShutdownErr error
|
||||||
cancelReadErr error
|
cancelReadErr error
|
||||||
resetRemotelyErr *StreamError
|
resetRemotelyErr StreamError
|
||||||
|
|
||||||
closedForShutdown bool // set when CloseForShutdown() is called
|
closedForShutdown bool // set when CloseForShutdown() is called
|
||||||
finRead bool // set once we read a frame with a Fin
|
finRead bool // set once we read a frame with a Fin
|
||||||
|
@ -197,7 +196,7 @@ func (s *receiveStream) dequeueNextFrame() {
|
||||||
s.readPosInFrame = 0
|
s.readPosInFrame = 0
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *receiveStream) CancelRead(errorCode StreamErrorCode) {
|
func (s *receiveStream) CancelRead(errorCode protocol.ApplicationErrorCode) {
|
||||||
s.mutex.Lock()
|
s.mutex.Lock()
|
||||||
completed := s.cancelReadImpl(errorCode)
|
completed := s.cancelReadImpl(errorCode)
|
||||||
s.mutex.Unlock()
|
s.mutex.Unlock()
|
||||||
|
@ -208,7 +207,7 @@ func (s *receiveStream) CancelRead(errorCode StreamErrorCode) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *receiveStream) cancelReadImpl(errorCode qerr.StreamErrorCode) bool /* completed */ {
|
func (s *receiveStream) cancelReadImpl(errorCode protocol.ApplicationErrorCode) bool /* completed */ {
|
||||||
if s.finRead || s.canceledRead || s.resetRemotely {
|
if s.finRead || s.canceledRead || s.resetRemotely {
|
||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
|
@ -282,9 +281,9 @@ func (s *receiveStream) handleResetStreamFrameImpl(frame *wire.ResetStreamFrame)
|
||||||
return false, nil
|
return false, nil
|
||||||
}
|
}
|
||||||
s.resetRemotely = true
|
s.resetRemotely = true
|
||||||
s.resetRemotelyErr = &StreamError{
|
s.resetRemotelyErr = streamCanceledError{
|
||||||
StreamID: s.streamID,
|
errorCode: frame.ErrorCode,
|
||||||
ErrorCode: frame.ErrorCode,
|
error: fmt.Errorf("stream %d was reset with error code %d", s.streamID, frame.ErrorCode),
|
||||||
}
|
}
|
||||||
s.signalRead()
|
s.signalRead()
|
||||||
return newlyRcvdFinalOffset, nil
|
return newlyRcvdFinalOffset, nil
|
||||||
|
|
|
@ -7,9 +7,9 @@ import (
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
"github.com/lucas-clemente/quic-go/internal/ackhandler"
|
"github.com/lucas-clemente/quic-go/internal/ackhandler"
|
||||||
|
|
||||||
"github.com/lucas-clemente/quic-go/internal/flowcontrol"
|
"github.com/lucas-clemente/quic-go/internal/flowcontrol"
|
||||||
"github.com/lucas-clemente/quic-go/internal/protocol"
|
"github.com/lucas-clemente/quic-go/internal/protocol"
|
||||||
"github.com/lucas-clemente/quic-go/internal/qerr"
|
|
||||||
"github.com/lucas-clemente/quic-go/internal/utils"
|
"github.com/lucas-clemente/quic-go/internal/utils"
|
||||||
"github.com/lucas-clemente/quic-go/internal/wire"
|
"github.com/lucas-clemente/quic-go/internal/wire"
|
||||||
)
|
)
|
||||||
|
@ -407,12 +407,12 @@ func (s *sendStream) Close() error {
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *sendStream) CancelWrite(errorCode StreamErrorCode) {
|
func (s *sendStream) CancelWrite(errorCode protocol.ApplicationErrorCode) {
|
||||||
s.cancelWriteImpl(errorCode, fmt.Errorf("Write on stream %d canceled with error code %d", s.streamID, errorCode))
|
s.cancelWriteImpl(errorCode, fmt.Errorf("Write on stream %d canceled with error code %d", s.streamID, errorCode))
|
||||||
}
|
}
|
||||||
|
|
||||||
// must be called after locking the mutex
|
// must be called after locking the mutex
|
||||||
func (s *sendStream) cancelWriteImpl(errorCode qerr.StreamErrorCode, writeErr error) {
|
func (s *sendStream) cancelWriteImpl(errorCode protocol.ApplicationErrorCode, writeErr error) {
|
||||||
s.mutex.Lock()
|
s.mutex.Lock()
|
||||||
if s.canceledWrite {
|
if s.canceledWrite {
|
||||||
s.mutex.Unlock()
|
s.mutex.Unlock()
|
||||||
|
@ -449,10 +449,11 @@ func (s *sendStream) updateSendWindow(limit protocol.ByteCount) {
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *sendStream) handleStopSendingFrame(frame *wire.StopSendingFrame) {
|
func (s *sendStream) handleStopSendingFrame(frame *wire.StopSendingFrame) {
|
||||||
s.cancelWriteImpl(frame.ErrorCode, &StreamError{
|
writeErr := streamCanceledError{
|
||||||
StreamID: s.streamID,
|
errorCode: frame.ErrorCode,
|
||||||
ErrorCode: frame.ErrorCode,
|
error: fmt.Errorf("stream %d was reset with error code %d", s.streamID, frame.ErrorCode),
|
||||||
})
|
}
|
||||||
|
s.cancelWriteImpl(frame.ErrorCode, writeErr)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *sendStream) Context() context.Context {
|
func (s *sendStream) Context() context.Context {
|
||||||
|
|
|
@ -87,7 +87,6 @@ type baseServer struct {
|
||||||
*handshake.TokenGenerator,
|
*handshake.TokenGenerator,
|
||||||
bool, /* enable 0-RTT */
|
bool, /* enable 0-RTT */
|
||||||
logging.ConnectionTracer,
|
logging.ConnectionTracer,
|
||||||
uint64,
|
|
||||||
utils.Logger,
|
utils.Logger,
|
||||||
protocol.VersionNumber,
|
protocol.VersionNumber,
|
||||||
) quicSession
|
) quicSession
|
||||||
|
@ -451,7 +450,6 @@ func (s *baseServer) handleInitialImpl(p *receivedPacket, hdr *wire.Header) erro
|
||||||
}
|
}
|
||||||
s.logger.Debugf("Changing connection ID to %s.", connID)
|
s.logger.Debugf("Changing connection ID to %s.", connID)
|
||||||
var sess quicSession
|
var sess quicSession
|
||||||
tracingID := nextSessionTracingID()
|
|
||||||
if added := s.sessionHandler.AddWithConnID(hdr.DestConnectionID, connID, func() packetHandler {
|
if added := s.sessionHandler.AddWithConnID(hdr.DestConnectionID, connID, func() packetHandler {
|
||||||
var tracer logging.ConnectionTracer
|
var tracer logging.ConnectionTracer
|
||||||
if s.config.Tracer != nil {
|
if s.config.Tracer != nil {
|
||||||
|
@ -460,11 +458,7 @@ func (s *baseServer) handleInitialImpl(p *receivedPacket, hdr *wire.Header) erro
|
||||||
if origDestConnID.Len() > 0 {
|
if origDestConnID.Len() > 0 {
|
||||||
connID = origDestConnID
|
connID = origDestConnID
|
||||||
}
|
}
|
||||||
tracer = s.config.Tracer.TracerForConnection(
|
tracer = s.config.Tracer.TracerForConnection(protocol.PerspectiveServer, connID)
|
||||||
context.WithValue(context.Background(), SessionTracingKey, tracingID),
|
|
||||||
protocol.PerspectiveServer,
|
|
||||||
connID,
|
|
||||||
)
|
|
||||||
}
|
}
|
||||||
sess = s.newSession(
|
sess = s.newSession(
|
||||||
newSendConn(s.conn, p.remoteAddr, p.info),
|
newSendConn(s.conn, p.remoteAddr, p.info),
|
||||||
|
@ -480,7 +474,6 @@ func (s *baseServer) handleInitialImpl(p *receivedPacket, hdr *wire.Header) erro
|
||||||
s.tokenGenerator,
|
s.tokenGenerator,
|
||||||
s.acceptEarlySessions,
|
s.acceptEarlySessions,
|
||||||
tracer,
|
tracer,
|
||||||
tracingID,
|
|
||||||
s.logger,
|
s.logger,
|
||||||
hdr.Version,
|
hdr.Version,
|
||||||
)
|
)
|
||||||
|
@ -600,12 +593,12 @@ func (s *baseServer) sendConnectionRefused(remoteAddr net.Addr, hdr *wire.Header
|
||||||
}
|
}
|
||||||
|
|
||||||
// sendError sends the error as a response to the packet received with header hdr
|
// sendError sends the error as a response to the packet received with header hdr
|
||||||
func (s *baseServer) sendError(remoteAddr net.Addr, hdr *wire.Header, sealer handshake.LongHeaderSealer, errorCode qerr.TransportErrorCode, info *packetInfo) error {
|
func (s *baseServer) sendError(remoteAddr net.Addr, hdr *wire.Header, sealer handshake.LongHeaderSealer, errorCode qerr.ErrorCode, info *packetInfo) error {
|
||||||
packetBuffer := getPacketBuffer()
|
packetBuffer := getPacketBuffer()
|
||||||
defer packetBuffer.Release()
|
defer packetBuffer.Release()
|
||||||
buf := bytes.NewBuffer(packetBuffer.Data)
|
buf := bytes.NewBuffer(packetBuffer.Data)
|
||||||
|
|
||||||
ccf := &wire.ConnectionCloseFrame{ErrorCode: uint64(errorCode)}
|
ccf := &wire.ConnectionCloseFrame{ErrorCode: errorCode}
|
||||||
|
|
||||||
replyHdr := &wire.ExtendedHeader{}
|
replyHdr := &wire.ExtendedHeader{}
|
||||||
replyHdr.IsLongHeader = true
|
replyHdr.IsLongHeader = true
|
||||||
|
|
|
@ -10,7 +10,6 @@ import (
|
||||||
"net"
|
"net"
|
||||||
"reflect"
|
"reflect"
|
||||||
"sync"
|
"sync"
|
||||||
"sync/atomic"
|
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
"github.com/lucas-clemente/quic-go/internal/ackhandler"
|
"github.com/lucas-clemente/quic-go/internal/ackhandler"
|
||||||
|
@ -123,17 +122,23 @@ type errCloseForRecreating struct {
|
||||||
nextVersion protocol.VersionNumber
|
nextVersion protocol.VersionNumber
|
||||||
}
|
}
|
||||||
|
|
||||||
func (e *errCloseForRecreating) Error() string {
|
func (errCloseForRecreating) Error() string {
|
||||||
return "closing session in order to recreate it"
|
return "closing session in order to recreate it"
|
||||||
}
|
}
|
||||||
|
|
||||||
func (e *errCloseForRecreating) Is(target error) bool {
|
func (errCloseForRecreating) Is(target error) bool {
|
||||||
_, ok := target.(*errCloseForRecreating)
|
_, ok := target.(errCloseForRecreating)
|
||||||
return ok
|
return ok
|
||||||
}
|
}
|
||||||
|
|
||||||
var sessionTracingID uint64 // to be accessed atomically
|
type errVersionNegotiation struct {
|
||||||
func nextSessionTracingID() uint64 { return atomic.AddUint64(&sessionTracingID, 1) }
|
ourVersions []protocol.VersionNumber
|
||||||
|
theirVersions []protocol.VersionNumber
|
||||||
|
}
|
||||||
|
|
||||||
|
func (e errVersionNegotiation) Error() string {
|
||||||
|
return fmt.Sprintf("no compatible QUIC version found (we support %s, server offered %s)", e.ourVersions, e.theirVersions)
|
||||||
|
}
|
||||||
|
|
||||||
// A Session is a QUIC session
|
// A Session is a QUIC session
|
||||||
type session struct {
|
type session struct {
|
||||||
|
@ -247,7 +252,6 @@ var newSession = func(
|
||||||
tokenGenerator *handshake.TokenGenerator,
|
tokenGenerator *handshake.TokenGenerator,
|
||||||
enable0RTT bool,
|
enable0RTT bool,
|
||||||
tracer logging.ConnectionTracer,
|
tracer logging.ConnectionTracer,
|
||||||
tracingID uint64,
|
|
||||||
logger utils.Logger,
|
logger utils.Logger,
|
||||||
v protocol.VersionNumber,
|
v protocol.VersionNumber,
|
||||||
) quicSession {
|
) quicSession {
|
||||||
|
@ -287,10 +291,8 @@ var newSession = func(
|
||||||
s.version,
|
s.version,
|
||||||
)
|
)
|
||||||
s.preSetup()
|
s.preSetup()
|
||||||
s.ctx, s.ctxCancel = context.WithCancel(context.WithValue(context.Background(), SessionTracingKey, tracingID))
|
|
||||||
s.sentPacketHandler, s.receivedPacketHandler = ackhandler.NewAckHandler(
|
s.sentPacketHandler, s.receivedPacketHandler = ackhandler.NewAckHandler(
|
||||||
0,
|
0,
|
||||||
getMaxPacketSize(s.conn.RemoteAddr()),
|
|
||||||
s.rttStats,
|
s.rttStats,
|
||||||
s.perspective,
|
s.perspective,
|
||||||
s.tracer,
|
s.tracer,
|
||||||
|
@ -378,7 +380,6 @@ var newClientSession = func(
|
||||||
enable0RTT bool,
|
enable0RTT bool,
|
||||||
hasNegotiatedVersion bool,
|
hasNegotiatedVersion bool,
|
||||||
tracer logging.ConnectionTracer,
|
tracer logging.ConnectionTracer,
|
||||||
tracingID uint64,
|
|
||||||
logger utils.Logger,
|
logger utils.Logger,
|
||||||
v protocol.VersionNumber,
|
v protocol.VersionNumber,
|
||||||
) quicSession {
|
) quicSession {
|
||||||
|
@ -414,10 +415,8 @@ var newClientSession = func(
|
||||||
s.version,
|
s.version,
|
||||||
)
|
)
|
||||||
s.preSetup()
|
s.preSetup()
|
||||||
s.ctx, s.ctxCancel = context.WithCancel(context.WithValue(context.Background(), SessionTracingKey, tracingID))
|
|
||||||
s.sentPacketHandler, s.receivedPacketHandler = ackhandler.NewAckHandler(
|
s.sentPacketHandler, s.receivedPacketHandler = ackhandler.NewAckHandler(
|
||||||
initialPacketNumber,
|
initialPacketNumber,
|
||||||
getMaxPacketSize(s.conn.RemoteAddr()),
|
|
||||||
s.rttStats,
|
s.rttStats,
|
||||||
s.perspective,
|
s.perspective,
|
||||||
s.tracer,
|
s.tracer,
|
||||||
|
@ -523,6 +522,7 @@ func (s *session) preSetup() {
|
||||||
s.receivedPackets = make(chan *receivedPacket, protocol.MaxSessionUnprocessedPackets)
|
s.receivedPackets = make(chan *receivedPacket, protocol.MaxSessionUnprocessedPackets)
|
||||||
s.closeChan = make(chan closeError, 1)
|
s.closeChan = make(chan closeError, 1)
|
||||||
s.sendingScheduled = make(chan struct{}, 1)
|
s.sendingScheduled = make(chan struct{}, 1)
|
||||||
|
s.ctx, s.ctxCancel = context.WithCancel(context.Background())
|
||||||
s.handshakeCtx, s.handshakeCtxCancel = context.WithCancel(context.Background())
|
s.handshakeCtx, s.handshakeCtxCancel = context.WithCancel(context.Background())
|
||||||
|
|
||||||
now := time.Now()
|
now := time.Now()
|
||||||
|
@ -608,6 +608,7 @@ runLoop:
|
||||||
// nothing to see here.
|
// nothing to see here.
|
||||||
case <-sendQueueAvailable:
|
case <-sendQueueAvailable:
|
||||||
case firstPacket := <-s.receivedPackets:
|
case firstPacket := <-s.receivedPackets:
|
||||||
|
s.sentPacketHandler.ReceivedBytes(firstPacket.Size())
|
||||||
wasProcessed := s.handlePacketImpl(firstPacket)
|
wasProcessed := s.handlePacketImpl(firstPacket)
|
||||||
// Don't set timers and send packets if the packet made us close the session.
|
// Don't set timers and send packets if the packet made us close the session.
|
||||||
select {
|
select {
|
||||||
|
@ -663,13 +664,19 @@ runLoop:
|
||||||
s.framer.QueueControlFrame(&wire.PingFrame{})
|
s.framer.QueueControlFrame(&wire.PingFrame{})
|
||||||
s.keepAlivePingSent = true
|
s.keepAlivePingSent = true
|
||||||
} else if !s.handshakeComplete && now.Sub(s.sessionCreationTime) >= s.config.handshakeTimeout() {
|
} else if !s.handshakeComplete && now.Sub(s.sessionCreationTime) >= s.config.handshakeTimeout() {
|
||||||
s.destroyImpl(qerr.ErrHandshakeTimeout)
|
if s.tracer != nil {
|
||||||
|
s.tracer.ClosedConnection(logging.NewTimeoutCloseReason(logging.TimeoutReasonHandshake))
|
||||||
|
}
|
||||||
|
s.destroyImpl(qerr.NewTimeoutError("Handshake did not complete in time"))
|
||||||
continue
|
continue
|
||||||
} else {
|
} else {
|
||||||
idleTimeoutStartTime := s.idleTimeoutStartTime()
|
idleTimeoutStartTime := s.idleTimeoutStartTime()
|
||||||
if (!s.handshakeComplete && now.Sub(idleTimeoutStartTime) >= s.config.HandshakeIdleTimeout) ||
|
if (!s.handshakeComplete && now.Sub(idleTimeoutStartTime) >= s.config.HandshakeIdleTimeout) ||
|
||||||
(s.handshakeComplete && now.Sub(idleTimeoutStartTime) >= s.idleTimeout) {
|
(s.handshakeComplete && now.Sub(idleTimeoutStartTime) >= s.idleTimeout) {
|
||||||
s.destroyImpl(qerr.ErrIdleTimeout)
|
if s.tracer != nil {
|
||||||
|
s.tracer.ClosedConnection(logging.NewTimeoutCloseReason(logging.TimeoutReasonIdle))
|
||||||
|
}
|
||||||
|
s.destroyImpl(qerr.NewTimeoutError("No recent network activity"))
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -690,8 +697,8 @@ runLoop:
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
s.handleCloseError(&closeErr)
|
s.handleCloseError(closeErr)
|
||||||
if !errors.Is(closeErr.err, &errCloseForRecreating{}) && s.tracer != nil {
|
if !errors.Is(closeErr.err, errCloseForRecreating{}) && s.tracer != nil {
|
||||||
s.tracer.Close()
|
s.tracer.Close()
|
||||||
}
|
}
|
||||||
s.logger.Infof("Connection %s closed.", s.logID)
|
s.logger.Infof("Connection %s closed.", s.logID)
|
||||||
|
@ -731,28 +738,25 @@ func (s *session) nextKeepAliveTime() time.Time {
|
||||||
if !s.config.KeepAlive || s.keepAlivePingSent || !s.firstAckElicitingPacketAfterIdleSentTime.IsZero() {
|
if !s.config.KeepAlive || s.keepAlivePingSent || !s.firstAckElicitingPacketAfterIdleSentTime.IsZero() {
|
||||||
return time.Time{}
|
return time.Time{}
|
||||||
}
|
}
|
||||||
return s.lastPacketReceivedTime.Add(s.keepAliveInterval)
|
return s.lastPacketReceivedTime.Add(s.keepAliveInterval / 2)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *session) maybeResetTimer() {
|
func (s *session) maybeResetTimer() {
|
||||||
var deadline time.Time
|
var deadline time.Time
|
||||||
if !s.handshakeComplete {
|
if !s.handshakeComplete {
|
||||||
deadline = utils.MinTime(
|
deadline = s.sessionCreationTime.Add(utils.MinDuration(s.config.handshakeTimeout(), s.config.HandshakeIdleTimeout))
|
||||||
s.sessionCreationTime.Add(s.config.handshakeTimeout()),
|
|
||||||
s.idleTimeoutStartTime().Add(s.config.HandshakeIdleTimeout),
|
|
||||||
)
|
|
||||||
} else {
|
} else {
|
||||||
if keepAliveTime := s.nextKeepAliveTime(); !keepAliveTime.IsZero() {
|
if keepAliveTime := s.nextKeepAliveTime(); !keepAliveTime.IsZero() {
|
||||||
deadline = keepAliveTime
|
deadline = keepAliveTime
|
||||||
} else {
|
} else {
|
||||||
deadline = s.idleTimeoutStartTime().Add(s.idleTimeout)
|
deadline = s.idleTimeoutStartTime().Add(s.idleTimeout)
|
||||||
}
|
}
|
||||||
}
|
if !s.config.DisablePathMTUDiscovery {
|
||||||
if s.handshakeConfirmed && !s.config.DisablePathMTUDiscovery {
|
|
||||||
if probeTime := s.mtuDiscoverer.NextProbeTime(); !probeTime.IsZero() {
|
if probeTime := s.mtuDiscoverer.NextProbeTime(); !probeTime.IsZero() {
|
||||||
deadline = utils.MinTime(deadline, probeTime)
|
deadline = utils.MinTime(deadline, probeTime)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if ackAlarm := s.receivedPacketHandler.GetAlarmTimeout(); !ackAlarm.IsZero() {
|
if ackAlarm := s.receivedPacketHandler.GetAlarmTimeout(); !ackAlarm.IsZero() {
|
||||||
deadline = utils.MinTime(deadline, ackAlarm)
|
deadline = utils.MinTime(deadline, ackAlarm)
|
||||||
|
@ -782,36 +786,6 @@ func (s *session) handleHandshakeComplete() {
|
||||||
s.connIDManager.SetHandshakeComplete()
|
s.connIDManager.SetHandshakeComplete()
|
||||||
s.connIDGenerator.SetHandshakeComplete()
|
s.connIDGenerator.SetHandshakeComplete()
|
||||||
|
|
||||||
if s.perspective == protocol.PerspectiveClient {
|
|
||||||
s.applyTransportParameters()
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
s.handleHandshakeConfirmed()
|
|
||||||
|
|
||||||
ticket, err := s.cryptoStreamHandler.GetSessionTicket()
|
|
||||||
if err != nil {
|
|
||||||
s.closeLocal(err)
|
|
||||||
}
|
|
||||||
if ticket != nil {
|
|
||||||
s.oneRTTStream.Write(ticket)
|
|
||||||
for s.oneRTTStream.HasData() {
|
|
||||||
s.queueControlFrame(s.oneRTTStream.PopCryptoFrame(protocol.MaxPostHandshakeCryptoFrameSize))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
token, err := s.tokenGenerator.NewToken(s.conn.RemoteAddr())
|
|
||||||
if err != nil {
|
|
||||||
s.closeLocal(err)
|
|
||||||
}
|
|
||||||
s.queueControlFrame(&wire.NewTokenFrame{Token: token})
|
|
||||||
s.queueControlFrame(&wire.HandshakeDoneFrame{})
|
|
||||||
}
|
|
||||||
|
|
||||||
func (s *session) handleHandshakeConfirmed() {
|
|
||||||
s.handshakeConfirmed = true
|
|
||||||
s.sentPacketHandler.SetHandshakeConfirmed()
|
|
||||||
s.cryptoStreamHandler.SetHandshakeConfirmed()
|
|
||||||
|
|
||||||
if !s.config.DisablePathMTUDiscovery {
|
if !s.config.DisablePathMTUDiscovery {
|
||||||
maxPacketSize := s.peerParams.MaxUDPPayloadSize
|
maxPacketSize := s.peerParams.MaxUDPPayloadSize
|
||||||
if maxPacketSize == 0 {
|
if maxPacketSize == 0 {
|
||||||
|
@ -828,11 +802,34 @@ func (s *session) handleHandshakeConfirmed() {
|
||||||
},
|
},
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if s.perspective == protocol.PerspectiveClient {
|
||||||
|
s.applyTransportParameters()
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
s.handshakeConfirmed = true
|
||||||
|
s.sentPacketHandler.SetHandshakeConfirmed()
|
||||||
|
ticket, err := s.cryptoStreamHandler.GetSessionTicket()
|
||||||
|
if err != nil {
|
||||||
|
s.closeLocal(err)
|
||||||
|
}
|
||||||
|
if ticket != nil {
|
||||||
|
s.oneRTTStream.Write(ticket)
|
||||||
|
for s.oneRTTStream.HasData() {
|
||||||
|
s.queueControlFrame(s.oneRTTStream.PopCryptoFrame(protocol.MaxPostHandshakeCryptoFrameSize))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
token, err := s.tokenGenerator.NewToken(s.conn.RemoteAddr())
|
||||||
|
if err != nil {
|
||||||
|
s.closeLocal(err)
|
||||||
|
}
|
||||||
|
s.queueControlFrame(&wire.NewTokenFrame{Token: token})
|
||||||
|
s.cryptoStreamHandler.SetHandshakeConfirmed()
|
||||||
|
s.queueControlFrame(&wire.HandshakeDoneFrame{})
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *session) handlePacketImpl(rp *receivedPacket) bool {
|
func (s *session) handlePacketImpl(rp *receivedPacket) bool {
|
||||||
s.sentPacketHandler.ReceivedBytes(rp.Size())
|
|
||||||
|
|
||||||
if wire.IsVersionNegotiationPacket(rp.data) {
|
if wire.IsVersionNegotiationPacket(rp.data) {
|
||||||
s.handleVersionNegotiationPacket(rp)
|
s.handleVersionNegotiationPacket(rp)
|
||||||
return false
|
return false
|
||||||
|
@ -943,10 +940,7 @@ func (s *session) handleSinglePacket(p *receivedPacket, hdr *wire.Header) bool /
|
||||||
wasQueued = true
|
wasQueued = true
|
||||||
s.tryQueueingUndecryptablePacket(p, hdr)
|
s.tryQueueingUndecryptablePacket(p, hdr)
|
||||||
case wire.ErrInvalidReservedBits:
|
case wire.ErrInvalidReservedBits:
|
||||||
s.closeLocal(&qerr.TransportError{
|
s.closeLocal(qerr.NewError(qerr.ProtocolViolation, err.Error()))
|
||||||
ErrorCode: qerr.ProtocolViolation,
|
|
||||||
ErrorMessage: err.Error(),
|
|
||||||
})
|
|
||||||
case handshake.ErrDecryptionFailed:
|
case handshake.ErrDecryptionFailed:
|
||||||
// This might be a packet injected by an attacker. Drop it.
|
// This might be a packet injected by an attacker. Drop it.
|
||||||
if s.tracer != nil {
|
if s.tracer != nil {
|
||||||
|
@ -1087,16 +1081,13 @@ func (s *session) handleVersionNegotiationPacket(p *receivedPacket) {
|
||||||
}
|
}
|
||||||
newVersion, ok := protocol.ChooseSupportedVersion(s.config.Versions, supportedVersions)
|
newVersion, ok := protocol.ChooseSupportedVersion(s.config.Versions, supportedVersions)
|
||||||
if !ok {
|
if !ok {
|
||||||
s.destroyImpl(&VersionNegotiationError{
|
s.destroyImpl(errVersionNegotiation{
|
||||||
Ours: s.config.Versions,
|
ourVersions: s.config.Versions,
|
||||||
Theirs: supportedVersions,
|
theirVersions: supportedVersions,
|
||||||
})
|
})
|
||||||
s.logger.Infof("No compatible QUIC version found.")
|
s.logger.Infof("No compatible QUIC version found.")
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
if s.tracer != nil {
|
|
||||||
s.tracer.NegotiatedVersion(newVersion, s.config.Versions, supportedVersions)
|
|
||||||
}
|
|
||||||
|
|
||||||
s.logger.Infof("Switching to QUIC version %s.", newVersion)
|
s.logger.Infof("Switching to QUIC version %s.", newVersion)
|
||||||
nextPN, _ := s.sentPacketHandler.PeekPacketNumber(protocol.EncryptionInitial)
|
nextPN, _ := s.sentPacketHandler.PeekPacketNumber(protocol.EncryptionInitial)
|
||||||
|
@ -1113,24 +1104,11 @@ func (s *session) handleUnpackedPacket(
|
||||||
packetSize protocol.ByteCount, // only for logging
|
packetSize protocol.ByteCount, // only for logging
|
||||||
) error {
|
) error {
|
||||||
if len(packet.data) == 0 {
|
if len(packet.data) == 0 {
|
||||||
return &qerr.TransportError{
|
return qerr.NewError(qerr.ProtocolViolation, "empty packet")
|
||||||
ErrorCode: qerr.ProtocolViolation,
|
|
||||||
ErrorMessage: "empty packet",
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if !s.receivedFirstPacket {
|
if !s.receivedFirstPacket {
|
||||||
s.receivedFirstPacket = true
|
s.receivedFirstPacket = true
|
||||||
if !s.versionNegotiated && s.tracer != nil {
|
|
||||||
var clientVersions, serverVersions []protocol.VersionNumber
|
|
||||||
switch s.perspective {
|
|
||||||
case protocol.PerspectiveClient:
|
|
||||||
clientVersions = s.config.Versions
|
|
||||||
case protocol.PerspectiveServer:
|
|
||||||
serverVersions = s.config.Versions
|
|
||||||
}
|
|
||||||
s.tracer.NegotiatedVersion(s.version, clientVersions, serverVersions)
|
|
||||||
}
|
|
||||||
// The server can change the source connection ID with the first Handshake packet.
|
// The server can change the source connection ID with the first Handshake packet.
|
||||||
if s.perspective == protocol.PerspectiveClient && packet.hdr.IsLongHeader && !packet.hdr.SrcConnectionID.Equal(s.handshakeDestConnID) {
|
if s.perspective == protocol.PerspectiveClient && packet.hdr.IsLongHeader && !packet.hdr.SrcConnectionID.Equal(s.handshakeDestConnID) {
|
||||||
cid := packet.hdr.SrcConnectionID
|
cid := packet.hdr.SrcConnectionID
|
||||||
|
@ -1152,6 +1130,7 @@ func (s *session) handleUnpackedPacket(
|
||||||
s.tracer.StartedConnection(
|
s.tracer.StartedConnection(
|
||||||
s.conn.LocalAddr(),
|
s.conn.LocalAddr(),
|
||||||
s.conn.RemoteAddr(),
|
s.conn.RemoteAddr(),
|
||||||
|
s.version,
|
||||||
packet.hdr.SrcConnectionID,
|
packet.hdr.SrcConnectionID,
|
||||||
packet.hdr.DestConnectionID,
|
packet.hdr.DestConnectionID,
|
||||||
)
|
)
|
||||||
|
@ -1267,20 +1246,13 @@ func (s *session) handlePacket(p *receivedPacket) {
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *session) handleConnectionCloseFrame(frame *wire.ConnectionCloseFrame) {
|
func (s *session) handleConnectionCloseFrame(frame *wire.ConnectionCloseFrame) {
|
||||||
|
var e error
|
||||||
if frame.IsApplicationError {
|
if frame.IsApplicationError {
|
||||||
s.closeRemote(&qerr.ApplicationError{
|
e = qerr.NewApplicationError(frame.ErrorCode, frame.ReasonPhrase)
|
||||||
Remote: true,
|
} else {
|
||||||
ErrorCode: qerr.ApplicationErrorCode(frame.ErrorCode),
|
e = qerr.NewError(frame.ErrorCode, frame.ReasonPhrase)
|
||||||
ErrorMessage: frame.ReasonPhrase,
|
|
||||||
})
|
|
||||||
return
|
|
||||||
}
|
}
|
||||||
s.closeRemote(&qerr.TransportError{
|
s.closeRemote(e)
|
||||||
Remote: true,
|
|
||||||
ErrorCode: qerr.TransportErrorCode(frame.ErrorCode),
|
|
||||||
FrameType: frame.FrameType,
|
|
||||||
ErrorMessage: frame.ReasonPhrase,
|
|
||||||
})
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *session) handleCryptoFrame(frame *wire.CryptoFrame, encLevel protocol.EncryptionLevel) error {
|
func (s *session) handleCryptoFrame(frame *wire.CryptoFrame, encLevel protocol.EncryptionLevel) error {
|
||||||
|
@ -1361,10 +1333,7 @@ func (s *session) handlePathChallengeFrame(frame *wire.PathChallengeFrame) {
|
||||||
|
|
||||||
func (s *session) handleNewTokenFrame(frame *wire.NewTokenFrame) error {
|
func (s *session) handleNewTokenFrame(frame *wire.NewTokenFrame) error {
|
||||||
if s.perspective == protocol.PerspectiveServer {
|
if s.perspective == protocol.PerspectiveServer {
|
||||||
return &qerr.TransportError{
|
return qerr.NewError(qerr.ProtocolViolation, "Received NEW_TOKEN frame from the client.")
|
||||||
ErrorCode: qerr.ProtocolViolation,
|
|
||||||
ErrorMessage: "received NEW_TOKEN frame from the client",
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
if s.config.TokenStore != nil {
|
if s.config.TokenStore != nil {
|
||||||
s.config.TokenStore.Put(s.tokenStoreKey, &ClientToken{data: frame.Token})
|
s.config.TokenStore.Put(s.tokenStoreKey, &ClientToken{data: frame.Token})
|
||||||
|
@ -1382,37 +1351,27 @@ func (s *session) handleRetireConnectionIDFrame(f *wire.RetireConnectionIDFrame,
|
||||||
|
|
||||||
func (s *session) handleHandshakeDoneFrame() error {
|
func (s *session) handleHandshakeDoneFrame() error {
|
||||||
if s.perspective == protocol.PerspectiveServer {
|
if s.perspective == protocol.PerspectiveServer {
|
||||||
return &qerr.TransportError{
|
return qerr.NewError(qerr.ProtocolViolation, "received a HANDSHAKE_DONE frame")
|
||||||
ErrorCode: qerr.ProtocolViolation,
|
|
||||||
ErrorMessage: "received a HANDSHAKE_DONE frame",
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if !s.handshakeConfirmed {
|
|
||||||
s.handleHandshakeConfirmed()
|
|
||||||
}
|
}
|
||||||
|
s.handshakeConfirmed = true
|
||||||
|
s.sentPacketHandler.SetHandshakeConfirmed()
|
||||||
|
s.cryptoStreamHandler.SetHandshakeConfirmed()
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *session) handleAckFrame(frame *wire.AckFrame, encLevel protocol.EncryptionLevel) error {
|
func (s *session) handleAckFrame(frame *wire.AckFrame, encLevel protocol.EncryptionLevel) error {
|
||||||
acked1RTTPacket, err := s.sentPacketHandler.ReceivedAck(frame, encLevel, s.lastPacketReceivedTime)
|
if err := s.sentPacketHandler.ReceivedAck(frame, encLevel, s.lastPacketReceivedTime); err != nil {
|
||||||
if err != nil {
|
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
if !acked1RTTPacket {
|
if encLevel != protocol.Encryption1RTT {
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
if s.perspective == protocol.PerspectiveClient && !s.handshakeConfirmed {
|
|
||||||
s.handleHandshakeConfirmed()
|
|
||||||
}
|
|
||||||
return s.cryptoStreamHandler.SetLargest1RTTAcked(frame.LargestAcked())
|
return s.cryptoStreamHandler.SetLargest1RTTAcked(frame.LargestAcked())
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *session) handleDatagramFrame(f *wire.DatagramFrame) error {
|
func (s *session) handleDatagramFrame(f *wire.DatagramFrame) error {
|
||||||
if f.Length(s.version) > protocol.MaxDatagramFrameSize {
|
if f.Length(s.version) > protocol.MaxDatagramFrameSize {
|
||||||
return &qerr.TransportError{
|
return qerr.NewError(qerr.ProtocolViolation, "DATAGRAM frame too large")
|
||||||
ErrorCode: qerr.ProtocolViolation,
|
|
||||||
ErrorMessage: "DATAGRAM frame too large",
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
s.datagramQueue.HandleDatagramFrame(f)
|
s.datagramQueue.HandleDatagramFrame(f)
|
||||||
return nil
|
return nil
|
||||||
|
@ -1461,48 +1420,44 @@ func (s *session) shutdown() {
|
||||||
<-s.ctx.Done()
|
<-s.ctx.Done()
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *session) CloseWithError(code ApplicationErrorCode, desc string) error {
|
func (s *session) CloseWithError(code protocol.ApplicationErrorCode, desc string) error {
|
||||||
s.closeLocal(&qerr.ApplicationError{
|
s.closeLocal(qerr.NewApplicationError(qerr.ErrorCode(code), desc))
|
||||||
ErrorCode: code,
|
|
||||||
ErrorMessage: desc,
|
|
||||||
})
|
|
||||||
<-s.ctx.Done()
|
<-s.ctx.Done()
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *session) handleCloseError(closeErr *closeError) {
|
func (s *session) handleCloseError(closeErr closeError) {
|
||||||
e := closeErr.err
|
if closeErr.err == nil {
|
||||||
if e == nil {
|
closeErr.err = qerr.NewApplicationError(0, "")
|
||||||
e = &qerr.ApplicationError{}
|
|
||||||
} else {
|
|
||||||
defer func() {
|
|
||||||
closeErr.err = e
|
|
||||||
}()
|
|
||||||
}
|
}
|
||||||
|
|
||||||
switch {
|
var quicErr *qerr.QuicError
|
||||||
case errors.Is(e, qerr.ErrIdleTimeout),
|
var ok bool
|
||||||
errors.Is(e, qerr.ErrHandshakeTimeout),
|
if quicErr, ok = closeErr.err.(*qerr.QuicError); !ok {
|
||||||
errors.Is(e, &StatelessResetError{}),
|
quicErr = qerr.ToQuicError(closeErr.err)
|
||||||
errors.Is(e, &VersionNegotiationError{}),
|
|
||||||
errors.Is(e, &errCloseForRecreating{}),
|
|
||||||
errors.Is(e, &qerr.ApplicationError{}),
|
|
||||||
errors.Is(e, &qerr.TransportError{}):
|
|
||||||
default:
|
|
||||||
e = &qerr.TransportError{
|
|
||||||
ErrorCode: qerr.InternalError,
|
|
||||||
ErrorMessage: e.Error(),
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
s.streamsMap.CloseWithError(e)
|
s.streamsMap.CloseWithError(quicErr)
|
||||||
s.connIDManager.Close()
|
s.connIDManager.Close()
|
||||||
if s.datagramQueue != nil {
|
if s.datagramQueue != nil {
|
||||||
s.datagramQueue.CloseWithError(e)
|
s.datagramQueue.CloseWithError(quicErr)
|
||||||
}
|
}
|
||||||
|
|
||||||
if s.tracer != nil && !errors.Is(e, &errCloseForRecreating{}) {
|
if s.tracer != nil {
|
||||||
s.tracer.ClosedConnection(e)
|
// timeout errors are logged as soon as they occur (to distinguish between handshake and idle timeouts)
|
||||||
|
if nerr, ok := closeErr.err.(net.Error); !ok || !nerr.Timeout() {
|
||||||
|
var resetErr statelessResetErr
|
||||||
|
var vnErr errVersionNegotiation
|
||||||
|
if errors.As(closeErr.err, &resetErr) {
|
||||||
|
s.tracer.ClosedConnection(logging.NewStatelessResetCloseReason(resetErr.token))
|
||||||
|
} else if errors.As(closeErr.err, &vnErr) {
|
||||||
|
s.tracer.ClosedConnection(logging.NewVersionNegotiationError(vnErr.theirVersions))
|
||||||
|
} else if quicErr.IsApplicationError() {
|
||||||
|
s.tracer.ClosedConnection(logging.NewApplicationCloseReason(quicErr.ErrorCode, closeErr.remote))
|
||||||
|
} else {
|
||||||
|
s.tracer.ClosedConnection(logging.NewTransportCloseReason(quicErr.ErrorCode, closeErr.remote))
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// If this is a remote close we're done here
|
// If this is a remote close we're done here
|
||||||
|
@ -1514,7 +1469,7 @@ func (s *session) handleCloseError(closeErr *closeError) {
|
||||||
s.connIDGenerator.RemoveAll()
|
s.connIDGenerator.RemoveAll()
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
connClosePacket, err := s.sendConnectionClose(e)
|
connClosePacket, err := s.sendConnectionClose(quicErr)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
s.logger.Debugf("Error sending CONNECTION_CLOSE: %s", err)
|
s.logger.Debugf("Error sending CONNECTION_CLOSE: %s", err)
|
||||||
}
|
}
|
||||||
|
@ -1553,10 +1508,7 @@ func (s *session) restoreTransportParameters(params *wire.TransportParameters) {
|
||||||
|
|
||||||
func (s *session) handleTransportParameters(params *wire.TransportParameters) {
|
func (s *session) handleTransportParameters(params *wire.TransportParameters) {
|
||||||
if err := s.checkTransportParameters(params); err != nil {
|
if err := s.checkTransportParameters(params); err != nil {
|
||||||
s.closeLocal(&qerr.TransportError{
|
s.closeLocal(err)
|
||||||
ErrorCode: qerr.TransportParameterError,
|
|
||||||
ErrorMessage: err.Error(),
|
|
||||||
})
|
|
||||||
}
|
}
|
||||||
s.peerParams = params
|
s.peerParams = params
|
||||||
// On the client side we have to wait for handshake completion.
|
// On the client side we have to wait for handshake completion.
|
||||||
|
@ -1579,7 +1531,7 @@ func (s *session) checkTransportParameters(params *wire.TransportParameters) err
|
||||||
|
|
||||||
// check the initial_source_connection_id
|
// check the initial_source_connection_id
|
||||||
if !params.InitialSourceConnectionID.Equal(s.handshakeDestConnID) {
|
if !params.InitialSourceConnectionID.Equal(s.handshakeDestConnID) {
|
||||||
return fmt.Errorf("expected initial_source_connection_id to equal %s, is %s", s.handshakeDestConnID, params.InitialSourceConnectionID)
|
return qerr.NewError(qerr.TransportParameterError, fmt.Sprintf("expected initial_source_connection_id to equal %s, is %s", s.handshakeDestConnID, params.InitialSourceConnectionID))
|
||||||
}
|
}
|
||||||
|
|
||||||
if s.perspective == protocol.PerspectiveServer {
|
if s.perspective == protocol.PerspectiveServer {
|
||||||
|
@ -1587,17 +1539,17 @@ func (s *session) checkTransportParameters(params *wire.TransportParameters) err
|
||||||
}
|
}
|
||||||
// check the original_destination_connection_id
|
// check the original_destination_connection_id
|
||||||
if !params.OriginalDestinationConnectionID.Equal(s.origDestConnID) {
|
if !params.OriginalDestinationConnectionID.Equal(s.origDestConnID) {
|
||||||
return fmt.Errorf("expected original_destination_connection_id to equal %s, is %s", s.origDestConnID, params.OriginalDestinationConnectionID)
|
return qerr.NewError(qerr.TransportParameterError, fmt.Sprintf("expected original_destination_connection_id to equal %s, is %s", s.origDestConnID, params.OriginalDestinationConnectionID))
|
||||||
}
|
}
|
||||||
if s.retrySrcConnID != nil { // a Retry was performed
|
if s.retrySrcConnID != nil { // a Retry was performed
|
||||||
if params.RetrySourceConnectionID == nil {
|
if params.RetrySourceConnectionID == nil {
|
||||||
return errors.New("missing retry_source_connection_id")
|
return qerr.NewError(qerr.TransportParameterError, "missing retry_source_connection_id")
|
||||||
}
|
}
|
||||||
if !(*params.RetrySourceConnectionID).Equal(*s.retrySrcConnID) {
|
if !(*params.RetrySourceConnectionID).Equal(*s.retrySrcConnID) {
|
||||||
return fmt.Errorf("expected retry_source_connection_id to equal %s, is %s", s.retrySrcConnID, *params.RetrySourceConnectionID)
|
return qerr.NewError(qerr.TransportParameterError, fmt.Sprintf("expected retry_source_connection_id to equal %s, is %s", s.retrySrcConnID, *params.RetrySourceConnectionID))
|
||||||
}
|
}
|
||||||
} else if params.RetrySourceConnectionID != nil {
|
} else if params.RetrySourceConnectionID != nil {
|
||||||
return errors.New("received retry_source_connection_id, although no Retry was performed")
|
return qerr.NewError(qerr.TransportParameterError, "received retry_source_connection_id, although no Retry was performed")
|
||||||
}
|
}
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
@ -1766,7 +1718,7 @@ func (s *session) sendPacket() (bool, error) {
|
||||||
s.sendQueue.Send(packet.buffer)
|
s.sendQueue.Send(packet.buffer)
|
||||||
return true, nil
|
return true, nil
|
||||||
}
|
}
|
||||||
if !s.config.DisablePathMTUDiscovery && s.mtuDiscoverer.ShouldSendProbe(now) {
|
if !s.config.DisablePathMTUDiscovery && s.handshakeComplete && s.mtuDiscoverer.ShouldSendProbe(now) {
|
||||||
packet, err := s.packer.PackMTUProbePacket(s.mtuDiscoverer.GetPing())
|
packet, err := s.packer.PackMTUProbePacket(s.mtuDiscoverer.GetPing())
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return false, err
|
return false, err
|
||||||
|
@ -1792,21 +1744,8 @@ func (s *session) sendPackedPacket(packet *packedPacket, now time.Time) {
|
||||||
s.sendQueue.Send(packet.buffer)
|
s.sendQueue.Send(packet.buffer)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *session) sendConnectionClose(e error) ([]byte, error) {
|
func (s *session) sendConnectionClose(quicErr *qerr.QuicError) ([]byte, error) {
|
||||||
var packet *coalescedPacket
|
packet, err := s.packer.PackConnectionClose(quicErr)
|
||||||
var err error
|
|
||||||
var transportErr *qerr.TransportError
|
|
||||||
var applicationErr *qerr.ApplicationError
|
|
||||||
if errors.As(e, &transportErr) {
|
|
||||||
packet, err = s.packer.PackConnectionClose(transportErr)
|
|
||||||
} else if errors.As(e, &applicationErr) {
|
|
||||||
packet, err = s.packer.PackApplicationClose(applicationErr)
|
|
||||||
} else {
|
|
||||||
packet, err = s.packer.PackConnectionClose(&qerr.TransportError{
|
|
||||||
ErrorCode: qerr.InternalError,
|
|
||||||
ErrorMessage: fmt.Sprintf("session BUG: unspecified error type (msg: %s)", e.Error()),
|
|
||||||
})
|
|
||||||
}
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
|
|
@ -86,6 +86,16 @@ type stream struct {
|
||||||
|
|
||||||
var _ Stream = &stream{}
|
var _ Stream = &stream{}
|
||||||
|
|
||||||
|
type streamCanceledError struct {
|
||||||
|
error
|
||||||
|
errorCode protocol.ApplicationErrorCode
|
||||||
|
}
|
||||||
|
|
||||||
|
func (streamCanceledError) Canceled() bool { return true }
|
||||||
|
func (e streamCanceledError) ErrorCode() protocol.ApplicationErrorCode { return e.errorCode }
|
||||||
|
|
||||||
|
var _ StreamError = &streamCanceledError{}
|
||||||
|
|
||||||
// newStream creates a new Stream
|
// newStream creates a new Stream
|
||||||
func newStream(streamID protocol.StreamID,
|
func newStream(streamID protocol.StreamID,
|
||||||
sender streamSender,
|
sender streamSender,
|
||||||
|
|
|
@ -209,10 +209,7 @@ func (m *streamsMap) DeleteStream(id protocol.StreamID) error {
|
||||||
func (m *streamsMap) GetOrOpenReceiveStream(id protocol.StreamID) (receiveStreamI, error) {
|
func (m *streamsMap) GetOrOpenReceiveStream(id protocol.StreamID) (receiveStreamI, error) {
|
||||||
str, err := m.getOrOpenReceiveStream(id)
|
str, err := m.getOrOpenReceiveStream(id)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, &qerr.TransportError{
|
return nil, qerr.NewError(qerr.StreamStateError, err.Error())
|
||||||
ErrorCode: qerr.StreamStateError,
|
|
||||||
ErrorMessage: err.Error(),
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
return str, nil
|
return str, nil
|
||||||
}
|
}
|
||||||
|
@ -243,10 +240,7 @@ func (m *streamsMap) getOrOpenReceiveStream(id protocol.StreamID) (receiveStream
|
||||||
func (m *streamsMap) GetOrOpenSendStream(id protocol.StreamID) (sendStreamI, error) {
|
func (m *streamsMap) GetOrOpenSendStream(id protocol.StreamID) (sendStreamI, error) {
|
||||||
str, err := m.getOrOpenSendStream(id)
|
str, err := m.getOrOpenSendStream(id)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, &qerr.TransportError{
|
return nil, qerr.NewError(qerr.StreamStateError, err.Error())
|
||||||
ErrorCode: qerr.StreamStateError,
|
|
||||||
ErrorMessage: err.Error(),
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
return str, nil
|
return str, nil
|
||||||
}
|
}
|
||||||
|
|
|
@ -145,7 +145,7 @@ func (m *incomingBidiStreamsMap) DeleteStream(num protocol.StreamNum) error {
|
||||||
func (m *incomingBidiStreamsMap) deleteStream(num protocol.StreamNum) error {
|
func (m *incomingBidiStreamsMap) deleteStream(num protocol.StreamNum) error {
|
||||||
if _, ok := m.streams[num]; !ok {
|
if _, ok := m.streams[num]; !ok {
|
||||||
return streamError{
|
return streamError{
|
||||||
message: "tried to delete unknown incoming stream %d",
|
message: "Tried to delete unknown incoming stream %d",
|
||||||
nums: []protocol.StreamNum{num},
|
nums: []protocol.StreamNum{num},
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -156,7 +156,7 @@ func (m *incomingBidiStreamsMap) deleteStream(num protocol.StreamNum) error {
|
||||||
entry, ok := m.streams[num]
|
entry, ok := m.streams[num]
|
||||||
if ok && entry.shouldDelete {
|
if ok && entry.shouldDelete {
|
||||||
return streamError{
|
return streamError{
|
||||||
message: "tried to delete incoming stream %d multiple times",
|
message: "Tried to delete incoming stream %d multiple times",
|
||||||
nums: []protocol.StreamNum{num},
|
nums: []protocol.StreamNum{num},
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -143,7 +143,7 @@ func (m *incomingItemsMap) DeleteStream(num protocol.StreamNum) error {
|
||||||
func (m *incomingItemsMap) deleteStream(num protocol.StreamNum) error {
|
func (m *incomingItemsMap) deleteStream(num protocol.StreamNum) error {
|
||||||
if _, ok := m.streams[num]; !ok {
|
if _, ok := m.streams[num]; !ok {
|
||||||
return streamError{
|
return streamError{
|
||||||
message: "tried to delete unknown incoming stream %d",
|
message: "Tried to delete unknown incoming stream %d",
|
||||||
nums: []protocol.StreamNum{num},
|
nums: []protocol.StreamNum{num},
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -154,7 +154,7 @@ func (m *incomingItemsMap) deleteStream(num protocol.StreamNum) error {
|
||||||
entry, ok := m.streams[num]
|
entry, ok := m.streams[num]
|
||||||
if ok && entry.shouldDelete {
|
if ok && entry.shouldDelete {
|
||||||
return streamError{
|
return streamError{
|
||||||
message: "tried to delete incoming stream %d multiple times",
|
message: "Tried to delete incoming stream %d multiple times",
|
||||||
nums: []protocol.StreamNum{num},
|
nums: []protocol.StreamNum{num},
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -145,7 +145,7 @@ func (m *incomingUniStreamsMap) DeleteStream(num protocol.StreamNum) error {
|
||||||
func (m *incomingUniStreamsMap) deleteStream(num protocol.StreamNum) error {
|
func (m *incomingUniStreamsMap) deleteStream(num protocol.StreamNum) error {
|
||||||
if _, ok := m.streams[num]; !ok {
|
if _, ok := m.streams[num]; !ok {
|
||||||
return streamError{
|
return streamError{
|
||||||
message: "tried to delete unknown incoming stream %d",
|
message: "Tried to delete unknown incoming stream %d",
|
||||||
nums: []protocol.StreamNum{num},
|
nums: []protocol.StreamNum{num},
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -156,7 +156,7 @@ func (m *incomingUniStreamsMap) deleteStream(num protocol.StreamNum) error {
|
||||||
entry, ok := m.streams[num]
|
entry, ok := m.streams[num]
|
||||||
if ok && entry.shouldDelete {
|
if ok && entry.shouldDelete {
|
||||||
return streamError{
|
return streamError{
|
||||||
message: "tried to delete incoming stream %d multiple times",
|
message: "Tried to delete incoming stream %d multiple times",
|
||||||
nums: []protocol.StreamNum{num},
|
nums: []protocol.StreamNum{num},
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -157,7 +157,7 @@ func (m *outgoingBidiStreamsMap) DeleteStream(num protocol.StreamNum) error {
|
||||||
|
|
||||||
if _, ok := m.streams[num]; !ok {
|
if _, ok := m.streams[num]; !ok {
|
||||||
return streamError{
|
return streamError{
|
||||||
message: "tried to delete unknown outgoing stream %d",
|
message: "Tried to delete unknown outgoing stream %d",
|
||||||
nums: []protocol.StreamNum{num},
|
nums: []protocol.StreamNum{num},
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -155,7 +155,7 @@ func (m *outgoingItemsMap) DeleteStream(num protocol.StreamNum) error {
|
||||||
|
|
||||||
if _, ok := m.streams[num]; !ok {
|
if _, ok := m.streams[num]; !ok {
|
||||||
return streamError{
|
return streamError{
|
||||||
message: "tried to delete unknown outgoing stream %d",
|
message: "Tried to delete unknown outgoing stream %d",
|
||||||
nums: []protocol.StreamNum{num},
|
nums: []protocol.StreamNum{num},
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -157,7 +157,7 @@ func (m *outgoingUniStreamsMap) DeleteStream(num protocol.StreamNum) error {
|
||||||
|
|
||||||
if _, ok := m.streams[num]; !ok {
|
if _, ok := m.streams[num]; !ok {
|
||||||
return streamError{
|
return streamError{
|
||||||
message: "tried to delete unknown outgoing stream %d",
|
message: "Tried to delete unknown outgoing stream %d",
|
||||||
nums: []protocol.StreamNum{num},
|
nums: []protocol.StreamNum{num},
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,27 +0,0 @@
|
||||||
Copyright (c) 2009 The Go Authors. All rights reserved.
|
|
||||||
|
|
||||||
Redistribution and use in source and binary forms, with or without
|
|
||||||
modification, are permitted provided that the following conditions are
|
|
||||||
met:
|
|
||||||
|
|
||||||
* Redistributions of source code must retain the above copyright
|
|
||||||
notice, this list of conditions and the following disclaimer.
|
|
||||||
* Redistributions in binary form must reproduce the above
|
|
||||||
copyright notice, this list of conditions and the following disclaimer
|
|
||||||
in the documentation and/or other materials provided with the
|
|
||||||
distribution.
|
|
||||||
* Neither the name of Google Inc. nor the names of its
|
|
||||||
contributors may be used to endorse or promote products derived from
|
|
||||||
this software without specific prior written permission.
|
|
||||||
|
|
||||||
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
|
||||||
"AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
|
||||||
LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
|
|
||||||
A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
|
|
||||||
OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
|
||||||
SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
|
||||||
LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
|
||||||
DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
|
||||||
THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
||||||
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
|
||||||
OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
|
@ -1,6 +0,0 @@
|
||||||
# qtls
|
|
||||||
|
|
||||||
[![Go Reference](https://pkg.go.dev/badge/github.com/marten-seemann/qtls-go1-17.svg)](https://pkg.go.dev/github.com/marten-seemann/qtls-go1-17)
|
|
||||||
[![.github/workflows/go-test.yml](https://github.com/marten-seemann/qtls-go1-17/actions/workflows/go-test.yml/badge.svg)](https://github.com/marten-seemann/qtls-go1-17/actions/workflows/go-test.yml)
|
|
||||||
|
|
||||||
This repository contains a modified version of the standard library's TLS implementation, modified for the QUIC protocol. It is used by [quic-go](https://github.com/lucas-clemente/quic-go).
|
|
|
@ -1,102 +0,0 @@
|
||||||
// Copyright 2009 The Go Authors. All rights reserved.
|
|
||||||
// Use of this source code is governed by a BSD-style
|
|
||||||
// license that can be found in the LICENSE file.
|
|
||||||
|
|
||||||
package qtls
|
|
||||||
|
|
||||||
import "strconv"
|
|
||||||
|
|
||||||
type alert uint8
|
|
||||||
|
|
||||||
// Alert is a TLS alert
|
|
||||||
type Alert = alert
|
|
||||||
|
|
||||||
const (
|
|
||||||
// alert level
|
|
||||||
alertLevelWarning = 1
|
|
||||||
alertLevelError = 2
|
|
||||||
)
|
|
||||||
|
|
||||||
const (
|
|
||||||
alertCloseNotify alert = 0
|
|
||||||
alertUnexpectedMessage alert = 10
|
|
||||||
alertBadRecordMAC alert = 20
|
|
||||||
alertDecryptionFailed alert = 21
|
|
||||||
alertRecordOverflow alert = 22
|
|
||||||
alertDecompressionFailure alert = 30
|
|
||||||
alertHandshakeFailure alert = 40
|
|
||||||
alertBadCertificate alert = 42
|
|
||||||
alertUnsupportedCertificate alert = 43
|
|
||||||
alertCertificateRevoked alert = 44
|
|
||||||
alertCertificateExpired alert = 45
|
|
||||||
alertCertificateUnknown alert = 46
|
|
||||||
alertIllegalParameter alert = 47
|
|
||||||
alertUnknownCA alert = 48
|
|
||||||
alertAccessDenied alert = 49
|
|
||||||
alertDecodeError alert = 50
|
|
||||||
alertDecryptError alert = 51
|
|
||||||
alertExportRestriction alert = 60
|
|
||||||
alertProtocolVersion alert = 70
|
|
||||||
alertInsufficientSecurity alert = 71
|
|
||||||
alertInternalError alert = 80
|
|
||||||
alertInappropriateFallback alert = 86
|
|
||||||
alertUserCanceled alert = 90
|
|
||||||
alertNoRenegotiation alert = 100
|
|
||||||
alertMissingExtension alert = 109
|
|
||||||
alertUnsupportedExtension alert = 110
|
|
||||||
alertCertificateUnobtainable alert = 111
|
|
||||||
alertUnrecognizedName alert = 112
|
|
||||||
alertBadCertificateStatusResponse alert = 113
|
|
||||||
alertBadCertificateHashValue alert = 114
|
|
||||||
alertUnknownPSKIdentity alert = 115
|
|
||||||
alertCertificateRequired alert = 116
|
|
||||||
alertNoApplicationProtocol alert = 120
|
|
||||||
)
|
|
||||||
|
|
||||||
var alertText = map[alert]string{
|
|
||||||
alertCloseNotify: "close notify",
|
|
||||||
alertUnexpectedMessage: "unexpected message",
|
|
||||||
alertBadRecordMAC: "bad record MAC",
|
|
||||||
alertDecryptionFailed: "decryption failed",
|
|
||||||
alertRecordOverflow: "record overflow",
|
|
||||||
alertDecompressionFailure: "decompression failure",
|
|
||||||
alertHandshakeFailure: "handshake failure",
|
|
||||||
alertBadCertificate: "bad certificate",
|
|
||||||
alertUnsupportedCertificate: "unsupported certificate",
|
|
||||||
alertCertificateRevoked: "revoked certificate",
|
|
||||||
alertCertificateExpired: "expired certificate",
|
|
||||||
alertCertificateUnknown: "unknown certificate",
|
|
||||||
alertIllegalParameter: "illegal parameter",
|
|
||||||
alertUnknownCA: "unknown certificate authority",
|
|
||||||
alertAccessDenied: "access denied",
|
|
||||||
alertDecodeError: "error decoding message",
|
|
||||||
alertDecryptError: "error decrypting message",
|
|
||||||
alertExportRestriction: "export restriction",
|
|
||||||
alertProtocolVersion: "protocol version not supported",
|
|
||||||
alertInsufficientSecurity: "insufficient security level",
|
|
||||||
alertInternalError: "internal error",
|
|
||||||
alertInappropriateFallback: "inappropriate fallback",
|
|
||||||
alertUserCanceled: "user canceled",
|
|
||||||
alertNoRenegotiation: "no renegotiation",
|
|
||||||
alertMissingExtension: "missing extension",
|
|
||||||
alertUnsupportedExtension: "unsupported extension",
|
|
||||||
alertCertificateUnobtainable: "certificate unobtainable",
|
|
||||||
alertUnrecognizedName: "unrecognized name",
|
|
||||||
alertBadCertificateStatusResponse: "bad certificate status response",
|
|
||||||
alertBadCertificateHashValue: "bad certificate hash value",
|
|
||||||
alertUnknownPSKIdentity: "unknown PSK identity",
|
|
||||||
alertCertificateRequired: "certificate required",
|
|
||||||
alertNoApplicationProtocol: "no application protocol",
|
|
||||||
}
|
|
||||||
|
|
||||||
func (e alert) String() string {
|
|
||||||
s, ok := alertText[e]
|
|
||||||
if ok {
|
|
||||||
return "tls: " + s
|
|
||||||
}
|
|
||||||
return "tls: alert(" + strconv.Itoa(int(e)) + ")"
|
|
||||||
}
|
|
||||||
|
|
||||||
func (e alert) Error() string {
|
|
||||||
return e.String()
|
|
||||||
}
|
|
|
@ -1,289 +0,0 @@
|
||||||
// Copyright 2017 The Go Authors. All rights reserved.
|
|
||||||
// Use of this source code is governed by a BSD-style
|
|
||||||
// license that can be found in the LICENSE file.
|
|
||||||
|
|
||||||
package qtls
|
|
||||||
|
|
||||||
import (
|
|
||||||
"bytes"
|
|
||||||
"crypto"
|
|
||||||
"crypto/ecdsa"
|
|
||||||
"crypto/ed25519"
|
|
||||||
"crypto/elliptic"
|
|
||||||
"crypto/rsa"
|
|
||||||
"errors"
|
|
||||||
"fmt"
|
|
||||||
"hash"
|
|
||||||
"io"
|
|
||||||
)
|
|
||||||
|
|
||||||
// verifyHandshakeSignature verifies a signature against pre-hashed
|
|
||||||
// (if required) handshake contents.
|
|
||||||
func verifyHandshakeSignature(sigType uint8, pubkey crypto.PublicKey, hashFunc crypto.Hash, signed, sig []byte) error {
|
|
||||||
switch sigType {
|
|
||||||
case signatureECDSA:
|
|
||||||
pubKey, ok := pubkey.(*ecdsa.PublicKey)
|
|
||||||
if !ok {
|
|
||||||
return fmt.Errorf("expected an ECDSA public key, got %T", pubkey)
|
|
||||||
}
|
|
||||||
if !ecdsa.VerifyASN1(pubKey, signed, sig) {
|
|
||||||
return errors.New("ECDSA verification failure")
|
|
||||||
}
|
|
||||||
case signatureEd25519:
|
|
||||||
pubKey, ok := pubkey.(ed25519.PublicKey)
|
|
||||||
if !ok {
|
|
||||||
return fmt.Errorf("expected an Ed25519 public key, got %T", pubkey)
|
|
||||||
}
|
|
||||||
if !ed25519.Verify(pubKey, signed, sig) {
|
|
||||||
return errors.New("Ed25519 verification failure")
|
|
||||||
}
|
|
||||||
case signaturePKCS1v15:
|
|
||||||
pubKey, ok := pubkey.(*rsa.PublicKey)
|
|
||||||
if !ok {
|
|
||||||
return fmt.Errorf("expected an RSA public key, got %T", pubkey)
|
|
||||||
}
|
|
||||||
if err := rsa.VerifyPKCS1v15(pubKey, hashFunc, signed, sig); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
case signatureRSAPSS:
|
|
||||||
pubKey, ok := pubkey.(*rsa.PublicKey)
|
|
||||||
if !ok {
|
|
||||||
return fmt.Errorf("expected an RSA public key, got %T", pubkey)
|
|
||||||
}
|
|
||||||
signOpts := &rsa.PSSOptions{SaltLength: rsa.PSSSaltLengthEqualsHash}
|
|
||||||
if err := rsa.VerifyPSS(pubKey, hashFunc, signed, sig, signOpts); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
default:
|
|
||||||
return errors.New("internal error: unknown signature type")
|
|
||||||
}
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
const (
|
|
||||||
serverSignatureContext = "TLS 1.3, server CertificateVerify\x00"
|
|
||||||
clientSignatureContext = "TLS 1.3, client CertificateVerify\x00"
|
|
||||||
)
|
|
||||||
|
|
||||||
var signaturePadding = []byte{
|
|
||||||
0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20,
|
|
||||||
0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20,
|
|
||||||
0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20,
|
|
||||||
0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20,
|
|
||||||
0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20,
|
|
||||||
0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20,
|
|
||||||
0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20,
|
|
||||||
0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20,
|
|
||||||
}
|
|
||||||
|
|
||||||
// signedMessage returns the pre-hashed (if necessary) message to be signed by
|
|
||||||
// certificate keys in TLS 1.3. See RFC 8446, Section 4.4.3.
|
|
||||||
func signedMessage(sigHash crypto.Hash, context string, transcript hash.Hash) []byte {
|
|
||||||
if sigHash == directSigning {
|
|
||||||
b := &bytes.Buffer{}
|
|
||||||
b.Write(signaturePadding)
|
|
||||||
io.WriteString(b, context)
|
|
||||||
b.Write(transcript.Sum(nil))
|
|
||||||
return b.Bytes()
|
|
||||||
}
|
|
||||||
h := sigHash.New()
|
|
||||||
h.Write(signaturePadding)
|
|
||||||
io.WriteString(h, context)
|
|
||||||
h.Write(transcript.Sum(nil))
|
|
||||||
return h.Sum(nil)
|
|
||||||
}
|
|
||||||
|
|
||||||
// typeAndHashFromSignatureScheme returns the corresponding signature type and
|
|
||||||
// crypto.Hash for a given TLS SignatureScheme.
|
|
||||||
func typeAndHashFromSignatureScheme(signatureAlgorithm SignatureScheme) (sigType uint8, hash crypto.Hash, err error) {
|
|
||||||
switch signatureAlgorithm {
|
|
||||||
case PKCS1WithSHA1, PKCS1WithSHA256, PKCS1WithSHA384, PKCS1WithSHA512:
|
|
||||||
sigType = signaturePKCS1v15
|
|
||||||
case PSSWithSHA256, PSSWithSHA384, PSSWithSHA512:
|
|
||||||
sigType = signatureRSAPSS
|
|
||||||
case ECDSAWithSHA1, ECDSAWithP256AndSHA256, ECDSAWithP384AndSHA384, ECDSAWithP521AndSHA512:
|
|
||||||
sigType = signatureECDSA
|
|
||||||
case Ed25519:
|
|
||||||
sigType = signatureEd25519
|
|
||||||
default:
|
|
||||||
return 0, 0, fmt.Errorf("unsupported signature algorithm: %v", signatureAlgorithm)
|
|
||||||
}
|
|
||||||
switch signatureAlgorithm {
|
|
||||||
case PKCS1WithSHA1, ECDSAWithSHA1:
|
|
||||||
hash = crypto.SHA1
|
|
||||||
case PKCS1WithSHA256, PSSWithSHA256, ECDSAWithP256AndSHA256:
|
|
||||||
hash = crypto.SHA256
|
|
||||||
case PKCS1WithSHA384, PSSWithSHA384, ECDSAWithP384AndSHA384:
|
|
||||||
hash = crypto.SHA384
|
|
||||||
case PKCS1WithSHA512, PSSWithSHA512, ECDSAWithP521AndSHA512:
|
|
||||||
hash = crypto.SHA512
|
|
||||||
case Ed25519:
|
|
||||||
hash = directSigning
|
|
||||||
default:
|
|
||||||
return 0, 0, fmt.Errorf("unsupported signature algorithm: %v", signatureAlgorithm)
|
|
||||||
}
|
|
||||||
return sigType, hash, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// legacyTypeAndHashFromPublicKey returns the fixed signature type and crypto.Hash for
|
|
||||||
// a given public key used with TLS 1.0 and 1.1, before the introduction of
|
|
||||||
// signature algorithm negotiation.
|
|
||||||
func legacyTypeAndHashFromPublicKey(pub crypto.PublicKey) (sigType uint8, hash crypto.Hash, err error) {
|
|
||||||
switch pub.(type) {
|
|
||||||
case *rsa.PublicKey:
|
|
||||||
return signaturePKCS1v15, crypto.MD5SHA1, nil
|
|
||||||
case *ecdsa.PublicKey:
|
|
||||||
return signatureECDSA, crypto.SHA1, nil
|
|
||||||
case ed25519.PublicKey:
|
|
||||||
// RFC 8422 specifies support for Ed25519 in TLS 1.0 and 1.1,
|
|
||||||
// but it requires holding on to a handshake transcript to do a
|
|
||||||
// full signature, and not even OpenSSL bothers with the
|
|
||||||
// complexity, so we can't even test it properly.
|
|
||||||
return 0, 0, fmt.Errorf("tls: Ed25519 public keys are not supported before TLS 1.2")
|
|
||||||
default:
|
|
||||||
return 0, 0, fmt.Errorf("tls: unsupported public key: %T", pub)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
var rsaSignatureSchemes = []struct {
|
|
||||||
scheme SignatureScheme
|
|
||||||
minModulusBytes int
|
|
||||||
maxVersion uint16
|
|
||||||
}{
|
|
||||||
// RSA-PSS is used with PSSSaltLengthEqualsHash, and requires
|
|
||||||
// emLen >= hLen + sLen + 2
|
|
||||||
{PSSWithSHA256, crypto.SHA256.Size()*2 + 2, VersionTLS13},
|
|
||||||
{PSSWithSHA384, crypto.SHA384.Size()*2 + 2, VersionTLS13},
|
|
||||||
{PSSWithSHA512, crypto.SHA512.Size()*2 + 2, VersionTLS13},
|
|
||||||
// PKCS #1 v1.5 uses prefixes from hashPrefixes in crypto/rsa, and requires
|
|
||||||
// emLen >= len(prefix) + hLen + 11
|
|
||||||
// TLS 1.3 dropped support for PKCS #1 v1.5 in favor of RSA-PSS.
|
|
||||||
{PKCS1WithSHA256, 19 + crypto.SHA256.Size() + 11, VersionTLS12},
|
|
||||||
{PKCS1WithSHA384, 19 + crypto.SHA384.Size() + 11, VersionTLS12},
|
|
||||||
{PKCS1WithSHA512, 19 + crypto.SHA512.Size() + 11, VersionTLS12},
|
|
||||||
{PKCS1WithSHA1, 15 + crypto.SHA1.Size() + 11, VersionTLS12},
|
|
||||||
}
|
|
||||||
|
|
||||||
// signatureSchemesForCertificate returns the list of supported SignatureSchemes
|
|
||||||
// for a given certificate, based on the public key and the protocol version,
|
|
||||||
// and optionally filtered by its explicit SupportedSignatureAlgorithms.
|
|
||||||
//
|
|
||||||
// This function must be kept in sync with supportedSignatureAlgorithms.
|
|
||||||
func signatureSchemesForCertificate(version uint16, cert *Certificate) []SignatureScheme {
|
|
||||||
priv, ok := cert.PrivateKey.(crypto.Signer)
|
|
||||||
if !ok {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
var sigAlgs []SignatureScheme
|
|
||||||
switch pub := priv.Public().(type) {
|
|
||||||
case *ecdsa.PublicKey:
|
|
||||||
if version != VersionTLS13 {
|
|
||||||
// In TLS 1.2 and earlier, ECDSA algorithms are not
|
|
||||||
// constrained to a single curve.
|
|
||||||
sigAlgs = []SignatureScheme{
|
|
||||||
ECDSAWithP256AndSHA256,
|
|
||||||
ECDSAWithP384AndSHA384,
|
|
||||||
ECDSAWithP521AndSHA512,
|
|
||||||
ECDSAWithSHA1,
|
|
||||||
}
|
|
||||||
break
|
|
||||||
}
|
|
||||||
switch pub.Curve {
|
|
||||||
case elliptic.P256():
|
|
||||||
sigAlgs = []SignatureScheme{ECDSAWithP256AndSHA256}
|
|
||||||
case elliptic.P384():
|
|
||||||
sigAlgs = []SignatureScheme{ECDSAWithP384AndSHA384}
|
|
||||||
case elliptic.P521():
|
|
||||||
sigAlgs = []SignatureScheme{ECDSAWithP521AndSHA512}
|
|
||||||
default:
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
case *rsa.PublicKey:
|
|
||||||
size := pub.Size()
|
|
||||||
sigAlgs = make([]SignatureScheme, 0, len(rsaSignatureSchemes))
|
|
||||||
for _, candidate := range rsaSignatureSchemes {
|
|
||||||
if size >= candidate.minModulusBytes && version <= candidate.maxVersion {
|
|
||||||
sigAlgs = append(sigAlgs, candidate.scheme)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
case ed25519.PublicKey:
|
|
||||||
sigAlgs = []SignatureScheme{Ed25519}
|
|
||||||
default:
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
if cert.SupportedSignatureAlgorithms != nil {
|
|
||||||
var filteredSigAlgs []SignatureScheme
|
|
||||||
for _, sigAlg := range sigAlgs {
|
|
||||||
if isSupportedSignatureAlgorithm(sigAlg, cert.SupportedSignatureAlgorithms) {
|
|
||||||
filteredSigAlgs = append(filteredSigAlgs, sigAlg)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return filteredSigAlgs
|
|
||||||
}
|
|
||||||
return sigAlgs
|
|
||||||
}
|
|
||||||
|
|
||||||
// selectSignatureScheme picks a SignatureScheme from the peer's preference list
|
|
||||||
// that works with the selected certificate. It's only called for protocol
|
|
||||||
// versions that support signature algorithms, so TLS 1.2 and 1.3.
|
|
||||||
func selectSignatureScheme(vers uint16, c *Certificate, peerAlgs []SignatureScheme) (SignatureScheme, error) {
|
|
||||||
supportedAlgs := signatureSchemesForCertificate(vers, c)
|
|
||||||
if len(supportedAlgs) == 0 {
|
|
||||||
return 0, unsupportedCertificateError(c)
|
|
||||||
}
|
|
||||||
if len(peerAlgs) == 0 && vers == VersionTLS12 {
|
|
||||||
// For TLS 1.2, if the client didn't send signature_algorithms then we
|
|
||||||
// can assume that it supports SHA1. See RFC 5246, Section 7.4.1.4.1.
|
|
||||||
peerAlgs = []SignatureScheme{PKCS1WithSHA1, ECDSAWithSHA1}
|
|
||||||
}
|
|
||||||
// Pick signature scheme in the peer's preference order, as our
|
|
||||||
// preference order is not configurable.
|
|
||||||
for _, preferredAlg := range peerAlgs {
|
|
||||||
if isSupportedSignatureAlgorithm(preferredAlg, supportedAlgs) {
|
|
||||||
return preferredAlg, nil
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return 0, errors.New("tls: peer doesn't support any of the certificate's signature algorithms")
|
|
||||||
}
|
|
||||||
|
|
||||||
// unsupportedCertificateError returns a helpful error for certificates with
|
|
||||||
// an unsupported private key.
|
|
||||||
func unsupportedCertificateError(cert *Certificate) error {
|
|
||||||
switch cert.PrivateKey.(type) {
|
|
||||||
case rsa.PrivateKey, ecdsa.PrivateKey:
|
|
||||||
return fmt.Errorf("tls: unsupported certificate: private key is %T, expected *%T",
|
|
||||||
cert.PrivateKey, cert.PrivateKey)
|
|
||||||
case *ed25519.PrivateKey:
|
|
||||||
return fmt.Errorf("tls: unsupported certificate: private key is *ed25519.PrivateKey, expected ed25519.PrivateKey")
|
|
||||||
}
|
|
||||||
|
|
||||||
signer, ok := cert.PrivateKey.(crypto.Signer)
|
|
||||||
if !ok {
|
|
||||||
return fmt.Errorf("tls: certificate private key (%T) does not implement crypto.Signer",
|
|
||||||
cert.PrivateKey)
|
|
||||||
}
|
|
||||||
|
|
||||||
switch pub := signer.Public().(type) {
|
|
||||||
case *ecdsa.PublicKey:
|
|
||||||
switch pub.Curve {
|
|
||||||
case elliptic.P256():
|
|
||||||
case elliptic.P384():
|
|
||||||
case elliptic.P521():
|
|
||||||
default:
|
|
||||||
return fmt.Errorf("tls: unsupported certificate curve (%s)", pub.Curve.Params().Name)
|
|
||||||
}
|
|
||||||
case *rsa.PublicKey:
|
|
||||||
return fmt.Errorf("tls: certificate RSA key size too small for supported signature algorithms")
|
|
||||||
case ed25519.PublicKey:
|
|
||||||
default:
|
|
||||||
return fmt.Errorf("tls: unsupported certificate key (%T)", pub)
|
|
||||||
}
|
|
||||||
|
|
||||||
if cert.SupportedSignatureAlgorithms != nil {
|
|
||||||
return fmt.Errorf("tls: peer doesn't support the certificate custom signature algorithms")
|
|
||||||
}
|
|
||||||
|
|
||||||
return fmt.Errorf("tls: internal error: unsupported key (%T)", cert.PrivateKey)
|
|
||||||
}
|
|
|
@ -1,705 +0,0 @@
|
||||||
// Copyright 2010 The Go Authors. All rights reserved.
|
|
||||||
// Use of this source code is governed by a BSD-style
|
|
||||||
// license that can be found in the LICENSE file.
|
|
||||||
|
|
||||||
package qtls
|
|
||||||
|
|
||||||
import (
|
|
||||||
"crypto"
|
|
||||||
"crypto/aes"
|
|
||||||
"crypto/cipher"
|
|
||||||
"crypto/des"
|
|
||||||
"crypto/hmac"
|
|
||||||
"crypto/rc4"
|
|
||||||
"crypto/sha1"
|
|
||||||
"crypto/sha256"
|
|
||||||
"fmt"
|
|
||||||
"hash"
|
|
||||||
"runtime"
|
|
||||||
|
|
||||||
"golang.org/x/crypto/chacha20poly1305"
|
|
||||||
"golang.org/x/sys/cpu"
|
|
||||||
)
|
|
||||||
|
|
||||||
// CipherSuite is a TLS cipher suite. Note that most functions in this package
|
|
||||||
// accept and expose cipher suite IDs instead of this type.
|
|
||||||
type CipherSuite struct {
|
|
||||||
ID uint16
|
|
||||||
Name string
|
|
||||||
|
|
||||||
// Supported versions is the list of TLS protocol versions that can
|
|
||||||
// negotiate this cipher suite.
|
|
||||||
SupportedVersions []uint16
|
|
||||||
|
|
||||||
// Insecure is true if the cipher suite has known security issues
|
|
||||||
// due to its primitives, design, or implementation.
|
|
||||||
Insecure bool
|
|
||||||
}
|
|
||||||
|
|
||||||
var (
|
|
||||||
supportedUpToTLS12 = []uint16{VersionTLS10, VersionTLS11, VersionTLS12}
|
|
||||||
supportedOnlyTLS12 = []uint16{VersionTLS12}
|
|
||||||
supportedOnlyTLS13 = []uint16{VersionTLS13}
|
|
||||||
)
|
|
||||||
|
|
||||||
// CipherSuites returns a list of cipher suites currently implemented by this
|
|
||||||
// package, excluding those with security issues, which are returned by
|
|
||||||
// InsecureCipherSuites.
|
|
||||||
//
|
|
||||||
// The list is sorted by ID. Note that the default cipher suites selected by
|
|
||||||
// this package might depend on logic that can't be captured by a static list,
|
|
||||||
// and might not match those returned by this function.
|
|
||||||
func CipherSuites() []*CipherSuite {
|
|
||||||
return []*CipherSuite{
|
|
||||||
{TLS_RSA_WITH_AES_128_CBC_SHA, "TLS_RSA_WITH_AES_128_CBC_SHA", supportedUpToTLS12, false},
|
|
||||||
{TLS_RSA_WITH_AES_256_CBC_SHA, "TLS_RSA_WITH_AES_256_CBC_SHA", supportedUpToTLS12, false},
|
|
||||||
{TLS_RSA_WITH_AES_128_GCM_SHA256, "TLS_RSA_WITH_AES_128_GCM_SHA256", supportedOnlyTLS12, false},
|
|
||||||
{TLS_RSA_WITH_AES_256_GCM_SHA384, "TLS_RSA_WITH_AES_256_GCM_SHA384", supportedOnlyTLS12, false},
|
|
||||||
|
|
||||||
{TLS_AES_128_GCM_SHA256, "TLS_AES_128_GCM_SHA256", supportedOnlyTLS13, false},
|
|
||||||
{TLS_AES_256_GCM_SHA384, "TLS_AES_256_GCM_SHA384", supportedOnlyTLS13, false},
|
|
||||||
{TLS_CHACHA20_POLY1305_SHA256, "TLS_CHACHA20_POLY1305_SHA256", supportedOnlyTLS13, false},
|
|
||||||
|
|
||||||
{TLS_ECDHE_ECDSA_WITH_AES_128_CBC_SHA, "TLS_ECDHE_ECDSA_WITH_AES_128_CBC_SHA", supportedUpToTLS12, false},
|
|
||||||
{TLS_ECDHE_ECDSA_WITH_AES_256_CBC_SHA, "TLS_ECDHE_ECDSA_WITH_AES_256_CBC_SHA", supportedUpToTLS12, false},
|
|
||||||
{TLS_ECDHE_RSA_WITH_AES_128_CBC_SHA, "TLS_ECDHE_RSA_WITH_AES_128_CBC_SHA", supportedUpToTLS12, false},
|
|
||||||
{TLS_ECDHE_RSA_WITH_AES_256_CBC_SHA, "TLS_ECDHE_RSA_WITH_AES_256_CBC_SHA", supportedUpToTLS12, false},
|
|
||||||
{TLS_ECDHE_ECDSA_WITH_AES_128_GCM_SHA256, "TLS_ECDHE_ECDSA_WITH_AES_128_GCM_SHA256", supportedOnlyTLS12, false},
|
|
||||||
{TLS_ECDHE_ECDSA_WITH_AES_256_GCM_SHA384, "TLS_ECDHE_ECDSA_WITH_AES_256_GCM_SHA384", supportedOnlyTLS12, false},
|
|
||||||
{TLS_ECDHE_RSA_WITH_AES_128_GCM_SHA256, "TLS_ECDHE_RSA_WITH_AES_128_GCM_SHA256", supportedOnlyTLS12, false},
|
|
||||||
{TLS_ECDHE_RSA_WITH_AES_256_GCM_SHA384, "TLS_ECDHE_RSA_WITH_AES_256_GCM_SHA384", supportedOnlyTLS12, false},
|
|
||||||
{TLS_ECDHE_RSA_WITH_CHACHA20_POLY1305_SHA256, "TLS_ECDHE_RSA_WITH_CHACHA20_POLY1305_SHA256", supportedOnlyTLS12, false},
|
|
||||||
{TLS_ECDHE_ECDSA_WITH_CHACHA20_POLY1305_SHA256, "TLS_ECDHE_ECDSA_WITH_CHACHA20_POLY1305_SHA256", supportedOnlyTLS12, false},
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// InsecureCipherSuites returns a list of cipher suites currently implemented by
|
|
||||||
// this package and which have security issues.
|
|
||||||
//
|
|
||||||
// Most applications should not use the cipher suites in this list, and should
|
|
||||||
// only use those returned by CipherSuites.
|
|
||||||
func InsecureCipherSuites() []*CipherSuite {
|
|
||||||
// This list includes RC4, CBC_SHA256, and 3DES cipher suites. See
|
|
||||||
// cipherSuitesPreferenceOrder for details.
|
|
||||||
return []*CipherSuite{
|
|
||||||
{TLS_RSA_WITH_RC4_128_SHA, "TLS_RSA_WITH_RC4_128_SHA", supportedUpToTLS12, true},
|
|
||||||
{TLS_RSA_WITH_3DES_EDE_CBC_SHA, "TLS_RSA_WITH_3DES_EDE_CBC_SHA", supportedUpToTLS12, true},
|
|
||||||
{TLS_RSA_WITH_AES_128_CBC_SHA256, "TLS_RSA_WITH_AES_128_CBC_SHA256", supportedOnlyTLS12, true},
|
|
||||||
{TLS_ECDHE_ECDSA_WITH_RC4_128_SHA, "TLS_ECDHE_ECDSA_WITH_RC4_128_SHA", supportedUpToTLS12, true},
|
|
||||||
{TLS_ECDHE_RSA_WITH_RC4_128_SHA, "TLS_ECDHE_RSA_WITH_RC4_128_SHA", supportedUpToTLS12, true},
|
|
||||||
{TLS_ECDHE_RSA_WITH_3DES_EDE_CBC_SHA, "TLS_ECDHE_RSA_WITH_3DES_EDE_CBC_SHA", supportedUpToTLS12, true},
|
|
||||||
{TLS_ECDHE_ECDSA_WITH_AES_128_CBC_SHA256, "TLS_ECDHE_ECDSA_WITH_AES_128_CBC_SHA256", supportedOnlyTLS12, true},
|
|
||||||
{TLS_ECDHE_RSA_WITH_AES_128_CBC_SHA256, "TLS_ECDHE_RSA_WITH_AES_128_CBC_SHA256", supportedOnlyTLS12, true},
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// CipherSuiteName returns the standard name for the passed cipher suite ID
|
|
||||||
// (e.g. "TLS_ECDHE_ECDSA_WITH_AES_128_GCM_SHA256"), or a fallback representation
|
|
||||||
// of the ID value if the cipher suite is not implemented by this package.
|
|
||||||
func CipherSuiteName(id uint16) string {
|
|
||||||
for _, c := range CipherSuites() {
|
|
||||||
if c.ID == id {
|
|
||||||
return c.Name
|
|
||||||
}
|
|
||||||
}
|
|
||||||
for _, c := range InsecureCipherSuites() {
|
|
||||||
if c.ID == id {
|
|
||||||
return c.Name
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return fmt.Sprintf("0x%04X", id)
|
|
||||||
}
|
|
||||||
|
|
||||||
const (
|
|
||||||
// suiteECDHE indicates that the cipher suite involves elliptic curve
|
|
||||||
// Diffie-Hellman. This means that it should only be selected when the
|
|
||||||
// client indicates that it supports ECC with a curve and point format
|
|
||||||
// that we're happy with.
|
|
||||||
suiteECDHE = 1 << iota
|
|
||||||
// suiteECSign indicates that the cipher suite involves an ECDSA or
|
|
||||||
// EdDSA signature and therefore may only be selected when the server's
|
|
||||||
// certificate is ECDSA or EdDSA. If this is not set then the cipher suite
|
|
||||||
// is RSA based.
|
|
||||||
suiteECSign
|
|
||||||
// suiteTLS12 indicates that the cipher suite should only be advertised
|
|
||||||
// and accepted when using TLS 1.2.
|
|
||||||
suiteTLS12
|
|
||||||
// suiteSHA384 indicates that the cipher suite uses SHA384 as the
|
|
||||||
// handshake hash.
|
|
||||||
suiteSHA384
|
|
||||||
)
|
|
||||||
|
|
||||||
// A cipherSuite is a TLS 1.0–1.2 cipher suite, and defines the key exchange
|
|
||||||
// mechanism, as well as the cipher+MAC pair or the AEAD.
|
|
||||||
type cipherSuite struct {
|
|
||||||
id uint16
|
|
||||||
// the lengths, in bytes, of the key material needed for each component.
|
|
||||||
keyLen int
|
|
||||||
macLen int
|
|
||||||
ivLen int
|
|
||||||
ka func(version uint16) keyAgreement
|
|
||||||
// flags is a bitmask of the suite* values, above.
|
|
||||||
flags int
|
|
||||||
cipher func(key, iv []byte, isRead bool) interface{}
|
|
||||||
mac func(key []byte) hash.Hash
|
|
||||||
aead func(key, fixedNonce []byte) aead
|
|
||||||
}
|
|
||||||
|
|
||||||
var cipherSuites = []*cipherSuite{ // TODO: replace with a map, since the order doesn't matter.
|
|
||||||
{TLS_ECDHE_RSA_WITH_CHACHA20_POLY1305, 32, 0, 12, ecdheRSAKA, suiteECDHE | suiteTLS12, nil, nil, aeadChaCha20Poly1305},
|
|
||||||
{TLS_ECDHE_ECDSA_WITH_CHACHA20_POLY1305, 32, 0, 12, ecdheECDSAKA, suiteECDHE | suiteECSign | suiteTLS12, nil, nil, aeadChaCha20Poly1305},
|
|
||||||
{TLS_ECDHE_RSA_WITH_AES_128_GCM_SHA256, 16, 0, 4, ecdheRSAKA, suiteECDHE | suiteTLS12, nil, nil, aeadAESGCM},
|
|
||||||
{TLS_ECDHE_ECDSA_WITH_AES_128_GCM_SHA256, 16, 0, 4, ecdheECDSAKA, suiteECDHE | suiteECSign | suiteTLS12, nil, nil, aeadAESGCM},
|
|
||||||
{TLS_ECDHE_RSA_WITH_AES_256_GCM_SHA384, 32, 0, 4, ecdheRSAKA, suiteECDHE | suiteTLS12 | suiteSHA384, nil, nil, aeadAESGCM},
|
|
||||||
{TLS_ECDHE_ECDSA_WITH_AES_256_GCM_SHA384, 32, 0, 4, ecdheECDSAKA, suiteECDHE | suiteECSign | suiteTLS12 | suiteSHA384, nil, nil, aeadAESGCM},
|
|
||||||
{TLS_ECDHE_RSA_WITH_AES_128_CBC_SHA256, 16, 32, 16, ecdheRSAKA, suiteECDHE | suiteTLS12, cipherAES, macSHA256, nil},
|
|
||||||
{TLS_ECDHE_RSA_WITH_AES_128_CBC_SHA, 16, 20, 16, ecdheRSAKA, suiteECDHE, cipherAES, macSHA1, nil},
|
|
||||||
{TLS_ECDHE_ECDSA_WITH_AES_128_CBC_SHA256, 16, 32, 16, ecdheECDSAKA, suiteECDHE | suiteECSign | suiteTLS12, cipherAES, macSHA256, nil},
|
|
||||||
{TLS_ECDHE_ECDSA_WITH_AES_128_CBC_SHA, 16, 20, 16, ecdheECDSAKA, suiteECDHE | suiteECSign, cipherAES, macSHA1, nil},
|
|
||||||
{TLS_ECDHE_RSA_WITH_AES_256_CBC_SHA, 32, 20, 16, ecdheRSAKA, suiteECDHE, cipherAES, macSHA1, nil},
|
|
||||||
{TLS_ECDHE_ECDSA_WITH_AES_256_CBC_SHA, 32, 20, 16, ecdheECDSAKA, suiteECDHE | suiteECSign, cipherAES, macSHA1, nil},
|
|
||||||
{TLS_RSA_WITH_AES_128_GCM_SHA256, 16, 0, 4, rsaKA, suiteTLS12, nil, nil, aeadAESGCM},
|
|
||||||
{TLS_RSA_WITH_AES_256_GCM_SHA384, 32, 0, 4, rsaKA, suiteTLS12 | suiteSHA384, nil, nil, aeadAESGCM},
|
|
||||||
{TLS_RSA_WITH_AES_128_CBC_SHA256, 16, 32, 16, rsaKA, suiteTLS12, cipherAES, macSHA256, nil},
|
|
||||||
{TLS_RSA_WITH_AES_128_CBC_SHA, 16, 20, 16, rsaKA, 0, cipherAES, macSHA1, nil},
|
|
||||||
{TLS_RSA_WITH_AES_256_CBC_SHA, 32, 20, 16, rsaKA, 0, cipherAES, macSHA1, nil},
|
|
||||||
{TLS_ECDHE_RSA_WITH_3DES_EDE_CBC_SHA, 24, 20, 8, ecdheRSAKA, suiteECDHE, cipher3DES, macSHA1, nil},
|
|
||||||
{TLS_RSA_WITH_3DES_EDE_CBC_SHA, 24, 20, 8, rsaKA, 0, cipher3DES, macSHA1, nil},
|
|
||||||
{TLS_RSA_WITH_RC4_128_SHA, 16, 20, 0, rsaKA, 0, cipherRC4, macSHA1, nil},
|
|
||||||
{TLS_ECDHE_RSA_WITH_RC4_128_SHA, 16, 20, 0, ecdheRSAKA, suiteECDHE, cipherRC4, macSHA1, nil},
|
|
||||||
{TLS_ECDHE_ECDSA_WITH_RC4_128_SHA, 16, 20, 0, ecdheECDSAKA, suiteECDHE | suiteECSign, cipherRC4, macSHA1, nil},
|
|
||||||
}
|
|
||||||
|
|
||||||
// selectCipherSuite returns the first TLS 1.0–1.2 cipher suite from ids which
|
|
||||||
// is also in supportedIDs and passes the ok filter.
|
|
||||||
func selectCipherSuite(ids, supportedIDs []uint16, ok func(*cipherSuite) bool) *cipherSuite {
|
|
||||||
for _, id := range ids {
|
|
||||||
candidate := cipherSuiteByID(id)
|
|
||||||
if candidate == nil || !ok(candidate) {
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
|
|
||||||
for _, suppID := range supportedIDs {
|
|
||||||
if id == suppID {
|
|
||||||
return candidate
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// A cipherSuiteTLS13 defines only the pair of the AEAD algorithm and hash
|
|
||||||
// algorithm to be used with HKDF. See RFC 8446, Appendix B.4.
|
|
||||||
type cipherSuiteTLS13 struct {
|
|
||||||
id uint16
|
|
||||||
keyLen int
|
|
||||||
aead func(key, fixedNonce []byte) aead
|
|
||||||
hash crypto.Hash
|
|
||||||
}
|
|
||||||
|
|
||||||
type CipherSuiteTLS13 struct {
|
|
||||||
ID uint16
|
|
||||||
KeyLen int
|
|
||||||
Hash crypto.Hash
|
|
||||||
AEAD func(key, fixedNonce []byte) cipher.AEAD
|
|
||||||
}
|
|
||||||
|
|
||||||
func (c *CipherSuiteTLS13) IVLen() int {
|
|
||||||
return aeadNonceLength
|
|
||||||
}
|
|
||||||
|
|
||||||
var cipherSuitesTLS13 = []*cipherSuiteTLS13{ // TODO: replace with a map.
|
|
||||||
{TLS_AES_128_GCM_SHA256, 16, aeadAESGCMTLS13, crypto.SHA256},
|
|
||||||
{TLS_CHACHA20_POLY1305_SHA256, 32, aeadChaCha20Poly1305, crypto.SHA256},
|
|
||||||
{TLS_AES_256_GCM_SHA384, 32, aeadAESGCMTLS13, crypto.SHA384},
|
|
||||||
}
|
|
||||||
|
|
||||||
// cipherSuitesPreferenceOrder is the order in which we'll select (on the
|
|
||||||
// server) or advertise (on the client) TLS 1.0–1.2 cipher suites.
|
|
||||||
//
|
|
||||||
// Cipher suites are filtered but not reordered based on the application and
|
|
||||||
// peer's preferences, meaning we'll never select a suite lower in this list if
|
|
||||||
// any higher one is available. This makes it more defensible to keep weaker
|
|
||||||
// cipher suites enabled, especially on the server side where we get the last
|
|
||||||
// word, since there are no known downgrade attacks on cipher suites selection.
|
|
||||||
//
|
|
||||||
// The list is sorted by applying the following priority rules, stopping at the
|
|
||||||
// first (most important) applicable one:
|
|
||||||
//
|
|
||||||
// - Anything else comes before RC4
|
|
||||||
//
|
|
||||||
// RC4 has practically exploitable biases. See https://www.rc4nomore.com.
|
|
||||||
//
|
|
||||||
// - Anything else comes before CBC_SHA256
|
|
||||||
//
|
|
||||||
// SHA-256 variants of the CBC ciphersuites don't implement any Lucky13
|
|
||||||
// countermeasures. See http://www.isg.rhul.ac.uk/tls/Lucky13.html and
|
|
||||||
// https://www.imperialviolet.org/2013/02/04/luckythirteen.html.
|
|
||||||
//
|
|
||||||
// - Anything else comes before 3DES
|
|
||||||
//
|
|
||||||
// 3DES has 64-bit blocks, which makes it fundamentally susceptible to
|
|
||||||
// birthday attacks. See https://sweet32.info.
|
|
||||||
//
|
|
||||||
// - ECDHE comes before anything else
|
|
||||||
//
|
|
||||||
// Once we got the broken stuff out of the way, the most important
|
|
||||||
// property a cipher suite can have is forward secrecy. We don't
|
|
||||||
// implement FFDHE, so that means ECDHE.
|
|
||||||
//
|
|
||||||
// - AEADs come before CBC ciphers
|
|
||||||
//
|
|
||||||
// Even with Lucky13 countermeasures, MAC-then-Encrypt CBC cipher suites
|
|
||||||
// are fundamentally fragile, and suffered from an endless sequence of
|
|
||||||
// padding oracle attacks. See https://eprint.iacr.org/2015/1129,
|
|
||||||
// https://www.imperialviolet.org/2014/12/08/poodleagain.html, and
|
|
||||||
// https://blog.cloudflare.com/yet-another-padding-oracle-in-openssl-cbc-ciphersuites/.
|
|
||||||
//
|
|
||||||
// - AES comes before ChaCha20
|
|
||||||
//
|
|
||||||
// When AES hardware is available, AES-128-GCM and AES-256-GCM are faster
|
|
||||||
// than ChaCha20Poly1305.
|
|
||||||
//
|
|
||||||
// When AES hardware is not available, AES-128-GCM is one or more of: much
|
|
||||||
// slower, way more complex, and less safe (because not constant time)
|
|
||||||
// than ChaCha20Poly1305.
|
|
||||||
//
|
|
||||||
// We use this list if we think both peers have AES hardware, and
|
|
||||||
// cipherSuitesPreferenceOrderNoAES otherwise.
|
|
||||||
//
|
|
||||||
// - AES-128 comes before AES-256
|
|
||||||
//
|
|
||||||
// The only potential advantages of AES-256 are better multi-target
|
|
||||||
// margins, and hypothetical post-quantum properties. Neither apply to
|
|
||||||
// TLS, and AES-256 is slower due to its four extra rounds (which don't
|
|
||||||
// contribute to the advantages above).
|
|
||||||
//
|
|
||||||
// - ECDSA comes before RSA
|
|
||||||
//
|
|
||||||
// The relative order of ECDSA and RSA cipher suites doesn't matter,
|
|
||||||
// as they depend on the certificate. Pick one to get a stable order.
|
|
||||||
//
|
|
||||||
var cipherSuitesPreferenceOrder = []uint16{
|
|
||||||
// AEADs w/ ECDHE
|
|
||||||
TLS_ECDHE_ECDSA_WITH_AES_128_GCM_SHA256, TLS_ECDHE_RSA_WITH_AES_128_GCM_SHA256,
|
|
||||||
TLS_ECDHE_ECDSA_WITH_AES_256_GCM_SHA384, TLS_ECDHE_RSA_WITH_AES_256_GCM_SHA384,
|
|
||||||
TLS_ECDHE_ECDSA_WITH_CHACHA20_POLY1305, TLS_ECDHE_RSA_WITH_CHACHA20_POLY1305,
|
|
||||||
|
|
||||||
// CBC w/ ECDHE
|
|
||||||
TLS_ECDHE_ECDSA_WITH_AES_128_CBC_SHA, TLS_ECDHE_RSA_WITH_AES_128_CBC_SHA,
|
|
||||||
TLS_ECDHE_ECDSA_WITH_AES_256_CBC_SHA, TLS_ECDHE_RSA_WITH_AES_256_CBC_SHA,
|
|
||||||
|
|
||||||
// AEADs w/o ECDHE
|
|
||||||
TLS_RSA_WITH_AES_128_GCM_SHA256,
|
|
||||||
TLS_RSA_WITH_AES_256_GCM_SHA384,
|
|
||||||
|
|
||||||
// CBC w/o ECDHE
|
|
||||||
TLS_RSA_WITH_AES_128_CBC_SHA,
|
|
||||||
TLS_RSA_WITH_AES_256_CBC_SHA,
|
|
||||||
|
|
||||||
// 3DES
|
|
||||||
TLS_ECDHE_RSA_WITH_3DES_EDE_CBC_SHA,
|
|
||||||
TLS_RSA_WITH_3DES_EDE_CBC_SHA,
|
|
||||||
|
|
||||||
// CBC_SHA256
|
|
||||||
TLS_ECDHE_ECDSA_WITH_AES_128_CBC_SHA256, TLS_ECDHE_RSA_WITH_AES_128_CBC_SHA256,
|
|
||||||
TLS_RSA_WITH_AES_128_CBC_SHA256,
|
|
||||||
|
|
||||||
// RC4
|
|
||||||
TLS_ECDHE_ECDSA_WITH_RC4_128_SHA, TLS_ECDHE_RSA_WITH_RC4_128_SHA,
|
|
||||||
TLS_RSA_WITH_RC4_128_SHA,
|
|
||||||
}
|
|
||||||
|
|
||||||
var cipherSuitesPreferenceOrderNoAES = []uint16{
|
|
||||||
// ChaCha20Poly1305
|
|
||||||
TLS_ECDHE_ECDSA_WITH_CHACHA20_POLY1305, TLS_ECDHE_RSA_WITH_CHACHA20_POLY1305,
|
|
||||||
|
|
||||||
// AES-GCM w/ ECDHE
|
|
||||||
TLS_ECDHE_ECDSA_WITH_AES_128_GCM_SHA256, TLS_ECDHE_RSA_WITH_AES_128_GCM_SHA256,
|
|
||||||
TLS_ECDHE_ECDSA_WITH_AES_256_GCM_SHA384, TLS_ECDHE_RSA_WITH_AES_256_GCM_SHA384,
|
|
||||||
|
|
||||||
// The rest of cipherSuitesPreferenceOrder.
|
|
||||||
TLS_ECDHE_ECDSA_WITH_AES_128_CBC_SHA, TLS_ECDHE_RSA_WITH_AES_128_CBC_SHA,
|
|
||||||
TLS_ECDHE_ECDSA_WITH_AES_256_CBC_SHA, TLS_ECDHE_RSA_WITH_AES_256_CBC_SHA,
|
|
||||||
TLS_RSA_WITH_AES_128_GCM_SHA256,
|
|
||||||
TLS_RSA_WITH_AES_256_GCM_SHA384,
|
|
||||||
TLS_RSA_WITH_AES_128_CBC_SHA,
|
|
||||||
TLS_RSA_WITH_AES_256_CBC_SHA,
|
|
||||||
TLS_ECDHE_RSA_WITH_3DES_EDE_CBC_SHA,
|
|
||||||
TLS_RSA_WITH_3DES_EDE_CBC_SHA,
|
|
||||||
TLS_ECDHE_ECDSA_WITH_AES_128_CBC_SHA256, TLS_ECDHE_RSA_WITH_AES_128_CBC_SHA256,
|
|
||||||
TLS_RSA_WITH_AES_128_CBC_SHA256,
|
|
||||||
TLS_ECDHE_ECDSA_WITH_RC4_128_SHA, TLS_ECDHE_RSA_WITH_RC4_128_SHA,
|
|
||||||
TLS_RSA_WITH_RC4_128_SHA,
|
|
||||||
}
|
|
||||||
|
|
||||||
// disabledCipherSuites are not used unless explicitly listed in
|
|
||||||
// Config.CipherSuites. They MUST be at the end of cipherSuitesPreferenceOrder.
|
|
||||||
var disabledCipherSuites = []uint16{
|
|
||||||
// CBC_SHA256
|
|
||||||
TLS_ECDHE_ECDSA_WITH_AES_128_CBC_SHA256, TLS_ECDHE_RSA_WITH_AES_128_CBC_SHA256,
|
|
||||||
TLS_RSA_WITH_AES_128_CBC_SHA256,
|
|
||||||
|
|
||||||
// RC4
|
|
||||||
TLS_ECDHE_ECDSA_WITH_RC4_128_SHA, TLS_ECDHE_RSA_WITH_RC4_128_SHA,
|
|
||||||
TLS_RSA_WITH_RC4_128_SHA,
|
|
||||||
}
|
|
||||||
|
|
||||||
var (
|
|
||||||
defaultCipherSuitesLen = len(cipherSuitesPreferenceOrder) - len(disabledCipherSuites)
|
|
||||||
defaultCipherSuites = cipherSuitesPreferenceOrder[:defaultCipherSuitesLen]
|
|
||||||
)
|
|
||||||
|
|
||||||
// defaultCipherSuitesTLS13 is also the preference order, since there are no
|
|
||||||
// disabled by default TLS 1.3 cipher suites. The same AES vs ChaCha20 logic as
|
|
||||||
// cipherSuitesPreferenceOrder applies.
|
|
||||||
var defaultCipherSuitesTLS13 = []uint16{
|
|
||||||
TLS_AES_128_GCM_SHA256,
|
|
||||||
TLS_AES_256_GCM_SHA384,
|
|
||||||
TLS_CHACHA20_POLY1305_SHA256,
|
|
||||||
}
|
|
||||||
|
|
||||||
var defaultCipherSuitesTLS13NoAES = []uint16{
|
|
||||||
TLS_CHACHA20_POLY1305_SHA256,
|
|
||||||
TLS_AES_128_GCM_SHA256,
|
|
||||||
TLS_AES_256_GCM_SHA384,
|
|
||||||
}
|
|
||||||
|
|
||||||
var (
|
|
||||||
hasGCMAsmAMD64 = cpu.X86.HasAES && cpu.X86.HasPCLMULQDQ
|
|
||||||
hasGCMAsmARM64 = cpu.ARM64.HasAES && cpu.ARM64.HasPMULL
|
|
||||||
// Keep in sync with crypto/aes/cipher_s390x.go.
|
|
||||||
hasGCMAsmS390X = cpu.S390X.HasAES && cpu.S390X.HasAESCBC && cpu.S390X.HasAESCTR &&
|
|
||||||
(cpu.S390X.HasGHASH || cpu.S390X.HasAESGCM)
|
|
||||||
|
|
||||||
hasAESGCMHardwareSupport = runtime.GOARCH == "amd64" && hasGCMAsmAMD64 ||
|
|
||||||
runtime.GOARCH == "arm64" && hasGCMAsmARM64 ||
|
|
||||||
runtime.GOARCH == "s390x" && hasGCMAsmS390X
|
|
||||||
)
|
|
||||||
|
|
||||||
var aesgcmCiphers = map[uint16]bool{
|
|
||||||
// TLS 1.2
|
|
||||||
TLS_ECDHE_RSA_WITH_AES_128_GCM_SHA256: true,
|
|
||||||
TLS_ECDHE_RSA_WITH_AES_256_GCM_SHA384: true,
|
|
||||||
TLS_ECDHE_ECDSA_WITH_AES_128_GCM_SHA256: true,
|
|
||||||
TLS_ECDHE_ECDSA_WITH_AES_256_GCM_SHA384: true,
|
|
||||||
// TLS 1.3
|
|
||||||
TLS_AES_128_GCM_SHA256: true,
|
|
||||||
TLS_AES_256_GCM_SHA384: true,
|
|
||||||
}
|
|
||||||
|
|
||||||
var nonAESGCMAEADCiphers = map[uint16]bool{
|
|
||||||
// TLS 1.2
|
|
||||||
TLS_ECDHE_RSA_WITH_CHACHA20_POLY1305: true,
|
|
||||||
TLS_ECDHE_ECDSA_WITH_CHACHA20_POLY1305: true,
|
|
||||||
// TLS 1.3
|
|
||||||
TLS_CHACHA20_POLY1305_SHA256: true,
|
|
||||||
}
|
|
||||||
|
|
||||||
// aesgcmPreferred returns whether the first known cipher in the preference list
|
|
||||||
// is an AES-GCM cipher, implying the peer has hardware support for it.
|
|
||||||
func aesgcmPreferred(ciphers []uint16) bool {
|
|
||||||
for _, cID := range ciphers {
|
|
||||||
if c := cipherSuiteByID(cID); c != nil {
|
|
||||||
return aesgcmCiphers[cID]
|
|
||||||
}
|
|
||||||
if c := cipherSuiteTLS13ByID(cID); c != nil {
|
|
||||||
return aesgcmCiphers[cID]
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
|
|
||||||
func cipherRC4(key, iv []byte, isRead bool) interface{} {
|
|
||||||
cipher, _ := rc4.NewCipher(key)
|
|
||||||
return cipher
|
|
||||||
}
|
|
||||||
|
|
||||||
func cipher3DES(key, iv []byte, isRead bool) interface{} {
|
|
||||||
block, _ := des.NewTripleDESCipher(key)
|
|
||||||
if isRead {
|
|
||||||
return cipher.NewCBCDecrypter(block, iv)
|
|
||||||
}
|
|
||||||
return cipher.NewCBCEncrypter(block, iv)
|
|
||||||
}
|
|
||||||
|
|
||||||
func cipherAES(key, iv []byte, isRead bool) interface{} {
|
|
||||||
block, _ := aes.NewCipher(key)
|
|
||||||
if isRead {
|
|
||||||
return cipher.NewCBCDecrypter(block, iv)
|
|
||||||
}
|
|
||||||
return cipher.NewCBCEncrypter(block, iv)
|
|
||||||
}
|
|
||||||
|
|
||||||
// macSHA1 returns a SHA-1 based constant time MAC.
|
|
||||||
func macSHA1(key []byte) hash.Hash {
|
|
||||||
return hmac.New(newConstantTimeHash(sha1.New), key)
|
|
||||||
}
|
|
||||||
|
|
||||||
// macSHA256 returns a SHA-256 based MAC. This is only supported in TLS 1.2 and
|
|
||||||
// is currently only used in disabled-by-default cipher suites.
|
|
||||||
func macSHA256(key []byte) hash.Hash {
|
|
||||||
return hmac.New(sha256.New, key)
|
|
||||||
}
|
|
||||||
|
|
||||||
type aead interface {
|
|
||||||
cipher.AEAD
|
|
||||||
|
|
||||||
// explicitNonceLen returns the number of bytes of explicit nonce
|
|
||||||
// included in each record. This is eight for older AEADs and
|
|
||||||
// zero for modern ones.
|
|
||||||
explicitNonceLen() int
|
|
||||||
}
|
|
||||||
|
|
||||||
const (
|
|
||||||
aeadNonceLength = 12
|
|
||||||
noncePrefixLength = 4
|
|
||||||
)
|
|
||||||
|
|
||||||
// prefixNonceAEAD wraps an AEAD and prefixes a fixed portion of the nonce to
|
|
||||||
// each call.
|
|
||||||
type prefixNonceAEAD struct {
|
|
||||||
// nonce contains the fixed part of the nonce in the first four bytes.
|
|
||||||
nonce [aeadNonceLength]byte
|
|
||||||
aead cipher.AEAD
|
|
||||||
}
|
|
||||||
|
|
||||||
func (f *prefixNonceAEAD) NonceSize() int { return aeadNonceLength - noncePrefixLength }
|
|
||||||
func (f *prefixNonceAEAD) Overhead() int { return f.aead.Overhead() }
|
|
||||||
func (f *prefixNonceAEAD) explicitNonceLen() int { return f.NonceSize() }
|
|
||||||
|
|
||||||
func (f *prefixNonceAEAD) Seal(out, nonce, plaintext, additionalData []byte) []byte {
|
|
||||||
copy(f.nonce[4:], nonce)
|
|
||||||
return f.aead.Seal(out, f.nonce[:], plaintext, additionalData)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (f *prefixNonceAEAD) Open(out, nonce, ciphertext, additionalData []byte) ([]byte, error) {
|
|
||||||
copy(f.nonce[4:], nonce)
|
|
||||||
return f.aead.Open(out, f.nonce[:], ciphertext, additionalData)
|
|
||||||
}
|
|
||||||
|
|
||||||
// xoredNonceAEAD wraps an AEAD by XORing in a fixed pattern to the nonce
|
|
||||||
// before each call.
|
|
||||||
type xorNonceAEAD struct {
|
|
||||||
nonceMask [aeadNonceLength]byte
|
|
||||||
aead cipher.AEAD
|
|
||||||
}
|
|
||||||
|
|
||||||
func (f *xorNonceAEAD) NonceSize() int { return 8 } // 64-bit sequence number
|
|
||||||
func (f *xorNonceAEAD) Overhead() int { return f.aead.Overhead() }
|
|
||||||
func (f *xorNonceAEAD) explicitNonceLen() int { return 0 }
|
|
||||||
|
|
||||||
func (f *xorNonceAEAD) Seal(out, nonce, plaintext, additionalData []byte) []byte {
|
|
||||||
for i, b := range nonce {
|
|
||||||
f.nonceMask[4+i] ^= b
|
|
||||||
}
|
|
||||||
result := f.aead.Seal(out, f.nonceMask[:], plaintext, additionalData)
|
|
||||||
for i, b := range nonce {
|
|
||||||
f.nonceMask[4+i] ^= b
|
|
||||||
}
|
|
||||||
|
|
||||||
return result
|
|
||||||
}
|
|
||||||
|
|
||||||
func (f *xorNonceAEAD) Open(out, nonce, ciphertext, additionalData []byte) ([]byte, error) {
|
|
||||||
for i, b := range nonce {
|
|
||||||
f.nonceMask[4+i] ^= b
|
|
||||||
}
|
|
||||||
result, err := f.aead.Open(out, f.nonceMask[:], ciphertext, additionalData)
|
|
||||||
for i, b := range nonce {
|
|
||||||
f.nonceMask[4+i] ^= b
|
|
||||||
}
|
|
||||||
|
|
||||||
return result, err
|
|
||||||
}
|
|
||||||
|
|
||||||
func aeadAESGCM(key, noncePrefix []byte) aead {
|
|
||||||
if len(noncePrefix) != noncePrefixLength {
|
|
||||||
panic("tls: internal error: wrong nonce length")
|
|
||||||
}
|
|
||||||
aes, err := aes.NewCipher(key)
|
|
||||||
if err != nil {
|
|
||||||
panic(err)
|
|
||||||
}
|
|
||||||
aead, err := cipher.NewGCM(aes)
|
|
||||||
if err != nil {
|
|
||||||
panic(err)
|
|
||||||
}
|
|
||||||
|
|
||||||
ret := &prefixNonceAEAD{aead: aead}
|
|
||||||
copy(ret.nonce[:], noncePrefix)
|
|
||||||
return ret
|
|
||||||
}
|
|
||||||
|
|
||||||
// AEADAESGCMTLS13 creates a new AES-GCM AEAD for TLS 1.3
|
|
||||||
func AEADAESGCMTLS13(key, fixedNonce []byte) cipher.AEAD {
|
|
||||||
return aeadAESGCMTLS13(key, fixedNonce)
|
|
||||||
}
|
|
||||||
|
|
||||||
func aeadAESGCMTLS13(key, nonceMask []byte) aead {
|
|
||||||
if len(nonceMask) != aeadNonceLength {
|
|
||||||
panic("tls: internal error: wrong nonce length")
|
|
||||||
}
|
|
||||||
aes, err := aes.NewCipher(key)
|
|
||||||
if err != nil {
|
|
||||||
panic(err)
|
|
||||||
}
|
|
||||||
aead, err := cipher.NewGCM(aes)
|
|
||||||
if err != nil {
|
|
||||||
panic(err)
|
|
||||||
}
|
|
||||||
|
|
||||||
ret := &xorNonceAEAD{aead: aead}
|
|
||||||
copy(ret.nonceMask[:], nonceMask)
|
|
||||||
return ret
|
|
||||||
}
|
|
||||||
|
|
||||||
func aeadChaCha20Poly1305(key, nonceMask []byte) aead {
|
|
||||||
if len(nonceMask) != aeadNonceLength {
|
|
||||||
panic("tls: internal error: wrong nonce length")
|
|
||||||
}
|
|
||||||
aead, err := chacha20poly1305.New(key)
|
|
||||||
if err != nil {
|
|
||||||
panic(err)
|
|
||||||
}
|
|
||||||
|
|
||||||
ret := &xorNonceAEAD{aead: aead}
|
|
||||||
copy(ret.nonceMask[:], nonceMask)
|
|
||||||
return ret
|
|
||||||
}
|
|
||||||
|
|
||||||
type constantTimeHash interface {
|
|
||||||
hash.Hash
|
|
||||||
ConstantTimeSum(b []byte) []byte
|
|
||||||
}
|
|
||||||
|
|
||||||
// cthWrapper wraps any hash.Hash that implements ConstantTimeSum, and replaces
|
|
||||||
// with that all calls to Sum. It's used to obtain a ConstantTimeSum-based HMAC.
|
|
||||||
type cthWrapper struct {
|
|
||||||
h constantTimeHash
|
|
||||||
}
|
|
||||||
|
|
||||||
func (c *cthWrapper) Size() int { return c.h.Size() }
|
|
||||||
func (c *cthWrapper) BlockSize() int { return c.h.BlockSize() }
|
|
||||||
func (c *cthWrapper) Reset() { c.h.Reset() }
|
|
||||||
func (c *cthWrapper) Write(p []byte) (int, error) { return c.h.Write(p) }
|
|
||||||
func (c *cthWrapper) Sum(b []byte) []byte { return c.h.ConstantTimeSum(b) }
|
|
||||||
|
|
||||||
func newConstantTimeHash(h func() hash.Hash) func() hash.Hash {
|
|
||||||
return func() hash.Hash {
|
|
||||||
return &cthWrapper{h().(constantTimeHash)}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// tls10MAC implements the TLS 1.0 MAC function. RFC 2246, Section 6.2.3.
|
|
||||||
func tls10MAC(h hash.Hash, out, seq, header, data, extra []byte) []byte {
|
|
||||||
h.Reset()
|
|
||||||
h.Write(seq)
|
|
||||||
h.Write(header)
|
|
||||||
h.Write(data)
|
|
||||||
res := h.Sum(out)
|
|
||||||
if extra != nil {
|
|
||||||
h.Write(extra)
|
|
||||||
}
|
|
||||||
return res
|
|
||||||
}
|
|
||||||
|
|
||||||
func rsaKA(version uint16) keyAgreement {
|
|
||||||
return rsaKeyAgreement{}
|
|
||||||
}
|
|
||||||
|
|
||||||
func ecdheECDSAKA(version uint16) keyAgreement {
|
|
||||||
return &ecdheKeyAgreement{
|
|
||||||
isRSA: false,
|
|
||||||
version: version,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func ecdheRSAKA(version uint16) keyAgreement {
|
|
||||||
return &ecdheKeyAgreement{
|
|
||||||
isRSA: true,
|
|
||||||
version: version,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// mutualCipherSuite returns a cipherSuite given a list of supported
|
|
||||||
// ciphersuites and the id requested by the peer.
|
|
||||||
func mutualCipherSuite(have []uint16, want uint16) *cipherSuite {
|
|
||||||
for _, id := range have {
|
|
||||||
if id == want {
|
|
||||||
return cipherSuiteByID(id)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func cipherSuiteByID(id uint16) *cipherSuite {
|
|
||||||
for _, cipherSuite := range cipherSuites {
|
|
||||||
if cipherSuite.id == id {
|
|
||||||
return cipherSuite
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func mutualCipherSuiteTLS13(have []uint16, want uint16) *cipherSuiteTLS13 {
|
|
||||||
for _, id := range have {
|
|
||||||
if id == want {
|
|
||||||
return cipherSuiteTLS13ByID(id)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func cipherSuiteTLS13ByID(id uint16) *cipherSuiteTLS13 {
|
|
||||||
for _, cipherSuite := range cipherSuitesTLS13 {
|
|
||||||
if cipherSuite.id == id {
|
|
||||||
return cipherSuite
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// A list of cipher suite IDs that are, or have been, implemented by this
|
|
||||||
// package.
|
|
||||||
//
|
|
||||||
// See https://www.iana.org/assignments/tls-parameters/tls-parameters.xml
|
|
||||||
const (
|
|
||||||
// TLS 1.0 - 1.2 cipher suites.
|
|
||||||
TLS_RSA_WITH_RC4_128_SHA uint16 = 0x0005
|
|
||||||
TLS_RSA_WITH_3DES_EDE_CBC_SHA uint16 = 0x000a
|
|
||||||
TLS_RSA_WITH_AES_128_CBC_SHA uint16 = 0x002f
|
|
||||||
TLS_RSA_WITH_AES_256_CBC_SHA uint16 = 0x0035
|
|
||||||
TLS_RSA_WITH_AES_128_CBC_SHA256 uint16 = 0x003c
|
|
||||||
TLS_RSA_WITH_AES_128_GCM_SHA256 uint16 = 0x009c
|
|
||||||
TLS_RSA_WITH_AES_256_GCM_SHA384 uint16 = 0x009d
|
|
||||||
TLS_ECDHE_ECDSA_WITH_RC4_128_SHA uint16 = 0xc007
|
|
||||||
TLS_ECDHE_ECDSA_WITH_AES_128_CBC_SHA uint16 = 0xc009
|
|
||||||
TLS_ECDHE_ECDSA_WITH_AES_256_CBC_SHA uint16 = 0xc00a
|
|
||||||
TLS_ECDHE_RSA_WITH_RC4_128_SHA uint16 = 0xc011
|
|
||||||
TLS_ECDHE_RSA_WITH_3DES_EDE_CBC_SHA uint16 = 0xc012
|
|
||||||
TLS_ECDHE_RSA_WITH_AES_128_CBC_SHA uint16 = 0xc013
|
|
||||||
TLS_ECDHE_RSA_WITH_AES_256_CBC_SHA uint16 = 0xc014
|
|
||||||
TLS_ECDHE_ECDSA_WITH_AES_128_CBC_SHA256 uint16 = 0xc023
|
|
||||||
TLS_ECDHE_RSA_WITH_AES_128_CBC_SHA256 uint16 = 0xc027
|
|
||||||
TLS_ECDHE_RSA_WITH_AES_128_GCM_SHA256 uint16 = 0xc02f
|
|
||||||
TLS_ECDHE_ECDSA_WITH_AES_128_GCM_SHA256 uint16 = 0xc02b
|
|
||||||
TLS_ECDHE_RSA_WITH_AES_256_GCM_SHA384 uint16 = 0xc030
|
|
||||||
TLS_ECDHE_ECDSA_WITH_AES_256_GCM_SHA384 uint16 = 0xc02c
|
|
||||||
TLS_ECDHE_RSA_WITH_CHACHA20_POLY1305_SHA256 uint16 = 0xcca8
|
|
||||||
TLS_ECDHE_ECDSA_WITH_CHACHA20_POLY1305_SHA256 uint16 = 0xcca9
|
|
||||||
|
|
||||||
// TLS 1.3 cipher suites.
|
|
||||||
TLS_AES_128_GCM_SHA256 uint16 = 0x1301
|
|
||||||
TLS_AES_256_GCM_SHA384 uint16 = 0x1302
|
|
||||||
TLS_CHACHA20_POLY1305_SHA256 uint16 = 0x1303
|
|
||||||
|
|
||||||
// TLS_FALLBACK_SCSV isn't a standard cipher suite but an indicator
|
|
||||||
// that the client is doing version fallback. See RFC 7507.
|
|
||||||
TLS_FALLBACK_SCSV uint16 = 0x5600
|
|
||||||
|
|
||||||
// Legacy names for the corresponding cipher suites with the correct _SHA256
|
|
||||||
// suffix, retained for backward compatibility.
|
|
||||||
TLS_ECDHE_RSA_WITH_CHACHA20_POLY1305 = TLS_ECDHE_RSA_WITH_CHACHA20_POLY1305_SHA256
|
|
||||||
TLS_ECDHE_ECDSA_WITH_CHACHA20_POLY1305 = TLS_ECDHE_ECDSA_WITH_CHACHA20_POLY1305_SHA256
|
|
||||||
)
|
|
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
|
@ -1,10 +0,0 @@
|
||||||
module github.com/marten-seemann/qtls-go1-17
|
|
||||||
|
|
||||||
go 1.17
|
|
||||||
|
|
||||||
require (
|
|
||||||
github.com/golang/mock v1.6.0
|
|
||||||
github.com/marten-seemann/qtls-go1-15 v0.1.4 // indirect
|
|
||||||
golang.org/x/crypto v0.0.0-20200221231518-2aa609cf4a9d
|
|
||||||
golang.org/x/sys v0.0.0-20210510120138-977fb7262007
|
|
||||||
)
|
|
|
@ -1,34 +0,0 @@
|
||||||
github.com/golang/mock v1.4.4/go.mod h1:l3mdAwkq5BuhzHwde/uurv3sEJeZMXNpwsxVWU71h+4=
|
|
||||||
github.com/golang/mock v1.6.0 h1:ErTB+efbowRARo13NNdxyJji2egdxLGQhRaY+DUumQc=
|
|
||||||
github.com/golang/mock v1.6.0/go.mod h1:p6yTPP+5HYm5mzsMV8JkE6ZKdX+/wYM6Hr+LicevLPs=
|
|
||||||
github.com/marten-seemann/qtls-go1-15 v0.1.4 h1:RehYMOyRW8hPVEja1KBVsFVNSm35Jj9Mvs5yNoZZ28A=
|
|
||||||
github.com/marten-seemann/qtls-go1-15 v0.1.4/go.mod h1:GyFwywLKkRt+6mfU99csTEY1joMZz5vmB1WNZH3P81I=
|
|
||||||
github.com/yuin/goldmark v1.3.5/go.mod h1:mwnBkeHKe2W/ZEtQ+71ViKU8L12m81fl3OWwC1Zlc8k=
|
|
||||||
golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w=
|
|
||||||
golang.org/x/crypto v0.0.0-20191011191535-87dc89f01550/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI=
|
|
||||||
golang.org/x/crypto v0.0.0-20200221231518-2aa609cf4a9d h1:1ZiEyfaQIg3Qh0EoqpwAakHVhecoE5wlSg5GjnafJGw=
|
|
||||||
golang.org/x/crypto v0.0.0-20200221231518-2aa609cf4a9d/go.mod h1:LzIPMQfyMNhhGPhUkYOs5KpL4U8rLKemX1yGLhDgUto=
|
|
||||||
golang.org/x/mod v0.4.2/go.mod h1:s0Qsj1ACt9ePp/hMypM3fl4fZqREWJwdYDEqhRiZZUA=
|
|
||||||
golang.org/x/net v0.0.0-20190311183353-d8887717615a/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg=
|
|
||||||
golang.org/x/net v0.0.0-20190404232315-eb5bcb51f2a3/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg=
|
|
||||||
golang.org/x/net v0.0.0-20190620200207-3b0461eec859/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s=
|
|
||||||
golang.org/x/net v0.0.0-20210405180319-a5a99cb37ef4/go.mod h1:p54w0d4576C0XHj96bSt6lcn1PtDYWL6XObtHCRCNQM=
|
|
||||||
golang.org/x/sync v0.0.0-20190423024810-112230192c58/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
|
|
||||||
golang.org/x/sync v0.0.0-20210220032951-036812b2e83c/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
|
|
||||||
golang.org/x/sys v0.0.0-20190215142949-d0b11bdaac8a/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
|
|
||||||
golang.org/x/sys v0.0.0-20190412213103-97732733099d/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
|
||||||
golang.org/x/sys v0.0.0-20200223170610-d5e6a3e2c0ae/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
|
||||||
golang.org/x/sys v0.0.0-20201119102817-f84b799fce68/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
|
||||||
golang.org/x/sys v0.0.0-20210330210617-4fbd30eecc44/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
|
||||||
golang.org/x/sys v0.0.0-20210510120138-977fb7262007 h1:gG67DSER+11cZvqIMb8S8bt0vZtiN6xWYARwirrOSfE=
|
|
||||||
golang.org/x/sys v0.0.0-20210510120138-977fb7262007/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
|
|
||||||
golang.org/x/term v0.0.0-20201126162022-7de9c90e9dd1/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo=
|
|
||||||
golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ=
|
|
||||||
golang.org/x/text v0.3.3/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ=
|
|
||||||
golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ=
|
|
||||||
golang.org/x/tools v0.0.0-20190425150028-36563e24a262/go.mod h1:RgjU9mgBXZiqYHBnxXauZ1Gv1EHHAz9KjViQ78xBX0Q=
|
|
||||||
golang.org/x/tools v0.0.0-20191119224855-298f0cb1881e/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo=
|
|
||||||
golang.org/x/tools v0.1.1/go.mod h1:o0xws9oXOQQZyjljx8fwUC0k7L1pTE6eaCbjGeHmOkk=
|
|
||||||
golang.org/x/xerrors v0.0.0-20190717185122-a985d3407aa7/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0=
|
|
||||||
golang.org/x/xerrors v0.0.0-20191011141410-1b5146add898/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0=
|
|
||||||
golang.org/x/xerrors v0.0.0-20200804184101-5ec99f83aff1/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0=
|
|
File diff suppressed because it is too large
Load Diff
|
@ -1,732 +0,0 @@
|
||||||
// Copyright 2018 The Go Authors. All rights reserved.
|
|
||||||
// Use of this source code is governed by a BSD-style
|
|
||||||
// license that can be found in the LICENSE file.
|
|
||||||
|
|
||||||
package qtls
|
|
||||||
|
|
||||||
import (
|
|
||||||
"bytes"
|
|
||||||
"context"
|
|
||||||
"crypto"
|
|
||||||
"crypto/hmac"
|
|
||||||
"crypto/rsa"
|
|
||||||
"encoding/binary"
|
|
||||||
"errors"
|
|
||||||
"hash"
|
|
||||||
"sync/atomic"
|
|
||||||
"time"
|
|
||||||
|
|
||||||
"golang.org/x/crypto/cryptobyte"
|
|
||||||
)
|
|
||||||
|
|
||||||
type clientHandshakeStateTLS13 struct {
|
|
||||||
c *Conn
|
|
||||||
ctx context.Context
|
|
||||||
serverHello *serverHelloMsg
|
|
||||||
hello *clientHelloMsg
|
|
||||||
ecdheParams ecdheParameters
|
|
||||||
|
|
||||||
session *clientSessionState
|
|
||||||
earlySecret []byte
|
|
||||||
binderKey []byte
|
|
||||||
|
|
||||||
certReq *certificateRequestMsgTLS13
|
|
||||||
usingPSK bool
|
|
||||||
sentDummyCCS bool
|
|
||||||
suite *cipherSuiteTLS13
|
|
||||||
transcript hash.Hash
|
|
||||||
masterSecret []byte
|
|
||||||
trafficSecret []byte // client_application_traffic_secret_0
|
|
||||||
}
|
|
||||||
|
|
||||||
// handshake requires hs.c, hs.hello, hs.serverHello, hs.ecdheParams, and,
|
|
||||||
// optionally, hs.session, hs.earlySecret and hs.binderKey to be set.
|
|
||||||
func (hs *clientHandshakeStateTLS13) handshake() error {
|
|
||||||
c := hs.c
|
|
||||||
|
|
||||||
// The server must not select TLS 1.3 in a renegotiation. See RFC 8446,
|
|
||||||
// sections 4.1.2 and 4.1.3.
|
|
||||||
if c.handshakes > 0 {
|
|
||||||
c.sendAlert(alertProtocolVersion)
|
|
||||||
return errors.New("tls: server selected TLS 1.3 in a renegotiation")
|
|
||||||
}
|
|
||||||
|
|
||||||
// Consistency check on the presence of a keyShare and its parameters.
|
|
||||||
if hs.ecdheParams == nil || len(hs.hello.keyShares) != 1 {
|
|
||||||
return c.sendAlert(alertInternalError)
|
|
||||||
}
|
|
||||||
|
|
||||||
if err := hs.checkServerHelloOrHRR(); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
hs.transcript = hs.suite.hash.New()
|
|
||||||
hs.transcript.Write(hs.hello.marshal())
|
|
||||||
|
|
||||||
if bytes.Equal(hs.serverHello.random, helloRetryRequestRandom) {
|
|
||||||
if err := hs.sendDummyChangeCipherSpec(); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
if err := hs.processHelloRetryRequest(); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
hs.transcript.Write(hs.serverHello.marshal())
|
|
||||||
|
|
||||||
c.buffering = true
|
|
||||||
if err := hs.processServerHello(); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
if err := hs.sendDummyChangeCipherSpec(); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
if err := hs.establishHandshakeKeys(); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
if err := hs.readServerParameters(); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
if err := hs.readServerCertificate(); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
if err := hs.readServerFinished(); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
if err := hs.sendClientCertificate(); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
if err := hs.sendClientFinished(); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
if _, err := c.flush(); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
atomic.StoreUint32(&c.handshakeStatus, 1)
|
|
||||||
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// checkServerHelloOrHRR does validity checks that apply to both ServerHello and
|
|
||||||
// HelloRetryRequest messages. It sets hs.suite.
|
|
||||||
func (hs *clientHandshakeStateTLS13) checkServerHelloOrHRR() error {
|
|
||||||
c := hs.c
|
|
||||||
|
|
||||||
if hs.serverHello.supportedVersion == 0 {
|
|
||||||
c.sendAlert(alertMissingExtension)
|
|
||||||
return errors.New("tls: server selected TLS 1.3 using the legacy version field")
|
|
||||||
}
|
|
||||||
|
|
||||||
if hs.serverHello.supportedVersion != VersionTLS13 {
|
|
||||||
c.sendAlert(alertIllegalParameter)
|
|
||||||
return errors.New("tls: server selected an invalid version after a HelloRetryRequest")
|
|
||||||
}
|
|
||||||
|
|
||||||
if hs.serverHello.vers != VersionTLS12 {
|
|
||||||
c.sendAlert(alertIllegalParameter)
|
|
||||||
return errors.New("tls: server sent an incorrect legacy version")
|
|
||||||
}
|
|
||||||
|
|
||||||
if hs.serverHello.ocspStapling ||
|
|
||||||
hs.serverHello.ticketSupported ||
|
|
||||||
hs.serverHello.secureRenegotiationSupported ||
|
|
||||||
len(hs.serverHello.secureRenegotiation) != 0 ||
|
|
||||||
len(hs.serverHello.alpnProtocol) != 0 ||
|
|
||||||
len(hs.serverHello.scts) != 0 {
|
|
||||||
c.sendAlert(alertUnsupportedExtension)
|
|
||||||
return errors.New("tls: server sent a ServerHello extension forbidden in TLS 1.3")
|
|
||||||
}
|
|
||||||
|
|
||||||
if !bytes.Equal(hs.hello.sessionId, hs.serverHello.sessionId) {
|
|
||||||
c.sendAlert(alertIllegalParameter)
|
|
||||||
return errors.New("tls: server did not echo the legacy session ID")
|
|
||||||
}
|
|
||||||
|
|
||||||
if hs.serverHello.compressionMethod != compressionNone {
|
|
||||||
c.sendAlert(alertIllegalParameter)
|
|
||||||
return errors.New("tls: server selected unsupported compression format")
|
|
||||||
}
|
|
||||||
|
|
||||||
selectedSuite := mutualCipherSuiteTLS13(hs.hello.cipherSuites, hs.serverHello.cipherSuite)
|
|
||||||
if hs.suite != nil && selectedSuite != hs.suite {
|
|
||||||
c.sendAlert(alertIllegalParameter)
|
|
||||||
return errors.New("tls: server changed cipher suite after a HelloRetryRequest")
|
|
||||||
}
|
|
||||||
if selectedSuite == nil {
|
|
||||||
c.sendAlert(alertIllegalParameter)
|
|
||||||
return errors.New("tls: server chose an unconfigured cipher suite")
|
|
||||||
}
|
|
||||||
hs.suite = selectedSuite
|
|
||||||
c.cipherSuite = hs.suite.id
|
|
||||||
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// sendDummyChangeCipherSpec sends a ChangeCipherSpec record for compatibility
|
|
||||||
// with middleboxes that didn't implement TLS correctly. See RFC 8446, Appendix D.4.
|
|
||||||
func (hs *clientHandshakeStateTLS13) sendDummyChangeCipherSpec() error {
|
|
||||||
if hs.sentDummyCCS {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
hs.sentDummyCCS = true
|
|
||||||
|
|
||||||
_, err := hs.c.writeRecord(recordTypeChangeCipherSpec, []byte{1})
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
// processHelloRetryRequest handles the HRR in hs.serverHello, modifies and
|
|
||||||
// resends hs.hello, and reads the new ServerHello into hs.serverHello.
|
|
||||||
func (hs *clientHandshakeStateTLS13) processHelloRetryRequest() error {
|
|
||||||
c := hs.c
|
|
||||||
|
|
||||||
// The first ClientHello gets double-hashed into the transcript upon a
|
|
||||||
// HelloRetryRequest. (The idea is that the server might offload transcript
|
|
||||||
// storage to the client in the cookie.) See RFC 8446, Section 4.4.1.
|
|
||||||
chHash := hs.transcript.Sum(nil)
|
|
||||||
hs.transcript.Reset()
|
|
||||||
hs.transcript.Write([]byte{typeMessageHash, 0, 0, uint8(len(chHash))})
|
|
||||||
hs.transcript.Write(chHash)
|
|
||||||
hs.transcript.Write(hs.serverHello.marshal())
|
|
||||||
|
|
||||||
// The only HelloRetryRequest extensions we support are key_share and
|
|
||||||
// cookie, and clients must abort the handshake if the HRR would not result
|
|
||||||
// in any change in the ClientHello.
|
|
||||||
if hs.serverHello.selectedGroup == 0 && hs.serverHello.cookie == nil {
|
|
||||||
c.sendAlert(alertIllegalParameter)
|
|
||||||
return errors.New("tls: server sent an unnecessary HelloRetryRequest message")
|
|
||||||
}
|
|
||||||
|
|
||||||
if hs.serverHello.cookie != nil {
|
|
||||||
hs.hello.cookie = hs.serverHello.cookie
|
|
||||||
}
|
|
||||||
|
|
||||||
if hs.serverHello.serverShare.group != 0 {
|
|
||||||
c.sendAlert(alertDecodeError)
|
|
||||||
return errors.New("tls: received malformed key_share extension")
|
|
||||||
}
|
|
||||||
|
|
||||||
// If the server sent a key_share extension selecting a group, ensure it's
|
|
||||||
// a group we advertised but did not send a key share for, and send a key
|
|
||||||
// share for it this time.
|
|
||||||
if curveID := hs.serverHello.selectedGroup; curveID != 0 {
|
|
||||||
curveOK := false
|
|
||||||
for _, id := range hs.hello.supportedCurves {
|
|
||||||
if id == curveID {
|
|
||||||
curveOK = true
|
|
||||||
break
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if !curveOK {
|
|
||||||
c.sendAlert(alertIllegalParameter)
|
|
||||||
return errors.New("tls: server selected unsupported group")
|
|
||||||
}
|
|
||||||
if hs.ecdheParams.CurveID() == curveID {
|
|
||||||
c.sendAlert(alertIllegalParameter)
|
|
||||||
return errors.New("tls: server sent an unnecessary HelloRetryRequest key_share")
|
|
||||||
}
|
|
||||||
if _, ok := curveForCurveID(curveID); curveID != X25519 && !ok {
|
|
||||||
c.sendAlert(alertInternalError)
|
|
||||||
return errors.New("tls: CurvePreferences includes unsupported curve")
|
|
||||||
}
|
|
||||||
params, err := generateECDHEParameters(c.config.rand(), curveID)
|
|
||||||
if err != nil {
|
|
||||||
c.sendAlert(alertInternalError)
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
hs.ecdheParams = params
|
|
||||||
hs.hello.keyShares = []keyShare{{group: curveID, data: params.PublicKey()}}
|
|
||||||
}
|
|
||||||
|
|
||||||
hs.hello.raw = nil
|
|
||||||
if len(hs.hello.pskIdentities) > 0 {
|
|
||||||
pskSuite := cipherSuiteTLS13ByID(hs.session.cipherSuite)
|
|
||||||
if pskSuite == nil {
|
|
||||||
return c.sendAlert(alertInternalError)
|
|
||||||
}
|
|
||||||
if pskSuite.hash == hs.suite.hash {
|
|
||||||
// Update binders and obfuscated_ticket_age.
|
|
||||||
ticketAge := uint32(c.config.time().Sub(hs.session.receivedAt) / time.Millisecond)
|
|
||||||
hs.hello.pskIdentities[0].obfuscatedTicketAge = ticketAge + hs.session.ageAdd
|
|
||||||
|
|
||||||
transcript := hs.suite.hash.New()
|
|
||||||
transcript.Write([]byte{typeMessageHash, 0, 0, uint8(len(chHash))})
|
|
||||||
transcript.Write(chHash)
|
|
||||||
transcript.Write(hs.serverHello.marshal())
|
|
||||||
transcript.Write(hs.hello.marshalWithoutBinders())
|
|
||||||
pskBinders := [][]byte{hs.suite.finishedHash(hs.binderKey, transcript)}
|
|
||||||
hs.hello.updateBinders(pskBinders)
|
|
||||||
} else {
|
|
||||||
// Server selected a cipher suite incompatible with the PSK.
|
|
||||||
hs.hello.pskIdentities = nil
|
|
||||||
hs.hello.pskBinders = nil
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if hs.hello.earlyData && c.extraConfig != nil && c.extraConfig.Rejected0RTT != nil {
|
|
||||||
c.extraConfig.Rejected0RTT()
|
|
||||||
}
|
|
||||||
hs.hello.earlyData = false // disable 0-RTT
|
|
||||||
|
|
||||||
hs.transcript.Write(hs.hello.marshal())
|
|
||||||
if _, err := c.writeRecord(recordTypeHandshake, hs.hello.marshal()); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
msg, err := c.readHandshake()
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
serverHello, ok := msg.(*serverHelloMsg)
|
|
||||||
if !ok {
|
|
||||||
c.sendAlert(alertUnexpectedMessage)
|
|
||||||
return unexpectedMessageError(serverHello, msg)
|
|
||||||
}
|
|
||||||
hs.serverHello = serverHello
|
|
||||||
|
|
||||||
if err := hs.checkServerHelloOrHRR(); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (hs *clientHandshakeStateTLS13) processServerHello() error {
|
|
||||||
c := hs.c
|
|
||||||
|
|
||||||
if bytes.Equal(hs.serverHello.random, helloRetryRequestRandom) {
|
|
||||||
c.sendAlert(alertUnexpectedMessage)
|
|
||||||
return errors.New("tls: server sent two HelloRetryRequest messages")
|
|
||||||
}
|
|
||||||
|
|
||||||
if len(hs.serverHello.cookie) != 0 {
|
|
||||||
c.sendAlert(alertUnsupportedExtension)
|
|
||||||
return errors.New("tls: server sent a cookie in a normal ServerHello")
|
|
||||||
}
|
|
||||||
|
|
||||||
if hs.serverHello.selectedGroup != 0 {
|
|
||||||
c.sendAlert(alertDecodeError)
|
|
||||||
return errors.New("tls: malformed key_share extension")
|
|
||||||
}
|
|
||||||
|
|
||||||
if hs.serverHello.serverShare.group == 0 {
|
|
||||||
c.sendAlert(alertIllegalParameter)
|
|
||||||
return errors.New("tls: server did not send a key share")
|
|
||||||
}
|
|
||||||
if hs.serverHello.serverShare.group != hs.ecdheParams.CurveID() {
|
|
||||||
c.sendAlert(alertIllegalParameter)
|
|
||||||
return errors.New("tls: server selected unsupported group")
|
|
||||||
}
|
|
||||||
|
|
||||||
if !hs.serverHello.selectedIdentityPresent {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
if int(hs.serverHello.selectedIdentity) >= len(hs.hello.pskIdentities) {
|
|
||||||
c.sendAlert(alertIllegalParameter)
|
|
||||||
return errors.New("tls: server selected an invalid PSK")
|
|
||||||
}
|
|
||||||
|
|
||||||
if len(hs.hello.pskIdentities) != 1 || hs.session == nil {
|
|
||||||
return c.sendAlert(alertInternalError)
|
|
||||||
}
|
|
||||||
pskSuite := cipherSuiteTLS13ByID(hs.session.cipherSuite)
|
|
||||||
if pskSuite == nil {
|
|
||||||
return c.sendAlert(alertInternalError)
|
|
||||||
}
|
|
||||||
if pskSuite.hash != hs.suite.hash {
|
|
||||||
c.sendAlert(alertIllegalParameter)
|
|
||||||
return errors.New("tls: server selected an invalid PSK and cipher suite pair")
|
|
||||||
}
|
|
||||||
|
|
||||||
hs.usingPSK = true
|
|
||||||
c.didResume = true
|
|
||||||
c.peerCertificates = hs.session.serverCertificates
|
|
||||||
c.verifiedChains = hs.session.verifiedChains
|
|
||||||
c.ocspResponse = hs.session.ocspResponse
|
|
||||||
c.scts = hs.session.scts
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (hs *clientHandshakeStateTLS13) establishHandshakeKeys() error {
|
|
||||||
c := hs.c
|
|
||||||
|
|
||||||
sharedKey := hs.ecdheParams.SharedKey(hs.serverHello.serverShare.data)
|
|
||||||
if sharedKey == nil {
|
|
||||||
c.sendAlert(alertIllegalParameter)
|
|
||||||
return errors.New("tls: invalid server key share")
|
|
||||||
}
|
|
||||||
|
|
||||||
earlySecret := hs.earlySecret
|
|
||||||
if !hs.usingPSK {
|
|
||||||
earlySecret = hs.suite.extract(nil, nil)
|
|
||||||
}
|
|
||||||
handshakeSecret := hs.suite.extract(sharedKey,
|
|
||||||
hs.suite.deriveSecret(earlySecret, "derived", nil))
|
|
||||||
|
|
||||||
clientSecret := hs.suite.deriveSecret(handshakeSecret,
|
|
||||||
clientHandshakeTrafficLabel, hs.transcript)
|
|
||||||
c.out.exportKey(EncryptionHandshake, hs.suite, clientSecret)
|
|
||||||
c.out.setTrafficSecret(hs.suite, clientSecret)
|
|
||||||
serverSecret := hs.suite.deriveSecret(handshakeSecret,
|
|
||||||
serverHandshakeTrafficLabel, hs.transcript)
|
|
||||||
c.in.exportKey(EncryptionHandshake, hs.suite, serverSecret)
|
|
||||||
c.in.setTrafficSecret(hs.suite, serverSecret)
|
|
||||||
|
|
||||||
err := c.config.writeKeyLog(keyLogLabelClientHandshake, hs.hello.random, clientSecret)
|
|
||||||
if err != nil {
|
|
||||||
c.sendAlert(alertInternalError)
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
err = c.config.writeKeyLog(keyLogLabelServerHandshake, hs.hello.random, serverSecret)
|
|
||||||
if err != nil {
|
|
||||||
c.sendAlert(alertInternalError)
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
hs.masterSecret = hs.suite.extract(nil,
|
|
||||||
hs.suite.deriveSecret(handshakeSecret, "derived", nil))
|
|
||||||
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (hs *clientHandshakeStateTLS13) readServerParameters() error {
|
|
||||||
c := hs.c
|
|
||||||
|
|
||||||
msg, err := c.readHandshake()
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
encryptedExtensions, ok := msg.(*encryptedExtensionsMsg)
|
|
||||||
if !ok {
|
|
||||||
c.sendAlert(alertUnexpectedMessage)
|
|
||||||
return unexpectedMessageError(encryptedExtensions, msg)
|
|
||||||
}
|
|
||||||
// Notify the caller if 0-RTT was rejected.
|
|
||||||
if !encryptedExtensions.earlyData && hs.hello.earlyData && c.extraConfig != nil && c.extraConfig.Rejected0RTT != nil {
|
|
||||||
c.extraConfig.Rejected0RTT()
|
|
||||||
}
|
|
||||||
c.used0RTT = encryptedExtensions.earlyData
|
|
||||||
if hs.c.extraConfig != nil && hs.c.extraConfig.ReceivedExtensions != nil {
|
|
||||||
hs.c.extraConfig.ReceivedExtensions(typeEncryptedExtensions, encryptedExtensions.additionalExtensions)
|
|
||||||
}
|
|
||||||
hs.transcript.Write(encryptedExtensions.marshal())
|
|
||||||
|
|
||||||
if err := checkALPN(hs.hello.alpnProtocols, encryptedExtensions.alpnProtocol); err != nil {
|
|
||||||
c.sendAlert(alertUnsupportedExtension)
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
c.clientProtocol = encryptedExtensions.alpnProtocol
|
|
||||||
|
|
||||||
if c.extraConfig != nil && c.extraConfig.EnforceNextProtoSelection {
|
|
||||||
if len(encryptedExtensions.alpnProtocol) == 0 {
|
|
||||||
// the server didn't select an ALPN
|
|
||||||
c.sendAlert(alertNoApplicationProtocol)
|
|
||||||
return errors.New("ALPN negotiation failed. Server didn't offer any protocols")
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (hs *clientHandshakeStateTLS13) readServerCertificate() error {
|
|
||||||
c := hs.c
|
|
||||||
|
|
||||||
// Either a PSK or a certificate is always used, but not both.
|
|
||||||
// See RFC 8446, Section 4.1.1.
|
|
||||||
if hs.usingPSK {
|
|
||||||
// Make sure the connection is still being verified whether or not this
|
|
||||||
// is a resumption. Resumptions currently don't reverify certificates so
|
|
||||||
// they don't call verifyServerCertificate. See Issue 31641.
|
|
||||||
if c.config.VerifyConnection != nil {
|
|
||||||
if err := c.config.VerifyConnection(c.connectionStateLocked()); err != nil {
|
|
||||||
c.sendAlert(alertBadCertificate)
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
msg, err := c.readHandshake()
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
certReq, ok := msg.(*certificateRequestMsgTLS13)
|
|
||||||
if ok {
|
|
||||||
hs.transcript.Write(certReq.marshal())
|
|
||||||
|
|
||||||
hs.certReq = certReq
|
|
||||||
|
|
||||||
msg, err = c.readHandshake()
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
certMsg, ok := msg.(*certificateMsgTLS13)
|
|
||||||
if !ok {
|
|
||||||
c.sendAlert(alertUnexpectedMessage)
|
|
||||||
return unexpectedMessageError(certMsg, msg)
|
|
||||||
}
|
|
||||||
if len(certMsg.certificate.Certificate) == 0 {
|
|
||||||
c.sendAlert(alertDecodeError)
|
|
||||||
return errors.New("tls: received empty certificates message")
|
|
||||||
}
|
|
||||||
hs.transcript.Write(certMsg.marshal())
|
|
||||||
|
|
||||||
c.scts = certMsg.certificate.SignedCertificateTimestamps
|
|
||||||
c.ocspResponse = certMsg.certificate.OCSPStaple
|
|
||||||
|
|
||||||
if err := c.verifyServerCertificate(certMsg.certificate.Certificate); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
msg, err = c.readHandshake()
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
certVerify, ok := msg.(*certificateVerifyMsg)
|
|
||||||
if !ok {
|
|
||||||
c.sendAlert(alertUnexpectedMessage)
|
|
||||||
return unexpectedMessageError(certVerify, msg)
|
|
||||||
}
|
|
||||||
|
|
||||||
// See RFC 8446, Section 4.4.3.
|
|
||||||
if !isSupportedSignatureAlgorithm(certVerify.signatureAlgorithm, supportedSignatureAlgorithms) {
|
|
||||||
c.sendAlert(alertIllegalParameter)
|
|
||||||
return errors.New("tls: certificate used with invalid signature algorithm")
|
|
||||||
}
|
|
||||||
sigType, sigHash, err := typeAndHashFromSignatureScheme(certVerify.signatureAlgorithm)
|
|
||||||
if err != nil {
|
|
||||||
return c.sendAlert(alertInternalError)
|
|
||||||
}
|
|
||||||
if sigType == signaturePKCS1v15 || sigHash == crypto.SHA1 {
|
|
||||||
c.sendAlert(alertIllegalParameter)
|
|
||||||
return errors.New("tls: certificate used with invalid signature algorithm")
|
|
||||||
}
|
|
||||||
signed := signedMessage(sigHash, serverSignatureContext, hs.transcript)
|
|
||||||
if err := verifyHandshakeSignature(sigType, c.peerCertificates[0].PublicKey,
|
|
||||||
sigHash, signed, certVerify.signature); err != nil {
|
|
||||||
c.sendAlert(alertDecryptError)
|
|
||||||
return errors.New("tls: invalid signature by the server certificate: " + err.Error())
|
|
||||||
}
|
|
||||||
|
|
||||||
hs.transcript.Write(certVerify.marshal())
|
|
||||||
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (hs *clientHandshakeStateTLS13) readServerFinished() error {
|
|
||||||
c := hs.c
|
|
||||||
|
|
||||||
msg, err := c.readHandshake()
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
finished, ok := msg.(*finishedMsg)
|
|
||||||
if !ok {
|
|
||||||
c.sendAlert(alertUnexpectedMessage)
|
|
||||||
return unexpectedMessageError(finished, msg)
|
|
||||||
}
|
|
||||||
|
|
||||||
expectedMAC := hs.suite.finishedHash(c.in.trafficSecret, hs.transcript)
|
|
||||||
if !hmac.Equal(expectedMAC, finished.verifyData) {
|
|
||||||
c.sendAlert(alertDecryptError)
|
|
||||||
return errors.New("tls: invalid server finished hash")
|
|
||||||
}
|
|
||||||
|
|
||||||
hs.transcript.Write(finished.marshal())
|
|
||||||
|
|
||||||
// Derive secrets that take context through the server Finished.
|
|
||||||
|
|
||||||
hs.trafficSecret = hs.suite.deriveSecret(hs.masterSecret,
|
|
||||||
clientApplicationTrafficLabel, hs.transcript)
|
|
||||||
serverSecret := hs.suite.deriveSecret(hs.masterSecret,
|
|
||||||
serverApplicationTrafficLabel, hs.transcript)
|
|
||||||
c.in.exportKey(EncryptionApplication, hs.suite, serverSecret)
|
|
||||||
c.in.setTrafficSecret(hs.suite, serverSecret)
|
|
||||||
|
|
||||||
err = c.config.writeKeyLog(keyLogLabelClientTraffic, hs.hello.random, hs.trafficSecret)
|
|
||||||
if err != nil {
|
|
||||||
c.sendAlert(alertInternalError)
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
err = c.config.writeKeyLog(keyLogLabelServerTraffic, hs.hello.random, serverSecret)
|
|
||||||
if err != nil {
|
|
||||||
c.sendAlert(alertInternalError)
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
c.ekm = hs.suite.exportKeyingMaterial(hs.masterSecret, hs.transcript)
|
|
||||||
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (hs *clientHandshakeStateTLS13) sendClientCertificate() error {
|
|
||||||
c := hs.c
|
|
||||||
|
|
||||||
if hs.certReq == nil {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
cert, err := c.getClientCertificate(toCertificateRequestInfo(&certificateRequestInfo{
|
|
||||||
AcceptableCAs: hs.certReq.certificateAuthorities,
|
|
||||||
SignatureSchemes: hs.certReq.supportedSignatureAlgorithms,
|
|
||||||
Version: c.vers,
|
|
||||||
ctx: hs.ctx,
|
|
||||||
}))
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
certMsg := new(certificateMsgTLS13)
|
|
||||||
|
|
||||||
certMsg.certificate = *cert
|
|
||||||
certMsg.scts = hs.certReq.scts && len(cert.SignedCertificateTimestamps) > 0
|
|
||||||
certMsg.ocspStapling = hs.certReq.ocspStapling && len(cert.OCSPStaple) > 0
|
|
||||||
|
|
||||||
hs.transcript.Write(certMsg.marshal())
|
|
||||||
if _, err := c.writeRecord(recordTypeHandshake, certMsg.marshal()); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
// If we sent an empty certificate message, skip the CertificateVerify.
|
|
||||||
if len(cert.Certificate) == 0 {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
certVerifyMsg := new(certificateVerifyMsg)
|
|
||||||
certVerifyMsg.hasSignatureAlgorithm = true
|
|
||||||
|
|
||||||
certVerifyMsg.signatureAlgorithm, err = selectSignatureScheme(c.vers, cert, hs.certReq.supportedSignatureAlgorithms)
|
|
||||||
if err != nil {
|
|
||||||
// getClientCertificate returned a certificate incompatible with the
|
|
||||||
// CertificateRequestInfo supported signature algorithms.
|
|
||||||
c.sendAlert(alertHandshakeFailure)
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
sigType, sigHash, err := typeAndHashFromSignatureScheme(certVerifyMsg.signatureAlgorithm)
|
|
||||||
if err != nil {
|
|
||||||
return c.sendAlert(alertInternalError)
|
|
||||||
}
|
|
||||||
|
|
||||||
signed := signedMessage(sigHash, clientSignatureContext, hs.transcript)
|
|
||||||
signOpts := crypto.SignerOpts(sigHash)
|
|
||||||
if sigType == signatureRSAPSS {
|
|
||||||
signOpts = &rsa.PSSOptions{SaltLength: rsa.PSSSaltLengthEqualsHash, Hash: sigHash}
|
|
||||||
}
|
|
||||||
sig, err := cert.PrivateKey.(crypto.Signer).Sign(c.config.rand(), signed, signOpts)
|
|
||||||
if err != nil {
|
|
||||||
c.sendAlert(alertInternalError)
|
|
||||||
return errors.New("tls: failed to sign handshake: " + err.Error())
|
|
||||||
}
|
|
||||||
certVerifyMsg.signature = sig
|
|
||||||
|
|
||||||
hs.transcript.Write(certVerifyMsg.marshal())
|
|
||||||
if _, err := c.writeRecord(recordTypeHandshake, certVerifyMsg.marshal()); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (hs *clientHandshakeStateTLS13) sendClientFinished() error {
|
|
||||||
c := hs.c
|
|
||||||
|
|
||||||
finished := &finishedMsg{
|
|
||||||
verifyData: hs.suite.finishedHash(c.out.trafficSecret, hs.transcript),
|
|
||||||
}
|
|
||||||
|
|
||||||
hs.transcript.Write(finished.marshal())
|
|
||||||
if _, err := c.writeRecord(recordTypeHandshake, finished.marshal()); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
c.out.exportKey(EncryptionApplication, hs.suite, hs.trafficSecret)
|
|
||||||
c.out.setTrafficSecret(hs.suite, hs.trafficSecret)
|
|
||||||
|
|
||||||
if !c.config.SessionTicketsDisabled && c.config.ClientSessionCache != nil {
|
|
||||||
c.resumptionSecret = hs.suite.deriveSecret(hs.masterSecret,
|
|
||||||
resumptionLabel, hs.transcript)
|
|
||||||
}
|
|
||||||
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (c *Conn) handleNewSessionTicket(msg *newSessionTicketMsgTLS13) error {
|
|
||||||
if !c.isClient {
|
|
||||||
c.sendAlert(alertUnexpectedMessage)
|
|
||||||
return errors.New("tls: received new session ticket from a client")
|
|
||||||
}
|
|
||||||
|
|
||||||
if c.config.SessionTicketsDisabled || c.config.ClientSessionCache == nil {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// See RFC 8446, Section 4.6.1.
|
|
||||||
if msg.lifetime == 0 {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
lifetime := time.Duration(msg.lifetime) * time.Second
|
|
||||||
if lifetime > maxSessionTicketLifetime {
|
|
||||||
c.sendAlert(alertIllegalParameter)
|
|
||||||
return errors.New("tls: received a session ticket with invalid lifetime")
|
|
||||||
}
|
|
||||||
|
|
||||||
cipherSuite := cipherSuiteTLS13ByID(c.cipherSuite)
|
|
||||||
if cipherSuite == nil || c.resumptionSecret == nil {
|
|
||||||
return c.sendAlert(alertInternalError)
|
|
||||||
}
|
|
||||||
|
|
||||||
// We need to save the max_early_data_size that the server sent us, in order
|
|
||||||
// to decide if we're going to try 0-RTT with this ticket.
|
|
||||||
// However, at the same time, the qtls.ClientSessionTicket needs to be equal to
|
|
||||||
// the tls.ClientSessionTicket, so we can't just add a new field to the struct.
|
|
||||||
// We therefore abuse the nonce field (which is a byte slice)
|
|
||||||
nonceWithEarlyData := make([]byte, len(msg.nonce)+4)
|
|
||||||
binary.BigEndian.PutUint32(nonceWithEarlyData, msg.maxEarlyData)
|
|
||||||
copy(nonceWithEarlyData[4:], msg.nonce)
|
|
||||||
|
|
||||||
var appData []byte
|
|
||||||
if c.extraConfig != nil && c.extraConfig.GetAppDataForSessionState != nil {
|
|
||||||
appData = c.extraConfig.GetAppDataForSessionState()
|
|
||||||
}
|
|
||||||
var b cryptobyte.Builder
|
|
||||||
b.AddUint16(clientSessionStateVersion) // revision
|
|
||||||
b.AddUint32(msg.maxEarlyData)
|
|
||||||
b.AddUint16LengthPrefixed(func(b *cryptobyte.Builder) {
|
|
||||||
b.AddBytes(appData)
|
|
||||||
})
|
|
||||||
b.AddUint16LengthPrefixed(func(b *cryptobyte.Builder) {
|
|
||||||
b.AddBytes(msg.nonce)
|
|
||||||
})
|
|
||||||
|
|
||||||
// Save the resumption_master_secret and nonce instead of deriving the PSK
|
|
||||||
// to do the least amount of work on NewSessionTicket messages before we
|
|
||||||
// know if the ticket will be used. Forward secrecy of resumed connections
|
|
||||||
// is guaranteed by the requirement for pskModeDHE.
|
|
||||||
session := &clientSessionState{
|
|
||||||
sessionTicket: msg.label,
|
|
||||||
vers: c.vers,
|
|
||||||
cipherSuite: c.cipherSuite,
|
|
||||||
masterSecret: c.resumptionSecret,
|
|
||||||
serverCertificates: c.peerCertificates,
|
|
||||||
verifiedChains: c.verifiedChains,
|
|
||||||
receivedAt: c.config.time(),
|
|
||||||
nonce: b.BytesOrPanic(),
|
|
||||||
useBy: c.config.time().Add(lifetime),
|
|
||||||
ageAdd: msg.ageAdd,
|
|
||||||
ocspResponse: c.ocspResponse,
|
|
||||||
scts: c.scts,
|
|
||||||
}
|
|
||||||
|
|
||||||
cacheKey := clientSessionCacheKey(c.conn.RemoteAddr(), c.config)
|
|
||||||
c.config.ClientSessionCache.Put(cacheKey, toClientSessionState(session))
|
|
||||||
|
|
||||||
return nil
|
|
||||||
}
|
|
File diff suppressed because it is too large
Load Diff
|
@ -1,905 +0,0 @@
|
||||||
// Copyright 2009 The Go Authors. All rights reserved.
|
|
||||||
// Use of this source code is governed by a BSD-style
|
|
||||||
// license that can be found in the LICENSE file.
|
|
||||||
|
|
||||||
package qtls
|
|
||||||
|
|
||||||
import (
|
|
||||||
"context"
|
|
||||||
"crypto"
|
|
||||||
"crypto/ecdsa"
|
|
||||||
"crypto/ed25519"
|
|
||||||
"crypto/rsa"
|
|
||||||
"crypto/subtle"
|
|
||||||
"crypto/x509"
|
|
||||||
"errors"
|
|
||||||
"fmt"
|
|
||||||
"hash"
|
|
||||||
"io"
|
|
||||||
"sync/atomic"
|
|
||||||
"time"
|
|
||||||
)
|
|
||||||
|
|
||||||
// serverHandshakeState contains details of a server handshake in progress.
|
|
||||||
// It's discarded once the handshake has completed.
|
|
||||||
type serverHandshakeState struct {
|
|
||||||
c *Conn
|
|
||||||
ctx context.Context
|
|
||||||
clientHello *clientHelloMsg
|
|
||||||
hello *serverHelloMsg
|
|
||||||
suite *cipherSuite
|
|
||||||
ecdheOk bool
|
|
||||||
ecSignOk bool
|
|
||||||
rsaDecryptOk bool
|
|
||||||
rsaSignOk bool
|
|
||||||
sessionState *sessionState
|
|
||||||
finishedHash finishedHash
|
|
||||||
masterSecret []byte
|
|
||||||
cert *Certificate
|
|
||||||
}
|
|
||||||
|
|
||||||
// serverHandshake performs a TLS handshake as a server.
|
|
||||||
func (c *Conn) serverHandshake(ctx context.Context) error {
|
|
||||||
c.setAlternativeRecordLayer()
|
|
||||||
|
|
||||||
clientHello, err := c.readClientHello(ctx)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
if c.vers == VersionTLS13 {
|
|
||||||
hs := serverHandshakeStateTLS13{
|
|
||||||
c: c,
|
|
||||||
ctx: ctx,
|
|
||||||
clientHello: clientHello,
|
|
||||||
}
|
|
||||||
return hs.handshake()
|
|
||||||
} else if c.extraConfig.usesAlternativeRecordLayer() {
|
|
||||||
// This should already have been caught by the check that the ClientHello doesn't
|
|
||||||
// offer any (supported) versions older than TLS 1.3.
|
|
||||||
// Check again to make sure we can't be tricked into using an older version.
|
|
||||||
c.sendAlert(alertProtocolVersion)
|
|
||||||
return errors.New("tls: negotiated TLS < 1.3 when using QUIC")
|
|
||||||
}
|
|
||||||
|
|
||||||
hs := serverHandshakeState{
|
|
||||||
c: c,
|
|
||||||
ctx: ctx,
|
|
||||||
clientHello: clientHello,
|
|
||||||
}
|
|
||||||
return hs.handshake()
|
|
||||||
}
|
|
||||||
|
|
||||||
func (hs *serverHandshakeState) handshake() error {
|
|
||||||
c := hs.c
|
|
||||||
|
|
||||||
if err := hs.processClientHello(); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
// For an overview of TLS handshaking, see RFC 5246, Section 7.3.
|
|
||||||
c.buffering = true
|
|
||||||
if hs.checkForResumption() {
|
|
||||||
// The client has included a session ticket and so we do an abbreviated handshake.
|
|
||||||
c.didResume = true
|
|
||||||
if err := hs.doResumeHandshake(); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
if err := hs.establishKeys(); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
if err := hs.sendSessionTicket(); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
if err := hs.sendFinished(c.serverFinished[:]); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
if _, err := c.flush(); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
c.clientFinishedIsFirst = false
|
|
||||||
if err := hs.readFinished(nil); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
// The client didn't include a session ticket, or it wasn't
|
|
||||||
// valid so we do a full handshake.
|
|
||||||
if err := hs.pickCipherSuite(); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
if err := hs.doFullHandshake(); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
if err := hs.establishKeys(); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
if err := hs.readFinished(c.clientFinished[:]); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
c.clientFinishedIsFirst = true
|
|
||||||
c.buffering = true
|
|
||||||
if err := hs.sendSessionTicket(); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
if err := hs.sendFinished(nil); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
if _, err := c.flush(); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
c.ekm = ekmFromMasterSecret(c.vers, hs.suite, hs.masterSecret, hs.clientHello.random, hs.hello.random)
|
|
||||||
atomic.StoreUint32(&c.handshakeStatus, 1)
|
|
||||||
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// readClientHello reads a ClientHello message and selects the protocol version.
|
|
||||||
func (c *Conn) readClientHello(ctx context.Context) (*clientHelloMsg, error) {
|
|
||||||
msg, err := c.readHandshake()
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
clientHello, ok := msg.(*clientHelloMsg)
|
|
||||||
if !ok {
|
|
||||||
c.sendAlert(alertUnexpectedMessage)
|
|
||||||
return nil, unexpectedMessageError(clientHello, msg)
|
|
||||||
}
|
|
||||||
|
|
||||||
var configForClient *config
|
|
||||||
originalConfig := c.config
|
|
||||||
if c.config.GetConfigForClient != nil {
|
|
||||||
chi := newClientHelloInfo(ctx, c, clientHello)
|
|
||||||
if cfc, err := c.config.GetConfigForClient(chi); err != nil {
|
|
||||||
c.sendAlert(alertInternalError)
|
|
||||||
return nil, err
|
|
||||||
} else if cfc != nil {
|
|
||||||
configForClient = fromConfig(cfc)
|
|
||||||
c.config = configForClient
|
|
||||||
}
|
|
||||||
}
|
|
||||||
c.ticketKeys = originalConfig.ticketKeys(configForClient)
|
|
||||||
|
|
||||||
clientVersions := clientHello.supportedVersions
|
|
||||||
if len(clientHello.supportedVersions) == 0 {
|
|
||||||
clientVersions = supportedVersionsFromMax(clientHello.vers)
|
|
||||||
}
|
|
||||||
if c.extraConfig.usesAlternativeRecordLayer() {
|
|
||||||
// In QUIC, the client MUST NOT offer any old TLS versions.
|
|
||||||
// Here, we can only check that none of the other supported versions of this library
|
|
||||||
// (TLS 1.0 - TLS 1.2) is offered. We don't check for any SSL versions here.
|
|
||||||
for _, ver := range clientVersions {
|
|
||||||
if ver == VersionTLS13 {
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
for _, v := range supportedVersions {
|
|
||||||
if ver == v {
|
|
||||||
c.sendAlert(alertProtocolVersion)
|
|
||||||
return nil, fmt.Errorf("tls: client offered old TLS version %#x", ver)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
// Make the config we're using allows us to use TLS 1.3.
|
|
||||||
if c.config.maxSupportedVersion() < VersionTLS13 {
|
|
||||||
c.sendAlert(alertInternalError)
|
|
||||||
return nil, errors.New("tls: MaxVersion prevents QUIC from using TLS 1.3")
|
|
||||||
}
|
|
||||||
}
|
|
||||||
c.vers, ok = c.config.mutualVersion(clientVersions)
|
|
||||||
if !ok {
|
|
||||||
c.sendAlert(alertProtocolVersion)
|
|
||||||
return nil, fmt.Errorf("tls: client offered only unsupported versions: %x", clientVersions)
|
|
||||||
}
|
|
||||||
c.haveVers = true
|
|
||||||
c.in.version = c.vers
|
|
||||||
c.out.version = c.vers
|
|
||||||
|
|
||||||
return clientHello, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (hs *serverHandshakeState) processClientHello() error {
|
|
||||||
c := hs.c
|
|
||||||
|
|
||||||
hs.hello = new(serverHelloMsg)
|
|
||||||
hs.hello.vers = c.vers
|
|
||||||
|
|
||||||
foundCompression := false
|
|
||||||
// We only support null compression, so check that the client offered it.
|
|
||||||
for _, compression := range hs.clientHello.compressionMethods {
|
|
||||||
if compression == compressionNone {
|
|
||||||
foundCompression = true
|
|
||||||
break
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if !foundCompression {
|
|
||||||
c.sendAlert(alertHandshakeFailure)
|
|
||||||
return errors.New("tls: client does not support uncompressed connections")
|
|
||||||
}
|
|
||||||
|
|
||||||
hs.hello.random = make([]byte, 32)
|
|
||||||
serverRandom := hs.hello.random
|
|
||||||
// Downgrade protection canaries. See RFC 8446, Section 4.1.3.
|
|
||||||
maxVers := c.config.maxSupportedVersion()
|
|
||||||
if maxVers >= VersionTLS12 && c.vers < maxVers || testingOnlyForceDowngradeCanary {
|
|
||||||
if c.vers == VersionTLS12 {
|
|
||||||
copy(serverRandom[24:], downgradeCanaryTLS12)
|
|
||||||
} else {
|
|
||||||
copy(serverRandom[24:], downgradeCanaryTLS11)
|
|
||||||
}
|
|
||||||
serverRandom = serverRandom[:24]
|
|
||||||
}
|
|
||||||
_, err := io.ReadFull(c.config.rand(), serverRandom)
|
|
||||||
if err != nil {
|
|
||||||
c.sendAlert(alertInternalError)
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
if len(hs.clientHello.secureRenegotiation) != 0 {
|
|
||||||
c.sendAlert(alertHandshakeFailure)
|
|
||||||
return errors.New("tls: initial handshake had non-empty renegotiation extension")
|
|
||||||
}
|
|
||||||
|
|
||||||
hs.hello.secureRenegotiationSupported = hs.clientHello.secureRenegotiationSupported
|
|
||||||
hs.hello.compressionMethod = compressionNone
|
|
||||||
if len(hs.clientHello.serverName) > 0 {
|
|
||||||
c.serverName = hs.clientHello.serverName
|
|
||||||
}
|
|
||||||
|
|
||||||
selectedProto, err := negotiateALPN(c.config.NextProtos, hs.clientHello.alpnProtocols)
|
|
||||||
if err != nil {
|
|
||||||
c.sendAlert(alertNoApplicationProtocol)
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
hs.hello.alpnProtocol = selectedProto
|
|
||||||
c.clientProtocol = selectedProto
|
|
||||||
|
|
||||||
hs.cert, err = c.config.getCertificate(newClientHelloInfo(hs.ctx, c, hs.clientHello))
|
|
||||||
if err != nil {
|
|
||||||
if err == errNoCertificates {
|
|
||||||
c.sendAlert(alertUnrecognizedName)
|
|
||||||
} else {
|
|
||||||
c.sendAlert(alertInternalError)
|
|
||||||
}
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
if hs.clientHello.scts {
|
|
||||||
hs.hello.scts = hs.cert.SignedCertificateTimestamps
|
|
||||||
}
|
|
||||||
|
|
||||||
hs.ecdheOk = supportsECDHE(c.config, hs.clientHello.supportedCurves, hs.clientHello.supportedPoints)
|
|
||||||
|
|
||||||
if hs.ecdheOk {
|
|
||||||
// Although omitting the ec_point_formats extension is permitted, some
|
|
||||||
// old OpenSSL version will refuse to handshake if not present.
|
|
||||||
//
|
|
||||||
// Per RFC 4492, section 5.1.2, implementations MUST support the
|
|
||||||
// uncompressed point format. See golang.org/issue/31943.
|
|
||||||
hs.hello.supportedPoints = []uint8{pointFormatUncompressed}
|
|
||||||
}
|
|
||||||
|
|
||||||
if priv, ok := hs.cert.PrivateKey.(crypto.Signer); ok {
|
|
||||||
switch priv.Public().(type) {
|
|
||||||
case *ecdsa.PublicKey:
|
|
||||||
hs.ecSignOk = true
|
|
||||||
case ed25519.PublicKey:
|
|
||||||
hs.ecSignOk = true
|
|
||||||
case *rsa.PublicKey:
|
|
||||||
hs.rsaSignOk = true
|
|
||||||
default:
|
|
||||||
c.sendAlert(alertInternalError)
|
|
||||||
return fmt.Errorf("tls: unsupported signing key type (%T)", priv.Public())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if priv, ok := hs.cert.PrivateKey.(crypto.Decrypter); ok {
|
|
||||||
switch priv.Public().(type) {
|
|
||||||
case *rsa.PublicKey:
|
|
||||||
hs.rsaDecryptOk = true
|
|
||||||
default:
|
|
||||||
c.sendAlert(alertInternalError)
|
|
||||||
return fmt.Errorf("tls: unsupported decryption key type (%T)", priv.Public())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// negotiateALPN picks a shared ALPN protocol that both sides support in server
|
|
||||||
// preference order. If ALPN is not configured or the peer doesn't support it,
|
|
||||||
// it returns "" and no error.
|
|
||||||
func negotiateALPN(serverProtos, clientProtos []string) (string, error) {
|
|
||||||
if len(serverProtos) == 0 || len(clientProtos) == 0 {
|
|
||||||
return "", nil
|
|
||||||
}
|
|
||||||
var http11fallback bool
|
|
||||||
for _, s := range serverProtos {
|
|
||||||
for _, c := range clientProtos {
|
|
||||||
if s == c {
|
|
||||||
return s, nil
|
|
||||||
}
|
|
||||||
if s == "h2" && c == "http/1.1" {
|
|
||||||
http11fallback = true
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
// As a special case, let http/1.1 clients connect to h2 servers as if they
|
|
||||||
// didn't support ALPN. We used not to enforce protocol overlap, so over
|
|
||||||
// time a number of HTTP servers were configured with only "h2", but
|
|
||||||
// expected to accept connections from "http/1.1" clients. See Issue 46310.
|
|
||||||
if http11fallback {
|
|
||||||
return "", nil
|
|
||||||
}
|
|
||||||
return "", fmt.Errorf("tls: client requested unsupported application protocols (%s)", clientProtos)
|
|
||||||
}
|
|
||||||
|
|
||||||
// supportsECDHE returns whether ECDHE key exchanges can be used with this
|
|
||||||
// pre-TLS 1.3 client.
|
|
||||||
func supportsECDHE(c *config, supportedCurves []CurveID, supportedPoints []uint8) bool {
|
|
||||||
supportsCurve := false
|
|
||||||
for _, curve := range supportedCurves {
|
|
||||||
if c.supportsCurve(curve) {
|
|
||||||
supportsCurve = true
|
|
||||||
break
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
supportsPointFormat := false
|
|
||||||
for _, pointFormat := range supportedPoints {
|
|
||||||
if pointFormat == pointFormatUncompressed {
|
|
||||||
supportsPointFormat = true
|
|
||||||
break
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return supportsCurve && supportsPointFormat
|
|
||||||
}
|
|
||||||
|
|
||||||
func (hs *serverHandshakeState) pickCipherSuite() error {
|
|
||||||
c := hs.c
|
|
||||||
|
|
||||||
preferenceOrder := cipherSuitesPreferenceOrder
|
|
||||||
if !hasAESGCMHardwareSupport || !aesgcmPreferred(hs.clientHello.cipherSuites) {
|
|
||||||
preferenceOrder = cipherSuitesPreferenceOrderNoAES
|
|
||||||
}
|
|
||||||
|
|
||||||
configCipherSuites := c.config.cipherSuites()
|
|
||||||
preferenceList := make([]uint16, 0, len(configCipherSuites))
|
|
||||||
for _, suiteID := range preferenceOrder {
|
|
||||||
for _, id := range configCipherSuites {
|
|
||||||
if id == suiteID {
|
|
||||||
preferenceList = append(preferenceList, id)
|
|
||||||
break
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
hs.suite = selectCipherSuite(preferenceList, hs.clientHello.cipherSuites, hs.cipherSuiteOk)
|
|
||||||
if hs.suite == nil {
|
|
||||||
c.sendAlert(alertHandshakeFailure)
|
|
||||||
return errors.New("tls: no cipher suite supported by both client and server")
|
|
||||||
}
|
|
||||||
c.cipherSuite = hs.suite.id
|
|
||||||
|
|
||||||
for _, id := range hs.clientHello.cipherSuites {
|
|
||||||
if id == TLS_FALLBACK_SCSV {
|
|
||||||
// The client is doing a fallback connection. See RFC 7507.
|
|
||||||
if hs.clientHello.vers < c.config.maxSupportedVersion() {
|
|
||||||
c.sendAlert(alertInappropriateFallback)
|
|
||||||
return errors.New("tls: client using inappropriate protocol fallback")
|
|
||||||
}
|
|
||||||
break
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (hs *serverHandshakeState) cipherSuiteOk(c *cipherSuite) bool {
|
|
||||||
if c.flags&suiteECDHE != 0 {
|
|
||||||
if !hs.ecdheOk {
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
if c.flags&suiteECSign != 0 {
|
|
||||||
if !hs.ecSignOk {
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
} else if !hs.rsaSignOk {
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
} else if !hs.rsaDecryptOk {
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
if hs.c.vers < VersionTLS12 && c.flags&suiteTLS12 != 0 {
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
|
|
||||||
// checkForResumption reports whether we should perform resumption on this connection.
|
|
||||||
func (hs *serverHandshakeState) checkForResumption() bool {
|
|
||||||
c := hs.c
|
|
||||||
|
|
||||||
if c.config.SessionTicketsDisabled {
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
|
|
||||||
plaintext, usedOldKey := c.decryptTicket(hs.clientHello.sessionTicket)
|
|
||||||
if plaintext == nil {
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
hs.sessionState = &sessionState{usedOldKey: usedOldKey}
|
|
||||||
ok := hs.sessionState.unmarshal(plaintext)
|
|
||||||
if !ok {
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
|
|
||||||
createdAt := time.Unix(int64(hs.sessionState.createdAt), 0)
|
|
||||||
if c.config.time().Sub(createdAt) > maxSessionTicketLifetime {
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
|
|
||||||
// Never resume a session for a different TLS version.
|
|
||||||
if c.vers != hs.sessionState.vers {
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
|
|
||||||
cipherSuiteOk := false
|
|
||||||
// Check that the client is still offering the ciphersuite in the session.
|
|
||||||
for _, id := range hs.clientHello.cipherSuites {
|
|
||||||
if id == hs.sessionState.cipherSuite {
|
|
||||||
cipherSuiteOk = true
|
|
||||||
break
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if !cipherSuiteOk {
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
|
|
||||||
// Check that we also support the ciphersuite from the session.
|
|
||||||
hs.suite = selectCipherSuite([]uint16{hs.sessionState.cipherSuite},
|
|
||||||
c.config.cipherSuites(), hs.cipherSuiteOk)
|
|
||||||
if hs.suite == nil {
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
|
|
||||||
sessionHasClientCerts := len(hs.sessionState.certificates) != 0
|
|
||||||
needClientCerts := requiresClientCert(c.config.ClientAuth)
|
|
||||||
if needClientCerts && !sessionHasClientCerts {
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
if sessionHasClientCerts && c.config.ClientAuth == NoClientCert {
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
|
|
||||||
func (hs *serverHandshakeState) doResumeHandshake() error {
|
|
||||||
c := hs.c
|
|
||||||
|
|
||||||
hs.hello.cipherSuite = hs.suite.id
|
|
||||||
c.cipherSuite = hs.suite.id
|
|
||||||
// We echo the client's session ID in the ServerHello to let it know
|
|
||||||
// that we're doing a resumption.
|
|
||||||
hs.hello.sessionId = hs.clientHello.sessionId
|
|
||||||
hs.hello.ticketSupported = hs.sessionState.usedOldKey
|
|
||||||
hs.finishedHash = newFinishedHash(c.vers, hs.suite)
|
|
||||||
hs.finishedHash.discardHandshakeBuffer()
|
|
||||||
hs.finishedHash.Write(hs.clientHello.marshal())
|
|
||||||
hs.finishedHash.Write(hs.hello.marshal())
|
|
||||||
if _, err := c.writeRecord(recordTypeHandshake, hs.hello.marshal()); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
if err := c.processCertsFromClient(Certificate{
|
|
||||||
Certificate: hs.sessionState.certificates,
|
|
||||||
}); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
if c.config.VerifyConnection != nil {
|
|
||||||
if err := c.config.VerifyConnection(c.connectionStateLocked()); err != nil {
|
|
||||||
c.sendAlert(alertBadCertificate)
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
hs.masterSecret = hs.sessionState.masterSecret
|
|
||||||
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (hs *serverHandshakeState) doFullHandshake() error {
|
|
||||||
c := hs.c
|
|
||||||
|
|
||||||
if hs.clientHello.ocspStapling && len(hs.cert.OCSPStaple) > 0 {
|
|
||||||
hs.hello.ocspStapling = true
|
|
||||||
}
|
|
||||||
|
|
||||||
hs.hello.ticketSupported = hs.clientHello.ticketSupported && !c.config.SessionTicketsDisabled
|
|
||||||
hs.hello.cipherSuite = hs.suite.id
|
|
||||||
|
|
||||||
hs.finishedHash = newFinishedHash(hs.c.vers, hs.suite)
|
|
||||||
if c.config.ClientAuth == NoClientCert {
|
|
||||||
// No need to keep a full record of the handshake if client
|
|
||||||
// certificates won't be used.
|
|
||||||
hs.finishedHash.discardHandshakeBuffer()
|
|
||||||
}
|
|
||||||
hs.finishedHash.Write(hs.clientHello.marshal())
|
|
||||||
hs.finishedHash.Write(hs.hello.marshal())
|
|
||||||
if _, err := c.writeRecord(recordTypeHandshake, hs.hello.marshal()); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
certMsg := new(certificateMsg)
|
|
||||||
certMsg.certificates = hs.cert.Certificate
|
|
||||||
hs.finishedHash.Write(certMsg.marshal())
|
|
||||||
if _, err := c.writeRecord(recordTypeHandshake, certMsg.marshal()); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
if hs.hello.ocspStapling {
|
|
||||||
certStatus := new(certificateStatusMsg)
|
|
||||||
certStatus.response = hs.cert.OCSPStaple
|
|
||||||
hs.finishedHash.Write(certStatus.marshal())
|
|
||||||
if _, err := c.writeRecord(recordTypeHandshake, certStatus.marshal()); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
keyAgreement := hs.suite.ka(c.vers)
|
|
||||||
skx, err := keyAgreement.generateServerKeyExchange(c.config, hs.cert, hs.clientHello, hs.hello)
|
|
||||||
if err != nil {
|
|
||||||
c.sendAlert(alertHandshakeFailure)
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
if skx != nil {
|
|
||||||
hs.finishedHash.Write(skx.marshal())
|
|
||||||
if _, err := c.writeRecord(recordTypeHandshake, skx.marshal()); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
var certReq *certificateRequestMsg
|
|
||||||
if c.config.ClientAuth >= RequestClientCert {
|
|
||||||
// Request a client certificate
|
|
||||||
certReq = new(certificateRequestMsg)
|
|
||||||
certReq.certificateTypes = []byte{
|
|
||||||
byte(certTypeRSASign),
|
|
||||||
byte(certTypeECDSASign),
|
|
||||||
}
|
|
||||||
if c.vers >= VersionTLS12 {
|
|
||||||
certReq.hasSignatureAlgorithm = true
|
|
||||||
certReq.supportedSignatureAlgorithms = supportedSignatureAlgorithms
|
|
||||||
}
|
|
||||||
|
|
||||||
// An empty list of certificateAuthorities signals to
|
|
||||||
// the client that it may send any certificate in response
|
|
||||||
// to our request. When we know the CAs we trust, then
|
|
||||||
// we can send them down, so that the client can choose
|
|
||||||
// an appropriate certificate to give to us.
|
|
||||||
if c.config.ClientCAs != nil {
|
|
||||||
certReq.certificateAuthorities = c.config.ClientCAs.Subjects()
|
|
||||||
}
|
|
||||||
hs.finishedHash.Write(certReq.marshal())
|
|
||||||
if _, err := c.writeRecord(recordTypeHandshake, certReq.marshal()); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
helloDone := new(serverHelloDoneMsg)
|
|
||||||
hs.finishedHash.Write(helloDone.marshal())
|
|
||||||
if _, err := c.writeRecord(recordTypeHandshake, helloDone.marshal()); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
if _, err := c.flush(); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
var pub crypto.PublicKey // public key for client auth, if any
|
|
||||||
|
|
||||||
msg, err := c.readHandshake()
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
// If we requested a client certificate, then the client must send a
|
|
||||||
// certificate message, even if it's empty.
|
|
||||||
if c.config.ClientAuth >= RequestClientCert {
|
|
||||||
certMsg, ok := msg.(*certificateMsg)
|
|
||||||
if !ok {
|
|
||||||
c.sendAlert(alertUnexpectedMessage)
|
|
||||||
return unexpectedMessageError(certMsg, msg)
|
|
||||||
}
|
|
||||||
hs.finishedHash.Write(certMsg.marshal())
|
|
||||||
|
|
||||||
if err := c.processCertsFromClient(Certificate{
|
|
||||||
Certificate: certMsg.certificates,
|
|
||||||
}); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
if len(certMsg.certificates) != 0 {
|
|
||||||
pub = c.peerCertificates[0].PublicKey
|
|
||||||
}
|
|
||||||
|
|
||||||
msg, err = c.readHandshake()
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if c.config.VerifyConnection != nil {
|
|
||||||
if err := c.config.VerifyConnection(c.connectionStateLocked()); err != nil {
|
|
||||||
c.sendAlert(alertBadCertificate)
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Get client key exchange
|
|
||||||
ckx, ok := msg.(*clientKeyExchangeMsg)
|
|
||||||
if !ok {
|
|
||||||
c.sendAlert(alertUnexpectedMessage)
|
|
||||||
return unexpectedMessageError(ckx, msg)
|
|
||||||
}
|
|
||||||
hs.finishedHash.Write(ckx.marshal())
|
|
||||||
|
|
||||||
preMasterSecret, err := keyAgreement.processClientKeyExchange(c.config, hs.cert, ckx, c.vers)
|
|
||||||
if err != nil {
|
|
||||||
c.sendAlert(alertHandshakeFailure)
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
hs.masterSecret = masterFromPreMasterSecret(c.vers, hs.suite, preMasterSecret, hs.clientHello.random, hs.hello.random)
|
|
||||||
if err := c.config.writeKeyLog(keyLogLabelTLS12, hs.clientHello.random, hs.masterSecret); err != nil {
|
|
||||||
c.sendAlert(alertInternalError)
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
// If we received a client cert in response to our certificate request message,
|
|
||||||
// the client will send us a certificateVerifyMsg immediately after the
|
|
||||||
// clientKeyExchangeMsg. This message is a digest of all preceding
|
|
||||||
// handshake-layer messages that is signed using the private key corresponding
|
|
||||||
// to the client's certificate. This allows us to verify that the client is in
|
|
||||||
// possession of the private key of the certificate.
|
|
||||||
if len(c.peerCertificates) > 0 {
|
|
||||||
msg, err = c.readHandshake()
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
certVerify, ok := msg.(*certificateVerifyMsg)
|
|
||||||
if !ok {
|
|
||||||
c.sendAlert(alertUnexpectedMessage)
|
|
||||||
return unexpectedMessageError(certVerify, msg)
|
|
||||||
}
|
|
||||||
|
|
||||||
var sigType uint8
|
|
||||||
var sigHash crypto.Hash
|
|
||||||
if c.vers >= VersionTLS12 {
|
|
||||||
if !isSupportedSignatureAlgorithm(certVerify.signatureAlgorithm, certReq.supportedSignatureAlgorithms) {
|
|
||||||
c.sendAlert(alertIllegalParameter)
|
|
||||||
return errors.New("tls: client certificate used with invalid signature algorithm")
|
|
||||||
}
|
|
||||||
sigType, sigHash, err = typeAndHashFromSignatureScheme(certVerify.signatureAlgorithm)
|
|
||||||
if err != nil {
|
|
||||||
return c.sendAlert(alertInternalError)
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
sigType, sigHash, err = legacyTypeAndHashFromPublicKey(pub)
|
|
||||||
if err != nil {
|
|
||||||
c.sendAlert(alertIllegalParameter)
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
signed := hs.finishedHash.hashForClientCertificate(sigType, sigHash, hs.masterSecret)
|
|
||||||
if err := verifyHandshakeSignature(sigType, pub, sigHash, signed, certVerify.signature); err != nil {
|
|
||||||
c.sendAlert(alertDecryptError)
|
|
||||||
return errors.New("tls: invalid signature by the client certificate: " + err.Error())
|
|
||||||
}
|
|
||||||
|
|
||||||
hs.finishedHash.Write(certVerify.marshal())
|
|
||||||
}
|
|
||||||
|
|
||||||
hs.finishedHash.discardHandshakeBuffer()
|
|
||||||
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (hs *serverHandshakeState) establishKeys() error {
|
|
||||||
c := hs.c
|
|
||||||
|
|
||||||
clientMAC, serverMAC, clientKey, serverKey, clientIV, serverIV :=
|
|
||||||
keysFromMasterSecret(c.vers, hs.suite, hs.masterSecret, hs.clientHello.random, hs.hello.random, hs.suite.macLen, hs.suite.keyLen, hs.suite.ivLen)
|
|
||||||
|
|
||||||
var clientCipher, serverCipher interface{}
|
|
||||||
var clientHash, serverHash hash.Hash
|
|
||||||
|
|
||||||
if hs.suite.aead == nil {
|
|
||||||
clientCipher = hs.suite.cipher(clientKey, clientIV, true /* for reading */)
|
|
||||||
clientHash = hs.suite.mac(clientMAC)
|
|
||||||
serverCipher = hs.suite.cipher(serverKey, serverIV, false /* not for reading */)
|
|
||||||
serverHash = hs.suite.mac(serverMAC)
|
|
||||||
} else {
|
|
||||||
clientCipher = hs.suite.aead(clientKey, clientIV)
|
|
||||||
serverCipher = hs.suite.aead(serverKey, serverIV)
|
|
||||||
}
|
|
||||||
|
|
||||||
c.in.prepareCipherSpec(c.vers, clientCipher, clientHash)
|
|
||||||
c.out.prepareCipherSpec(c.vers, serverCipher, serverHash)
|
|
||||||
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (hs *serverHandshakeState) readFinished(out []byte) error {
|
|
||||||
c := hs.c
|
|
||||||
|
|
||||||
if err := c.readChangeCipherSpec(); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
msg, err := c.readHandshake()
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
clientFinished, ok := msg.(*finishedMsg)
|
|
||||||
if !ok {
|
|
||||||
c.sendAlert(alertUnexpectedMessage)
|
|
||||||
return unexpectedMessageError(clientFinished, msg)
|
|
||||||
}
|
|
||||||
|
|
||||||
verify := hs.finishedHash.clientSum(hs.masterSecret)
|
|
||||||
if len(verify) != len(clientFinished.verifyData) ||
|
|
||||||
subtle.ConstantTimeCompare(verify, clientFinished.verifyData) != 1 {
|
|
||||||
c.sendAlert(alertHandshakeFailure)
|
|
||||||
return errors.New("tls: client's Finished message is incorrect")
|
|
||||||
}
|
|
||||||
|
|
||||||
hs.finishedHash.Write(clientFinished.marshal())
|
|
||||||
copy(out, verify)
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (hs *serverHandshakeState) sendSessionTicket() error {
|
|
||||||
// ticketSupported is set in a resumption handshake if the
|
|
||||||
// ticket from the client was encrypted with an old session
|
|
||||||
// ticket key and thus a refreshed ticket should be sent.
|
|
||||||
if !hs.hello.ticketSupported {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
c := hs.c
|
|
||||||
m := new(newSessionTicketMsg)
|
|
||||||
|
|
||||||
createdAt := uint64(c.config.time().Unix())
|
|
||||||
if hs.sessionState != nil {
|
|
||||||
// If this is re-wrapping an old key, then keep
|
|
||||||
// the original time it was created.
|
|
||||||
createdAt = hs.sessionState.createdAt
|
|
||||||
}
|
|
||||||
|
|
||||||
var certsFromClient [][]byte
|
|
||||||
for _, cert := range c.peerCertificates {
|
|
||||||
certsFromClient = append(certsFromClient, cert.Raw)
|
|
||||||
}
|
|
||||||
state := sessionState{
|
|
||||||
vers: c.vers,
|
|
||||||
cipherSuite: hs.suite.id,
|
|
||||||
createdAt: createdAt,
|
|
||||||
masterSecret: hs.masterSecret,
|
|
||||||
certificates: certsFromClient,
|
|
||||||
}
|
|
||||||
var err error
|
|
||||||
m.ticket, err = c.encryptTicket(state.marshal())
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
hs.finishedHash.Write(m.marshal())
|
|
||||||
if _, err := c.writeRecord(recordTypeHandshake, m.marshal()); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (hs *serverHandshakeState) sendFinished(out []byte) error {
|
|
||||||
c := hs.c
|
|
||||||
|
|
||||||
if _, err := c.writeRecord(recordTypeChangeCipherSpec, []byte{1}); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
finished := new(finishedMsg)
|
|
||||||
finished.verifyData = hs.finishedHash.serverSum(hs.masterSecret)
|
|
||||||
hs.finishedHash.Write(finished.marshal())
|
|
||||||
if _, err := c.writeRecord(recordTypeHandshake, finished.marshal()); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
copy(out, finished.verifyData)
|
|
||||||
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// processCertsFromClient takes a chain of client certificates either from a
|
|
||||||
// Certificates message or from a sessionState and verifies them. It returns
|
|
||||||
// the public key of the leaf certificate.
|
|
||||||
func (c *Conn) processCertsFromClient(certificate Certificate) error {
|
|
||||||
certificates := certificate.Certificate
|
|
||||||
certs := make([]*x509.Certificate, len(certificates))
|
|
||||||
var err error
|
|
||||||
for i, asn1Data := range certificates {
|
|
||||||
if certs[i], err = x509.ParseCertificate(asn1Data); err != nil {
|
|
||||||
c.sendAlert(alertBadCertificate)
|
|
||||||
return errors.New("tls: failed to parse client certificate: " + err.Error())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if len(certs) == 0 && requiresClientCert(c.config.ClientAuth) {
|
|
||||||
c.sendAlert(alertBadCertificate)
|
|
||||||
return errors.New("tls: client didn't provide a certificate")
|
|
||||||
}
|
|
||||||
|
|
||||||
if c.config.ClientAuth >= VerifyClientCertIfGiven && len(certs) > 0 {
|
|
||||||
opts := x509.VerifyOptions{
|
|
||||||
Roots: c.config.ClientCAs,
|
|
||||||
CurrentTime: c.config.time(),
|
|
||||||
Intermediates: x509.NewCertPool(),
|
|
||||||
KeyUsages: []x509.ExtKeyUsage{x509.ExtKeyUsageClientAuth},
|
|
||||||
}
|
|
||||||
|
|
||||||
for _, cert := range certs[1:] {
|
|
||||||
opts.Intermediates.AddCert(cert)
|
|
||||||
}
|
|
||||||
|
|
||||||
chains, err := certs[0].Verify(opts)
|
|
||||||
if err != nil {
|
|
||||||
c.sendAlert(alertBadCertificate)
|
|
||||||
return errors.New("tls: failed to verify client certificate: " + err.Error())
|
|
||||||
}
|
|
||||||
|
|
||||||
c.verifiedChains = chains
|
|
||||||
}
|
|
||||||
|
|
||||||
c.peerCertificates = certs
|
|
||||||
c.ocspResponse = certificate.OCSPStaple
|
|
||||||
c.scts = certificate.SignedCertificateTimestamps
|
|
||||||
|
|
||||||
if len(certs) > 0 {
|
|
||||||
switch certs[0].PublicKey.(type) {
|
|
||||||
case *ecdsa.PublicKey, *rsa.PublicKey, ed25519.PublicKey:
|
|
||||||
default:
|
|
||||||
c.sendAlert(alertUnsupportedCertificate)
|
|
||||||
return fmt.Errorf("tls: client certificate contains an unsupported public key of type %T", certs[0].PublicKey)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if c.config.VerifyPeerCertificate != nil {
|
|
||||||
if err := c.config.VerifyPeerCertificate(certificates, c.verifiedChains); err != nil {
|
|
||||||
c.sendAlert(alertBadCertificate)
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func newClientHelloInfo(ctx context.Context, c *Conn, clientHello *clientHelloMsg) *ClientHelloInfo {
|
|
||||||
supportedVersions := clientHello.supportedVersions
|
|
||||||
if len(clientHello.supportedVersions) == 0 {
|
|
||||||
supportedVersions = supportedVersionsFromMax(clientHello.vers)
|
|
||||||
}
|
|
||||||
|
|
||||||
return toClientHelloInfo(&clientHelloInfo{
|
|
||||||
CipherSuites: clientHello.cipherSuites,
|
|
||||||
ServerName: clientHello.serverName,
|
|
||||||
SupportedCurves: clientHello.supportedCurves,
|
|
||||||
SupportedPoints: clientHello.supportedPoints,
|
|
||||||
SignatureSchemes: clientHello.supportedSignatureAlgorithms,
|
|
||||||
SupportedProtos: clientHello.alpnProtocols,
|
|
||||||
SupportedVersions: supportedVersions,
|
|
||||||
Conn: c.conn,
|
|
||||||
config: toConfig(c.config),
|
|
||||||
ctx: ctx,
|
|
||||||
})
|
|
||||||
}
|
|
|
@ -1,895 +0,0 @@
|
||||||
// Copyright 2018 The Go Authors. All rights reserved.
|
|
||||||
// Use of this source code is governed by a BSD-style
|
|
||||||
// license that can be found in the LICENSE file.
|
|
||||||
|
|
||||||
package qtls
|
|
||||||
|
|
||||||
import (
|
|
||||||
"bytes"
|
|
||||||
"context"
|
|
||||||
"crypto"
|
|
||||||
"crypto/hmac"
|
|
||||||
"crypto/rsa"
|
|
||||||
"errors"
|
|
||||||
"hash"
|
|
||||||
"io"
|
|
||||||
"sync/atomic"
|
|
||||||
"time"
|
|
||||||
)
|
|
||||||
|
|
||||||
// maxClientPSKIdentities is the number of client PSK identities the server will
|
|
||||||
// attempt to validate. It will ignore the rest not to let cheap ClientHello
|
|
||||||
// messages cause too much work in session ticket decryption attempts.
|
|
||||||
const maxClientPSKIdentities = 5
|
|
||||||
|
|
||||||
type serverHandshakeStateTLS13 struct {
|
|
||||||
c *Conn
|
|
||||||
ctx context.Context
|
|
||||||
clientHello *clientHelloMsg
|
|
||||||
hello *serverHelloMsg
|
|
||||||
alpnNegotiationErr error
|
|
||||||
encryptedExtensions *encryptedExtensionsMsg
|
|
||||||
sentDummyCCS bool
|
|
||||||
usingPSK bool
|
|
||||||
suite *cipherSuiteTLS13
|
|
||||||
cert *Certificate
|
|
||||||
sigAlg SignatureScheme
|
|
||||||
earlySecret []byte
|
|
||||||
sharedKey []byte
|
|
||||||
handshakeSecret []byte
|
|
||||||
masterSecret []byte
|
|
||||||
trafficSecret []byte // client_application_traffic_secret_0
|
|
||||||
transcript hash.Hash
|
|
||||||
clientFinished []byte
|
|
||||||
}
|
|
||||||
|
|
||||||
func (hs *serverHandshakeStateTLS13) handshake() error {
|
|
||||||
c := hs.c
|
|
||||||
|
|
||||||
// For an overview of the TLS 1.3 handshake, see RFC 8446, Section 2.
|
|
||||||
if err := hs.processClientHello(); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
if err := hs.checkForResumption(); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
if err := hs.pickCertificate(); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
c.buffering = true
|
|
||||||
if err := hs.sendServerParameters(); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
if err := hs.sendServerCertificate(); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
if err := hs.sendServerFinished(); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
// Note that at this point we could start sending application data without
|
|
||||||
// waiting for the client's second flight, but the application might not
|
|
||||||
// expect the lack of replay protection of the ClientHello parameters.
|
|
||||||
if _, err := c.flush(); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
if err := hs.readClientCertificate(); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
if err := hs.readClientFinished(); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
atomic.StoreUint32(&c.handshakeStatus, 1)
|
|
||||||
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (hs *serverHandshakeStateTLS13) processClientHello() error {
|
|
||||||
c := hs.c
|
|
||||||
|
|
||||||
hs.hello = new(serverHelloMsg)
|
|
||||||
hs.encryptedExtensions = new(encryptedExtensionsMsg)
|
|
||||||
|
|
||||||
// TLS 1.3 froze the ServerHello.legacy_version field, and uses
|
|
||||||
// supported_versions instead. See RFC 8446, sections 4.1.3 and 4.2.1.
|
|
||||||
hs.hello.vers = VersionTLS12
|
|
||||||
hs.hello.supportedVersion = c.vers
|
|
||||||
|
|
||||||
if len(hs.clientHello.supportedVersions) == 0 {
|
|
||||||
c.sendAlert(alertIllegalParameter)
|
|
||||||
return errors.New("tls: client used the legacy version field to negotiate TLS 1.3")
|
|
||||||
}
|
|
||||||
|
|
||||||
// Abort if the client is doing a fallback and landing lower than what we
|
|
||||||
// support. See RFC 7507, which however does not specify the interaction
|
|
||||||
// with supported_versions. The only difference is that with
|
|
||||||
// supported_versions a client has a chance to attempt a [TLS 1.2, TLS 1.4]
|
|
||||||
// handshake in case TLS 1.3 is broken but 1.2 is not. Alas, in that case,
|
|
||||||
// it will have to drop the TLS_FALLBACK_SCSV protection if it falls back to
|
|
||||||
// TLS 1.2, because a TLS 1.3 server would abort here. The situation before
|
|
||||||
// supported_versions was not better because there was just no way to do a
|
|
||||||
// TLS 1.4 handshake without risking the server selecting TLS 1.3.
|
|
||||||
for _, id := range hs.clientHello.cipherSuites {
|
|
||||||
if id == TLS_FALLBACK_SCSV {
|
|
||||||
// Use c.vers instead of max(supported_versions) because an attacker
|
|
||||||
// could defeat this by adding an arbitrary high version otherwise.
|
|
||||||
if c.vers < c.config.maxSupportedVersion() {
|
|
||||||
c.sendAlert(alertInappropriateFallback)
|
|
||||||
return errors.New("tls: client using inappropriate protocol fallback")
|
|
||||||
}
|
|
||||||
break
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if len(hs.clientHello.compressionMethods) != 1 ||
|
|
||||||
hs.clientHello.compressionMethods[0] != compressionNone {
|
|
||||||
c.sendAlert(alertIllegalParameter)
|
|
||||||
return errors.New("tls: TLS 1.3 client supports illegal compression methods")
|
|
||||||
}
|
|
||||||
|
|
||||||
hs.hello.random = make([]byte, 32)
|
|
||||||
if _, err := io.ReadFull(c.config.rand(), hs.hello.random); err != nil {
|
|
||||||
c.sendAlert(alertInternalError)
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
if len(hs.clientHello.secureRenegotiation) != 0 {
|
|
||||||
c.sendAlert(alertHandshakeFailure)
|
|
||||||
return errors.New("tls: initial handshake had non-empty renegotiation extension")
|
|
||||||
}
|
|
||||||
|
|
||||||
hs.hello.sessionId = hs.clientHello.sessionId
|
|
||||||
hs.hello.compressionMethod = compressionNone
|
|
||||||
|
|
||||||
if hs.suite == nil {
|
|
||||||
var preferenceList []uint16
|
|
||||||
for _, suiteID := range c.config.CipherSuites {
|
|
||||||
for _, suite := range cipherSuitesTLS13 {
|
|
||||||
if suite.id == suiteID {
|
|
||||||
preferenceList = append(preferenceList, suiteID)
|
|
||||||
break
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if len(preferenceList) == 0 {
|
|
||||||
preferenceList = defaultCipherSuitesTLS13
|
|
||||||
if !hasAESGCMHardwareSupport || !aesgcmPreferred(hs.clientHello.cipherSuites) {
|
|
||||||
preferenceList = defaultCipherSuitesTLS13NoAES
|
|
||||||
}
|
|
||||||
}
|
|
||||||
for _, suiteID := range preferenceList {
|
|
||||||
hs.suite = mutualCipherSuiteTLS13(hs.clientHello.cipherSuites, suiteID)
|
|
||||||
if hs.suite != nil {
|
|
||||||
break
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if hs.suite == nil {
|
|
||||||
c.sendAlert(alertHandshakeFailure)
|
|
||||||
return errors.New("tls: no cipher suite supported by both client and server")
|
|
||||||
}
|
|
||||||
c.cipherSuite = hs.suite.id
|
|
||||||
hs.hello.cipherSuite = hs.suite.id
|
|
||||||
hs.transcript = hs.suite.hash.New()
|
|
||||||
|
|
||||||
// Pick the ECDHE group in server preference order, but give priority to
|
|
||||||
// groups with a key share, to avoid a HelloRetryRequest round-trip.
|
|
||||||
var selectedGroup CurveID
|
|
||||||
var clientKeyShare *keyShare
|
|
||||||
GroupSelection:
|
|
||||||
for _, preferredGroup := range c.config.curvePreferences() {
|
|
||||||
for _, ks := range hs.clientHello.keyShares {
|
|
||||||
if ks.group == preferredGroup {
|
|
||||||
selectedGroup = ks.group
|
|
||||||
clientKeyShare = &ks
|
|
||||||
break GroupSelection
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if selectedGroup != 0 {
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
for _, group := range hs.clientHello.supportedCurves {
|
|
||||||
if group == preferredGroup {
|
|
||||||
selectedGroup = group
|
|
||||||
break
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if selectedGroup == 0 {
|
|
||||||
c.sendAlert(alertHandshakeFailure)
|
|
||||||
return errors.New("tls: no ECDHE curve supported by both client and server")
|
|
||||||
}
|
|
||||||
if clientKeyShare == nil {
|
|
||||||
if err := hs.doHelloRetryRequest(selectedGroup); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
clientKeyShare = &hs.clientHello.keyShares[0]
|
|
||||||
}
|
|
||||||
|
|
||||||
if _, ok := curveForCurveID(selectedGroup); selectedGroup != X25519 && !ok {
|
|
||||||
c.sendAlert(alertInternalError)
|
|
||||||
return errors.New("tls: CurvePreferences includes unsupported curve")
|
|
||||||
}
|
|
||||||
params, err := generateECDHEParameters(c.config.rand(), selectedGroup)
|
|
||||||
if err != nil {
|
|
||||||
c.sendAlert(alertInternalError)
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
hs.hello.serverShare = keyShare{group: selectedGroup, data: params.PublicKey()}
|
|
||||||
hs.sharedKey = params.SharedKey(clientKeyShare.data)
|
|
||||||
if hs.sharedKey == nil {
|
|
||||||
c.sendAlert(alertIllegalParameter)
|
|
||||||
return errors.New("tls: invalid client key share")
|
|
||||||
}
|
|
||||||
|
|
||||||
c.serverName = hs.clientHello.serverName
|
|
||||||
|
|
||||||
if c.extraConfig != nil && c.extraConfig.ReceivedExtensions != nil {
|
|
||||||
c.extraConfig.ReceivedExtensions(typeClientHello, hs.clientHello.additionalExtensions)
|
|
||||||
}
|
|
||||||
|
|
||||||
selectedProto, err := negotiateALPN(c.config.NextProtos, hs.clientHello.alpnProtocols)
|
|
||||||
if err != nil {
|
|
||||||
hs.alpnNegotiationErr = err
|
|
||||||
}
|
|
||||||
hs.encryptedExtensions.alpnProtocol = selectedProto
|
|
||||||
c.clientProtocol = selectedProto
|
|
||||||
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (hs *serverHandshakeStateTLS13) checkForResumption() error {
|
|
||||||
c := hs.c
|
|
||||||
|
|
||||||
if c.config.SessionTicketsDisabled {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
modeOK := false
|
|
||||||
for _, mode := range hs.clientHello.pskModes {
|
|
||||||
if mode == pskModeDHE {
|
|
||||||
modeOK = true
|
|
||||||
break
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if !modeOK {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
if len(hs.clientHello.pskIdentities) != len(hs.clientHello.pskBinders) {
|
|
||||||
c.sendAlert(alertIllegalParameter)
|
|
||||||
return errors.New("tls: invalid or missing PSK binders")
|
|
||||||
}
|
|
||||||
if len(hs.clientHello.pskIdentities) == 0 {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
for i, identity := range hs.clientHello.pskIdentities {
|
|
||||||
if i >= maxClientPSKIdentities {
|
|
||||||
break
|
|
||||||
}
|
|
||||||
|
|
||||||
plaintext, _ := c.decryptTicket(identity.label)
|
|
||||||
if plaintext == nil {
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
sessionState := new(sessionStateTLS13)
|
|
||||||
if ok := sessionState.unmarshal(plaintext); !ok {
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
|
|
||||||
if hs.clientHello.earlyData {
|
|
||||||
if sessionState.maxEarlyData == 0 {
|
|
||||||
c.sendAlert(alertUnsupportedExtension)
|
|
||||||
return errors.New("tls: client sent unexpected early data")
|
|
||||||
}
|
|
||||||
|
|
||||||
if hs.alpnNegotiationErr == nil && sessionState.alpn == c.clientProtocol &&
|
|
||||||
c.extraConfig != nil && c.extraConfig.MaxEarlyData > 0 &&
|
|
||||||
c.extraConfig.Accept0RTT != nil && c.extraConfig.Accept0RTT(sessionState.appData) {
|
|
||||||
hs.encryptedExtensions.earlyData = true
|
|
||||||
c.used0RTT = true
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
createdAt := time.Unix(int64(sessionState.createdAt), 0)
|
|
||||||
if c.config.time().Sub(createdAt) > maxSessionTicketLifetime {
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
|
|
||||||
// We don't check the obfuscated ticket age because it's affected by
|
|
||||||
// clock skew and it's only a freshness signal useful for shrinking the
|
|
||||||
// window for replay attacks, which don't affect us as we don't do 0-RTT.
|
|
||||||
|
|
||||||
pskSuite := cipherSuiteTLS13ByID(sessionState.cipherSuite)
|
|
||||||
if pskSuite == nil || pskSuite.hash != hs.suite.hash {
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
|
|
||||||
// PSK connections don't re-establish client certificates, but carry
|
|
||||||
// them over in the session ticket. Ensure the presence of client certs
|
|
||||||
// in the ticket is consistent with the configured requirements.
|
|
||||||
sessionHasClientCerts := len(sessionState.certificate.Certificate) != 0
|
|
||||||
needClientCerts := requiresClientCert(c.config.ClientAuth)
|
|
||||||
if needClientCerts && !sessionHasClientCerts {
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
if sessionHasClientCerts && c.config.ClientAuth == NoClientCert {
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
|
|
||||||
psk := hs.suite.expandLabel(sessionState.resumptionSecret, "resumption",
|
|
||||||
nil, hs.suite.hash.Size())
|
|
||||||
hs.earlySecret = hs.suite.extract(psk, nil)
|
|
||||||
binderKey := hs.suite.deriveSecret(hs.earlySecret, resumptionBinderLabel, nil)
|
|
||||||
// Clone the transcript in case a HelloRetryRequest was recorded.
|
|
||||||
transcript := cloneHash(hs.transcript, hs.suite.hash)
|
|
||||||
if transcript == nil {
|
|
||||||
c.sendAlert(alertInternalError)
|
|
||||||
return errors.New("tls: internal error: failed to clone hash")
|
|
||||||
}
|
|
||||||
transcript.Write(hs.clientHello.marshalWithoutBinders())
|
|
||||||
pskBinder := hs.suite.finishedHash(binderKey, transcript)
|
|
||||||
if !hmac.Equal(hs.clientHello.pskBinders[i], pskBinder) {
|
|
||||||
c.sendAlert(alertDecryptError)
|
|
||||||
return errors.New("tls: invalid PSK binder")
|
|
||||||
}
|
|
||||||
|
|
||||||
c.didResume = true
|
|
||||||
if err := c.processCertsFromClient(sessionState.certificate); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
h := cloneHash(hs.transcript, hs.suite.hash)
|
|
||||||
h.Write(hs.clientHello.marshal())
|
|
||||||
if hs.encryptedExtensions.earlyData {
|
|
||||||
clientEarlySecret := hs.suite.deriveSecret(hs.earlySecret, "c e traffic", h)
|
|
||||||
c.in.exportKey(Encryption0RTT, hs.suite, clientEarlySecret)
|
|
||||||
if err := c.config.writeKeyLog(keyLogLabelEarlyTraffic, hs.clientHello.random, clientEarlySecret); err != nil {
|
|
||||||
c.sendAlert(alertInternalError)
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
hs.hello.selectedIdentityPresent = true
|
|
||||||
hs.hello.selectedIdentity = uint16(i)
|
|
||||||
hs.usingPSK = true
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// cloneHash uses the encoding.BinaryMarshaler and encoding.BinaryUnmarshaler
|
|
||||||
// interfaces implemented by standard library hashes to clone the state of in
|
|
||||||
// to a new instance of h. It returns nil if the operation fails.
|
|
||||||
func cloneHash(in hash.Hash, h crypto.Hash) hash.Hash {
|
|
||||||
// Recreate the interface to avoid importing encoding.
|
|
||||||
type binaryMarshaler interface {
|
|
||||||
MarshalBinary() (data []byte, err error)
|
|
||||||
UnmarshalBinary(data []byte) error
|
|
||||||
}
|
|
||||||
marshaler, ok := in.(binaryMarshaler)
|
|
||||||
if !ok {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
state, err := marshaler.MarshalBinary()
|
|
||||||
if err != nil {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
out := h.New()
|
|
||||||
unmarshaler, ok := out.(binaryMarshaler)
|
|
||||||
if !ok {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
if err := unmarshaler.UnmarshalBinary(state); err != nil {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
return out
|
|
||||||
}
|
|
||||||
|
|
||||||
func (hs *serverHandshakeStateTLS13) pickCertificate() error {
|
|
||||||
c := hs.c
|
|
||||||
|
|
||||||
// Only one of PSK and certificates are used at a time.
|
|
||||||
if hs.usingPSK {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// signature_algorithms is required in TLS 1.3. See RFC 8446, Section 4.2.3.
|
|
||||||
if len(hs.clientHello.supportedSignatureAlgorithms) == 0 {
|
|
||||||
return c.sendAlert(alertMissingExtension)
|
|
||||||
}
|
|
||||||
|
|
||||||
certificate, err := c.config.getCertificate(newClientHelloInfo(hs.ctx, c, hs.clientHello))
|
|
||||||
if err != nil {
|
|
||||||
if err == errNoCertificates {
|
|
||||||
c.sendAlert(alertUnrecognizedName)
|
|
||||||
} else {
|
|
||||||
c.sendAlert(alertInternalError)
|
|
||||||
}
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
hs.sigAlg, err = selectSignatureScheme(c.vers, certificate, hs.clientHello.supportedSignatureAlgorithms)
|
|
||||||
if err != nil {
|
|
||||||
// getCertificate returned a certificate that is unsupported or
|
|
||||||
// incompatible with the client's signature algorithms.
|
|
||||||
c.sendAlert(alertHandshakeFailure)
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
hs.cert = certificate
|
|
||||||
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// sendDummyChangeCipherSpec sends a ChangeCipherSpec record for compatibility
|
|
||||||
// with middleboxes that didn't implement TLS correctly. See RFC 8446, Appendix D.4.
|
|
||||||
func (hs *serverHandshakeStateTLS13) sendDummyChangeCipherSpec() error {
|
|
||||||
if hs.sentDummyCCS {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
hs.sentDummyCCS = true
|
|
||||||
|
|
||||||
_, err := hs.c.writeRecord(recordTypeChangeCipherSpec, []byte{1})
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
func (hs *serverHandshakeStateTLS13) doHelloRetryRequest(selectedGroup CurveID) error {
|
|
||||||
c := hs.c
|
|
||||||
|
|
||||||
// The first ClientHello gets double-hashed into the transcript upon a
|
|
||||||
// HelloRetryRequest. See RFC 8446, Section 4.4.1.
|
|
||||||
hs.transcript.Write(hs.clientHello.marshal())
|
|
||||||
chHash := hs.transcript.Sum(nil)
|
|
||||||
hs.transcript.Reset()
|
|
||||||
hs.transcript.Write([]byte{typeMessageHash, 0, 0, uint8(len(chHash))})
|
|
||||||
hs.transcript.Write(chHash)
|
|
||||||
|
|
||||||
helloRetryRequest := &serverHelloMsg{
|
|
||||||
vers: hs.hello.vers,
|
|
||||||
random: helloRetryRequestRandom,
|
|
||||||
sessionId: hs.hello.sessionId,
|
|
||||||
cipherSuite: hs.hello.cipherSuite,
|
|
||||||
compressionMethod: hs.hello.compressionMethod,
|
|
||||||
supportedVersion: hs.hello.supportedVersion,
|
|
||||||
selectedGroup: selectedGroup,
|
|
||||||
}
|
|
||||||
|
|
||||||
hs.transcript.Write(helloRetryRequest.marshal())
|
|
||||||
if _, err := c.writeRecord(recordTypeHandshake, helloRetryRequest.marshal()); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
if err := hs.sendDummyChangeCipherSpec(); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
msg, err := c.readHandshake()
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
clientHello, ok := msg.(*clientHelloMsg)
|
|
||||||
if !ok {
|
|
||||||
c.sendAlert(alertUnexpectedMessage)
|
|
||||||
return unexpectedMessageError(clientHello, msg)
|
|
||||||
}
|
|
||||||
|
|
||||||
if len(clientHello.keyShares) != 1 || clientHello.keyShares[0].group != selectedGroup {
|
|
||||||
c.sendAlert(alertIllegalParameter)
|
|
||||||
return errors.New("tls: client sent invalid key share in second ClientHello")
|
|
||||||
}
|
|
||||||
|
|
||||||
if clientHello.earlyData {
|
|
||||||
c.sendAlert(alertIllegalParameter)
|
|
||||||
return errors.New("tls: client indicated early data in second ClientHello")
|
|
||||||
}
|
|
||||||
|
|
||||||
if illegalClientHelloChange(clientHello, hs.clientHello) {
|
|
||||||
c.sendAlert(alertIllegalParameter)
|
|
||||||
return errors.New("tls: client illegally modified second ClientHello")
|
|
||||||
}
|
|
||||||
|
|
||||||
if clientHello.earlyData {
|
|
||||||
c.sendAlert(alertIllegalParameter)
|
|
||||||
return errors.New("tls: client offered 0-RTT data in second ClientHello")
|
|
||||||
}
|
|
||||||
|
|
||||||
hs.clientHello = clientHello
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// illegalClientHelloChange reports whether the two ClientHello messages are
|
|
||||||
// different, with the exception of the changes allowed before and after a
|
|
||||||
// HelloRetryRequest. See RFC 8446, Section 4.1.2.
|
|
||||||
func illegalClientHelloChange(ch, ch1 *clientHelloMsg) bool {
|
|
||||||
if len(ch.supportedVersions) != len(ch1.supportedVersions) ||
|
|
||||||
len(ch.cipherSuites) != len(ch1.cipherSuites) ||
|
|
||||||
len(ch.supportedCurves) != len(ch1.supportedCurves) ||
|
|
||||||
len(ch.supportedSignatureAlgorithms) != len(ch1.supportedSignatureAlgorithms) ||
|
|
||||||
len(ch.supportedSignatureAlgorithmsCert) != len(ch1.supportedSignatureAlgorithmsCert) ||
|
|
||||||
len(ch.alpnProtocols) != len(ch1.alpnProtocols) {
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
for i := range ch.supportedVersions {
|
|
||||||
if ch.supportedVersions[i] != ch1.supportedVersions[i] {
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
}
|
|
||||||
for i := range ch.cipherSuites {
|
|
||||||
if ch.cipherSuites[i] != ch1.cipherSuites[i] {
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
}
|
|
||||||
for i := range ch.supportedCurves {
|
|
||||||
if ch.supportedCurves[i] != ch1.supportedCurves[i] {
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
}
|
|
||||||
for i := range ch.supportedSignatureAlgorithms {
|
|
||||||
if ch.supportedSignatureAlgorithms[i] != ch1.supportedSignatureAlgorithms[i] {
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
}
|
|
||||||
for i := range ch.supportedSignatureAlgorithmsCert {
|
|
||||||
if ch.supportedSignatureAlgorithmsCert[i] != ch1.supportedSignatureAlgorithmsCert[i] {
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
}
|
|
||||||
for i := range ch.alpnProtocols {
|
|
||||||
if ch.alpnProtocols[i] != ch1.alpnProtocols[i] {
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return ch.vers != ch1.vers ||
|
|
||||||
!bytes.Equal(ch.random, ch1.random) ||
|
|
||||||
!bytes.Equal(ch.sessionId, ch1.sessionId) ||
|
|
||||||
!bytes.Equal(ch.compressionMethods, ch1.compressionMethods) ||
|
|
||||||
ch.serverName != ch1.serverName ||
|
|
||||||
ch.ocspStapling != ch1.ocspStapling ||
|
|
||||||
!bytes.Equal(ch.supportedPoints, ch1.supportedPoints) ||
|
|
||||||
ch.ticketSupported != ch1.ticketSupported ||
|
|
||||||
!bytes.Equal(ch.sessionTicket, ch1.sessionTicket) ||
|
|
||||||
ch.secureRenegotiationSupported != ch1.secureRenegotiationSupported ||
|
|
||||||
!bytes.Equal(ch.secureRenegotiation, ch1.secureRenegotiation) ||
|
|
||||||
ch.scts != ch1.scts ||
|
|
||||||
!bytes.Equal(ch.cookie, ch1.cookie) ||
|
|
||||||
!bytes.Equal(ch.pskModes, ch1.pskModes)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (hs *serverHandshakeStateTLS13) sendServerParameters() error {
|
|
||||||
c := hs.c
|
|
||||||
|
|
||||||
hs.transcript.Write(hs.clientHello.marshal())
|
|
||||||
hs.transcript.Write(hs.hello.marshal())
|
|
||||||
if _, err := c.writeRecord(recordTypeHandshake, hs.hello.marshal()); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
if err := hs.sendDummyChangeCipherSpec(); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
earlySecret := hs.earlySecret
|
|
||||||
if earlySecret == nil {
|
|
||||||
earlySecret = hs.suite.extract(nil, nil)
|
|
||||||
}
|
|
||||||
hs.handshakeSecret = hs.suite.extract(hs.sharedKey,
|
|
||||||
hs.suite.deriveSecret(earlySecret, "derived", nil))
|
|
||||||
|
|
||||||
clientSecret := hs.suite.deriveSecret(hs.handshakeSecret,
|
|
||||||
clientHandshakeTrafficLabel, hs.transcript)
|
|
||||||
c.in.exportKey(EncryptionHandshake, hs.suite, clientSecret)
|
|
||||||
c.in.setTrafficSecret(hs.suite, clientSecret)
|
|
||||||
serverSecret := hs.suite.deriveSecret(hs.handshakeSecret,
|
|
||||||
serverHandshakeTrafficLabel, hs.transcript)
|
|
||||||
c.out.exportKey(EncryptionHandshake, hs.suite, serverSecret)
|
|
||||||
c.out.setTrafficSecret(hs.suite, serverSecret)
|
|
||||||
|
|
||||||
err := c.config.writeKeyLog(keyLogLabelClientHandshake, hs.clientHello.random, clientSecret)
|
|
||||||
if err != nil {
|
|
||||||
c.sendAlert(alertInternalError)
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
err = c.config.writeKeyLog(keyLogLabelServerHandshake, hs.clientHello.random, serverSecret)
|
|
||||||
if err != nil {
|
|
||||||
c.sendAlert(alertInternalError)
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
if hs.alpnNegotiationErr != nil {
|
|
||||||
c.sendAlert(alertNoApplicationProtocol)
|
|
||||||
return hs.alpnNegotiationErr
|
|
||||||
}
|
|
||||||
if hs.c.extraConfig != nil && hs.c.extraConfig.GetExtensions != nil {
|
|
||||||
hs.encryptedExtensions.additionalExtensions = hs.c.extraConfig.GetExtensions(typeEncryptedExtensions)
|
|
||||||
}
|
|
||||||
|
|
||||||
hs.transcript.Write(hs.encryptedExtensions.marshal())
|
|
||||||
if _, err := c.writeRecord(recordTypeHandshake, hs.encryptedExtensions.marshal()); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (hs *serverHandshakeStateTLS13) requestClientCert() bool {
|
|
||||||
return hs.c.config.ClientAuth >= RequestClientCert && !hs.usingPSK
|
|
||||||
}
|
|
||||||
|
|
||||||
func (hs *serverHandshakeStateTLS13) sendServerCertificate() error {
|
|
||||||
c := hs.c
|
|
||||||
|
|
||||||
// Only one of PSK and certificates are used at a time.
|
|
||||||
if hs.usingPSK {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
if hs.requestClientCert() {
|
|
||||||
// Request a client certificate
|
|
||||||
certReq := new(certificateRequestMsgTLS13)
|
|
||||||
certReq.ocspStapling = true
|
|
||||||
certReq.scts = true
|
|
||||||
certReq.supportedSignatureAlgorithms = supportedSignatureAlgorithms
|
|
||||||
if c.config.ClientCAs != nil {
|
|
||||||
certReq.certificateAuthorities = c.config.ClientCAs.Subjects()
|
|
||||||
}
|
|
||||||
|
|
||||||
hs.transcript.Write(certReq.marshal())
|
|
||||||
if _, err := c.writeRecord(recordTypeHandshake, certReq.marshal()); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
certMsg := new(certificateMsgTLS13)
|
|
||||||
|
|
||||||
certMsg.certificate = *hs.cert
|
|
||||||
certMsg.scts = hs.clientHello.scts && len(hs.cert.SignedCertificateTimestamps) > 0
|
|
||||||
certMsg.ocspStapling = hs.clientHello.ocspStapling && len(hs.cert.OCSPStaple) > 0
|
|
||||||
|
|
||||||
hs.transcript.Write(certMsg.marshal())
|
|
||||||
if _, err := c.writeRecord(recordTypeHandshake, certMsg.marshal()); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
certVerifyMsg := new(certificateVerifyMsg)
|
|
||||||
certVerifyMsg.hasSignatureAlgorithm = true
|
|
||||||
certVerifyMsg.signatureAlgorithm = hs.sigAlg
|
|
||||||
|
|
||||||
sigType, sigHash, err := typeAndHashFromSignatureScheme(hs.sigAlg)
|
|
||||||
if err != nil {
|
|
||||||
return c.sendAlert(alertInternalError)
|
|
||||||
}
|
|
||||||
|
|
||||||
signed := signedMessage(sigHash, serverSignatureContext, hs.transcript)
|
|
||||||
signOpts := crypto.SignerOpts(sigHash)
|
|
||||||
if sigType == signatureRSAPSS {
|
|
||||||
signOpts = &rsa.PSSOptions{SaltLength: rsa.PSSSaltLengthEqualsHash, Hash: sigHash}
|
|
||||||
}
|
|
||||||
sig, err := hs.cert.PrivateKey.(crypto.Signer).Sign(c.config.rand(), signed, signOpts)
|
|
||||||
if err != nil {
|
|
||||||
public := hs.cert.PrivateKey.(crypto.Signer).Public()
|
|
||||||
if rsaKey, ok := public.(*rsa.PublicKey); ok && sigType == signatureRSAPSS &&
|
|
||||||
rsaKey.N.BitLen()/8 < sigHash.Size()*2+2 { // key too small for RSA-PSS
|
|
||||||
c.sendAlert(alertHandshakeFailure)
|
|
||||||
} else {
|
|
||||||
c.sendAlert(alertInternalError)
|
|
||||||
}
|
|
||||||
return errors.New("tls: failed to sign handshake: " + err.Error())
|
|
||||||
}
|
|
||||||
certVerifyMsg.signature = sig
|
|
||||||
|
|
||||||
hs.transcript.Write(certVerifyMsg.marshal())
|
|
||||||
if _, err := c.writeRecord(recordTypeHandshake, certVerifyMsg.marshal()); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (hs *serverHandshakeStateTLS13) sendServerFinished() error {
|
|
||||||
c := hs.c
|
|
||||||
|
|
||||||
finished := &finishedMsg{
|
|
||||||
verifyData: hs.suite.finishedHash(c.out.trafficSecret, hs.transcript),
|
|
||||||
}
|
|
||||||
|
|
||||||
hs.transcript.Write(finished.marshal())
|
|
||||||
if _, err := c.writeRecord(recordTypeHandshake, finished.marshal()); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
// Derive secrets that take context through the server Finished.
|
|
||||||
|
|
||||||
hs.masterSecret = hs.suite.extract(nil,
|
|
||||||
hs.suite.deriveSecret(hs.handshakeSecret, "derived", nil))
|
|
||||||
|
|
||||||
hs.trafficSecret = hs.suite.deriveSecret(hs.masterSecret,
|
|
||||||
clientApplicationTrafficLabel, hs.transcript)
|
|
||||||
serverSecret := hs.suite.deriveSecret(hs.masterSecret,
|
|
||||||
serverApplicationTrafficLabel, hs.transcript)
|
|
||||||
c.out.exportKey(EncryptionApplication, hs.suite, serverSecret)
|
|
||||||
c.out.setTrafficSecret(hs.suite, serverSecret)
|
|
||||||
|
|
||||||
err := c.config.writeKeyLog(keyLogLabelClientTraffic, hs.clientHello.random, hs.trafficSecret)
|
|
||||||
if err != nil {
|
|
||||||
c.sendAlert(alertInternalError)
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
err = c.config.writeKeyLog(keyLogLabelServerTraffic, hs.clientHello.random, serverSecret)
|
|
||||||
if err != nil {
|
|
||||||
c.sendAlert(alertInternalError)
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
c.ekm = hs.suite.exportKeyingMaterial(hs.masterSecret, hs.transcript)
|
|
||||||
|
|
||||||
// If we did not request client certificates, at this point we can
|
|
||||||
// precompute the client finished and roll the transcript forward to send
|
|
||||||
// session tickets in our first flight.
|
|
||||||
if !hs.requestClientCert() {
|
|
||||||
if err := hs.sendSessionTickets(); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (hs *serverHandshakeStateTLS13) shouldSendSessionTickets() bool {
|
|
||||||
if hs.c.config.SessionTicketsDisabled {
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
|
|
||||||
// Don't send tickets the client wouldn't use. See RFC 8446, Section 4.2.9.
|
|
||||||
for _, pskMode := range hs.clientHello.pskModes {
|
|
||||||
if pskMode == pskModeDHE {
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
|
|
||||||
func (hs *serverHandshakeStateTLS13) sendSessionTickets() error {
|
|
||||||
c := hs.c
|
|
||||||
|
|
||||||
hs.clientFinished = hs.suite.finishedHash(c.in.trafficSecret, hs.transcript)
|
|
||||||
finishedMsg := &finishedMsg{
|
|
||||||
verifyData: hs.clientFinished,
|
|
||||||
}
|
|
||||||
hs.transcript.Write(finishedMsg.marshal())
|
|
||||||
|
|
||||||
if !hs.shouldSendSessionTickets() {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
c.resumptionSecret = hs.suite.deriveSecret(hs.masterSecret,
|
|
||||||
resumptionLabel, hs.transcript)
|
|
||||||
|
|
||||||
// Don't send session tickets when the alternative record layer is set.
|
|
||||||
// Instead, save the resumption secret on the Conn.
|
|
||||||
// Session tickets can then be generated by calling Conn.GetSessionTicket().
|
|
||||||
if hs.c.extraConfig != nil && hs.c.extraConfig.AlternativeRecordLayer != nil {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
m, err := hs.c.getSessionTicketMsg(nil)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
if _, err := c.writeRecord(recordTypeHandshake, m.marshal()); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (hs *serverHandshakeStateTLS13) readClientCertificate() error {
|
|
||||||
c := hs.c
|
|
||||||
|
|
||||||
if !hs.requestClientCert() {
|
|
||||||
// Make sure the connection is still being verified whether or not
|
|
||||||
// the server requested a client certificate.
|
|
||||||
if c.config.VerifyConnection != nil {
|
|
||||||
if err := c.config.VerifyConnection(c.connectionStateLocked()); err != nil {
|
|
||||||
c.sendAlert(alertBadCertificate)
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// If we requested a client certificate, then the client must send a
|
|
||||||
// certificate message. If it's empty, no CertificateVerify is sent.
|
|
||||||
|
|
||||||
msg, err := c.readHandshake()
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
certMsg, ok := msg.(*certificateMsgTLS13)
|
|
||||||
if !ok {
|
|
||||||
c.sendAlert(alertUnexpectedMessage)
|
|
||||||
return unexpectedMessageError(certMsg, msg)
|
|
||||||
}
|
|
||||||
hs.transcript.Write(certMsg.marshal())
|
|
||||||
|
|
||||||
if err := c.processCertsFromClient(certMsg.certificate); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
if c.config.VerifyConnection != nil {
|
|
||||||
if err := c.config.VerifyConnection(c.connectionStateLocked()); err != nil {
|
|
||||||
c.sendAlert(alertBadCertificate)
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if len(certMsg.certificate.Certificate) != 0 {
|
|
||||||
msg, err = c.readHandshake()
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
certVerify, ok := msg.(*certificateVerifyMsg)
|
|
||||||
if !ok {
|
|
||||||
c.sendAlert(alertUnexpectedMessage)
|
|
||||||
return unexpectedMessageError(certVerify, msg)
|
|
||||||
}
|
|
||||||
|
|
||||||
// See RFC 8446, Section 4.4.3.
|
|
||||||
if !isSupportedSignatureAlgorithm(certVerify.signatureAlgorithm, supportedSignatureAlgorithms) {
|
|
||||||
c.sendAlert(alertIllegalParameter)
|
|
||||||
return errors.New("tls: client certificate used with invalid signature algorithm")
|
|
||||||
}
|
|
||||||
sigType, sigHash, err := typeAndHashFromSignatureScheme(certVerify.signatureAlgorithm)
|
|
||||||
if err != nil {
|
|
||||||
return c.sendAlert(alertInternalError)
|
|
||||||
}
|
|
||||||
if sigType == signaturePKCS1v15 || sigHash == crypto.SHA1 {
|
|
||||||
c.sendAlert(alertIllegalParameter)
|
|
||||||
return errors.New("tls: client certificate used with invalid signature algorithm")
|
|
||||||
}
|
|
||||||
signed := signedMessage(sigHash, clientSignatureContext, hs.transcript)
|
|
||||||
if err := verifyHandshakeSignature(sigType, c.peerCertificates[0].PublicKey,
|
|
||||||
sigHash, signed, certVerify.signature); err != nil {
|
|
||||||
c.sendAlert(alertDecryptError)
|
|
||||||
return errors.New("tls: invalid signature by the client certificate: " + err.Error())
|
|
||||||
}
|
|
||||||
|
|
||||||
hs.transcript.Write(certVerify.marshal())
|
|
||||||
}
|
|
||||||
|
|
||||||
// If we waited until the client certificates to send session tickets, we
|
|
||||||
// are ready to do it now.
|
|
||||||
if err := hs.sendSessionTickets(); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (hs *serverHandshakeStateTLS13) readClientFinished() error {
|
|
||||||
c := hs.c
|
|
||||||
|
|
||||||
msg, err := c.readHandshake()
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
finished, ok := msg.(*finishedMsg)
|
|
||||||
if !ok {
|
|
||||||
c.sendAlert(alertUnexpectedMessage)
|
|
||||||
return unexpectedMessageError(finished, msg)
|
|
||||||
}
|
|
||||||
|
|
||||||
if !hmac.Equal(hs.clientFinished, finished.verifyData) {
|
|
||||||
c.sendAlert(alertDecryptError)
|
|
||||||
return errors.New("tls: invalid client finished hash")
|
|
||||||
}
|
|
||||||
|
|
||||||
c.in.exportKey(EncryptionApplication, hs.suite, hs.trafficSecret)
|
|
||||||
c.in.setTrafficSecret(hs.suite, hs.trafficSecret)
|
|
||||||
|
|
||||||
return nil
|
|
||||||
}
|
|
|
@ -1,353 +0,0 @@
|
||||||
// Copyright 2010 The Go Authors. All rights reserved.
|
|
||||||
// Use of this source code is governed by a BSD-style
|
|
||||||
// license that can be found in the LICENSE file.
|
|
||||||
|
|
||||||
package qtls
|
|
||||||
|
|
||||||
import (
|
|
||||||
"crypto"
|
|
||||||
"crypto/md5"
|
|
||||||
"crypto/rsa"
|
|
||||||
"crypto/sha1"
|
|
||||||
"crypto/x509"
|
|
||||||
"errors"
|
|
||||||
"fmt"
|
|
||||||
"io"
|
|
||||||
)
|
|
||||||
|
|
||||||
// a keyAgreement implements the client and server side of a TLS key agreement
|
|
||||||
// protocol by generating and processing key exchange messages.
|
|
||||||
type keyAgreement interface {
|
|
||||||
// On the server side, the first two methods are called in order.
|
|
||||||
|
|
||||||
// In the case that the key agreement protocol doesn't use a
|
|
||||||
// ServerKeyExchange message, generateServerKeyExchange can return nil,
|
|
||||||
// nil.
|
|
||||||
generateServerKeyExchange(*config, *Certificate, *clientHelloMsg, *serverHelloMsg) (*serverKeyExchangeMsg, error)
|
|
||||||
processClientKeyExchange(*config, *Certificate, *clientKeyExchangeMsg, uint16) ([]byte, error)
|
|
||||||
|
|
||||||
// On the client side, the next two methods are called in order.
|
|
||||||
|
|
||||||
// This method may not be called if the server doesn't send a
|
|
||||||
// ServerKeyExchange message.
|
|
||||||
processServerKeyExchange(*config, *clientHelloMsg, *serverHelloMsg, *x509.Certificate, *serverKeyExchangeMsg) error
|
|
||||||
generateClientKeyExchange(*config, *clientHelloMsg, *x509.Certificate) ([]byte, *clientKeyExchangeMsg, error)
|
|
||||||
}
|
|
||||||
|
|
||||||
var errClientKeyExchange = errors.New("tls: invalid ClientKeyExchange message")
|
|
||||||
var errServerKeyExchange = errors.New("tls: invalid ServerKeyExchange message")
|
|
||||||
|
|
||||||
// rsaKeyAgreement implements the standard TLS key agreement where the client
|
|
||||||
// encrypts the pre-master secret to the server's public key.
|
|
||||||
type rsaKeyAgreement struct{}
|
|
||||||
|
|
||||||
func (ka rsaKeyAgreement) generateServerKeyExchange(config *config, cert *Certificate, clientHello *clientHelloMsg, hello *serverHelloMsg) (*serverKeyExchangeMsg, error) {
|
|
||||||
return nil, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (ka rsaKeyAgreement) processClientKeyExchange(config *config, cert *Certificate, ckx *clientKeyExchangeMsg, version uint16) ([]byte, error) {
|
|
||||||
if len(ckx.ciphertext) < 2 {
|
|
||||||
return nil, errClientKeyExchange
|
|
||||||
}
|
|
||||||
ciphertextLen := int(ckx.ciphertext[0])<<8 | int(ckx.ciphertext[1])
|
|
||||||
if ciphertextLen != len(ckx.ciphertext)-2 {
|
|
||||||
return nil, errClientKeyExchange
|
|
||||||
}
|
|
||||||
ciphertext := ckx.ciphertext[2:]
|
|
||||||
|
|
||||||
priv, ok := cert.PrivateKey.(crypto.Decrypter)
|
|
||||||
if !ok {
|
|
||||||
return nil, errors.New("tls: certificate private key does not implement crypto.Decrypter")
|
|
||||||
}
|
|
||||||
// Perform constant time RSA PKCS #1 v1.5 decryption
|
|
||||||
preMasterSecret, err := priv.Decrypt(config.rand(), ciphertext, &rsa.PKCS1v15DecryptOptions{SessionKeyLen: 48})
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
// We don't check the version number in the premaster secret. For one,
|
|
||||||
// by checking it, we would leak information about the validity of the
|
|
||||||
// encrypted pre-master secret. Secondly, it provides only a small
|
|
||||||
// benefit against a downgrade attack and some implementations send the
|
|
||||||
// wrong version anyway. See the discussion at the end of section
|
|
||||||
// 7.4.7.1 of RFC 4346.
|
|
||||||
return preMasterSecret, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (ka rsaKeyAgreement) processServerKeyExchange(config *config, clientHello *clientHelloMsg, serverHello *serverHelloMsg, cert *x509.Certificate, skx *serverKeyExchangeMsg) error {
|
|
||||||
return errors.New("tls: unexpected ServerKeyExchange")
|
|
||||||
}
|
|
||||||
|
|
||||||
func (ka rsaKeyAgreement) generateClientKeyExchange(config *config, clientHello *clientHelloMsg, cert *x509.Certificate) ([]byte, *clientKeyExchangeMsg, error) {
|
|
||||||
preMasterSecret := make([]byte, 48)
|
|
||||||
preMasterSecret[0] = byte(clientHello.vers >> 8)
|
|
||||||
preMasterSecret[1] = byte(clientHello.vers)
|
|
||||||
_, err := io.ReadFull(config.rand(), preMasterSecret[2:])
|
|
||||||
if err != nil {
|
|
||||||
return nil, nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
encrypted, err := rsa.EncryptPKCS1v15(config.rand(), cert.PublicKey.(*rsa.PublicKey), preMasterSecret)
|
|
||||||
if err != nil {
|
|
||||||
return nil, nil, err
|
|
||||||
}
|
|
||||||
ckx := new(clientKeyExchangeMsg)
|
|
||||||
ckx.ciphertext = make([]byte, len(encrypted)+2)
|
|
||||||
ckx.ciphertext[0] = byte(len(encrypted) >> 8)
|
|
||||||
ckx.ciphertext[1] = byte(len(encrypted))
|
|
||||||
copy(ckx.ciphertext[2:], encrypted)
|
|
||||||
return preMasterSecret, ckx, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// sha1Hash calculates a SHA1 hash over the given byte slices.
|
|
||||||
func sha1Hash(slices [][]byte) []byte {
|
|
||||||
hsha1 := sha1.New()
|
|
||||||
for _, slice := range slices {
|
|
||||||
hsha1.Write(slice)
|
|
||||||
}
|
|
||||||
return hsha1.Sum(nil)
|
|
||||||
}
|
|
||||||
|
|
||||||
// md5SHA1Hash implements TLS 1.0's hybrid hash function which consists of the
|
|
||||||
// concatenation of an MD5 and SHA1 hash.
|
|
||||||
func md5SHA1Hash(slices [][]byte) []byte {
|
|
||||||
md5sha1 := make([]byte, md5.Size+sha1.Size)
|
|
||||||
hmd5 := md5.New()
|
|
||||||
for _, slice := range slices {
|
|
||||||
hmd5.Write(slice)
|
|
||||||
}
|
|
||||||
copy(md5sha1, hmd5.Sum(nil))
|
|
||||||
copy(md5sha1[md5.Size:], sha1Hash(slices))
|
|
||||||
return md5sha1
|
|
||||||
}
|
|
||||||
|
|
||||||
// hashForServerKeyExchange hashes the given slices and returns their digest
|
|
||||||
// using the given hash function (for >= TLS 1.2) or using a default based on
|
|
||||||
// the sigType (for earlier TLS versions). For Ed25519 signatures, which don't
|
|
||||||
// do pre-hashing, it returns the concatenation of the slices.
|
|
||||||
func hashForServerKeyExchange(sigType uint8, hashFunc crypto.Hash, version uint16, slices ...[]byte) []byte {
|
|
||||||
if sigType == signatureEd25519 {
|
|
||||||
var signed []byte
|
|
||||||
for _, slice := range slices {
|
|
||||||
signed = append(signed, slice...)
|
|
||||||
}
|
|
||||||
return signed
|
|
||||||
}
|
|
||||||
if version >= VersionTLS12 {
|
|
||||||
h := hashFunc.New()
|
|
||||||
for _, slice := range slices {
|
|
||||||
h.Write(slice)
|
|
||||||
}
|
|
||||||
digest := h.Sum(nil)
|
|
||||||
return digest
|
|
||||||
}
|
|
||||||
if sigType == signatureECDSA {
|
|
||||||
return sha1Hash(slices)
|
|
||||||
}
|
|
||||||
return md5SHA1Hash(slices)
|
|
||||||
}
|
|
||||||
|
|
||||||
// ecdheKeyAgreement implements a TLS key agreement where the server
|
|
||||||
// generates an ephemeral EC public/private key pair and signs it. The
|
|
||||||
// pre-master secret is then calculated using ECDH. The signature may
|
|
||||||
// be ECDSA, Ed25519 or RSA.
|
|
||||||
type ecdheKeyAgreement struct {
|
|
||||||
version uint16
|
|
||||||
isRSA bool
|
|
||||||
params ecdheParameters
|
|
||||||
|
|
||||||
// ckx and preMasterSecret are generated in processServerKeyExchange
|
|
||||||
// and returned in generateClientKeyExchange.
|
|
||||||
ckx *clientKeyExchangeMsg
|
|
||||||
preMasterSecret []byte
|
|
||||||
}
|
|
||||||
|
|
||||||
func (ka *ecdheKeyAgreement) generateServerKeyExchange(config *config, cert *Certificate, clientHello *clientHelloMsg, hello *serverHelloMsg) (*serverKeyExchangeMsg, error) {
|
|
||||||
var curveID CurveID
|
|
||||||
for _, c := range clientHello.supportedCurves {
|
|
||||||
if config.supportsCurve(c) {
|
|
||||||
curveID = c
|
|
||||||
break
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if curveID == 0 {
|
|
||||||
return nil, errors.New("tls: no supported elliptic curves offered")
|
|
||||||
}
|
|
||||||
if _, ok := curveForCurveID(curveID); curveID != X25519 && !ok {
|
|
||||||
return nil, errors.New("tls: CurvePreferences includes unsupported curve")
|
|
||||||
}
|
|
||||||
|
|
||||||
params, err := generateECDHEParameters(config.rand(), curveID)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
ka.params = params
|
|
||||||
|
|
||||||
// See RFC 4492, Section 5.4.
|
|
||||||
ecdhePublic := params.PublicKey()
|
|
||||||
serverECDHEParams := make([]byte, 1+2+1+len(ecdhePublic))
|
|
||||||
serverECDHEParams[0] = 3 // named curve
|
|
||||||
serverECDHEParams[1] = byte(curveID >> 8)
|
|
||||||
serverECDHEParams[2] = byte(curveID)
|
|
||||||
serverECDHEParams[3] = byte(len(ecdhePublic))
|
|
||||||
copy(serverECDHEParams[4:], ecdhePublic)
|
|
||||||
|
|
||||||
priv, ok := cert.PrivateKey.(crypto.Signer)
|
|
||||||
if !ok {
|
|
||||||
return nil, fmt.Errorf("tls: certificate private key of type %T does not implement crypto.Signer", cert.PrivateKey)
|
|
||||||
}
|
|
||||||
|
|
||||||
var signatureAlgorithm SignatureScheme
|
|
||||||
var sigType uint8
|
|
||||||
var sigHash crypto.Hash
|
|
||||||
if ka.version >= VersionTLS12 {
|
|
||||||
signatureAlgorithm, err = selectSignatureScheme(ka.version, cert, clientHello.supportedSignatureAlgorithms)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
sigType, sigHash, err = typeAndHashFromSignatureScheme(signatureAlgorithm)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
sigType, sigHash, err = legacyTypeAndHashFromPublicKey(priv.Public())
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (sigType == signaturePKCS1v15 || sigType == signatureRSAPSS) != ka.isRSA {
|
|
||||||
return nil, errors.New("tls: certificate cannot be used with the selected cipher suite")
|
|
||||||
}
|
|
||||||
|
|
||||||
signed := hashForServerKeyExchange(sigType, sigHash, ka.version, clientHello.random, hello.random, serverECDHEParams)
|
|
||||||
|
|
||||||
signOpts := crypto.SignerOpts(sigHash)
|
|
||||||
if sigType == signatureRSAPSS {
|
|
||||||
signOpts = &rsa.PSSOptions{SaltLength: rsa.PSSSaltLengthEqualsHash, Hash: sigHash}
|
|
||||||
}
|
|
||||||
sig, err := priv.Sign(config.rand(), signed, signOpts)
|
|
||||||
if err != nil {
|
|
||||||
return nil, errors.New("tls: failed to sign ECDHE parameters: " + err.Error())
|
|
||||||
}
|
|
||||||
|
|
||||||
skx := new(serverKeyExchangeMsg)
|
|
||||||
sigAndHashLen := 0
|
|
||||||
if ka.version >= VersionTLS12 {
|
|
||||||
sigAndHashLen = 2
|
|
||||||
}
|
|
||||||
skx.key = make([]byte, len(serverECDHEParams)+sigAndHashLen+2+len(sig))
|
|
||||||
copy(skx.key, serverECDHEParams)
|
|
||||||
k := skx.key[len(serverECDHEParams):]
|
|
||||||
if ka.version >= VersionTLS12 {
|
|
||||||
k[0] = byte(signatureAlgorithm >> 8)
|
|
||||||
k[1] = byte(signatureAlgorithm)
|
|
||||||
k = k[2:]
|
|
||||||
}
|
|
||||||
k[0] = byte(len(sig) >> 8)
|
|
||||||
k[1] = byte(len(sig))
|
|
||||||
copy(k[2:], sig)
|
|
||||||
|
|
||||||
return skx, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (ka *ecdheKeyAgreement) processClientKeyExchange(config *config, cert *Certificate, ckx *clientKeyExchangeMsg, version uint16) ([]byte, error) {
|
|
||||||
if len(ckx.ciphertext) == 0 || int(ckx.ciphertext[0]) != len(ckx.ciphertext)-1 {
|
|
||||||
return nil, errClientKeyExchange
|
|
||||||
}
|
|
||||||
|
|
||||||
preMasterSecret := ka.params.SharedKey(ckx.ciphertext[1:])
|
|
||||||
if preMasterSecret == nil {
|
|
||||||
return nil, errClientKeyExchange
|
|
||||||
}
|
|
||||||
|
|
||||||
return preMasterSecret, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (ka *ecdheKeyAgreement) processServerKeyExchange(config *config, clientHello *clientHelloMsg, serverHello *serverHelloMsg, cert *x509.Certificate, skx *serverKeyExchangeMsg) error {
|
|
||||||
if len(skx.key) < 4 {
|
|
||||||
return errServerKeyExchange
|
|
||||||
}
|
|
||||||
if skx.key[0] != 3 { // named curve
|
|
||||||
return errors.New("tls: server selected unsupported curve")
|
|
||||||
}
|
|
||||||
curveID := CurveID(skx.key[1])<<8 | CurveID(skx.key[2])
|
|
||||||
|
|
||||||
publicLen := int(skx.key[3])
|
|
||||||
if publicLen+4 > len(skx.key) {
|
|
||||||
return errServerKeyExchange
|
|
||||||
}
|
|
||||||
serverECDHEParams := skx.key[:4+publicLen]
|
|
||||||
publicKey := serverECDHEParams[4:]
|
|
||||||
|
|
||||||
sig := skx.key[4+publicLen:]
|
|
||||||
if len(sig) < 2 {
|
|
||||||
return errServerKeyExchange
|
|
||||||
}
|
|
||||||
|
|
||||||
if _, ok := curveForCurveID(curveID); curveID != X25519 && !ok {
|
|
||||||
return errors.New("tls: server selected unsupported curve")
|
|
||||||
}
|
|
||||||
|
|
||||||
params, err := generateECDHEParameters(config.rand(), curveID)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
ka.params = params
|
|
||||||
|
|
||||||
ka.preMasterSecret = params.SharedKey(publicKey)
|
|
||||||
if ka.preMasterSecret == nil {
|
|
||||||
return errServerKeyExchange
|
|
||||||
}
|
|
||||||
|
|
||||||
ourPublicKey := params.PublicKey()
|
|
||||||
ka.ckx = new(clientKeyExchangeMsg)
|
|
||||||
ka.ckx.ciphertext = make([]byte, 1+len(ourPublicKey))
|
|
||||||
ka.ckx.ciphertext[0] = byte(len(ourPublicKey))
|
|
||||||
copy(ka.ckx.ciphertext[1:], ourPublicKey)
|
|
||||||
|
|
||||||
var sigType uint8
|
|
||||||
var sigHash crypto.Hash
|
|
||||||
if ka.version >= VersionTLS12 {
|
|
||||||
signatureAlgorithm := SignatureScheme(sig[0])<<8 | SignatureScheme(sig[1])
|
|
||||||
sig = sig[2:]
|
|
||||||
if len(sig) < 2 {
|
|
||||||
return errServerKeyExchange
|
|
||||||
}
|
|
||||||
|
|
||||||
if !isSupportedSignatureAlgorithm(signatureAlgorithm, clientHello.supportedSignatureAlgorithms) {
|
|
||||||
return errors.New("tls: certificate used with invalid signature algorithm")
|
|
||||||
}
|
|
||||||
sigType, sigHash, err = typeAndHashFromSignatureScheme(signatureAlgorithm)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
sigType, sigHash, err = legacyTypeAndHashFromPublicKey(cert.PublicKey)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (sigType == signaturePKCS1v15 || sigType == signatureRSAPSS) != ka.isRSA {
|
|
||||||
return errServerKeyExchange
|
|
||||||
}
|
|
||||||
|
|
||||||
sigLen := int(sig[0])<<8 | int(sig[1])
|
|
||||||
if sigLen+2 != len(sig) {
|
|
||||||
return errServerKeyExchange
|
|
||||||
}
|
|
||||||
sig = sig[2:]
|
|
||||||
|
|
||||||
signed := hashForServerKeyExchange(sigType, sigHash, ka.version, clientHello.random, serverHello.random, serverECDHEParams)
|
|
||||||
if err := verifyHandshakeSignature(sigType, cert.PublicKey, sigHash, signed, sig); err != nil {
|
|
||||||
return errors.New("tls: invalid signature by the server certificate: " + err.Error())
|
|
||||||
}
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (ka *ecdheKeyAgreement) generateClientKeyExchange(config *config, clientHello *clientHelloMsg, cert *x509.Certificate) ([]byte, *clientKeyExchangeMsg, error) {
|
|
||||||
if ka.ckx == nil {
|
|
||||||
return nil, nil, errors.New("tls: missing ServerKeyExchange message")
|
|
||||||
}
|
|
||||||
|
|
||||||
return ka.preMasterSecret, ka.ckx, nil
|
|
||||||
}
|
|
|
@ -1,199 +0,0 @@
|
||||||
// Copyright 2018 The Go Authors. All rights reserved.
|
|
||||||
// Use of this source code is governed by a BSD-style
|
|
||||||
// license that can be found in the LICENSE file.
|
|
||||||
|
|
||||||
package qtls
|
|
||||||
|
|
||||||
import (
|
|
||||||
"crypto/elliptic"
|
|
||||||
"crypto/hmac"
|
|
||||||
"errors"
|
|
||||||
"hash"
|
|
||||||
"io"
|
|
||||||
"math/big"
|
|
||||||
|
|
||||||
"golang.org/x/crypto/cryptobyte"
|
|
||||||
"golang.org/x/crypto/curve25519"
|
|
||||||
"golang.org/x/crypto/hkdf"
|
|
||||||
)
|
|
||||||
|
|
||||||
// This file contains the functions necessary to compute the TLS 1.3 key
|
|
||||||
// schedule. See RFC 8446, Section 7.
|
|
||||||
|
|
||||||
const (
|
|
||||||
resumptionBinderLabel = "res binder"
|
|
||||||
clientHandshakeTrafficLabel = "c hs traffic"
|
|
||||||
serverHandshakeTrafficLabel = "s hs traffic"
|
|
||||||
clientApplicationTrafficLabel = "c ap traffic"
|
|
||||||
serverApplicationTrafficLabel = "s ap traffic"
|
|
||||||
exporterLabel = "exp master"
|
|
||||||
resumptionLabel = "res master"
|
|
||||||
trafficUpdateLabel = "traffic upd"
|
|
||||||
)
|
|
||||||
|
|
||||||
// expandLabel implements HKDF-Expand-Label from RFC 8446, Section 7.1.
|
|
||||||
func (c *cipherSuiteTLS13) expandLabel(secret []byte, label string, context []byte, length int) []byte {
|
|
||||||
var hkdfLabel cryptobyte.Builder
|
|
||||||
hkdfLabel.AddUint16(uint16(length))
|
|
||||||
hkdfLabel.AddUint8LengthPrefixed(func(b *cryptobyte.Builder) {
|
|
||||||
b.AddBytes([]byte("tls13 "))
|
|
||||||
b.AddBytes([]byte(label))
|
|
||||||
})
|
|
||||||
hkdfLabel.AddUint8LengthPrefixed(func(b *cryptobyte.Builder) {
|
|
||||||
b.AddBytes(context)
|
|
||||||
})
|
|
||||||
out := make([]byte, length)
|
|
||||||
n, err := hkdf.Expand(c.hash.New, secret, hkdfLabel.BytesOrPanic()).Read(out)
|
|
||||||
if err != nil || n != length {
|
|
||||||
panic("tls: HKDF-Expand-Label invocation failed unexpectedly")
|
|
||||||
}
|
|
||||||
return out
|
|
||||||
}
|
|
||||||
|
|
||||||
// deriveSecret implements Derive-Secret from RFC 8446, Section 7.1.
|
|
||||||
func (c *cipherSuiteTLS13) deriveSecret(secret []byte, label string, transcript hash.Hash) []byte {
|
|
||||||
if transcript == nil {
|
|
||||||
transcript = c.hash.New()
|
|
||||||
}
|
|
||||||
return c.expandLabel(secret, label, transcript.Sum(nil), c.hash.Size())
|
|
||||||
}
|
|
||||||
|
|
||||||
// extract implements HKDF-Extract with the cipher suite hash.
|
|
||||||
func (c *cipherSuiteTLS13) extract(newSecret, currentSecret []byte) []byte {
|
|
||||||
if newSecret == nil {
|
|
||||||
newSecret = make([]byte, c.hash.Size())
|
|
||||||
}
|
|
||||||
return hkdf.Extract(c.hash.New, newSecret, currentSecret)
|
|
||||||
}
|
|
||||||
|
|
||||||
// nextTrafficSecret generates the next traffic secret, given the current one,
|
|
||||||
// according to RFC 8446, Section 7.2.
|
|
||||||
func (c *cipherSuiteTLS13) nextTrafficSecret(trafficSecret []byte) []byte {
|
|
||||||
return c.expandLabel(trafficSecret, trafficUpdateLabel, nil, c.hash.Size())
|
|
||||||
}
|
|
||||||
|
|
||||||
// trafficKey generates traffic keys according to RFC 8446, Section 7.3.
|
|
||||||
func (c *cipherSuiteTLS13) trafficKey(trafficSecret []byte) (key, iv []byte) {
|
|
||||||
key = c.expandLabel(trafficSecret, "key", nil, c.keyLen)
|
|
||||||
iv = c.expandLabel(trafficSecret, "iv", nil, aeadNonceLength)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
// finishedHash generates the Finished verify_data or PskBinderEntry according
|
|
||||||
// to RFC 8446, Section 4.4.4. See sections 4.4 and 4.2.11.2 for the baseKey
|
|
||||||
// selection.
|
|
||||||
func (c *cipherSuiteTLS13) finishedHash(baseKey []byte, transcript hash.Hash) []byte {
|
|
||||||
finishedKey := c.expandLabel(baseKey, "finished", nil, c.hash.Size())
|
|
||||||
verifyData := hmac.New(c.hash.New, finishedKey)
|
|
||||||
verifyData.Write(transcript.Sum(nil))
|
|
||||||
return verifyData.Sum(nil)
|
|
||||||
}
|
|
||||||
|
|
||||||
// exportKeyingMaterial implements RFC5705 exporters for TLS 1.3 according to
|
|
||||||
// RFC 8446, Section 7.5.
|
|
||||||
func (c *cipherSuiteTLS13) exportKeyingMaterial(masterSecret []byte, transcript hash.Hash) func(string, []byte, int) ([]byte, error) {
|
|
||||||
expMasterSecret := c.deriveSecret(masterSecret, exporterLabel, transcript)
|
|
||||||
return func(label string, context []byte, length int) ([]byte, error) {
|
|
||||||
secret := c.deriveSecret(expMasterSecret, label, nil)
|
|
||||||
h := c.hash.New()
|
|
||||||
h.Write(context)
|
|
||||||
return c.expandLabel(secret, "exporter", h.Sum(nil), length), nil
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// ecdheParameters implements Diffie-Hellman with either NIST curves or X25519,
|
|
||||||
// according to RFC 8446, Section 4.2.8.2.
|
|
||||||
type ecdheParameters interface {
|
|
||||||
CurveID() CurveID
|
|
||||||
PublicKey() []byte
|
|
||||||
SharedKey(peerPublicKey []byte) []byte
|
|
||||||
}
|
|
||||||
|
|
||||||
func generateECDHEParameters(rand io.Reader, curveID CurveID) (ecdheParameters, error) {
|
|
||||||
if curveID == X25519 {
|
|
||||||
privateKey := make([]byte, curve25519.ScalarSize)
|
|
||||||
if _, err := io.ReadFull(rand, privateKey); err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
publicKey, err := curve25519.X25519(privateKey, curve25519.Basepoint)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
return &x25519Parameters{privateKey: privateKey, publicKey: publicKey}, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
curve, ok := curveForCurveID(curveID)
|
|
||||||
if !ok {
|
|
||||||
return nil, errors.New("tls: internal error: unsupported curve")
|
|
||||||
}
|
|
||||||
|
|
||||||
p := &nistParameters{curveID: curveID}
|
|
||||||
var err error
|
|
||||||
p.privateKey, p.x, p.y, err = elliptic.GenerateKey(curve, rand)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
return p, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func curveForCurveID(id CurveID) (elliptic.Curve, bool) {
|
|
||||||
switch id {
|
|
||||||
case CurveP256:
|
|
||||||
return elliptic.P256(), true
|
|
||||||
case CurveP384:
|
|
||||||
return elliptic.P384(), true
|
|
||||||
case CurveP521:
|
|
||||||
return elliptic.P521(), true
|
|
||||||
default:
|
|
||||||
return nil, false
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
type nistParameters struct {
|
|
||||||
privateKey []byte
|
|
||||||
x, y *big.Int // public key
|
|
||||||
curveID CurveID
|
|
||||||
}
|
|
||||||
|
|
||||||
func (p *nistParameters) CurveID() CurveID {
|
|
||||||
return p.curveID
|
|
||||||
}
|
|
||||||
|
|
||||||
func (p *nistParameters) PublicKey() []byte {
|
|
||||||
curve, _ := curveForCurveID(p.curveID)
|
|
||||||
return elliptic.Marshal(curve, p.x, p.y)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (p *nistParameters) SharedKey(peerPublicKey []byte) []byte {
|
|
||||||
curve, _ := curveForCurveID(p.curveID)
|
|
||||||
// Unmarshal also checks whether the given point is on the curve.
|
|
||||||
x, y := elliptic.Unmarshal(curve, peerPublicKey)
|
|
||||||
if x == nil {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
xShared, _ := curve.ScalarMult(x, y, p.privateKey)
|
|
||||||
sharedKey := make([]byte, (curve.Params().BitSize+7)/8)
|
|
||||||
return xShared.FillBytes(sharedKey)
|
|
||||||
}
|
|
||||||
|
|
||||||
type x25519Parameters struct {
|
|
||||||
privateKey []byte
|
|
||||||
publicKey []byte
|
|
||||||
}
|
|
||||||
|
|
||||||
func (p *x25519Parameters) CurveID() CurveID {
|
|
||||||
return X25519
|
|
||||||
}
|
|
||||||
|
|
||||||
func (p *x25519Parameters) PublicKey() []byte {
|
|
||||||
return p.publicKey[:]
|
|
||||||
}
|
|
||||||
|
|
||||||
func (p *x25519Parameters) SharedKey(peerPublicKey []byte) []byte {
|
|
||||||
sharedKey, err := curve25519.X25519(p.privateKey, peerPublicKey)
|
|
||||||
if err != nil {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
return sharedKey
|
|
||||||
}
|
|
|
@ -1,283 +0,0 @@
|
||||||
// Copyright 2009 The Go Authors. All rights reserved.
|
|
||||||
// Use of this source code is governed by a BSD-style
|
|
||||||
// license that can be found in the LICENSE file.
|
|
||||||
|
|
||||||
package qtls
|
|
||||||
|
|
||||||
import (
|
|
||||||
"crypto"
|
|
||||||
"crypto/hmac"
|
|
||||||
"crypto/md5"
|
|
||||||
"crypto/sha1"
|
|
||||||
"crypto/sha256"
|
|
||||||
"crypto/sha512"
|
|
||||||
"errors"
|
|
||||||
"fmt"
|
|
||||||
"hash"
|
|
||||||
)
|
|
||||||
|
|
||||||
// Split a premaster secret in two as specified in RFC 4346, Section 5.
|
|
||||||
func splitPreMasterSecret(secret []byte) (s1, s2 []byte) {
|
|
||||||
s1 = secret[0 : (len(secret)+1)/2]
|
|
||||||
s2 = secret[len(secret)/2:]
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
// pHash implements the P_hash function, as defined in RFC 4346, Section 5.
|
|
||||||
func pHash(result, secret, seed []byte, hash func() hash.Hash) {
|
|
||||||
h := hmac.New(hash, secret)
|
|
||||||
h.Write(seed)
|
|
||||||
a := h.Sum(nil)
|
|
||||||
|
|
||||||
j := 0
|
|
||||||
for j < len(result) {
|
|
||||||
h.Reset()
|
|
||||||
h.Write(a)
|
|
||||||
h.Write(seed)
|
|
||||||
b := h.Sum(nil)
|
|
||||||
copy(result[j:], b)
|
|
||||||
j += len(b)
|
|
||||||
|
|
||||||
h.Reset()
|
|
||||||
h.Write(a)
|
|
||||||
a = h.Sum(nil)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// prf10 implements the TLS 1.0 pseudo-random function, as defined in RFC 2246, Section 5.
|
|
||||||
func prf10(result, secret, label, seed []byte) {
|
|
||||||
hashSHA1 := sha1.New
|
|
||||||
hashMD5 := md5.New
|
|
||||||
|
|
||||||
labelAndSeed := make([]byte, len(label)+len(seed))
|
|
||||||
copy(labelAndSeed, label)
|
|
||||||
copy(labelAndSeed[len(label):], seed)
|
|
||||||
|
|
||||||
s1, s2 := splitPreMasterSecret(secret)
|
|
||||||
pHash(result, s1, labelAndSeed, hashMD5)
|
|
||||||
result2 := make([]byte, len(result))
|
|
||||||
pHash(result2, s2, labelAndSeed, hashSHA1)
|
|
||||||
|
|
||||||
for i, b := range result2 {
|
|
||||||
result[i] ^= b
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// prf12 implements the TLS 1.2 pseudo-random function, as defined in RFC 5246, Section 5.
|
|
||||||
func prf12(hashFunc func() hash.Hash) func(result, secret, label, seed []byte) {
|
|
||||||
return func(result, secret, label, seed []byte) {
|
|
||||||
labelAndSeed := make([]byte, len(label)+len(seed))
|
|
||||||
copy(labelAndSeed, label)
|
|
||||||
copy(labelAndSeed[len(label):], seed)
|
|
||||||
|
|
||||||
pHash(result, secret, labelAndSeed, hashFunc)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
const (
|
|
||||||
masterSecretLength = 48 // Length of a master secret in TLS 1.1.
|
|
||||||
finishedVerifyLength = 12 // Length of verify_data in a Finished message.
|
|
||||||
)
|
|
||||||
|
|
||||||
var masterSecretLabel = []byte("master secret")
|
|
||||||
var keyExpansionLabel = []byte("key expansion")
|
|
||||||
var clientFinishedLabel = []byte("client finished")
|
|
||||||
var serverFinishedLabel = []byte("server finished")
|
|
||||||
|
|
||||||
func prfAndHashForVersion(version uint16, suite *cipherSuite) (func(result, secret, label, seed []byte), crypto.Hash) {
|
|
||||||
switch version {
|
|
||||||
case VersionTLS10, VersionTLS11:
|
|
||||||
return prf10, crypto.Hash(0)
|
|
||||||
case VersionTLS12:
|
|
||||||
if suite.flags&suiteSHA384 != 0 {
|
|
||||||
return prf12(sha512.New384), crypto.SHA384
|
|
||||||
}
|
|
||||||
return prf12(sha256.New), crypto.SHA256
|
|
||||||
default:
|
|
||||||
panic("unknown version")
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func prfForVersion(version uint16, suite *cipherSuite) func(result, secret, label, seed []byte) {
|
|
||||||
prf, _ := prfAndHashForVersion(version, suite)
|
|
||||||
return prf
|
|
||||||
}
|
|
||||||
|
|
||||||
// masterFromPreMasterSecret generates the master secret from the pre-master
|
|
||||||
// secret. See RFC 5246, Section 8.1.
|
|
||||||
func masterFromPreMasterSecret(version uint16, suite *cipherSuite, preMasterSecret, clientRandom, serverRandom []byte) []byte {
|
|
||||||
seed := make([]byte, 0, len(clientRandom)+len(serverRandom))
|
|
||||||
seed = append(seed, clientRandom...)
|
|
||||||
seed = append(seed, serverRandom...)
|
|
||||||
|
|
||||||
masterSecret := make([]byte, masterSecretLength)
|
|
||||||
prfForVersion(version, suite)(masterSecret, preMasterSecret, masterSecretLabel, seed)
|
|
||||||
return masterSecret
|
|
||||||
}
|
|
||||||
|
|
||||||
// keysFromMasterSecret generates the connection keys from the master
|
|
||||||
// secret, given the lengths of the MAC key, cipher key and IV, as defined in
|
|
||||||
// RFC 2246, Section 6.3.
|
|
||||||
func keysFromMasterSecret(version uint16, suite *cipherSuite, masterSecret, clientRandom, serverRandom []byte, macLen, keyLen, ivLen int) (clientMAC, serverMAC, clientKey, serverKey, clientIV, serverIV []byte) {
|
|
||||||
seed := make([]byte, 0, len(serverRandom)+len(clientRandom))
|
|
||||||
seed = append(seed, serverRandom...)
|
|
||||||
seed = append(seed, clientRandom...)
|
|
||||||
|
|
||||||
n := 2*macLen + 2*keyLen + 2*ivLen
|
|
||||||
keyMaterial := make([]byte, n)
|
|
||||||
prfForVersion(version, suite)(keyMaterial, masterSecret, keyExpansionLabel, seed)
|
|
||||||
clientMAC = keyMaterial[:macLen]
|
|
||||||
keyMaterial = keyMaterial[macLen:]
|
|
||||||
serverMAC = keyMaterial[:macLen]
|
|
||||||
keyMaterial = keyMaterial[macLen:]
|
|
||||||
clientKey = keyMaterial[:keyLen]
|
|
||||||
keyMaterial = keyMaterial[keyLen:]
|
|
||||||
serverKey = keyMaterial[:keyLen]
|
|
||||||
keyMaterial = keyMaterial[keyLen:]
|
|
||||||
clientIV = keyMaterial[:ivLen]
|
|
||||||
keyMaterial = keyMaterial[ivLen:]
|
|
||||||
serverIV = keyMaterial[:ivLen]
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
func newFinishedHash(version uint16, cipherSuite *cipherSuite) finishedHash {
|
|
||||||
var buffer []byte
|
|
||||||
if version >= VersionTLS12 {
|
|
||||||
buffer = []byte{}
|
|
||||||
}
|
|
||||||
|
|
||||||
prf, hash := prfAndHashForVersion(version, cipherSuite)
|
|
||||||
if hash != 0 {
|
|
||||||
return finishedHash{hash.New(), hash.New(), nil, nil, buffer, version, prf}
|
|
||||||
}
|
|
||||||
|
|
||||||
return finishedHash{sha1.New(), sha1.New(), md5.New(), md5.New(), buffer, version, prf}
|
|
||||||
}
|
|
||||||
|
|
||||||
// A finishedHash calculates the hash of a set of handshake messages suitable
|
|
||||||
// for including in a Finished message.
|
|
||||||
type finishedHash struct {
|
|
||||||
client hash.Hash
|
|
||||||
server hash.Hash
|
|
||||||
|
|
||||||
// Prior to TLS 1.2, an additional MD5 hash is required.
|
|
||||||
clientMD5 hash.Hash
|
|
||||||
serverMD5 hash.Hash
|
|
||||||
|
|
||||||
// In TLS 1.2, a full buffer is sadly required.
|
|
||||||
buffer []byte
|
|
||||||
|
|
||||||
version uint16
|
|
||||||
prf func(result, secret, label, seed []byte)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (h *finishedHash) Write(msg []byte) (n int, err error) {
|
|
||||||
h.client.Write(msg)
|
|
||||||
h.server.Write(msg)
|
|
||||||
|
|
||||||
if h.version < VersionTLS12 {
|
|
||||||
h.clientMD5.Write(msg)
|
|
||||||
h.serverMD5.Write(msg)
|
|
||||||
}
|
|
||||||
|
|
||||||
if h.buffer != nil {
|
|
||||||
h.buffer = append(h.buffer, msg...)
|
|
||||||
}
|
|
||||||
|
|
||||||
return len(msg), nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (h finishedHash) Sum() []byte {
|
|
||||||
if h.version >= VersionTLS12 {
|
|
||||||
return h.client.Sum(nil)
|
|
||||||
}
|
|
||||||
|
|
||||||
out := make([]byte, 0, md5.Size+sha1.Size)
|
|
||||||
out = h.clientMD5.Sum(out)
|
|
||||||
return h.client.Sum(out)
|
|
||||||
}
|
|
||||||
|
|
||||||
// clientSum returns the contents of the verify_data member of a client's
|
|
||||||
// Finished message.
|
|
||||||
func (h finishedHash) clientSum(masterSecret []byte) []byte {
|
|
||||||
out := make([]byte, finishedVerifyLength)
|
|
||||||
h.prf(out, masterSecret, clientFinishedLabel, h.Sum())
|
|
||||||
return out
|
|
||||||
}
|
|
||||||
|
|
||||||
// serverSum returns the contents of the verify_data member of a server's
|
|
||||||
// Finished message.
|
|
||||||
func (h finishedHash) serverSum(masterSecret []byte) []byte {
|
|
||||||
out := make([]byte, finishedVerifyLength)
|
|
||||||
h.prf(out, masterSecret, serverFinishedLabel, h.Sum())
|
|
||||||
return out
|
|
||||||
}
|
|
||||||
|
|
||||||
// hashForClientCertificate returns the handshake messages so far, pre-hashed if
|
|
||||||
// necessary, suitable for signing by a TLS client certificate.
|
|
||||||
func (h finishedHash) hashForClientCertificate(sigType uint8, hashAlg crypto.Hash, masterSecret []byte) []byte {
|
|
||||||
if (h.version >= VersionTLS12 || sigType == signatureEd25519) && h.buffer == nil {
|
|
||||||
panic("tls: handshake hash for a client certificate requested after discarding the handshake buffer")
|
|
||||||
}
|
|
||||||
|
|
||||||
if sigType == signatureEd25519 {
|
|
||||||
return h.buffer
|
|
||||||
}
|
|
||||||
|
|
||||||
if h.version >= VersionTLS12 {
|
|
||||||
hash := hashAlg.New()
|
|
||||||
hash.Write(h.buffer)
|
|
||||||
return hash.Sum(nil)
|
|
||||||
}
|
|
||||||
|
|
||||||
if sigType == signatureECDSA {
|
|
||||||
return h.server.Sum(nil)
|
|
||||||
}
|
|
||||||
|
|
||||||
return h.Sum()
|
|
||||||
}
|
|
||||||
|
|
||||||
// discardHandshakeBuffer is called when there is no more need to
|
|
||||||
// buffer the entirety of the handshake messages.
|
|
||||||
func (h *finishedHash) discardHandshakeBuffer() {
|
|
||||||
h.buffer = nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// noExportedKeyingMaterial is used as a value of
|
|
||||||
// ConnectionState.ekm when renegotiation is enabled and thus
|
|
||||||
// we wish to fail all key-material export requests.
|
|
||||||
func noExportedKeyingMaterial(label string, context []byte, length int) ([]byte, error) {
|
|
||||||
return nil, errors.New("crypto/tls: ExportKeyingMaterial is unavailable when renegotiation is enabled")
|
|
||||||
}
|
|
||||||
|
|
||||||
// ekmFromMasterSecret generates exported keying material as defined in RFC 5705.
|
|
||||||
func ekmFromMasterSecret(version uint16, suite *cipherSuite, masterSecret, clientRandom, serverRandom []byte) func(string, []byte, int) ([]byte, error) {
|
|
||||||
return func(label string, context []byte, length int) ([]byte, error) {
|
|
||||||
switch label {
|
|
||||||
case "client finished", "server finished", "master secret", "key expansion":
|
|
||||||
// These values are reserved and may not be used.
|
|
||||||
return nil, fmt.Errorf("crypto/tls: reserved ExportKeyingMaterial label: %s", label)
|
|
||||||
}
|
|
||||||
|
|
||||||
seedLen := len(serverRandom) + len(clientRandom)
|
|
||||||
if context != nil {
|
|
||||||
seedLen += 2 + len(context)
|
|
||||||
}
|
|
||||||
seed := make([]byte, 0, seedLen)
|
|
||||||
|
|
||||||
seed = append(seed, clientRandom...)
|
|
||||||
seed = append(seed, serverRandom...)
|
|
||||||
|
|
||||||
if context != nil {
|
|
||||||
if len(context) >= 1<<16 {
|
|
||||||
return nil, fmt.Errorf("crypto/tls: ExportKeyingMaterial context too long")
|
|
||||||
}
|
|
||||||
seed = append(seed, byte(len(context)>>8), byte(len(context)))
|
|
||||||
seed = append(seed, context...)
|
|
||||||
}
|
|
||||||
|
|
||||||
keyMaterial := make([]byte, length)
|
|
||||||
prfForVersion(version, suite)(keyMaterial, masterSecret, []byte(label), seed)
|
|
||||||
return keyMaterial, nil
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -1,259 +0,0 @@
|
||||||
// Copyright 2012 The Go Authors. All rights reserved.
|
|
||||||
// Use of this source code is governed by a BSD-style
|
|
||||||
// license that can be found in the LICENSE file.
|
|
||||||
|
|
||||||
package qtls
|
|
||||||
|
|
||||||
import (
|
|
||||||
"bytes"
|
|
||||||
"crypto/aes"
|
|
||||||
"crypto/cipher"
|
|
||||||
"crypto/hmac"
|
|
||||||
"crypto/sha256"
|
|
||||||
"crypto/subtle"
|
|
||||||
"errors"
|
|
||||||
"io"
|
|
||||||
"time"
|
|
||||||
|
|
||||||
"golang.org/x/crypto/cryptobyte"
|
|
||||||
)
|
|
||||||
|
|
||||||
// sessionState contains the information that is serialized into a session
|
|
||||||
// ticket in order to later resume a connection.
|
|
||||||
type sessionState struct {
|
|
||||||
vers uint16
|
|
||||||
cipherSuite uint16
|
|
||||||
createdAt uint64
|
|
||||||
masterSecret []byte // opaque master_secret<1..2^16-1>;
|
|
||||||
// struct { opaque certificate<1..2^24-1> } Certificate;
|
|
||||||
certificates [][]byte // Certificate certificate_list<0..2^24-1>;
|
|
||||||
|
|
||||||
// usedOldKey is true if the ticket from which this session came from
|
|
||||||
// was encrypted with an older key and thus should be refreshed.
|
|
||||||
usedOldKey bool
|
|
||||||
}
|
|
||||||
|
|
||||||
func (m *sessionState) marshal() []byte {
|
|
||||||
var b cryptobyte.Builder
|
|
||||||
b.AddUint16(m.vers)
|
|
||||||
b.AddUint16(m.cipherSuite)
|
|
||||||
addUint64(&b, m.createdAt)
|
|
||||||
b.AddUint16LengthPrefixed(func(b *cryptobyte.Builder) {
|
|
||||||
b.AddBytes(m.masterSecret)
|
|
||||||
})
|
|
||||||
b.AddUint24LengthPrefixed(func(b *cryptobyte.Builder) {
|
|
||||||
for _, cert := range m.certificates {
|
|
||||||
b.AddUint24LengthPrefixed(func(b *cryptobyte.Builder) {
|
|
||||||
b.AddBytes(cert)
|
|
||||||
})
|
|
||||||
}
|
|
||||||
})
|
|
||||||
return b.BytesOrPanic()
|
|
||||||
}
|
|
||||||
|
|
||||||
func (m *sessionState) unmarshal(data []byte) bool {
|
|
||||||
*m = sessionState{usedOldKey: m.usedOldKey}
|
|
||||||
s := cryptobyte.String(data)
|
|
||||||
if ok := s.ReadUint16(&m.vers) &&
|
|
||||||
s.ReadUint16(&m.cipherSuite) &&
|
|
||||||
readUint64(&s, &m.createdAt) &&
|
|
||||||
readUint16LengthPrefixed(&s, &m.masterSecret) &&
|
|
||||||
len(m.masterSecret) != 0; !ok {
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
var certList cryptobyte.String
|
|
||||||
if !s.ReadUint24LengthPrefixed(&certList) {
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
for !certList.Empty() {
|
|
||||||
var cert []byte
|
|
||||||
if !readUint24LengthPrefixed(&certList, &cert) {
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
m.certificates = append(m.certificates, cert)
|
|
||||||
}
|
|
||||||
return s.Empty()
|
|
||||||
}
|
|
||||||
|
|
||||||
// sessionStateTLS13 is the content of a TLS 1.3 session ticket. Its first
|
|
||||||
// version (revision = 0) doesn't carry any of the information needed for 0-RTT
|
|
||||||
// validation and the nonce is always empty.
|
|
||||||
// version (revision = 1) carries the max_early_data_size sent in the ticket.
|
|
||||||
// version (revision = 2) carries the ALPN sent in the ticket.
|
|
||||||
type sessionStateTLS13 struct {
|
|
||||||
// uint8 version = 0x0304;
|
|
||||||
// uint8 revision = 2;
|
|
||||||
cipherSuite uint16
|
|
||||||
createdAt uint64
|
|
||||||
resumptionSecret []byte // opaque resumption_master_secret<1..2^8-1>;
|
|
||||||
certificate Certificate // CertificateEntry certificate_list<0..2^24-1>;
|
|
||||||
maxEarlyData uint32
|
|
||||||
alpn string
|
|
||||||
|
|
||||||
appData []byte
|
|
||||||
}
|
|
||||||
|
|
||||||
func (m *sessionStateTLS13) marshal() []byte {
|
|
||||||
var b cryptobyte.Builder
|
|
||||||
b.AddUint16(VersionTLS13)
|
|
||||||
b.AddUint8(2) // revision
|
|
||||||
b.AddUint16(m.cipherSuite)
|
|
||||||
addUint64(&b, m.createdAt)
|
|
||||||
b.AddUint8LengthPrefixed(func(b *cryptobyte.Builder) {
|
|
||||||
b.AddBytes(m.resumptionSecret)
|
|
||||||
})
|
|
||||||
marshalCertificate(&b, m.certificate)
|
|
||||||
b.AddUint32(m.maxEarlyData)
|
|
||||||
b.AddUint8LengthPrefixed(func(b *cryptobyte.Builder) {
|
|
||||||
b.AddBytes([]byte(m.alpn))
|
|
||||||
})
|
|
||||||
b.AddUint16LengthPrefixed(func(b *cryptobyte.Builder) {
|
|
||||||
b.AddBytes(m.appData)
|
|
||||||
})
|
|
||||||
return b.BytesOrPanic()
|
|
||||||
}
|
|
||||||
|
|
||||||
func (m *sessionStateTLS13) unmarshal(data []byte) bool {
|
|
||||||
*m = sessionStateTLS13{}
|
|
||||||
s := cryptobyte.String(data)
|
|
||||||
var version uint16
|
|
||||||
var revision uint8
|
|
||||||
var alpn []byte
|
|
||||||
ret := s.ReadUint16(&version) &&
|
|
||||||
version == VersionTLS13 &&
|
|
||||||
s.ReadUint8(&revision) &&
|
|
||||||
revision == 2 &&
|
|
||||||
s.ReadUint16(&m.cipherSuite) &&
|
|
||||||
readUint64(&s, &m.createdAt) &&
|
|
||||||
readUint8LengthPrefixed(&s, &m.resumptionSecret) &&
|
|
||||||
len(m.resumptionSecret) != 0 &&
|
|
||||||
unmarshalCertificate(&s, &m.certificate) &&
|
|
||||||
s.ReadUint32(&m.maxEarlyData) &&
|
|
||||||
readUint8LengthPrefixed(&s, &alpn) &&
|
|
||||||
readUint16LengthPrefixed(&s, &m.appData) &&
|
|
||||||
s.Empty()
|
|
||||||
m.alpn = string(alpn)
|
|
||||||
return ret
|
|
||||||
}
|
|
||||||
|
|
||||||
func (c *Conn) encryptTicket(state []byte) ([]byte, error) {
|
|
||||||
if len(c.ticketKeys) == 0 {
|
|
||||||
return nil, errors.New("tls: internal error: session ticket keys unavailable")
|
|
||||||
}
|
|
||||||
|
|
||||||
encrypted := make([]byte, ticketKeyNameLen+aes.BlockSize+len(state)+sha256.Size)
|
|
||||||
keyName := encrypted[:ticketKeyNameLen]
|
|
||||||
iv := encrypted[ticketKeyNameLen : ticketKeyNameLen+aes.BlockSize]
|
|
||||||
macBytes := encrypted[len(encrypted)-sha256.Size:]
|
|
||||||
|
|
||||||
if _, err := io.ReadFull(c.config.rand(), iv); err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
key := c.ticketKeys[0]
|
|
||||||
copy(keyName, key.keyName[:])
|
|
||||||
block, err := aes.NewCipher(key.aesKey[:])
|
|
||||||
if err != nil {
|
|
||||||
return nil, errors.New("tls: failed to create cipher while encrypting ticket: " + err.Error())
|
|
||||||
}
|
|
||||||
cipher.NewCTR(block, iv).XORKeyStream(encrypted[ticketKeyNameLen+aes.BlockSize:], state)
|
|
||||||
|
|
||||||
mac := hmac.New(sha256.New, key.hmacKey[:])
|
|
||||||
mac.Write(encrypted[:len(encrypted)-sha256.Size])
|
|
||||||
mac.Sum(macBytes[:0])
|
|
||||||
|
|
||||||
return encrypted, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (c *Conn) decryptTicket(encrypted []byte) (plaintext []byte, usedOldKey bool) {
|
|
||||||
if len(encrypted) < ticketKeyNameLen+aes.BlockSize+sha256.Size {
|
|
||||||
return nil, false
|
|
||||||
}
|
|
||||||
|
|
||||||
keyName := encrypted[:ticketKeyNameLen]
|
|
||||||
iv := encrypted[ticketKeyNameLen : ticketKeyNameLen+aes.BlockSize]
|
|
||||||
macBytes := encrypted[len(encrypted)-sha256.Size:]
|
|
||||||
ciphertext := encrypted[ticketKeyNameLen+aes.BlockSize : len(encrypted)-sha256.Size]
|
|
||||||
|
|
||||||
keyIndex := -1
|
|
||||||
for i, candidateKey := range c.ticketKeys {
|
|
||||||
if bytes.Equal(keyName, candidateKey.keyName[:]) {
|
|
||||||
keyIndex = i
|
|
||||||
break
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if keyIndex == -1 {
|
|
||||||
return nil, false
|
|
||||||
}
|
|
||||||
key := &c.ticketKeys[keyIndex]
|
|
||||||
|
|
||||||
mac := hmac.New(sha256.New, key.hmacKey[:])
|
|
||||||
mac.Write(encrypted[:len(encrypted)-sha256.Size])
|
|
||||||
expected := mac.Sum(nil)
|
|
||||||
|
|
||||||
if subtle.ConstantTimeCompare(macBytes, expected) != 1 {
|
|
||||||
return nil, false
|
|
||||||
}
|
|
||||||
|
|
||||||
block, err := aes.NewCipher(key.aesKey[:])
|
|
||||||
if err != nil {
|
|
||||||
return nil, false
|
|
||||||
}
|
|
||||||
plaintext = make([]byte, len(ciphertext))
|
|
||||||
cipher.NewCTR(block, iv).XORKeyStream(plaintext, ciphertext)
|
|
||||||
|
|
||||||
return plaintext, keyIndex > 0
|
|
||||||
}
|
|
||||||
|
|
||||||
func (c *Conn) getSessionTicketMsg(appData []byte) (*newSessionTicketMsgTLS13, error) {
|
|
||||||
m := new(newSessionTicketMsgTLS13)
|
|
||||||
|
|
||||||
var certsFromClient [][]byte
|
|
||||||
for _, cert := range c.peerCertificates {
|
|
||||||
certsFromClient = append(certsFromClient, cert.Raw)
|
|
||||||
}
|
|
||||||
state := sessionStateTLS13{
|
|
||||||
cipherSuite: c.cipherSuite,
|
|
||||||
createdAt: uint64(c.config.time().Unix()),
|
|
||||||
resumptionSecret: c.resumptionSecret,
|
|
||||||
certificate: Certificate{
|
|
||||||
Certificate: certsFromClient,
|
|
||||||
OCSPStaple: c.ocspResponse,
|
|
||||||
SignedCertificateTimestamps: c.scts,
|
|
||||||
},
|
|
||||||
appData: appData,
|
|
||||||
alpn: c.clientProtocol,
|
|
||||||
}
|
|
||||||
if c.extraConfig != nil {
|
|
||||||
state.maxEarlyData = c.extraConfig.MaxEarlyData
|
|
||||||
}
|
|
||||||
var err error
|
|
||||||
m.label, err = c.encryptTicket(state.marshal())
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
m.lifetime = uint32(maxSessionTicketLifetime / time.Second)
|
|
||||||
if c.extraConfig != nil {
|
|
||||||
m.maxEarlyData = c.extraConfig.MaxEarlyData
|
|
||||||
}
|
|
||||||
return m, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// GetSessionTicket generates a new session ticket.
|
|
||||||
// It should only be called after the handshake completes.
|
|
||||||
// It can only be used for servers, and only if the alternative record layer is set.
|
|
||||||
// The ticket may be nil if config.SessionTicketsDisabled is set,
|
|
||||||
// or if the client isn't able to receive session tickets.
|
|
||||||
func (c *Conn) GetSessionTicket(appData []byte) ([]byte, error) {
|
|
||||||
if c.isClient || !c.handshakeComplete() || c.extraConfig == nil || c.extraConfig.AlternativeRecordLayer == nil {
|
|
||||||
return nil, errors.New("GetSessionTicket is only valid for servers after completion of the handshake, and if an alternative record layer is set.")
|
|
||||||
}
|
|
||||||
if c.config.SessionTicketsDisabled {
|
|
||||||
return nil, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
m, err := c.getSessionTicketMsg(appData)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
return m.marshal(), nil
|
|
||||||
}
|
|
|
@ -1,362 +0,0 @@
|
||||||
// Copyright 2009 The Go Authors. All rights reserved.
|
|
||||||
// Use of this source code is governed by a BSD-style
|
|
||||||
// license that can be found in the LICENSE file.
|
|
||||||
|
|
||||||
// package qtls partially implements TLS 1.2, as specified in RFC 5246,
|
|
||||||
// and TLS 1.3, as specified in RFC 8446.
|
|
||||||
package qtls
|
|
||||||
|
|
||||||
// BUG(agl): The crypto/tls package only implements some countermeasures
|
|
||||||
// against Lucky13 attacks on CBC-mode encryption, and only on SHA1
|
|
||||||
// variants. See http://www.isg.rhul.ac.uk/tls/TLStiming.pdf and
|
|
||||||
// https://www.imperialviolet.org/2013/02/04/luckythirteen.html.
|
|
||||||
|
|
||||||
import (
|
|
||||||
"bytes"
|
|
||||||
"context"
|
|
||||||
"crypto"
|
|
||||||
"crypto/ecdsa"
|
|
||||||
"crypto/ed25519"
|
|
||||||
"crypto/rsa"
|
|
||||||
"crypto/x509"
|
|
||||||
"encoding/pem"
|
|
||||||
"errors"
|
|
||||||
"fmt"
|
|
||||||
"net"
|
|
||||||
"os"
|
|
||||||
"strings"
|
|
||||||
)
|
|
||||||
|
|
||||||
// Server returns a new TLS server side connection
|
|
||||||
// using conn as the underlying transport.
|
|
||||||
// The configuration config must be non-nil and must include
|
|
||||||
// at least one certificate or else set GetCertificate.
|
|
||||||
func Server(conn net.Conn, config *Config, extraConfig *ExtraConfig) *Conn {
|
|
||||||
c := &Conn{
|
|
||||||
conn: conn,
|
|
||||||
config: fromConfig(config),
|
|
||||||
extraConfig: extraConfig,
|
|
||||||
}
|
|
||||||
c.handshakeFn = c.serverHandshake
|
|
||||||
return c
|
|
||||||
}
|
|
||||||
|
|
||||||
// Client returns a new TLS client side connection
|
|
||||||
// using conn as the underlying transport.
|
|
||||||
// The config cannot be nil: users must set either ServerName or
|
|
||||||
// InsecureSkipVerify in the config.
|
|
||||||
func Client(conn net.Conn, config *Config, extraConfig *ExtraConfig) *Conn {
|
|
||||||
c := &Conn{
|
|
||||||
conn: conn,
|
|
||||||
config: fromConfig(config),
|
|
||||||
extraConfig: extraConfig,
|
|
||||||
isClient: true,
|
|
||||||
}
|
|
||||||
c.handshakeFn = c.clientHandshake
|
|
||||||
return c
|
|
||||||
}
|
|
||||||
|
|
||||||
// A listener implements a network listener (net.Listener) for TLS connections.
|
|
||||||
type listener struct {
|
|
||||||
net.Listener
|
|
||||||
config *Config
|
|
||||||
extraConfig *ExtraConfig
|
|
||||||
}
|
|
||||||
|
|
||||||
// Accept waits for and returns the next incoming TLS connection.
|
|
||||||
// The returned connection is of type *Conn.
|
|
||||||
func (l *listener) Accept() (net.Conn, error) {
|
|
||||||
c, err := l.Listener.Accept()
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
return Server(c, l.config, l.extraConfig), nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// NewListener creates a Listener which accepts connections from an inner
|
|
||||||
// Listener and wraps each connection with Server.
|
|
||||||
// The configuration config must be non-nil and must include
|
|
||||||
// at least one certificate or else set GetCertificate.
|
|
||||||
func NewListener(inner net.Listener, config *Config, extraConfig *ExtraConfig) net.Listener {
|
|
||||||
l := new(listener)
|
|
||||||
l.Listener = inner
|
|
||||||
l.config = config
|
|
||||||
l.extraConfig = extraConfig
|
|
||||||
return l
|
|
||||||
}
|
|
||||||
|
|
||||||
// Listen creates a TLS listener accepting connections on the
|
|
||||||
// given network address using net.Listen.
|
|
||||||
// The configuration config must be non-nil and must include
|
|
||||||
// at least one certificate or else set GetCertificate.
|
|
||||||
func Listen(network, laddr string, config *Config, extraConfig *ExtraConfig) (net.Listener, error) {
|
|
||||||
if config == nil || len(config.Certificates) == 0 &&
|
|
||||||
config.GetCertificate == nil && config.GetConfigForClient == nil {
|
|
||||||
return nil, errors.New("tls: neither Certificates, GetCertificate, nor GetConfigForClient set in Config")
|
|
||||||
}
|
|
||||||
l, err := net.Listen(network, laddr)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
return NewListener(l, config, extraConfig), nil
|
|
||||||
}
|
|
||||||
|
|
||||||
type timeoutError struct{}
|
|
||||||
|
|
||||||
func (timeoutError) Error() string { return "tls: DialWithDialer timed out" }
|
|
||||||
func (timeoutError) Timeout() bool { return true }
|
|
||||||
func (timeoutError) Temporary() bool { return true }
|
|
||||||
|
|
||||||
// DialWithDialer connects to the given network address using dialer.Dial and
|
|
||||||
// then initiates a TLS handshake, returning the resulting TLS connection. Any
|
|
||||||
// timeout or deadline given in the dialer apply to connection and TLS
|
|
||||||
// handshake as a whole.
|
|
||||||
//
|
|
||||||
// DialWithDialer interprets a nil configuration as equivalent to the zero
|
|
||||||
// configuration; see the documentation of Config for the defaults.
|
|
||||||
//
|
|
||||||
// DialWithDialer uses context.Background internally; to specify the context,
|
|
||||||
// use Dialer.DialContext with NetDialer set to the desired dialer.
|
|
||||||
func DialWithDialer(dialer *net.Dialer, network, addr string, config *Config, extraConfig *ExtraConfig) (*Conn, error) {
|
|
||||||
return dial(context.Background(), dialer, network, addr, config, extraConfig)
|
|
||||||
}
|
|
||||||
|
|
||||||
func dial(ctx context.Context, netDialer *net.Dialer, network, addr string, config *Config, extraConfig *ExtraConfig) (*Conn, error) {
|
|
||||||
if netDialer.Timeout != 0 {
|
|
||||||
var cancel context.CancelFunc
|
|
||||||
ctx, cancel = context.WithTimeout(ctx, netDialer.Timeout)
|
|
||||||
defer cancel()
|
|
||||||
}
|
|
||||||
|
|
||||||
if !netDialer.Deadline.IsZero() {
|
|
||||||
var cancel context.CancelFunc
|
|
||||||
ctx, cancel = context.WithDeadline(ctx, netDialer.Deadline)
|
|
||||||
defer cancel()
|
|
||||||
}
|
|
||||||
|
|
||||||
rawConn, err := netDialer.DialContext(ctx, network, addr)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
colonPos := strings.LastIndex(addr, ":")
|
|
||||||
if colonPos == -1 {
|
|
||||||
colonPos = len(addr)
|
|
||||||
}
|
|
||||||
hostname := addr[:colonPos]
|
|
||||||
|
|
||||||
if config == nil {
|
|
||||||
config = defaultConfig()
|
|
||||||
}
|
|
||||||
// If no ServerName is set, infer the ServerName
|
|
||||||
// from the hostname we're connecting to.
|
|
||||||
if config.ServerName == "" {
|
|
||||||
// Make a copy to avoid polluting argument or default.
|
|
||||||
c := config.Clone()
|
|
||||||
c.ServerName = hostname
|
|
||||||
config = c
|
|
||||||
}
|
|
||||||
|
|
||||||
conn := Client(rawConn, config, extraConfig)
|
|
||||||
if err := conn.HandshakeContext(ctx); err != nil {
|
|
||||||
rawConn.Close()
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
return conn, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// Dial connects to the given network address using net.Dial
|
|
||||||
// and then initiates a TLS handshake, returning the resulting
|
|
||||||
// TLS connection.
|
|
||||||
// Dial interprets a nil configuration as equivalent to
|
|
||||||
// the zero configuration; see the documentation of Config
|
|
||||||
// for the defaults.
|
|
||||||
func Dial(network, addr string, config *Config, extraConfig *ExtraConfig) (*Conn, error) {
|
|
||||||
return DialWithDialer(new(net.Dialer), network, addr, config, extraConfig)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Dialer dials TLS connections given a configuration and a Dialer for the
|
|
||||||
// underlying connection.
|
|
||||||
type Dialer struct {
|
|
||||||
// NetDialer is the optional dialer to use for the TLS connections'
|
|
||||||
// underlying TCP connections.
|
|
||||||
// A nil NetDialer is equivalent to the net.Dialer zero value.
|
|
||||||
NetDialer *net.Dialer
|
|
||||||
|
|
||||||
// Config is the TLS configuration to use for new connections.
|
|
||||||
// A nil configuration is equivalent to the zero
|
|
||||||
// configuration; see the documentation of Config for the
|
|
||||||
// defaults.
|
|
||||||
Config *Config
|
|
||||||
|
|
||||||
ExtraConfig *ExtraConfig
|
|
||||||
}
|
|
||||||
|
|
||||||
// Dial connects to the given network address and initiates a TLS
|
|
||||||
// handshake, returning the resulting TLS connection.
|
|
||||||
//
|
|
||||||
// The returned Conn, if any, will always be of type *Conn.
|
|
||||||
//
|
|
||||||
// Dial uses context.Background internally; to specify the context,
|
|
||||||
// use DialContext.
|
|
||||||
func (d *Dialer) Dial(network, addr string) (net.Conn, error) {
|
|
||||||
return d.DialContext(context.Background(), network, addr)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (d *Dialer) netDialer() *net.Dialer {
|
|
||||||
if d.NetDialer != nil {
|
|
||||||
return d.NetDialer
|
|
||||||
}
|
|
||||||
return new(net.Dialer)
|
|
||||||
}
|
|
||||||
|
|
||||||
// DialContext connects to the given network address and initiates a TLS
|
|
||||||
// handshake, returning the resulting TLS connection.
|
|
||||||
//
|
|
||||||
// The provided Context must be non-nil. If the context expires before
|
|
||||||
// the connection is complete, an error is returned. Once successfully
|
|
||||||
// connected, any expiration of the context will not affect the
|
|
||||||
// connection.
|
|
||||||
//
|
|
||||||
// The returned Conn, if any, will always be of type *Conn.
|
|
||||||
func (d *Dialer) DialContext(ctx context.Context, network, addr string) (net.Conn, error) {
|
|
||||||
c, err := dial(ctx, d.netDialer(), network, addr, d.Config, d.ExtraConfig)
|
|
||||||
if err != nil {
|
|
||||||
// Don't return c (a typed nil) in an interface.
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
return c, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// LoadX509KeyPair reads and parses a public/private key pair from a pair
|
|
||||||
// of files. The files must contain PEM encoded data. The certificate file
|
|
||||||
// may contain intermediate certificates following the leaf certificate to
|
|
||||||
// form a certificate chain. On successful return, Certificate.Leaf will
|
|
||||||
// be nil because the parsed form of the certificate is not retained.
|
|
||||||
func LoadX509KeyPair(certFile, keyFile string) (Certificate, error) {
|
|
||||||
certPEMBlock, err := os.ReadFile(certFile)
|
|
||||||
if err != nil {
|
|
||||||
return Certificate{}, err
|
|
||||||
}
|
|
||||||
keyPEMBlock, err := os.ReadFile(keyFile)
|
|
||||||
if err != nil {
|
|
||||||
return Certificate{}, err
|
|
||||||
}
|
|
||||||
return X509KeyPair(certPEMBlock, keyPEMBlock)
|
|
||||||
}
|
|
||||||
|
|
||||||
// X509KeyPair parses a public/private key pair from a pair of
|
|
||||||
// PEM encoded data. On successful return, Certificate.Leaf will be nil because
|
|
||||||
// the parsed form of the certificate is not retained.
|
|
||||||
func X509KeyPair(certPEMBlock, keyPEMBlock []byte) (Certificate, error) {
|
|
||||||
fail := func(err error) (Certificate, error) { return Certificate{}, err }
|
|
||||||
|
|
||||||
var cert Certificate
|
|
||||||
var skippedBlockTypes []string
|
|
||||||
for {
|
|
||||||
var certDERBlock *pem.Block
|
|
||||||
certDERBlock, certPEMBlock = pem.Decode(certPEMBlock)
|
|
||||||
if certDERBlock == nil {
|
|
||||||
break
|
|
||||||
}
|
|
||||||
if certDERBlock.Type == "CERTIFICATE" {
|
|
||||||
cert.Certificate = append(cert.Certificate, certDERBlock.Bytes)
|
|
||||||
} else {
|
|
||||||
skippedBlockTypes = append(skippedBlockTypes, certDERBlock.Type)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if len(cert.Certificate) == 0 {
|
|
||||||
if len(skippedBlockTypes) == 0 {
|
|
||||||
return fail(errors.New("tls: failed to find any PEM data in certificate input"))
|
|
||||||
}
|
|
||||||
if len(skippedBlockTypes) == 1 && strings.HasSuffix(skippedBlockTypes[0], "PRIVATE KEY") {
|
|
||||||
return fail(errors.New("tls: failed to find certificate PEM data in certificate input, but did find a private key; PEM inputs may have been switched"))
|
|
||||||
}
|
|
||||||
return fail(fmt.Errorf("tls: failed to find \"CERTIFICATE\" PEM block in certificate input after skipping PEM blocks of the following types: %v", skippedBlockTypes))
|
|
||||||
}
|
|
||||||
|
|
||||||
skippedBlockTypes = skippedBlockTypes[:0]
|
|
||||||
var keyDERBlock *pem.Block
|
|
||||||
for {
|
|
||||||
keyDERBlock, keyPEMBlock = pem.Decode(keyPEMBlock)
|
|
||||||
if keyDERBlock == nil {
|
|
||||||
if len(skippedBlockTypes) == 0 {
|
|
||||||
return fail(errors.New("tls: failed to find any PEM data in key input"))
|
|
||||||
}
|
|
||||||
if len(skippedBlockTypes) == 1 && skippedBlockTypes[0] == "CERTIFICATE" {
|
|
||||||
return fail(errors.New("tls: found a certificate rather than a key in the PEM for the private key"))
|
|
||||||
}
|
|
||||||
return fail(fmt.Errorf("tls: failed to find PEM block with type ending in \"PRIVATE KEY\" in key input after skipping PEM blocks of the following types: %v", skippedBlockTypes))
|
|
||||||
}
|
|
||||||
if keyDERBlock.Type == "PRIVATE KEY" || strings.HasSuffix(keyDERBlock.Type, " PRIVATE KEY") {
|
|
||||||
break
|
|
||||||
}
|
|
||||||
skippedBlockTypes = append(skippedBlockTypes, keyDERBlock.Type)
|
|
||||||
}
|
|
||||||
|
|
||||||
// We don't need to parse the public key for TLS, but we so do anyway
|
|
||||||
// to check that it looks sane and matches the private key.
|
|
||||||
x509Cert, err := x509.ParseCertificate(cert.Certificate[0])
|
|
||||||
if err != nil {
|
|
||||||
return fail(err)
|
|
||||||
}
|
|
||||||
|
|
||||||
cert.PrivateKey, err = parsePrivateKey(keyDERBlock.Bytes)
|
|
||||||
if err != nil {
|
|
||||||
return fail(err)
|
|
||||||
}
|
|
||||||
|
|
||||||
switch pub := x509Cert.PublicKey.(type) {
|
|
||||||
case *rsa.PublicKey:
|
|
||||||
priv, ok := cert.PrivateKey.(*rsa.PrivateKey)
|
|
||||||
if !ok {
|
|
||||||
return fail(errors.New("tls: private key type does not match public key type"))
|
|
||||||
}
|
|
||||||
if pub.N.Cmp(priv.N) != 0 {
|
|
||||||
return fail(errors.New("tls: private key does not match public key"))
|
|
||||||
}
|
|
||||||
case *ecdsa.PublicKey:
|
|
||||||
priv, ok := cert.PrivateKey.(*ecdsa.PrivateKey)
|
|
||||||
if !ok {
|
|
||||||
return fail(errors.New("tls: private key type does not match public key type"))
|
|
||||||
}
|
|
||||||
if pub.X.Cmp(priv.X) != 0 || pub.Y.Cmp(priv.Y) != 0 {
|
|
||||||
return fail(errors.New("tls: private key does not match public key"))
|
|
||||||
}
|
|
||||||
case ed25519.PublicKey:
|
|
||||||
priv, ok := cert.PrivateKey.(ed25519.PrivateKey)
|
|
||||||
if !ok {
|
|
||||||
return fail(errors.New("tls: private key type does not match public key type"))
|
|
||||||
}
|
|
||||||
if !bytes.Equal(priv.Public().(ed25519.PublicKey), pub) {
|
|
||||||
return fail(errors.New("tls: private key does not match public key"))
|
|
||||||
}
|
|
||||||
default:
|
|
||||||
return fail(errors.New("tls: unknown public key algorithm"))
|
|
||||||
}
|
|
||||||
|
|
||||||
return cert, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// Attempt to parse the given private key DER block. OpenSSL 0.9.8 generates
|
|
||||||
// PKCS #1 private keys by default, while OpenSSL 1.0.0 generates PKCS #8 keys.
|
|
||||||
// OpenSSL ecparam generates SEC1 EC private keys for ECDSA. We try all three.
|
|
||||||
func parsePrivateKey(der []byte) (crypto.PrivateKey, error) {
|
|
||||||
if key, err := x509.ParsePKCS1PrivateKey(der); err == nil {
|
|
||||||
return key, nil
|
|
||||||
}
|
|
||||||
if key, err := x509.ParsePKCS8PrivateKey(der); err == nil {
|
|
||||||
switch key := key.(type) {
|
|
||||||
case *rsa.PrivateKey, *ecdsa.PrivateKey, ed25519.PrivateKey:
|
|
||||||
return key, nil
|
|
||||||
default:
|
|
||||||
return nil, errors.New("tls: found unknown private key type in PKCS#8 wrapping")
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if key, err := x509.ParseECPrivateKey(der); err == nil {
|
|
||||||
return key, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
return nil, errors.New("tls: failed to parse private key")
|
|
||||||
}
|
|
|
@ -1,96 +0,0 @@
|
||||||
package qtls
|
|
||||||
|
|
||||||
import (
|
|
||||||
"crypto/tls"
|
|
||||||
"reflect"
|
|
||||||
"unsafe"
|
|
||||||
)
|
|
||||||
|
|
||||||
func init() {
|
|
||||||
if !structsEqual(&tls.ConnectionState{}, &connectionState{}) {
|
|
||||||
panic("qtls.ConnectionState doesn't match")
|
|
||||||
}
|
|
||||||
if !structsEqual(&tls.ClientSessionState{}, &clientSessionState{}) {
|
|
||||||
panic("qtls.ClientSessionState doesn't match")
|
|
||||||
}
|
|
||||||
if !structsEqual(&tls.CertificateRequestInfo{}, &certificateRequestInfo{}) {
|
|
||||||
panic("qtls.CertificateRequestInfo doesn't match")
|
|
||||||
}
|
|
||||||
if !structsEqual(&tls.Config{}, &config{}) {
|
|
||||||
panic("qtls.Config doesn't match")
|
|
||||||
}
|
|
||||||
if !structsEqual(&tls.ClientHelloInfo{}, &clientHelloInfo{}) {
|
|
||||||
panic("qtls.ClientHelloInfo doesn't match")
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func toConnectionState(c connectionState) ConnectionState {
|
|
||||||
return *(*ConnectionState)(unsafe.Pointer(&c))
|
|
||||||
}
|
|
||||||
|
|
||||||
func toClientSessionState(s *clientSessionState) *ClientSessionState {
|
|
||||||
return (*ClientSessionState)(unsafe.Pointer(s))
|
|
||||||
}
|
|
||||||
|
|
||||||
func fromClientSessionState(s *ClientSessionState) *clientSessionState {
|
|
||||||
return (*clientSessionState)(unsafe.Pointer(s))
|
|
||||||
}
|
|
||||||
|
|
||||||
func toCertificateRequestInfo(i *certificateRequestInfo) *CertificateRequestInfo {
|
|
||||||
return (*CertificateRequestInfo)(unsafe.Pointer(i))
|
|
||||||
}
|
|
||||||
|
|
||||||
func toConfig(c *config) *Config {
|
|
||||||
return (*Config)(unsafe.Pointer(c))
|
|
||||||
}
|
|
||||||
|
|
||||||
func fromConfig(c *Config) *config {
|
|
||||||
return (*config)(unsafe.Pointer(c))
|
|
||||||
}
|
|
||||||
|
|
||||||
func toClientHelloInfo(chi *clientHelloInfo) *ClientHelloInfo {
|
|
||||||
return (*ClientHelloInfo)(unsafe.Pointer(chi))
|
|
||||||
}
|
|
||||||
|
|
||||||
func structsEqual(a, b interface{}) bool {
|
|
||||||
return compare(reflect.ValueOf(a), reflect.ValueOf(b))
|
|
||||||
}
|
|
||||||
|
|
||||||
func compare(a, b reflect.Value) bool {
|
|
||||||
sa := a.Elem()
|
|
||||||
sb := b.Elem()
|
|
||||||
if sa.NumField() != sb.NumField() {
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
for i := 0; i < sa.NumField(); i++ {
|
|
||||||
fa := sa.Type().Field(i)
|
|
||||||
fb := sb.Type().Field(i)
|
|
||||||
if !reflect.DeepEqual(fa.Index, fb.Index) || fa.Name != fb.Name || fa.Anonymous != fb.Anonymous || fa.Offset != fb.Offset || !reflect.DeepEqual(fa.Type, fb.Type) {
|
|
||||||
if fa.Type.Kind() != fb.Type.Kind() {
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
if fa.Type.Kind() == reflect.Slice {
|
|
||||||
if !compareStruct(fa.Type.Elem(), fb.Type.Elem()) {
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
|
|
||||||
func compareStruct(a, b reflect.Type) bool {
|
|
||||||
if a.NumField() != b.NumField() {
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
for i := 0; i < a.NumField(); i++ {
|
|
||||||
fa := a.Field(i)
|
|
||||||
fb := b.Field(i)
|
|
||||||
if !reflect.DeepEqual(fa.Index, fb.Index) || fa.Name != fb.Name || fa.Anonymous != fb.Anonymous || fa.Offset != fb.Offset || !reflect.DeepEqual(fa.Type, fb.Type) {
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return true
|
|
||||||
}
|
|
|
@ -2,7 +2,6 @@
|
||||||
// Use of this source code is governed by a BSD-style
|
// Use of this source code is governed by a BSD-style
|
||||||
// license that can be found in the LICENSE file.
|
// license that can be found in the LICENSE file.
|
||||||
|
|
||||||
//go:build go1.7
|
|
||||||
// +build go1.7
|
// +build go1.7
|
||||||
|
|
||||||
package context
|
package context
|
||||||
|
|
|
@ -2,7 +2,6 @@
|
||||||
// Use of this source code is governed by a BSD-style
|
// Use of this source code is governed by a BSD-style
|
||||||
// license that can be found in the LICENSE file.
|
// license that can be found in the LICENSE file.
|
||||||
|
|
||||||
//go:build go1.9
|
|
||||||
// +build go1.9
|
// +build go1.9
|
||||||
|
|
||||||
package context
|
package context
|
||||||
|
|
|
@ -2,7 +2,6 @@
|
||||||
// Use of this source code is governed by a BSD-style
|
// Use of this source code is governed by a BSD-style
|
||||||
// license that can be found in the LICENSE file.
|
// license that can be found in the LICENSE file.
|
||||||
|
|
||||||
//go:build !go1.7
|
|
||||||
// +build !go1.7
|
// +build !go1.7
|
||||||
|
|
||||||
package context
|
package context
|
||||||
|
|
|
@ -2,7 +2,6 @@
|
||||||
// Use of this source code is governed by a BSD-style
|
// Use of this source code is governed by a BSD-style
|
||||||
// license that can be found in the LICENSE file.
|
// license that can be found in the LICENSE file.
|
||||||
|
|
||||||
//go:build !go1.9
|
|
||||||
// +build !go1.9
|
// +build !go1.9
|
||||||
|
|
||||||
package context
|
package context
|
||||||
|
|
|
@ -2,7 +2,6 @@
|
||||||
// Use of this source code is governed by a BSD-style
|
// Use of this source code is governed by a BSD-style
|
||||||
// license that can be found in the LICENSE file.
|
// license that can be found in the LICENSE file.
|
||||||
|
|
||||||
//go:build go1.11
|
|
||||||
// +build go1.11
|
// +build go1.11
|
||||||
|
|
||||||
package http2
|
package http2
|
||||||
|
|
|
@ -2,7 +2,6 @@
|
||||||
// Use of this source code is governed by a BSD-style
|
// Use of this source code is governed by a BSD-style
|
||||||
// license that can be found in the LICENSE file.
|
// license that can be found in the LICENSE file.
|
||||||
|
|
||||||
//go:build !go1.11
|
|
||||||
// +build !go1.11
|
// +build !go1.11
|
||||||
|
|
||||||
package http2
|
package http2
|
||||||
|
|
|
@ -1293,9 +1293,7 @@ func (sc *serverConn) startGracefulShutdown() {
|
||||||
sc.shutdownOnce.Do(func() { sc.sendServeMsg(gracefulShutdownMsg) })
|
sc.shutdownOnce.Do(func() { sc.sendServeMsg(gracefulShutdownMsg) })
|
||||||
}
|
}
|
||||||
|
|
||||||
// After sending GOAWAY with an error code (non-graceful shutdown), the
|
// After sending GOAWAY, the connection will close after goAwayTimeout.
|
||||||
// connection will close after goAwayTimeout.
|
|
||||||
//
|
|
||||||
// If we close the connection immediately after sending GOAWAY, there may
|
// If we close the connection immediately after sending GOAWAY, there may
|
||||||
// be unsent data in our kernel receive buffer, which will cause the kernel
|
// be unsent data in our kernel receive buffer, which will cause the kernel
|
||||||
// to send a TCP RST on close() instead of a FIN. This RST will abort the
|
// to send a TCP RST on close() instead of a FIN. This RST will abort the
|
||||||
|
@ -1631,37 +1629,23 @@ func (sc *serverConn) processSettingInitialWindowSize(val uint32) error {
|
||||||
|
|
||||||
func (sc *serverConn) processData(f *DataFrame) error {
|
func (sc *serverConn) processData(f *DataFrame) error {
|
||||||
sc.serveG.check()
|
sc.serveG.check()
|
||||||
id := f.Header().StreamID
|
if sc.inGoAway && sc.goAwayCode != ErrCodeNo {
|
||||||
if sc.inGoAway && (sc.goAwayCode != ErrCodeNo || id > sc.maxClientStreamID) {
|
|
||||||
// Discard all DATA frames if the GOAWAY is due to an
|
|
||||||
// error, or:
|
|
||||||
//
|
|
||||||
// Section 6.8: After sending a GOAWAY frame, the sender
|
|
||||||
// can discard frames for streams initiated by the
|
|
||||||
// receiver with identifiers higher than the identified
|
|
||||||
// last stream.
|
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
data := f.Data()
|
data := f.Data()
|
||||||
|
|
||||||
|
// "If a DATA frame is received whose stream is not in "open"
|
||||||
|
// or "half closed (local)" state, the recipient MUST respond
|
||||||
|
// with a stream error (Section 5.4.2) of type STREAM_CLOSED."
|
||||||
|
id := f.Header().StreamID
|
||||||
state, st := sc.state(id)
|
state, st := sc.state(id)
|
||||||
if id == 0 || state == stateIdle {
|
if id == 0 || state == stateIdle {
|
||||||
// Section 6.1: "DATA frames MUST be associated with a
|
|
||||||
// stream. If a DATA frame is received whose stream
|
|
||||||
// identifier field is 0x0, the recipient MUST respond
|
|
||||||
// with a connection error (Section 5.4.1) of type
|
|
||||||
// PROTOCOL_ERROR."
|
|
||||||
//
|
|
||||||
// Section 5.1: "Receiving any frame other than HEADERS
|
// Section 5.1: "Receiving any frame other than HEADERS
|
||||||
// or PRIORITY on a stream in this state MUST be
|
// or PRIORITY on a stream in this state MUST be
|
||||||
// treated as a connection error (Section 5.4.1) of
|
// treated as a connection error (Section 5.4.1) of
|
||||||
// type PROTOCOL_ERROR."
|
// type PROTOCOL_ERROR."
|
||||||
return ConnectionError(ErrCodeProtocol)
|
return ConnectionError(ErrCodeProtocol)
|
||||||
}
|
}
|
||||||
|
|
||||||
// "If a DATA frame is received whose stream is not in "open"
|
|
||||||
// or "half closed (local)" state, the recipient MUST respond
|
|
||||||
// with a stream error (Section 5.4.2) of type STREAM_CLOSED."
|
|
||||||
if st == nil || state != stateOpen || st.gotTrailerHeader || st.resetQueued {
|
if st == nil || state != stateOpen || st.gotTrailerHeader || st.resetQueued {
|
||||||
// This includes sending a RST_STREAM if the stream is
|
// This includes sending a RST_STREAM if the stream is
|
||||||
// in stateHalfClosedLocal (which currently means that
|
// in stateHalfClosedLocal (which currently means that
|
||||||
|
@ -1710,7 +1694,6 @@ func (sc *serverConn) processData(f *DataFrame) error {
|
||||||
if len(data) > 0 {
|
if len(data) > 0 {
|
||||||
wrote, err := st.body.Write(data)
|
wrote, err := st.body.Write(data)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
sc.sendWindowUpdate(nil, int(f.Length)-wrote)
|
|
||||||
return streamError(id, ErrCodeStreamClosed)
|
return streamError(id, ErrCodeStreamClosed)
|
||||||
}
|
}
|
||||||
if wrote != len(data) {
|
if wrote != len(data) {
|
||||||
|
|
|
@ -154,21 +154,12 @@ func (t *Transport) pingTimeout() time.Duration {
|
||||||
|
|
||||||
// ConfigureTransport configures a net/http HTTP/1 Transport to use HTTP/2.
|
// ConfigureTransport configures a net/http HTTP/1 Transport to use HTTP/2.
|
||||||
// It returns an error if t1 has already been HTTP/2-enabled.
|
// It returns an error if t1 has already been HTTP/2-enabled.
|
||||||
//
|
|
||||||
// Use ConfigureTransports instead to configure the HTTP/2 Transport.
|
|
||||||
func ConfigureTransport(t1 *http.Transport) error {
|
func ConfigureTransport(t1 *http.Transport) error {
|
||||||
_, err := ConfigureTransports(t1)
|
_, err := configureTransport(t1)
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
// ConfigureTransports configures a net/http HTTP/1 Transport to use HTTP/2.
|
func configureTransport(t1 *http.Transport) (*Transport, error) {
|
||||||
// It returns a new HTTP/2 Transport for further configuration.
|
|
||||||
// It returns an error if t1 has already been HTTP/2-enabled.
|
|
||||||
func ConfigureTransports(t1 *http.Transport) (*Transport, error) {
|
|
||||||
return configureTransports(t1)
|
|
||||||
}
|
|
||||||
|
|
||||||
func configureTransports(t1 *http.Transport) (*Transport, error) {
|
|
||||||
connPool := new(clientConnPool)
|
connPool := new(clientConnPool)
|
||||||
t2 := &Transport{
|
t2 := &Transport{
|
||||||
ConnPool: noDialClientConnPool{connPool},
|
ConnPool: noDialClientConnPool{connPool},
|
||||||
|
@ -698,7 +689,6 @@ func (t *Transport) newClientConn(c net.Conn, singleUse bool) (*ClientConn, erro
|
||||||
cc.inflow.add(transportDefaultConnFlow + initialWindowSize)
|
cc.inflow.add(transportDefaultConnFlow + initialWindowSize)
|
||||||
cc.bw.Flush()
|
cc.bw.Flush()
|
||||||
if cc.werr != nil {
|
if cc.werr != nil {
|
||||||
cc.Close()
|
|
||||||
return nil, cc.werr
|
return nil, cc.werr
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1090,15 +1080,6 @@ func (cc *ClientConn) roundTrip(req *http.Request) (res *http.Response, gotErrAf
|
||||||
bodyWriter := cc.t.getBodyWriterState(cs, body)
|
bodyWriter := cc.t.getBodyWriterState(cs, body)
|
||||||
cs.on100 = bodyWriter.on100
|
cs.on100 = bodyWriter.on100
|
||||||
|
|
||||||
defer func() {
|
|
||||||
cc.wmu.Lock()
|
|
||||||
werr := cc.werr
|
|
||||||
cc.wmu.Unlock()
|
|
||||||
if werr != nil {
|
|
||||||
cc.Close()
|
|
||||||
}
|
|
||||||
}()
|
|
||||||
|
|
||||||
cc.wmu.Lock()
|
cc.wmu.Lock()
|
||||||
endStream := !hasBody && !hasTrailers
|
endStream := !hasBody && !hasTrailers
|
||||||
werr := cc.writeHeaders(cs.ID, endStream, int(cc.maxFrameSize), hdrs)
|
werr := cc.writeHeaders(cs.ID, endStream, int(cc.maxFrameSize), hdrs)
|
||||||
|
@ -1148,9 +1129,6 @@ func (cc *ClientConn) roundTrip(req *http.Request) (res *http.Response, gotErrAf
|
||||||
// we can keep it.
|
// we can keep it.
|
||||||
bodyWriter.cancel()
|
bodyWriter.cancel()
|
||||||
cs.abortRequestBodyWrite(errStopReqBodyWrite)
|
cs.abortRequestBodyWrite(errStopReqBodyWrite)
|
||||||
if hasBody && !bodyWritten {
|
|
||||||
<-bodyWriter.resc
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
if re.err != nil {
|
if re.err != nil {
|
||||||
cc.forgetStreamID(cs.ID)
|
cc.forgetStreamID(cs.ID)
|
||||||
|
@ -1171,7 +1149,6 @@ func (cc *ClientConn) roundTrip(req *http.Request) (res *http.Response, gotErrAf
|
||||||
} else {
|
} else {
|
||||||
bodyWriter.cancel()
|
bodyWriter.cancel()
|
||||||
cs.abortRequestBodyWrite(errStopReqBodyWriteAndCancel)
|
cs.abortRequestBodyWrite(errStopReqBodyWriteAndCancel)
|
||||||
<-bodyWriter.resc
|
|
||||||
}
|
}
|
||||||
cc.forgetStreamID(cs.ID)
|
cc.forgetStreamID(cs.ID)
|
||||||
return nil, cs.getStartedWrite(), errTimeout
|
return nil, cs.getStartedWrite(), errTimeout
|
||||||
|
@ -1181,7 +1158,6 @@ func (cc *ClientConn) roundTrip(req *http.Request) (res *http.Response, gotErrAf
|
||||||
} else {
|
} else {
|
||||||
bodyWriter.cancel()
|
bodyWriter.cancel()
|
||||||
cs.abortRequestBodyWrite(errStopReqBodyWriteAndCancel)
|
cs.abortRequestBodyWrite(errStopReqBodyWriteAndCancel)
|
||||||
<-bodyWriter.resc
|
|
||||||
}
|
}
|
||||||
cc.forgetStreamID(cs.ID)
|
cc.forgetStreamID(cs.ID)
|
||||||
return nil, cs.getStartedWrite(), ctx.Err()
|
return nil, cs.getStartedWrite(), ctx.Err()
|
||||||
|
@ -1191,7 +1167,6 @@ func (cc *ClientConn) roundTrip(req *http.Request) (res *http.Response, gotErrAf
|
||||||
} else {
|
} else {
|
||||||
bodyWriter.cancel()
|
bodyWriter.cancel()
|
||||||
cs.abortRequestBodyWrite(errStopReqBodyWriteAndCancel)
|
cs.abortRequestBodyWrite(errStopReqBodyWriteAndCancel)
|
||||||
<-bodyWriter.resc
|
|
||||||
}
|
}
|
||||||
cc.forgetStreamID(cs.ID)
|
cc.forgetStreamID(cs.ID)
|
||||||
return nil, cs.getStartedWrite(), errRequestCanceled
|
return nil, cs.getStartedWrite(), errRequestCanceled
|
||||||
|
@ -1201,7 +1176,6 @@ func (cc *ClientConn) roundTrip(req *http.Request) (res *http.Response, gotErrAf
|
||||||
// forgetStreamID.
|
// forgetStreamID.
|
||||||
return nil, cs.getStartedWrite(), cs.resetErr
|
return nil, cs.getStartedWrite(), cs.resetErr
|
||||||
case err := <-bodyWriter.resc:
|
case err := <-bodyWriter.resc:
|
||||||
bodyWritten = true
|
|
||||||
// Prefer the read loop's response, if available. Issue 16102.
|
// Prefer the read loop's response, if available. Issue 16102.
|
||||||
select {
|
select {
|
||||||
case re := <-readLoopResCh:
|
case re := <-readLoopResCh:
|
||||||
|
@ -1212,6 +1186,7 @@ func (cc *ClientConn) roundTrip(req *http.Request) (res *http.Response, gotErrAf
|
||||||
cc.forgetStreamID(cs.ID)
|
cc.forgetStreamID(cs.ID)
|
||||||
return nil, cs.getStartedWrite(), err
|
return nil, cs.getStartedWrite(), err
|
||||||
}
|
}
|
||||||
|
bodyWritten = true
|
||||||
if d := cc.responseHeaderTimeout(); d != 0 {
|
if d := cc.responseHeaderTimeout(); d != 0 {
|
||||||
timer := time.NewTimer(d)
|
timer := time.NewTimer(d)
|
||||||
defer timer.Stop()
|
defer timer.Stop()
|
||||||
|
@ -2632,9 +2607,7 @@ func (t *Transport) getBodyWriterState(cs *clientStream, body io.Reader) (s body
|
||||||
|
|
||||||
func (s bodyWriterState) cancel() {
|
func (s bodyWriterState) cancel() {
|
||||||
if s.timer != nil {
|
if s.timer != nil {
|
||||||
if s.timer.Stop() {
|
s.timer.Stop()
|
||||||
s.resc <- nil
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -4,7 +4,6 @@
|
||||||
// Use of this source code is governed by a BSD-style
|
// Use of this source code is governed by a BSD-style
|
||||||
// license that can be found in the LICENSE file.
|
// license that can be found in the LICENSE file.
|
||||||
|
|
||||||
//go:build go1.10
|
|
||||||
// +build go1.10
|
// +build go1.10
|
||||||
|
|
||||||
// Package idna implements IDNA2008 using the compatibility processing
|
// Package idna implements IDNA2008 using the compatibility processing
|
||||||
|
|
|
@ -4,7 +4,6 @@
|
||||||
// Use of this source code is governed by a BSD-style
|
// Use of this source code is governed by a BSD-style
|
||||||
// license that can be found in the LICENSE file.
|
// license that can be found in the LICENSE file.
|
||||||
|
|
||||||
//go:build !go1.10
|
|
||||||
// +build !go1.10
|
// +build !go1.10
|
||||||
|
|
||||||
// Package idna implements IDNA2008 using the compatibility processing
|
// Package idna implements IDNA2008 using the compatibility processing
|
||||||
|
|
|
@ -1,6 +1,5 @@
|
||||||
// Code generated by running "go generate" in golang.org/x/text. DO NOT EDIT.
|
// Code generated by running "go generate" in golang.org/x/text. DO NOT EDIT.
|
||||||
|
|
||||||
//go:build go1.10 && !go1.13
|
|
||||||
// +build go1.10,!go1.13
|
// +build go1.10,!go1.13
|
||||||
|
|
||||||
package idna
|
package idna
|
||||||
|
|
|
@ -1,6 +1,5 @@
|
||||||
// Code generated by running "go generate" in golang.org/x/text. DO NOT EDIT.
|
// Code generated by running "go generate" in golang.org/x/text. DO NOT EDIT.
|
||||||
|
|
||||||
//go:build go1.13 && !go1.14
|
|
||||||
// +build go1.13,!go1.14
|
// +build go1.13,!go1.14
|
||||||
|
|
||||||
package idna
|
package idna
|
||||||
|
|
|
@ -1,7 +1,6 @@
|
||||||
// Code generated by running "go generate" in golang.org/x/text. DO NOT EDIT.
|
// Code generated by running "go generate" in golang.org/x/text. DO NOT EDIT.
|
||||||
|
|
||||||
//go:build go1.14 && !go1.16
|
// +build go1.14
|
||||||
// +build go1.14,!go1.16
|
|
||||||
|
|
||||||
package idna
|
package idna
|
||||||
|
|
File diff suppressed because it is too large
Load Diff
|
@ -1,6 +1,5 @@
|
||||||
// Code generated by running "go generate" in golang.org/x/text. DO NOT EDIT.
|
// Code generated by running "go generate" in golang.org/x/text. DO NOT EDIT.
|
||||||
|
|
||||||
//go:build !go1.10
|
|
||||||
// +build !go1.10
|
// +build !go1.10
|
||||||
|
|
||||||
package idna
|
package idna
|
||||||
|
|
|
@ -2,8 +2,7 @@
|
||||||
// Use of this source code is governed by a BSD-style
|
// Use of this source code is governed by a BSD-style
|
||||||
// license that can be found in the LICENSE file.
|
// license that can be found in the LICENSE file.
|
||||||
|
|
||||||
//go:build aix || darwin || dragonfly || freebsd || linux || netbsd || openbsd || solaris || zos
|
// +build aix darwin dragonfly freebsd linux netbsd openbsd solaris
|
||||||
// +build aix darwin dragonfly freebsd linux netbsd openbsd solaris zos
|
|
||||||
|
|
||||||
package socket
|
package socket
|
||||||
|
|
||||||
|
|
|
@ -2,7 +2,6 @@
|
||||||
// Use of this source code is governed by a BSD-style
|
// Use of this source code is governed by a BSD-style
|
||||||
// license that can be found in the LICENSE file.
|
// license that can be found in the LICENSE file.
|
||||||
|
|
||||||
//go:build aix || darwin || dragonfly || freebsd || netbsd || openbsd
|
|
||||||
// +build aix darwin dragonfly freebsd netbsd openbsd
|
// +build aix darwin dragonfly freebsd netbsd openbsd
|
||||||
|
|
||||||
package socket
|
package socket
|
||||||
|
|
|
@ -2,7 +2,6 @@
|
||||||
// Use of this source code is governed by a BSD-style
|
// Use of this source code is governed by a BSD-style
|
||||||
// license that can be found in the LICENSE file.
|
// license that can be found in the LICENSE file.
|
||||||
|
|
||||||
//go:build (arm || mips || mipsle || 386) && linux
|
|
||||||
// +build arm mips mipsle 386
|
// +build arm mips mipsle 386
|
||||||
// +build linux
|
// +build linux
|
||||||
|
|
||||||
|
|
|
@ -2,7 +2,6 @@
|
||||||
// Use of this source code is governed by a BSD-style
|
// Use of this source code is governed by a BSD-style
|
||||||
// license that can be found in the LICENSE file.
|
// license that can be found in the LICENSE file.
|
||||||
|
|
||||||
//go:build (arm64 || amd64 || ppc64 || ppc64le || mips64 || mips64le || riscv64 || s390x) && linux
|
|
||||||
// +build arm64 amd64 ppc64 ppc64le mips64 mips64le riscv64 s390x
|
// +build arm64 amd64 ppc64 ppc64le mips64 mips64le riscv64 s390x
|
||||||
// +build linux
|
// +build linux
|
||||||
|
|
||||||
|
|
|
@ -2,8 +2,8 @@
|
||||||
// Use of this source code is governed by a BSD-style
|
// Use of this source code is governed by a BSD-style
|
||||||
// license that can be found in the LICENSE file.
|
// license that can be found in the LICENSE file.
|
||||||
|
|
||||||
//go:build amd64 && solaris
|
// +build amd64
|
||||||
// +build amd64,solaris
|
// +build solaris
|
||||||
|
|
||||||
package socket
|
package socket
|
||||||
|
|
||||||
|
|
Some files were not shown because too many files have changed in this diff Show More
Loading…
Reference in New Issue