Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add switching between encoded and decoded URL formats. #233

Merged
merged 10 commits into from
Feb 23, 2020
Merged

Add switching between encoded and decoded URL formats. #233

merged 10 commits into from
Feb 23, 2020

Conversation

MiSikora
Copy link
Contributor

@MiSikora MiSikora commented Feb 9, 2020

📷 Screenshots

Show us what you've changed, we love images.

Screenshot_1581281305
Screenshot_1581280198
Screenshot_1581280204
Screenshot_1581280210

📄 Context

Why did you change something? Is there an issue to link here? Or an extarnal link?

There is an issue - #155

📝 Changes

Which code did you change? How?

From the exsiting code this PR touches mostly ViewModels and Entities. I added a way to display encoded or decoded URL (or its parts) with a wrapper FormattedUrl class.

⚠️ Breaking

Is there something breaking in the API? Any class or method signature changed?

No.

📝 How to test

Is there a special case to test your changes?

Run the sample app. Use checkbox in the top right corner to switch between encoded and decoded URLs. The option should persist between application restarts.

🔮 Next steps

Do we have to plan something else after the merge?

No.

Closes #155

@vbuberen
Copy link
Collaborator

vbuberen commented Feb 9, 2020

Thanks for your contribution.

I haven't checked the code yet, but would like to provide some feedback on UI.
For me it looks not good from UX point of view to have a checkbox in a dropdown.
I would switch to some icon with 2 states (off/on) just inside toolbar to reduce number of actions required for switching.
We will have space there especially after my future planned update where share/save operations will move into FAB.

As to code review - will do in nearest days, since have quite limited availability at the moment.

P.S. Big thumbs up for tests)) Soon we will have coverage reports here in Chucker with requirements of some specific coverage level, so you are helping a lot.

@vbuberen
Copy link
Collaborator

vbuberen commented Feb 9, 2020

Side note: welcome to our #chucker channel in Kotlinlang Slack. We discuss some plans and ideas there as well as share some things to find out if it is worth implementation.
We could discuss UX topic of your changes there if needed, for example.

@MiSikora
Copy link
Contributor Author

MiSikora commented Feb 9, 2020

Ok, thanks for the feedback. I wanted to use a toggle icon button but couldn't figure out how to represent HTTP encoding with an image. Maybe I'll come up with something.

Also, I'm thinking now it makes sense to make this option configurable from code. Users will probably expect to have a default option on app reinstallation as well. Since this is a QA plugin the apps will be reinstalled all the time. Not sure what is the best place for that. Chucker with a static function? Some option on ChukcerInterceptor? I'll have to update README.md.

@vbuberen
Copy link
Collaborator

vbuberen commented Feb 9, 2020

Yes, it makes sense to provide users with option to configure default preference while setting up Chucker, like they can with retention period.

@vbuberen
Copy link
Collaborator

vbuberen commented Feb 9, 2020

Ok, thanks for the feedback. I wanted to use a toggle icon button but couldn't figure out how to represent HTTP encoding with an image. Maybe I'll come up with something.

I will look for some icons as well and share ideas to not be just critic)))

@cortinico
Copy link
Member

Also, I'm thinking now it makes sense to make this option configurable from code. Users will probably expect to have a default option on app reinstallation as well. Since this is a QA plugin the apps will be reinstalled all the time. Not sure what is the best place for that. Chucker with a static function? Some option on ChukcerInterceptor? I'll have to update README.md.

I'd rather avoid this approach. Offering this as a configurable feature has the side effect of extending the API surface for a limited benefit in terms of features.

I think we can just offer some UI controls (a button, checkbox, radio button, menu entry, etc...) to toggle the visibility of encoded/decoded URL.

@vbuberen
Copy link
Collaborator

What about saving state of selection (show encoded or decoded URLs)? Will we save the state or just make users switch after every reinstall?
Also, do you see it as a switch for every individual request or general in list of items?

@MiSikora
Copy link
Contributor Author

MiSikora commented Feb 11, 2020

State is already saved to preferences on switching it. The main issue I see users have to switch it every time whenever their app is reinstalled. Which I assume would be often given the nature of the library.

@vbuberen
Copy link
Collaborator

vbuberen commented Feb 11, 2020

Sorry @MiSikora, I addressed this question to @cortinico. I also referred case about reinstalls, which happens quite a lot during development.

@cortinico
Copy link
Member

What about saving state of selection (show encoded or decoded URLs)? Will we save the state or just make users switch after every reinstall?

👍 for this approach. We can definitely use SharedPreferences or a table in the Room Database to store key value pairs with the current state.

Also, do you see it as a switch for every individual request or general in list of items?

If I would have to develop this feature just for my self I'll do it stateless and only in the "detail" view (e.g. you click on the encoded URL and you magically decode it). I can't think of any meaningful use case that would push me to develop this feature for the list view as well (at least in my experience). I can anyway understand how this can be helpful for some so it's probably worth to investigate/consider.

Please also note that this has implications on how we search through the list (like if you decode URLs in, you probably want to offer search through the decoded URLs as well?)

@vbuberen
Copy link
Collaborator

@MiSikora Sorry for not checking your code. Would like to come to some agreement regarding this feature first to not make you and me do some back-and-forth check/fix steps to not spend even more time. I will check as soon as we get the final vision of this feature flow.

@cortinico Makes sense. I agree that I would prefer to have this so-called switch encode/decode in each individual transaction, without saving states between sessions at all, since also can't remember scenario where I needed to decode all urls at once.
But I don't want to push too hard here, since we might be breaking the rule You are not your user.

Probably, we should do some sort of survey somewhere and get some meaningful responses.

@cortinico
Copy link
Member

without saving states between sessions at all, since also can't remember scenario where I needed to decode all urls at once.

Exactly. That's my strong point. I'd try to make this stateless to avoid complication and just provide a simple way to decode a URL, nothing more.

@vbuberen
Copy link
Collaborator

Ok, in such case let's skip the idea of doing survey etc. to not waste too much time.
@MiSikora could I ask you to update your code in a way that we will just have an options menu item in transaction details, which will switch encoded/decoded view without any state savings in SharedPreferences or db?

@vbuberen
Copy link
Collaborator

vbuberen commented Feb 14, 2020

Also, I promised to provide icon options for the toggle.
My suggestion is to use eye icons:
https://materialdesignicons.com/icon/eye
https://materialdesignicons.com/icon/eye-off

The meaning behind this icon something like convert item into human readable text or not.

Having no state might be fine and we will definitely look for the feedback from users and change the way this feature works depending on it.

@MiSikora
Copy link
Contributor Author

Hey, sorry for being silent. I'm a bit occupied lately. I will update the code to reflect discussion some time soon

@vbuberen
Copy link
Collaborator

@MiSikora Sure, take your time. We already appreciate you doing this PR and absolutely no pressure :)

@MiSikora
Copy link
Contributor Author

MiSikora commented Feb 15, 2020

I adapted the code. One thing that was not clear to me was HttpTransaction handling from the database. In the title LiveData it was assumed that the value might be null but in the transaction LiveData it was not. I changed signature to expliclitly say that the value might be null in the database and handled it in the code.

Below are the screenshots with the redesigned approach. By default URLs are not encoded. The eye icon is only visible when the overview tab is active.

Screenshot_1581764109
Screenshot_1581764114

@vbuberen
Copy link
Collaborator

Tried to run the sample app from your repo and wasn't able to see mentioned switch between encoding/decoding. Probably, I did something wrong?
Here is the sequence of actions:

  1. Added redirectTo("https://ascii.cl?parameter=%22Click+on+%27URL+Encode%27%21%22").enqueue(cb) into HttpBinClient.
  2. Opened this transaction in the sample app and saw that it is already decoded and clicks on eye icon do nothing.
    Here is how it looks like.
    chucker_encoding

@vbuberen
Copy link
Collaborator

Also, after trying it out myself it seems for me that we might improve the UX. Would be awesome if we could detect if url is encoded and show the eye icon only in such cases. For decoded or normal urls, like we have in our sample app show no eye icon, since I see no reason for willing to encode the url, which wasn't encoded originally.

@vbuberen
Copy link
Collaborator

And it would be awesome if you add this url for check encode/decode functionality into the sample app.

@MiSikora
Copy link
Contributor Author

Wow, I'm sorry about that. It doesn't seems to work for me now as well. I must have messed up something with pushing and reverting commits. I'll double check my local commits and also I'll apply your suggestion about not showing eye icon for URLs which do not require decoding.

@vbuberen
Copy link
Collaborator

No worries here. You are doing a great amount of work for Chucker and it is fine that there might be some typos.

@vbuberen vbuberen added this to the 3.2.0 milestone Feb 22, 2020
@MiSikora
Copy link
Contributor Author

MiSikora commented Feb 22, 2020

Ok, I rebased to accommodate for the LiveDataRecord in tests and fixed URL not displaying in encoded form. I also added mentioned suggestion about showin the icon.

@vbuberen
Copy link
Collaborator

Yeah, can confirm that everything works.

Wouldn't you mind adding redirectTo("https://ascii.cl?parameter=%22Click+on+%27URL+Encode%27%21%22").enqueue(cb) into HttpBinClient in Chucker sample app so we could easily test this functionality on a regular basis?

Also found out that since the eye icon is available only in Overview after switching state user will see encoded/decoded url on all 3 tabs. It is not a big deal andI believe it is not worth it to add some additional logic for handling this, but just be aware.

@MiSikora
Copy link
Contributor Author

Ok, I added the samples.

Not switching back on other tabs is intentional as there is share functionality which also uses encoding/decoding based on the users' preference. Should this feature not affect sharing?

path = ("/${url.pathSegments().joinToString("/")}${url.query()?.let { "?$it" } ?: ""}")
scheme = url.scheme()
val formattedUrl = FormattedUrl.fromHttpUrl(url, encoded = false)
this.url = formattedUrl.url
Copy link
Collaborator

Choose a reason for hiding this comment

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

Maybe we could rename url parameter for this function, so there will be no need in using this.url for this line?

Copy link
Contributor Author

Choose a reason for hiding this comment

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

How about httpUrl?

Copy link
Collaborator

Choose a reason for hiding this comment

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

Works for me 👍

val transactionTitle: LiveData<String> = RepositoryProvider.transaction()
.getTransaction(transactionId)
.combineLatest(encodeUrl) { transaction, encodeUrl ->
if (transaction != null) "${transaction.method} ${transaction.getFormattedPath(encode = encodeUrl)}" else ""
Copy link
Collaborator

Choose a reason for hiding this comment

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

This line seems too long.

Suggested change
if (transaction != null) "${transaction.method} ${transaction.getFormattedPath(encode = encodeUrl)}" else ""
if (transaction != null) {
"${transaction.method} ${transaction.getFormattedPath(encode = encodeUrl)}"
} else {
""
}

Copy link
Contributor Author

Choose a reason for hiding this comment

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

It looks long to me as well but it does not break Detekt rules. I can change it but I would rather rely on static analysis for such stuff. Otherwise what's the point of it?

Copy link
Collaborator

Choose a reason for hiding this comment

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

Ok, as you wish. I will update rules for static analysis soon.

@vbuberen
Copy link
Collaborator

Ok, I finished code review. Everything seems fine and works like it should.

@vbuberen
Copy link
Collaborator

vbuberen commented Feb 23, 2020

The only request for you - in the following PRs could you, please, wait for the whole review to pass, so the reviewer could leave all his comments? I think it is not that convenient that you do multiple commits with single line changes as well as for reviewer there is also need to refresh PR after every such commit.

This is not only more convenient, but also allows to not spin up CI too often and wait for checks to pass as well as burn some energy 😄

@MiSikora
Copy link
Contributor Author

MiSikora commented Feb 23, 2020

Sure. But wouldn't it be easier for both sides to leave all comments at once instead of one by one? There is such a feature in GH while doing CR, and this way I know when the code is reviewed as a whole. :)

@vbuberen
Copy link
Collaborator

I left comments one by one during one review. It took that much time, because I also checked some things on a device etc.
I would leave the final review anyway after my whole review.

@MiSikora
Copy link
Contributor Author

MiSikora commented Feb 23, 2020

Ok, makes sense. I'll try to be less eager with fixes between reviews in the future. :)

@vbuberen
Copy link
Collaborator

I would use one of the following options anyway, so you will see everything and it won't be one by one as you say.
Screen Shot 2020-02-23 at 12 18 49

@vbuberen
Copy link
Collaborator

Thanks for your time and for implementation of this feature request 👍 ❤️

@vbuberen vbuberen merged commit bee8813 into ChuckerTeam:develop Feb 23, 2020
@vbuberen
Copy link
Collaborator

And yeah, I won't use Add a single comment button in future as well. Thanks for pointing out.

@MiSikora MiSikora deleted the url-formatting branch February 23, 2020 10:29
vbuberen added a commit that referenced this pull request Apr 4, 2020
* Remove scroll flags (#210)

* Fix/gradle properties (#211)

* Allow Gradle parallel build
* Fix version name

* Fix for curl command (#214)

* Fix R8 minification crash on TransactionViewModel creation. (#219)

* Big resources renaming (#216)

* Fix clear action crash when application is dead (#222)

* Fix for crash on Save transaction action (#221)

* Show warning is transaction is null, fix crash in Save action

* Uncomment sample transactions

* Replace mulptiple returning with multiple throw due to detekt issue

* Add message into IOException, update string for request being not ready

* Fix for NPE in NotificationHelper (#223)

* Add additional check fo transaction being not null before getting its notificationText

* Extract transaction item from transactionBuffer

* ViewModel refactoring (#220)

* Update ViewModel dependency, refactor TransactionViewModel
* Dependencies clean up
* Switch to ViewModel on the main screen

* Fix depleting bytes from the response. (#226)

* Use HttpUrl instead of Uri for parsing URL data.
* Do not read image sources directly from the response.
* Simplify gzip logic.
* Move gzip checks from IoUtils to OkHttpUtils.
* Remove unused 'Response.hasBody()' extension.
* Update library/src/main/java/com/chuckerteam/chucker/internal/support/OkHttpUtils.kt

* Revert resource renaming (#227)

* Revert renaming

* Add changelgos for 3.1.2 (#230)

* Add missing  section to release 3.1.1 and 3.1.2 (#232)

* Update Github templates (#235)

* Update templates
* Remove redundant dot
* Remove default `no` from the checkbox

* Switch to platform defined HTTP code constants (#238)

* Add instruction for checkbox selection (#237)

* Fix HttpHeader serialization when obfuscation is enabled (#240)

* Update README (#243)

* Add Action to validate the Gradle Wrapper (#245)

* Gradle to 6.2 (#247)

* Do not refresh transaction when it is not being affected. (#246)

* Do not refresh transaction when it is not being affected.
* Use correct null-aware comparison for HttpTransaction.

* Add switching between encoded and decoded URL formats. (#233)

* Add switching between encoded and decoded URL formats.
* Make URL encoding transaction specific.
* Change test name for upcoming #244 PR.
* Use LiveDataRecord for combineLatest tests.
* Properly switch encoding and decoding URL.
* Show encoding icon only when it is viable.
* Add encoded URL samples to HttpBinClient.
* Avoid using 'this' scoping mechanism for URL.

* Fix typo in feature request comment (#251)

* RTL Support (#208)

* Remove ltr forcing and replace ScrollView in Overview
* Replace Overview layout, add rtl support for it
* Add textDirection and textAlignment property for API 21+
* Fix host textview constraints
* Replace android:src with app:srcCompat
* Update ids for layouts to avoid clashes
* Update Material components to stable
* Remove supportsRTL tag from Manifest, update Gradle plugin
* Styles update
* Remove supportsRTL from library manifest
* Revert usage of supportVectorDrawables to avoid crashes on APIs 16-19 due to notifications icons
* Fix lint issue with vector drawable

* Response search fixes (#256)

* Fix response search to be case insensitive
* Add minimum required symbols
* Fix invalid options menu in response fragment

* Feature/tls info (#252)

* Add UI for TLS info

* Implement logic for retrieving TLS info

* Address code review feedback

* Switch views to ViewBinding (#253)

* Switch to ViewBinding in activities
* Switch to ViewBinding in ErrorsAdapter, add formattedDate field into Throwable models
* Transaction adapter switch to ViewBinding
* Remove variable for formatted date from models
* Switch to ViewBinding in TransactionPayloadAdapter
* Switch to ViewBinding in TransactionPaayload and TransactionOverviewFragments
* Switch list fragments to ViewBinding
* Fix link for tutorial opening
* Rename views
* Address code review feedback
* Hide SSL field if isSSL is null

* Libs update (#260)

* Update tools versions
* JUnit update

* Feature/truth (#258)

* Add Truth, update part of test classes
* Convert other tests to use Truth, fix date parser test
* Add Truth assertions to FormatUtilsTest, fix ktlint issue
* Update assertions to a proper syntax

* Add missing ThrowableViewModel (#257)

* Add Error ViewModel, update title in TransactionActivity in onCreate
* Switch from errors to throwable naming to have a uniform naming style
* Rename toolbar title

* Migrating from Travis to GH Actions (#262)

* Setup GH Actions

* Run only on Linux

* Remove Travis File

* Run only gradlew directly

* Update targetSDK and Kotlin (#264)

* Add stale builds canceller (#265)

* Add filters
* Update Gradle wrapper validation workflow
* Update pre-merge workflow

* Fixed various Lints (#268)

* fixed typos
* fixed KDocs

* Replace Travis badge with GH Actions badge (#269)

* Remove redundant JvmName (#274)

* Fix margins and paddings for payload items (#277)

* Add selective interception. (#279)

* Add selective interception.
* Update README.md.
* Align formatting in README with other points.
* Avoid header name duplication.
* Strip interception header also in the no-op library.

* UX improvements (#275)

* Add icon for non-https transactions
* Update secondary color to be more contrast
* Simplify protocol resources setting

* Add tests to format utils (#281)

* add tests to format utils

* fixes after code review

* formatting fix

Co-authored-by: adammasyk <adam.masyk@programisci.eu>

* format utils test refactor (#285)

* format utils test refactor
* share text test refactor

* Migrate to Kotlin Coroutines (#270)

* Add coroutine as a dependency in build.gradle
* Migrate AsyncTasks to kotlin coroutines
* Migrate executors with the coroutines in repositories

* Multi cast upstream response for Chucker consumption. (#267)

* Multi cast response upstream for Chucker consumption.

* Read buffer prefix before potentially gunzipping it.

* Inform Chucker about unprocessed responses.

* Simplify multi casting logic.

* Move read offset to a variable.

* Inline one-line method.

* Give better control over TeeSource read results.

* Add documentation to TeeSource.

* Close side channel when capacity is exceeded.

Co-authored-by: Volodymyr Buberenko <vbuberen@users.noreply.github.com>

* Remove unnecessary mock method. (#289)

* removed redundant Gson configurations (#290)

* increased test coverage for format utils (#291)

Co-authored-by: Karthik R <karthr@paypal.com>

* added few test cases for json formatting (#295)

* Properly handle unexhausted network responses (#288)

* Handle properly not consumed upstream body.
* Handle IO issues while reading from file.

* Update dependencies (#296)

* Update depencies

* Update OkHttp to 3.12.10

* Handle empty and missing response bodies. (#250)

* Add failing test cases.
* Remove unused const.
* Gzip response body if it was gunzipped.
* Add test cases for regular bodies in Chucker.
* Fix rule formatting.
* Use proper name for application interceptor.
* Return original response downstream.
* Account for no content with gzip encoding.
* Use Truth for Chucker tests.
* Honor empty plaintext bodies.
* Revert changes to HttpBinClient.
* Update library/src/test/java/com/chuckerteam/chucker/ChuckerInterceptorDelegate.kt
* Update library/src/main/java/com/chuckerteam/chucker/internal/support/OkHttpUtils.kt
Co-authored-by: Nicola Corti <corti.nico@gmail.com>
Co-authored-by: Volodymyr Buberenko <vbuberen@users.noreply.github.com>

* Add hasFixed size to RecyclerViews (#297)

* Detekt to 1.7.4 (#299)

* Revert "Add selective interception. (#279)" (#300)

This reverts commit d14ed64.

* Prepare 3.2.0 (#298)

* Update versions info

* Update Changelog

* Fix links and update date

Co-authored-by: Michał Sikora <michalsikora90@gmail.com>
Co-authored-by: Nicola Corti <corti.nico@gmail.com>
Co-authored-by: Sergey Chelombitko <119192+technoir42@users.noreply.github.com>
Co-authored-by: Michał Sikora <me@mehow.io>
Co-authored-by: Hitanshu Dhawan <hitanshudhawan1996@gmail.com>
Co-authored-by: adammasyk <masiol91@gmail.com>
Co-authored-by: adammasyk <adam.masyk@programisci.eu>
Co-authored-by: Nikhil Chaudhari <nikhyl777@gmail.com>
Co-authored-by: karthik rk <karthik.rk18@gmail.com>
Co-authored-by: Karthik R <karthr@paypal.com>
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

URL Encoder/Decoder
3 participants