temporarily implemented client authentication using client certificates (mTLS)
This commit is contained in:
		
					parent
					
						
							
								8f7f9d4257
							
						
					
				
			
			
				commit
				
					
						d7193af4e6
					
				
			
		
					 21 changed files with 326 additions and 40 deletions
				
			
		
							
								
								
									
										58
									
								
								README.md
									
										
									
									
									
								
							
							
						
						
									
										58
									
								
								README.md
									
										
									
									
									
								
							|  | @ -128,7 +128,7 @@ listen_port_tls = 443 | |||
| 
 | ||||
| [apps."app_name"] | ||||
| server_name = 'app1.example.com' | ||||
| tls = { tls_cert_path = 'localhost.crt',  tls_cert_key_path = 'localhost.key' } | ||||
| tls = { tls_cert_path = 'server.crt',  tls_cert_key_path = 'server.key' } | ||||
| reverse_proxy = [{ upstream = [{ location = 'app1.local:8080' }] }] | ||||
| ``` | ||||
| 
 | ||||
|  | @ -141,7 +141,7 @@ We should note that the private key specified by `tls_cert_key_path` must be *in | |||
| In the current Web, we believe it is common to serve everything through HTTPS rather than HTTP, and hence *https redirection* is often used for HTTP requests. When you specify both `listen_port` and `listen_port_tls`, you can enable an option of such  redirection by making `https_redirection` true. | ||||
| 
 | ||||
| ```toml | ||||
| tls = { https_redirection = true, tls_cert_path = 'localhost.crt', tls_cert_key_path = 'localhost.key' } | ||||
| tls = { https_redirection = true, tls_cert_path = 'server.crt', tls_cert_key_path = 'server.key' } | ||||
| ``` | ||||
| 
 | ||||
| If it is true, `rpxy` returns the status code `301` to the cleartext request with new location `https://<requested_host>/<requested_query_and_path>` served over TLS. | ||||
|  | @ -155,7 +155,7 @@ listen_port_tls = 443 | |||
| 
 | ||||
| [apps.app1] | ||||
| server_name = 'app1.example.com' | ||||
| tls = { https_redirection = true, tls_cert_path = 'localhost.crt', tls_cert_key_path = 'localhost.key' } | ||||
| tls = { https_redirection = true, tls_cert_path = 'server.crt', tls_cert_key_path = 'server.key' } | ||||
| 
 | ||||
| [[apps.app1.reverse_proxy]] | ||||
| upstream = [ | ||||
|  | @ -235,13 +235,63 @@ If you obtain certificates and private keys from [Let's Encrypt](https://letsenc | |||
| The easiest way is to use `openssl` by | ||||
| 
 | ||||
| ```bash | ||||
| openssl pkcs8 -topk8 -nocrypt \ | ||||
| $ openssl pkcs8 -topk8 -nocrypt \ | ||||
|     -in yoru_domain_from_le.key \ | ||||
|     -inform PEM \ | ||||
|     -out your_domain_pkcs8.key.pem \ | ||||
|     -outform PEM | ||||
| ``` | ||||
| 
 | ||||
| ### Client Authentication using Client Certificate Signed by Your Own Root CA | ||||
| 
 | ||||
| First, you need to prepare a CA certificate used to verify client certificate. If you do not have one, you can generate CA key and certificate by OpenSSL commands as follows. *Note that `rustls` accepts X509v3 certificates and reject SHA-1, and that `rpxy` relys on Version 3 extension fields of `KeyID`s of `Subject Key Identifier` and `Authority Key Identifier`.* | ||||
| 
 | ||||
| 1. Generate CA key of `secp256v1`, CSR, and then generate CA certificate that will be set for `tls.client_ca_cert_path` for each server app in `config.toml`. | ||||
| 
 | ||||
|   ```bash | ||||
|   $ openssl genpkey -algorithm EC -pkeyopt ec_paramgen_curve:prime256v1 -out client.ca.key | ||||
| 
 | ||||
|   $ openssl req -new -key client.ca.key -out client.ca.csr | ||||
|   ... | ||||
|   ----- | ||||
|   Country Name (2 letter code) []: ... | ||||
|   State or Province Name (full name) []: ... | ||||
|   Locality Name (eg, city) []: ... | ||||
|   Organization Name (eg, company) []: ... | ||||
|   Organizational Unit Name (eg, section) []: ... | ||||
|   Common Name (eg, fully qualified host name) []: <Should not input CN> | ||||
|   Email Address []: ... | ||||
| 
 | ||||
|   $ openssl x509 -req -days 3650 -sha256 -in client.ca.csr -signkey client.ca.key -out client.ca.crt -extfile client.ca.ext | ||||
|   ``` | ||||
| 
 | ||||
| 2. Generate a client key of `secp256v1` and certificate signed by CA key. | ||||
| 
 | ||||
|   ```bash | ||||
|   $ openssl genpkey -algorithm EC -pkeyopt ec_paramgen_curve:prime256v1 -out client.key | ||||
| 
 | ||||
|   $ openssl req -new -key client.key -out client.csr | ||||
|   ... | ||||
|   ----- | ||||
|   Country Name (2 letter code) []: | ||||
|   State or Province Name (full name) []: | ||||
|   Locality Name (eg, city) []: | ||||
|   Organization Name (eg, company) []: | ||||
|   Organizational Unit Name (eg, section) []: | ||||
|   Common Name (eg, fully qualified host name) []: <Should not input CN> | ||||
|   Email Address []: | ||||
| 
 | ||||
|   $ openssl x509 -req -days 365 -sha256 -in client.csr -CA client.ca.crt -CAkey client.ca.key -CAcreateserial -out client.crt -extfile client.ext | ||||
|   ``` | ||||
| 
 | ||||
|   Now you have a client key `client.key` and certificate `client.crt` (version 3). `p12` file can be retrieved as | ||||
| 
 | ||||
|   ```bash | ||||
|   $ openssl pkcs12 -export -inkey client.key -in client.crt -certfile client.ca.crt -out client.pfx | ||||
|   ``` | ||||
| 
 | ||||
|   All of sample certificate files are found in `./example-certs/` directory. | ||||
| 
 | ||||
| ### (Work Around) Deployment on Ubuntu 22.04LTS using docker behind `ufw` | ||||
| 
 | ||||
| Basically, docker automatically manage your iptables if you use the port-mapping option, i.e., `--publish` for `docker run` or `ports` in `docker-compose.yml`. This means you do not need to manually expose your port, e.g., 443 TCP/UDP for HTTPS, using `ufw`-like management command. | ||||
|  |  | |||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue
	
	 Jun Kurihara
				Jun Kurihara