Skip to content
This repository was archived by the owner on Mar 8, 2024. It is now read-only.

feat: httpsRequired flag #301

Merged
merged 0 commits into from
Apr 29, 2020
Merged

feat: httpsRequired flag #301

merged 0 commits into from
Apr 29, 2020

Conversation

dino-rodriguez
Copy link
Member

High Level Overview of Change

NOTE: Breaking out my URL validation PR into various self-contained PRs as it
was getting too big.

Adds httpsRequired flag to configuration.

Updates getPaymentInfo middleware to use req.protocol to test the
implementation.

Add unit tests for urlToPayId to make sure this works.

Context of Change

Currently, we are hardcoding https for every PayId, and not actually checking
the request protocol.

This could be dangerous, as it means the PayID code claims to use https, but
supports the insecure http.

By adding a flag, and updating the relevant code, we can get secure PayID for
production (https), and insecure PayID for local development (http).

Type of Change

  • Bug fix (non-breaking change which fixes an issue)
  • New feature (non-breaking change which adds functionality)
  • Tests (You added tests for code that already exists, or your new feature included in this PR)

Before / After

Before: All PayIDs automatically "upgraded" to https
After: Use req.protocol and check against https or http, depending on
config

Test Plan

Added unit tests for urlToPayId that covers usage of the new configuration.

@@ -25,6 +25,7 @@ const config = {
privateAPIPort: Number(process.env.PRIVATE_API_PORT) || 8081,
version: process.env.VERSION ?? Version.V1,
log_level: process.env.LOG_LEVEL ?? 'INFO',
httpsRequired: process.env.HTTPS_REQUIRED === 'true',
Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Default set to false. Added a comment in this file to indicate why.

@@ -96,8 +96,8 @@ describe('payIdToUrl', function (): void {
describe('urlToPayId', function (): void {
it('throws an error on inputs that clearly are not PayID URLs', function (): void {
// GIVEN a badly formed PayID URL (no leading https://)
const url = 'http://hansbergren.example.com'
const expectedErrorMessage = 'Bad input. PayID URLs must be HTTPS.'
const url = 'example.com+alice'
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
const url = 'example.com+alice'
const url = 'example.com/alice'

Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Why the + here?

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This test says: 'throws an error on inputs that clearly are not PayID URLs'

I didn't think that http fell into that category, but I also thought this was a good test

So I kept it by making it something that was clearly not a PayID url

And tested http down below

*
* NOTE: The defaults are developer defaults. This configuration is NOT a valid
* production configuration. Please refer to example.production.env for
* reference.
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Should you add httpsRequired to example.production.env?

Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Oh, I didn't realize example.production.env already had the HTTPS_REQUIRED env var in it.

// WHEN we attempt converting it to a PayID
const actualPayId = urlToPayId(url, false)

// THEN we get our expected error
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

No error here, right?

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Updated these test comments.

Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think you may have forgotten to push your new test comments.

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Haha, the comments above that were messed up so I fixed all of those, but it seems the one you actually commented on I didn’t fix

Should be fixed now. Also the comments on other tests that were already there were incorrect, so fixed those too.

throw new Error('Bad input. PayID URLs must be HTTPS.')
}

if (!url.startsWith(HTTPS) && !url.startsWith(HTTP)) {
Copy link
Collaborator

@0xASK 0xASK Apr 29, 2020

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Can we break this into 2 errors indicating to the client if the server is requiring HTTP or HTTPS?

if (!url.startsWith(HTTPS) && httpsRequired) {
  ...
} else if (!url.startsWith(HTTP) && !httpsRequired) {
  ...
}

Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

That if statement is slightly different logic than what Dino has.

In general, if a website requires HTTPS, then you can't connect to it over http. But some websites have a SSL certificate but still allow standard http connections.

So if httpsRequired === false, then both HTTP and HTTPS are valid, right?

Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

ya, good point

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

So in the else case, we wouldn't return an error, but rather continue as normal because:

if the client requires https, you can only use https
if the client does not require https, you can use either http or https

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

whoops, didn't refresh and re-answered. thanks @hbergren

Copy link
Collaborator

@0xASK 0xASK left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

looks good overall, just one comment about breaking up an if statement into two that gives greater clarity to the requesting client on the required protocol

Copy link
Collaborator

@0xCLARITY 0xCLARITY left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

LGTM! Good catch on the test comments besides the tests you were actually writing!

@0xASK 0xASK self-requested a review April 29, 2020 19:24
@dino-rodriguez dino-rodriguez merged this pull request into master Apr 29, 2020
@dino-rodriguez dino-rodriguez deleted the dr-https-required branch April 29, 2020 20:26
dino-rodriguez pushed a commit that referenced this pull request Jun 17, 2020
* feat: add httpsRequired to config

* fix: code comment

* feat: urlToPayID handles http and https

* test: tests for https/http flag

* fix: getPaymentInfo uses req.protocol

* docs: developer defaults comment

* refactor: rename example .env

* fix: update test comments

* fix: more incorrect code comments
Sign up for free to subscribe to this conversation on GitHub. Already have an account? Sign in.
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

3 participants