Compare commits
	
		
			40 Commits
		
	
	
		
			v0.3.1
			...
			3dd609864b
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 3dd609864b | |||
| 13d3ed85ad | |||
| b23c36a25f | |||
|  | 87d8cfd2e5 | ||
|  | 85b869d28b | ||
| 04f842cdfe | |||
| c72fcb3a49 | |||
| 9c819ad4c2 | |||
| 6d5439a00a | |||
| 6d0bd5906a | |||
|  | f3c0793a88 | ||
|  | e99be27300 | ||
|  | 480994bfa9 | ||
| e7ad54b0a4 | |||
| 3f67ba1895 | |||
| 0f441b26d1 | |||
| f9d35d01c6 | |||
| c183098b7d | |||
| e4ff32cc94 | |||
| c4757ce109 | |||
| 8e2fc7264b | |||
| aa7856ea7b | |||
| 8b25a1519e | |||
| cfa43451d8 | |||
| cf2fc0f54f | |||
| bc6261d444 | |||
| e1266ebf64 | |||
| 989e7079a5 | |||
| edfd4e2e06 | |||
| 1c8d5146fb | |||
| 0208cbd960 | |||
| 08dffb3e3a | |||
| 680e2dcba2 | |||
| 6e63797167 | |||
| 787a2b9636 | |||
| 39198a40d6 | |||
| 26eb2c104a | |||
| 041ad9e393 | |||
| f9e9235726 | |||
| f8ea8a9276 | 
							
								
								
									
										22
									
								
								.drone.yml
									
									
									
									
									
								
							
							
						
						
									
										22
									
								
								.drone.yml
									
									
									
									
									
								
							| @@ -15,13 +15,24 @@ steps: | ||||
|       - push | ||||
|       - pull_request | ||||
|       - tag | ||||
|   - name: build-dev | ||||
|     image: golang:1.18.3-alpine3.16 | ||||
|     commands: | ||||
|       - "apk add --update --no-cache make" | ||||
|       - "make build" | ||||
|     when: | ||||
|       event: | ||||
|       - push | ||||
|     resources: | ||||
|       limits: | ||||
|         cpu: 1000 | ||||
|         memory: 1024MiB | ||||
|   - name: build | ||||
|     image: goreleaser/goreleaser | ||||
|     commands: | ||||
|       - goreleaser build --snapshot | ||||
|     when: | ||||
|       event: | ||||
|       - push | ||||
|       - pull_request | ||||
|     resources: | ||||
|       limits: | ||||
| @@ -41,7 +52,16 @@ steps: | ||||
|     environment: | ||||
|       GITEA_TOKEN: | ||||
|         from_secret: gitea_token | ||||
|       GPG_PRIVATE_KEY: | ||||
|         from_secret: GPG_PRIVATE_KEY | ||||
|       GPG_FINGERPRINT: | ||||
|         from_secret: GPG_FINGERPRINT | ||||
|       GPG_PRIVATE_KEY_BASE64: | ||||
|         from_secret: GPG_PRIVATE_KEY_BASE64 | ||||
|     commands: | ||||
|       - apk add gpg-agent | ||||
|       - gpg-agent --daemon --default-cache-ttl 7200 | ||||
|       - echo $GPG_PRIVATE_KEY_BASE64 | base64 -d | gpg --import --batch --no-tty | ||||
|       - goreleaser release | ||||
|     resources: | ||||
|       limits: | ||||
|   | ||||
							
								
								
									
										28
									
								
								.github/ISSUE_TEMPLATE/bug_report.md
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										28
									
								
								.github/ISSUE_TEMPLATE/bug_report.md
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,28 @@ | ||||
| --- | ||||
| name: Bug report | ||||
| about: Create a report to improve the provider | ||||
| title: '' | ||||
| labels: 'bug' | ||||
| assignees: '' | ||||
| --- | ||||
|  | ||||
| **Describe the bug** | ||||
| A clear and concise description of what the bug is. | ||||
|  | ||||
| **To Reproduce** | ||||
| Steps to reproduce the behavior: | ||||
|  | ||||
| **Expected behavior** | ||||
| A clear and concise description of what you expected to happen. | ||||
|  | ||||
| **Log Output** | ||||
| If applicable, add logs to help explain your problem. | ||||
|  | ||||
| **Additional Data** | ||||
| Important for reproducability. | ||||
|  | ||||
| - Terraform Version | ||||
|  | ||||
| - Operating System | ||||
|  | ||||
| - Provider Version | ||||
							
								
								
									
										20
									
								
								.github/ISSUE_TEMPLATE/feature_request.md
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										20
									
								
								.github/ISSUE_TEMPLATE/feature_request.md
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,20 @@ | ||||
| --- | ||||
| name: Feature request | ||||
| about: Suggest an idea for this provider | ||||
| title: '' | ||||
| labels: 'enhancement' | ||||
| assignees: '' | ||||
|  | ||||
| --- | ||||
|  | ||||
| **Is your feature request related to a problem? Please describe.** | ||||
| A clear and concise description of what the problem is. Ex. I'm always frustrated when [...] | ||||
|  | ||||
| **Describe the solution you'd like** | ||||
| A clear and concise description of what you want to happen. | ||||
|  | ||||
| **Describe alternatives you've considered** | ||||
| A clear and concise description of any alternative solutions or features you've considered. | ||||
|  | ||||
| **Additional context** | ||||
| Add any other context or screenshots about the feature request here. | ||||
							
								
								
									
										9
									
								
								.github/workflows/release.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										9
									
								
								.github/workflows/release.yml
									
									
									
									
										vendored
									
									
								
							| @@ -32,11 +32,10 @@ jobs: | ||||
|       - | ||||
|         name: Import GPG key | ||||
|         id: import_gpg | ||||
|         uses: hashicorp/ghaction-import-gpg@v2.1.0 | ||||
|         env: | ||||
|           # These secrets will need to be configured for the repository: | ||||
|           GPG_PRIVATE_KEY: ${{ secrets.GPG_PRIVATE_KEY }} | ||||
|           PASSPHRASE: ${{ secrets.PASSPHRASE }} | ||||
|         uses: crazy-max/ghaction-import-gpg@v5.0.0 | ||||
|         with: | ||||
|           gpg_private_key: ${{ secrets.GPG_PRIVATE_KEY }} | ||||
|           passphrase: ${{ secrets.PASSPHRASE }} | ||||
|       - | ||||
|         name: Run GoReleaser | ||||
|         uses: goreleaser/goreleaser-action@v3.0.0 | ||||
|   | ||||
| @@ -4,6 +4,9 @@ before: | ||||
|   hooks: | ||||
|     # this is just an example and not a requirement for provider building/publishing | ||||
|     - go mod tidy | ||||
| gitea_urls: | ||||
|   api: https://git.uploadfilter24.eu/api/v1/ | ||||
|   download: https://git.uploadfilter24.eu | ||||
| builds: | ||||
| - env: | ||||
|     # goreleaser does not work with CGO, it could also complicate | ||||
| @@ -51,6 +54,9 @@ signs: | ||||
|       - "--detach-sign" | ||||
|       - "${artifact}" | ||||
| release: | ||||
|   gitea: | ||||
|     owner: lerentis | ||||
|     name: terraform-provider-gitea | ||||
|   extra_files: | ||||
|     - glob: 'terraform-registry-manifest.json' | ||||
|       name_template: '{{ .ProjectName }}_{{ .Version }}_manifest.json' | ||||
|   | ||||
							
								
								
									
										128
									
								
								CODE_OF_CONDUCT.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										128
									
								
								CODE_OF_CONDUCT.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,128 @@ | ||||
| # Contributor Covenant Code of Conduct | ||||
|  | ||||
| ## Our Pledge | ||||
|  | ||||
| We as members, contributors, and leaders pledge to make participation in our | ||||
| community a harassment-free experience for everyone, regardless of age, body | ||||
| size, visible or invisible disability, ethnicity, sex characteristics, gender | ||||
| identity and expression, level of experience, education, socio-economic status, | ||||
| nationality, personal appearance, race, religion, or sexual identity | ||||
| and orientation. | ||||
|  | ||||
| We pledge to act and interact in ways that contribute to an open, welcoming, | ||||
| diverse, inclusive, and healthy community. | ||||
|  | ||||
| ## Our Standards | ||||
|  | ||||
| Examples of behavior that contributes to a positive environment for our | ||||
| community include: | ||||
|  | ||||
| * Demonstrating empathy and kindness toward other people | ||||
| * Being respectful of differing opinions, viewpoints, and experiences | ||||
| * Giving and gracefully accepting constructive feedback | ||||
| * Accepting responsibility and apologizing to those affected by our mistakes, | ||||
|   and learning from the experience | ||||
| * Focusing on what is best not just for us as individuals, but for the | ||||
|   overall community | ||||
|  | ||||
| Examples of unacceptable behavior include: | ||||
|  | ||||
| * The use of sexualized language or imagery, and sexual attention or | ||||
|   advances of any kind | ||||
| * Trolling, insulting or derogatory comments, and personal or political attacks | ||||
| * Public or private harassment | ||||
| * Publishing others' private information, such as a physical or email | ||||
|   address, without their explicit permission | ||||
| * Other conduct which could reasonably be considered inappropriate in a | ||||
|   professional setting | ||||
|  | ||||
| ## Enforcement Responsibilities | ||||
|  | ||||
| Community leaders are responsible for clarifying and enforcing our standards of | ||||
| acceptable behavior and will take appropriate and fair corrective action in | ||||
| response to any behavior that they deem inappropriate, threatening, offensive, | ||||
| or harmful. | ||||
|  | ||||
| Community leaders have the right and responsibility to remove, edit, or reject | ||||
| comments, commits, code, wiki edits, issues, and other contributions that are | ||||
| not aligned to this Code of Conduct, and will communicate reasons for moderation | ||||
| decisions when appropriate. | ||||
|  | ||||
| ## Scope | ||||
|  | ||||
| This Code of Conduct applies within all community spaces, and also applies when | ||||
| an individual is officially representing the community in public spaces. | ||||
| Examples of representing our community include using an official e-mail address, | ||||
| posting via an official social media account, or acting as an appointed | ||||
| representative at an online or offline event. | ||||
|  | ||||
| ## Enforcement | ||||
|  | ||||
| Instances of abusive, harassing, or otherwise unacceptable behavior may be | ||||
| reported to the community leaders responsible for enforcement at | ||||
| `lerentis at uploadfilter24 dot eu`. | ||||
| All complaints will be reviewed and investigated promptly and fairly. | ||||
|  | ||||
| All community leaders are obligated to respect the privacy and security of the | ||||
| reporter of any incident. | ||||
|  | ||||
| ## Enforcement Guidelines | ||||
|  | ||||
| Community leaders will follow these Community Impact Guidelines in determining | ||||
| the consequences for any action they deem in violation of this Code of Conduct: | ||||
|  | ||||
| ### 1. Correction | ||||
|  | ||||
| **Community Impact**: Use of inappropriate language or other behavior deemed | ||||
| unprofessional or unwelcome in the community. | ||||
|  | ||||
| **Consequence**: A private, written warning from community leaders, providing | ||||
| clarity around the nature of the violation and an explanation of why the | ||||
| behavior was inappropriate. A public apology may be requested. | ||||
|  | ||||
| ### 2. Warning | ||||
|  | ||||
| **Community Impact**: A violation through a single incident or series | ||||
| of actions. | ||||
|  | ||||
| **Consequence**: A warning with consequences for continued behavior. No | ||||
| interaction with the people involved, including unsolicited interaction with | ||||
| those enforcing the Code of Conduct, for a specified period of time. This | ||||
| includes avoiding interactions in community spaces as well as external channels | ||||
| like social media. Violating these terms may lead to a temporary or | ||||
| permanent ban. | ||||
|  | ||||
| ### 3. Temporary Ban | ||||
|  | ||||
| **Community Impact**: A serious violation of community standards, including | ||||
| sustained inappropriate behavior. | ||||
|  | ||||
| **Consequence**: A temporary ban from any sort of interaction or public | ||||
| communication with the community for a specified period of time. No public or | ||||
| private interaction with the people involved, including unsolicited interaction | ||||
| with those enforcing the Code of Conduct, is allowed during this period. | ||||
| Violating these terms may lead to a permanent ban. | ||||
|  | ||||
| ### 4. Permanent Ban | ||||
|  | ||||
| **Community Impact**: Demonstrating a pattern of violation of community | ||||
| standards, including sustained inappropriate behavior,  harassment of an | ||||
| individual, or aggression toward or disparagement of classes of individuals. | ||||
|  | ||||
| **Consequence**: A permanent ban from any sort of public interaction within | ||||
| the community. | ||||
|  | ||||
| ## Attribution | ||||
|  | ||||
| This Code of Conduct is adapted from the [Contributor Covenant][homepage], | ||||
| version 2.0, available at | ||||
| https://www.contributor-covenant.org/version/2/0/code_of_conduct.html. | ||||
|  | ||||
| Community Impact Guidelines were inspired by [Mozilla's code of conduct | ||||
| enforcement ladder](https://github.com/mozilla/diversity). | ||||
|  | ||||
| [homepage]: https://www.contributor-covenant.org | ||||
|  | ||||
| For answers to common questions about this code of conduct, see the FAQ at | ||||
| https://www.contributor-covenant.org/faq. Translations are available at | ||||
| https://www.contributor-covenant.org/translations. | ||||
							
								
								
									
										22
									
								
								LICENSE
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										22
									
								
								LICENSE
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,22 @@ | ||||
| The MIT License (MIT) | ||||
|  | ||||
| Copyright (c) 2022 lerentis, https://git.uploadfilter24.eu/lerentis | ||||
|  | ||||
| Permission is hereby granted, free of charge, to any person obtaining | ||||
| a copy of this software and associated documentation files (the | ||||
| "Software"), to deal in the Software without restriction, including | ||||
| without limitation the rights to use, copy, modify, merge, publish, | ||||
| distribute, sublicense, and/or sell copies of the Software, and to | ||||
| permit persons to whom the Software is furnished to do so, subject to | ||||
| the following conditions: | ||||
|  | ||||
| The above copyright notice and this permission notice shall be | ||||
| included in all copies or substantial portions of the Software. | ||||
|  | ||||
| THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, | ||||
| EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF | ||||
| MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND | ||||
| NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE | ||||
| LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION | ||||
| OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION | ||||
| WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. | ||||
							
								
								
									
										4
									
								
								Makefile
									
									
									
									
									
								
							
							
						
						
									
										4
									
								
								Makefile
									
									
									
									
									
								
							| @@ -3,7 +3,7 @@ GOFMT_FILES?=$$(find . -name '*.go' |grep -v vendor) | ||||
|  | ||||
| GOFMT ?= gofmt -s | ||||
|  | ||||
| VERSION = 0.3.0 | ||||
| VERSION = 0.11.1 | ||||
|  | ||||
| test: fmt-check | ||||
| 	go test -i $(TEST) || exit 1 | ||||
| @@ -37,3 +37,5 @@ install: build | ||||
| 	@echo ~/.terraform.d/plugins/terraform.local/lerentis/gitea/${VERSION}/linux_amd64/terraform-provider-gitea_${VERSION} | ||||
| 	@mkdir -p ~/.terraform.d/plugins/terraform.local/lerentis/gitea/${VERSION}/linux_amd64 | ||||
| 	@mv terraform-provider-gitea_${VERSION} ~/.terraform.d/plugins/terraform.local/lerentis/gitea/${VERSION}/linux_amd64/terraform-provider-gitea_${VERSION} | ||||
| doc: | ||||
| 	tfplugindocs | ||||
|   | ||||
| @@ -17,7 +17,7 @@ terraform { | ||||
|   required_providers { | ||||
|     gitea = { | ||||
|       source = "Lerentis/gitea" | ||||
|       version = "0.3.0" | ||||
|       version = "0.11.1" | ||||
|     } | ||||
|   } | ||||
| } | ||||
|   | ||||
| @@ -17,7 +17,7 @@ terraform { | ||||
|   required_providers { | ||||
|     gitea = { | ||||
|       source = "Lerentis/gitea" | ||||
|       version = "0.3.0" | ||||
|       version = "0.11.1" | ||||
|     } | ||||
|   } | ||||
| } | ||||
|   | ||||
							
								
								
									
										61
									
								
								docs/resources/fork.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										61
									
								
								docs/resources/fork.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,61 @@ | ||||
| --- | ||||
| # generated by https://github.com/hashicorp/terraform-plugin-docs | ||||
| page_title: "gitea_fork Resource - terraform-provider-gitea" | ||||
| subcategory: "" | ||||
| description: |- | ||||
|   gitea_fork manages repository fork to the current user or an organisation | ||||
|   Forking a repository to a dedicated user is currently unsupported | ||||
|   Creating a fork using this resource without an organisation will create the fork in the executors name | ||||
| --- | ||||
|  | ||||
| # gitea_fork (Resource) | ||||
|  | ||||
| `gitea_fork` manages repository fork to the current user or an organisation | ||||
| Forking a repository to a dedicated user is currently unsupported | ||||
| Creating a fork using this resource without an organisation will create the fork in the executors name | ||||
|  | ||||
| ## Example Usage | ||||
|  | ||||
| ```terraform | ||||
| resource "gitea_org" "org1" { | ||||
|   name = "org1" | ||||
| } | ||||
|  | ||||
| resource "gitea_org" "org2" { | ||||
|   name = "org2" | ||||
| } | ||||
|  | ||||
| resource "gitea_repository" "repo1_in_org1" { | ||||
|   username = gitea_org.org1.name | ||||
|   name     = "repo1-in-org1" | ||||
| } | ||||
|  | ||||
| resource "gitea_fork" "user_fork_of_repo1_in_org1" { | ||||
|   owner = gitea_org.org1.name | ||||
|   repo  = gitea_repository.repo1_in_org1.name | ||||
| } | ||||
|  | ||||
| resource "gitea_fork" "org2_fork_of_repo1_in_org1" { | ||||
|   owner        = gitea_org.org1.name | ||||
|   repo         = gitea_repository.repo1_in_org1.name | ||||
|   organization = gitea_org.org2.name | ||||
| } | ||||
| ``` | ||||
|  | ||||
| <!-- schema generated by tfplugindocs --> | ||||
| ## Schema | ||||
|  | ||||
| ### Required | ||||
|  | ||||
| - `owner` (String) The owner or owning organization of the repository to fork | ||||
| - `repo` (String) The name of the repository to fork | ||||
|  | ||||
| ### Optional | ||||
|  | ||||
| - `organization` (String) The organization that owns the forked repo | ||||
|  | ||||
| ### Read-Only | ||||
|  | ||||
| - `id` (String) The ID of this resource. | ||||
|  | ||||
|  | ||||
							
								
								
									
										55
									
								
								docs/resources/git_hook.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										55
									
								
								docs/resources/git_hook.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,55 @@ | ||||
| --- | ||||
| # generated by https://github.com/hashicorp/terraform-plugin-docs | ||||
| page_title: "gitea_git_hook Resource - terraform-provider-gitea" | ||||
| subcategory: "" | ||||
| description: |- | ||||
|   gitea_git_hook manages git hooks on a repository. | ||||
|   import is currently not supported | ||||
|   WARNING: using this resource requires to enable server side hookswhich are known to cause security issues https://github.com/go-gitea/gitea/pull/13058! | ||||
|   if you want to procede, you need to enable server side hooks as stated here https://docs.gitea.io/en-us/config-cheat-sheet/#security-security | ||||
| --- | ||||
|  | ||||
| # gitea_git_hook (Resource) | ||||
|  | ||||
| `gitea_git_hook` manages git hooks on a repository. | ||||
| import is currently not supported | ||||
|  | ||||
| WARNING: using this resource requires to enable server side hookswhich are known to cause [security issues](https://github.com/go-gitea/gitea/pull/13058)! | ||||
|  | ||||
| if you want to procede, you need to enable server side hooks as stated [here](https://docs.gitea.io/en-us/config-cheat-sheet/#security-security) | ||||
|  | ||||
| ## Example Usage | ||||
|  | ||||
| ```terraform | ||||
| resource "gitea_org" "test_org" { | ||||
|   name = "test-org" | ||||
| } | ||||
|  | ||||
| resource "gitea_repository" "org_repo" { | ||||
|   username = gitea_org.test_org.name | ||||
|   name     = "org-test-repo" | ||||
| } | ||||
|  | ||||
| resource "gitea_git_hook" "org_repo_post_receive" { | ||||
|   name    = "post-receive" | ||||
|   user    = gitea_org.test_org.name | ||||
|   repo    = gitea_repository.org_repo.name | ||||
|   content = file("${path.module}/post-receive.sh") | ||||
| } | ||||
| ``` | ||||
|  | ||||
| <!-- schema generated by tfplugindocs --> | ||||
| ## Schema | ||||
|  | ||||
| ### Required | ||||
|  | ||||
| - `content` (String) Content of the git hook | ||||
| - `name` (String) Name of the git hook to configure | ||||
| - `repo` (String) The repository that this hook belongs too. | ||||
| - `user` (String) The user (or organisation) owning the repo this hook belongs too | ||||
|  | ||||
| ### Read-Only | ||||
|  | ||||
| - `id` (String) The ID of this resource. | ||||
|  | ||||
|  | ||||
							
								
								
									
										52
									
								
								docs/resources/public_key.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										52
									
								
								docs/resources/public_key.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,52 @@ | ||||
| --- | ||||
| # generated by https://github.com/hashicorp/terraform-plugin-docs | ||||
| page_title: "gitea_public_key Resource - terraform-provider-gitea" | ||||
| subcategory: "" | ||||
| description: |- | ||||
|   gitea_public_key manages ssh key that are associated with users. | ||||
| --- | ||||
|  | ||||
| # gitea_public_key (Resource) | ||||
|  | ||||
| `gitea_public_key` manages ssh key that are associated with users. | ||||
|  | ||||
| ## Example Usage | ||||
|  | ||||
| ```terraform | ||||
| resource "gitea_user" "test" { | ||||
|   username             = "test" | ||||
|   login_name           = "test" | ||||
|   password             = "Geheim1!" | ||||
|   email                = "test@user.dev" | ||||
|   must_change_password = false | ||||
| } | ||||
|  | ||||
|  | ||||
| resource "gitea_public_key" "test_user_key" { | ||||
|   title     = "test" | ||||
|   key       = file("${path.module}/id_ed25519.pub") | ||||
|   username  = gitea_user.test.username | ||||
| } | ||||
| ``` | ||||
|  | ||||
| <!-- schema generated by tfplugindocs --> | ||||
| ## Schema | ||||
|  | ||||
| ### Required | ||||
|  | ||||
| - `key` (String, Sensitive) An armored SSH key to add | ||||
| - `title` (String) Title of the key to add | ||||
| - `username` (String) User to associate with the added key | ||||
|  | ||||
| ### Optional | ||||
|  | ||||
| - `read_only` (Boolean) Describe if the key has only read access or read/write | ||||
|  | ||||
| ### Read-Only | ||||
|  | ||||
| - `created` (String) | ||||
| - `fingerprint` (String) | ||||
| - `id` (String) The ID of this resource. | ||||
| - `type` (String) | ||||
|  | ||||
|  | ||||
| @@ -72,7 +72,8 @@ Need to exist in the gitea instance | ||||
| Need to exist in the gitea instance | ||||
| - `license` (String) The license under which the source code of this repository should be. | ||||
| Need to exist in the gitea instance | ||||
| - `migration_clone_addresse` (String) | ||||
| - `migration_clone_address` (String) | ||||
| - `migration_clone_addresse` (String) DEPRECATED in favor of `migration_clone_address` | ||||
| - `migration_issue_labels` (Boolean) | ||||
| - `migration_lfs` (Boolean) | ||||
| - `migration_lfs_endpoint` (String) | ||||
| @@ -92,7 +93,10 @@ Need to exist in the gitea instance | ||||
| ### Read-Only | ||||
|  | ||||
| - `created` (String) | ||||
| - `clone_url` (String) | ||||
| - `html_url` (String) | ||||
| - `id` (String) The ID of this resource. | ||||
| - `ssh_url` (String) | ||||
| - `permission_admin` (Boolean) | ||||
| - `permission_pull` (Boolean) | ||||
| - `permission_push` (Boolean) | ||||
|   | ||||
							
								
								
									
										62
									
								
								docs/resources/team.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										62
									
								
								docs/resources/team.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,62 @@ | ||||
| --- | ||||
| # generated by https://github.com/hashicorp/terraform-plugin-docs | ||||
| page_title: "gitea_team Resource - terraform-provider-gitea" | ||||
| subcategory: "" | ||||
| description: |- | ||||
|   gitea_team manages Team that are part of an organisation. | ||||
| --- | ||||
|  | ||||
| # gitea_team (Resource) | ||||
|  | ||||
| `gitea_team` manages Team that are part of an organisation. | ||||
|  | ||||
| ## Example Usage | ||||
|  | ||||
| ```terraform | ||||
| resource "gitea_org" "test_org" { | ||||
|   name = "test-org" | ||||
| } | ||||
|  | ||||
| resource "gitea_user" "test" { | ||||
|   username             = "test" | ||||
|   login_name           = "test" | ||||
|   password             = "Geheim1!" | ||||
|   email                = "test@user.dev" | ||||
|   must_change_password = false | ||||
|   admin                = true | ||||
| } | ||||
|  | ||||
|  | ||||
| resource "gitea_team" "test_team" { | ||||
|   name         = "Devs" | ||||
|   organisation = gitea_org.test_org.name | ||||
|   description  = "Devs of Test Org" | ||||
|   permission   = "write" | ||||
|   members      = [gitea_user.test.username] | ||||
| } | ||||
| ``` | ||||
|  | ||||
| <!-- schema generated by tfplugindocs --> | ||||
| ## Schema | ||||
|  | ||||
| ### Required | ||||
|  | ||||
| - `name` (String) Name of the Team | ||||
| - `organisation` (String) The organisation which this Team is part of. | ||||
|  | ||||
| ### Optional | ||||
|  | ||||
| - `can_create_repos` (Boolean) Flag if the Teams members should be able to create Rpositories in the Organisation | ||||
| - `description` (String) Description of the Team | ||||
| - `include_all_repositories` (Boolean) Flag if the Teams members should have access to all Repositories in the Organisation | ||||
| - `members` (List of String) List of Users that should be part of this team | ||||
| - `permission` (String) Permissions associated with this Team | ||||
| Can be `none`, `read`, `write`, `admin` or `owner` | ||||
| - `units` (String) List of types of Repositories that should be allowed to be created from Team members. | ||||
| Can be `repo.code`, `repo.issues`, `repo.ext_issues`, `repo.wiki`, `repo.pulls`, `repo.releases`, `repo.projects` and/or `repo.ext_wiki` | ||||
|  | ||||
| ### Read-Only | ||||
|  | ||||
| - `id` (String) The ID of this resource. | ||||
|  | ||||
|  | ||||
							
								
								
									
										67
									
								
								docs/resources/token.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										67
									
								
								docs/resources/token.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,67 @@ | ||||
| --- | ||||
| # generated by https://github.com/hashicorp/terraform-plugin-docs | ||||
| page_title: "gitea_token Resource - terraform-provider-gitea" | ||||
| subcategory: "" | ||||
| description: |- | ||||
|   gitea_token manages gitea Access Tokens. | ||||
|   Due to upstream limitations (see https://gitea.com/gitea/go-sdk/issues/610) this resource | ||||
|   can only be used with username/password provider configuration. | ||||
|   WARNING: | ||||
|   Tokens will be stored in the terraform state! | ||||
| --- | ||||
|  | ||||
| # gitea_token (Resource) | ||||
|  | ||||
| `gitea_token` manages gitea Access Tokens. | ||||
|  | ||||
| Due to upstream limitations (see https://gitea.com/gitea/go-sdk/issues/610) this resource | ||||
| can only be used with username/password provider configuration. | ||||
|  | ||||
| WARNING: | ||||
| Tokens will be stored in the terraform state! | ||||
|  | ||||
| ## Example Usage | ||||
|  | ||||
| ```terraform | ||||
| provider "gitea" { | ||||
|   base_url = var.gitea_url | ||||
|   # Token Auth can not be used with this resource | ||||
|   username = var.gitea_username | ||||
|   password = var.gitea_password | ||||
| } | ||||
|  | ||||
| resource "gitea_user" "test" { | ||||
|   username             = "test" | ||||
|   login_name           = "test" | ||||
|   password             = "Geheim1!" | ||||
|   email                = "test@user.dev" | ||||
|   must_change_password = false | ||||
|   admin                = true | ||||
| } | ||||
|  | ||||
| resource "gitea_token" "test_token" { | ||||
|   username = resource.gitea_user.test.username | ||||
|   name     = "test-token" | ||||
| } | ||||
|  | ||||
| output "token" { | ||||
|   value     = resource.gitea_token.test_token.token | ||||
|   sensitive = true | ||||
| } | ||||
| ``` | ||||
|  | ||||
| <!-- schema generated by tfplugindocs --> | ||||
| ## Schema | ||||
|  | ||||
| ### Required | ||||
|  | ||||
| - `name` (String) The name of the Access Token | ||||
| - `username` (String) The owner of the Access Token | ||||
|  | ||||
| ### Read-Only | ||||
|  | ||||
| - `id` (String) The ID of this resource. | ||||
| - `last_eight` (String) | ||||
| - `token` (String, Sensitive) The actual Access Token | ||||
|  | ||||
|  | ||||
							
								
								
									
										60
									
								
								docs/resources/user.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										60
									
								
								docs/resources/user.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,60 @@ | ||||
| --- | ||||
| # generated by https://github.com/hashicorp/terraform-plugin-docs | ||||
| page_title: "gitea_user Resource - terraform-provider-gitea" | ||||
| subcategory: "" | ||||
| description: |- | ||||
|   gitea_user manages a native gitea user. | ||||
|   If you are using OIDC or other kinds of authentication mechanisms you can still try to managessh keys or other ressources this way | ||||
| --- | ||||
|  | ||||
| # gitea_user (Resource) | ||||
|  | ||||
| `gitea_user` manages a native gitea user. | ||||
|  | ||||
| If you are using OIDC or other kinds of authentication mechanisms you can still try to managessh keys or other ressources this way | ||||
|  | ||||
| ## Example Usage | ||||
|  | ||||
| ```terraform | ||||
| resource "gitea_user" "test" { | ||||
|   username             = "test" | ||||
|   login_name           = "test" | ||||
|   password             = "Geheim1!" | ||||
|   email                = "test@user.dev" | ||||
|   must_change_password = false | ||||
| } | ||||
| ``` | ||||
|  | ||||
| <!-- schema generated by tfplugindocs --> | ||||
| ## Schema | ||||
|  | ||||
| ### Required | ||||
|  | ||||
| - `email` (String) E-Mail Address of the user | ||||
| - `login_name` (String) The login name can differ from the username | ||||
| - `password` (String, Sensitive) Password to be set for the user | ||||
| - `username` (String) Username of the user to be created | ||||
|  | ||||
| ### Optional | ||||
|  | ||||
| - `active` (Boolean) Flag if this user should be active or not | ||||
| - `admin` (Boolean) Flag if this user should be an administrator or not | ||||
| - `allow_create_organization` (Boolean) | ||||
| - `allow_git_hook` (Boolean) | ||||
| - `allow_import_local` (Boolean) | ||||
| - `description` (String) A description of the user | ||||
| - `force_password_change` (Boolean) Flag if the user defined password should be overwritten or not | ||||
| - `full_name` (String) Full name of the user | ||||
| - `location` (String) | ||||
| - `max_repo_creation` (Number) | ||||
| - `must_change_password` (Boolean) Flag if the user should change the password after first login | ||||
| - `prohibit_login` (Boolean) Flag if the user should not be allowed to log in (bot user) | ||||
| - `restricted` (Boolean) | ||||
| - `send_notification` (Boolean) Flag to send a notification about the user creation to the defined `email` | ||||
| - `visibility` (String) Visibility of the user. Can be `public`, `limited` or `private` | ||||
|  | ||||
| ### Read-Only | ||||
|  | ||||
| - `id` (String) The ID of this resource. | ||||
|  | ||||
|  | ||||
							
								
								
									
										1
									
								
								examples/.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								examples/.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -3,3 +3,4 @@ | ||||
| terraform.tfstate | ||||
| terraform.tfstate.backup | ||||
| *.tfvars | ||||
| id_ed25519 | ||||
| @@ -12,16 +12,96 @@ resource "gitea_repository" "mirror" { | ||||
|   name                         = "terraform-provider-gitea-mirror" | ||||
|   description                  = "Mirror of Terraform Provider" | ||||
|   mirror                       = true | ||||
|   migration_clone_addresse     = "https://git.uploadfilter24.eu/lerentis/terraform-provider-gitea.git" | ||||
|   migration_clone_address      = "https://git.uploadfilter24.eu/lerentis/terraform-provider-gitea.git" | ||||
|   migration_service            = "gitea" | ||||
|   migration_service_auth_token = var.gitea_mirror_token | ||||
| } | ||||
|  | ||||
| resource "gitea_org" "test_org" { | ||||
|   name        = "test-org" | ||||
|   description = "test description" | ||||
| } | ||||
|  | ||||
| resource "gitea_repository" "org_repo" { | ||||
|   username = gitea_org.test_org.name | ||||
|   name     = "org-test-repo" | ||||
| } | ||||
|  | ||||
| data "gitea_user" "me" { | ||||
|   username = "lerentis" | ||||
| } | ||||
|  | ||||
| resource "gitea_user" "test" { | ||||
|   username             = "test" | ||||
|   login_name           = "test" | ||||
|   password             = "Geheim1!" | ||||
|   email                = "test@user.dev" | ||||
|   must_change_password = false | ||||
|   admin                = true | ||||
| } | ||||
|  | ||||
|  | ||||
| resource "gitea_public_key" "test_user_key" { | ||||
|   title     = "test" | ||||
|   key       = file("${path.module}/resources/gitea_public_key/id_ed25519.pub") | ||||
|   read_only = true | ||||
|   username  = gitea_user.test.username | ||||
| } | ||||
|  | ||||
|  | ||||
| resource "gitea_team" "test_team" { | ||||
|   name         = "Devs" | ||||
|   organisation = gitea_org.test_org.name | ||||
|   description  = "Devs of Test Org" | ||||
|   permission   = "write" | ||||
|   members      = [gitea_user.test.username] | ||||
| } | ||||
|  | ||||
| resource "gitea_team" "admin_team" { | ||||
|   name         = "Admins" | ||||
|   organisation = gitea_org.test_org.name | ||||
|   description  = "Admins of Test Org" | ||||
|   permission   = "admin" | ||||
|   members      = [data.gitea_user.me.username] | ||||
| } | ||||
|  | ||||
| resource "gitea_git_hook" "org_repo_pre_receive" { | ||||
|   name    = "pre-receive" | ||||
|   user    = gitea_org.test_org.name | ||||
|   repo    = gitea_repository.org_repo.name | ||||
|   content = file("${path.module}/pre-receive.sh") | ||||
| } | ||||
|  | ||||
| resource "gitea_org" "org1" { | ||||
|   name = "org1" | ||||
| } | ||||
|  | ||||
| resource "gitea_org" "org2" { | ||||
|   name = "org2" | ||||
| } | ||||
|  | ||||
| resource "gitea_repository" "repo1_in_org1" { | ||||
|   username = gitea_org.org1.name | ||||
|   name     = "repo1-in-org1" | ||||
| } | ||||
|  | ||||
| resource "gitea_fork" "user_fork_of_repo1_in_org1" { | ||||
|   owner = gitea_org.org1.name | ||||
|   repo  = gitea_repository.repo1_in_org1.name | ||||
| } | ||||
|  | ||||
| resource "gitea_fork" "org2_fork_of_repo1_in_org1" { | ||||
|   owner        = gitea_org.org1.name | ||||
|   repo         = gitea_repository.repo1_in_org1.name | ||||
|   organization = gitea_org.org2.name | ||||
| } | ||||
|  | ||||
| resource "gitea_token" "test_token" { | ||||
|   username = data.gitea_user.me.username | ||||
|   name     = "test-token" | ||||
| } | ||||
|  | ||||
| output "token" { | ||||
|   value = resource.gitea_token.test_token.token | ||||
|   sensitive = true | ||||
| } | ||||
							
								
								
									
										9
									
								
								examples/pre-receive.sh
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										9
									
								
								examples/pre-receive.sh
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,9 @@ | ||||
| #!/bin/bash | ||||
| while read oldrev newrev refname | ||||
| do | ||||
|     branch=$(git rev-parse --symbolic --abbrev-ref $refname) | ||||
|     if [ "master" = "$branch" ]; then | ||||
|         echo "wrong branch" | ||||
|         exit 1 | ||||
|     fi | ||||
| done | ||||
							
								
								
									
										15
									
								
								examples/provider.tf
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										15
									
								
								examples/provider.tf
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,15 @@ | ||||
| terraform { | ||||
|   required_providers { | ||||
|     gitea = { | ||||
|       source = "terraform.local/lerentis/gitea" | ||||
|       version = "0.11.1" | ||||
|     } | ||||
|   } | ||||
| } | ||||
|  | ||||
| provider "gitea" { | ||||
|   base_url = var.gitea_url | ||||
|   username = "lerentis" | ||||
|   password = var.gitea_password | ||||
|   #token    = var.gitea_token | ||||
| } | ||||
| @@ -2,7 +2,7 @@ terraform { | ||||
|   required_providers { | ||||
|     gitea = { | ||||
|       source = "Lerentis/gitea" | ||||
|       version = "0.3.0" | ||||
|       version = "0.11.1" | ||||
|     } | ||||
|   } | ||||
| } | ||||
|   | ||||
							
								
								
									
										23
									
								
								examples/resources/gitea_fork/resource.tf
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										23
									
								
								examples/resources/gitea_fork/resource.tf
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,23 @@ | ||||
| resource "gitea_org" "org1" { | ||||
|   name = "org1" | ||||
| } | ||||
|  | ||||
| resource "gitea_org" "org2" { | ||||
|   name = "org2" | ||||
| } | ||||
|  | ||||
| resource "gitea_repository" "repo1_in_org1" { | ||||
|   username = gitea_org.org1.name | ||||
|   name     = "repo1-in-org1" | ||||
| } | ||||
|  | ||||
| resource "gitea_fork" "user_fork_of_repo1_in_org1" { | ||||
|   owner = gitea_org.org1.name | ||||
|   repo  = gitea_repository.repo1_in_org1.name | ||||
| } | ||||
|  | ||||
| resource "gitea_fork" "org2_fork_of_repo1_in_org1" { | ||||
|   owner        = gitea_org.org1.name | ||||
|   repo         = gitea_repository.repo1_in_org1.name | ||||
|   organization = gitea_org.org2.name | ||||
| } | ||||
							
								
								
									
										8
									
								
								examples/resources/gitea_git_hook/post-receive.sh
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										8
									
								
								examples/resources/gitea_git_hook/post-receive.sh
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,8 @@ | ||||
| #!/bin/bash | ||||
| while read oldrev newrev refname | ||||
| do | ||||
|     branch=$(git rev-parse --symbolic --abbrev-ref $refname) | ||||
|     if [ "master" = "$branch" ]; then | ||||
|         # Do something | ||||
|     fi | ||||
| done | ||||
							
								
								
									
										15
									
								
								examples/resources/gitea_git_hook/resource.tf
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										15
									
								
								examples/resources/gitea_git_hook/resource.tf
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,15 @@ | ||||
| resource "gitea_org" "test_org" { | ||||
|   name = "test-org" | ||||
| } | ||||
|  | ||||
| resource "gitea_repository" "org_repo" { | ||||
|   username = gitea_org.test_org.name | ||||
|   name     = "org-test-repo" | ||||
| } | ||||
|  | ||||
| resource "gitea_git_hook" "org_repo_post_receive" { | ||||
|   name    = "post-receive" | ||||
|   user    = gitea_org.test_org.name | ||||
|   repo    = gitea_repository.org_repo.name | ||||
|   content = file("${path.module}/post-receive.sh") | ||||
| } | ||||
							
								
								
									
										1
									
								
								examples/resources/gitea_public_key/id_ed25519.pub
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								examples/resources/gitea_public_key/id_ed25519.pub
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1 @@ | ||||
| ssh-ed25519 AAAAC3NzaC1lZDI1NTE5AAAAINn6hAP48oKz6MVWjYvn0fne2YeaOv/zC6zuvFXlJKf2 test@dev.local | ||||
							
								
								
									
										14
									
								
								examples/resources/gitea_public_key/resource.tf
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										14
									
								
								examples/resources/gitea_public_key/resource.tf
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,14 @@ | ||||
| resource "gitea_user" "test" { | ||||
|   username             = "test" | ||||
|   login_name           = "test" | ||||
|   password             = "Geheim1!" | ||||
|   email                = "test@user.dev" | ||||
|   must_change_password = false | ||||
| } | ||||
|  | ||||
|  | ||||
| resource "gitea_public_key" "test_user_key" { | ||||
|   title     = "test" | ||||
|   key       = file("${path.module}/id_ed25519.pub") | ||||
|   username  = gitea_user.test.username | ||||
| } | ||||
							
								
								
									
										21
									
								
								examples/resources/gitea_team/resource.tf
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										21
									
								
								examples/resources/gitea_team/resource.tf
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,21 @@ | ||||
| resource "gitea_org" "test_org" { | ||||
|   name = "test-org" | ||||
| } | ||||
|  | ||||
| resource "gitea_user" "test" { | ||||
|   username             = "test" | ||||
|   login_name           = "test" | ||||
|   password             = "Geheim1!" | ||||
|   email                = "test@user.dev" | ||||
|   must_change_password = false | ||||
|   admin                = true | ||||
| } | ||||
|  | ||||
|  | ||||
| resource "gitea_team" "test_team" { | ||||
|   name         = "Devs" | ||||
|   organisation = gitea_org.test_org.name | ||||
|   description  = "Devs of Test Org" | ||||
|   permission   = "write" | ||||
|   members      = [gitea_user.test.username] | ||||
| } | ||||
							
								
								
									
										25
									
								
								examples/resources/gitea_token/resource.tf
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										25
									
								
								examples/resources/gitea_token/resource.tf
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,25 @@ | ||||
| provider "gitea" { | ||||
|   base_url = var.gitea_url | ||||
|   # Token Auth can not be used with this resource | ||||
|   username = var.gitea_username | ||||
|   password = var.gitea_password | ||||
| } | ||||
|  | ||||
| resource "gitea_user" "test" { | ||||
|   username             = "test" | ||||
|   login_name           = "test" | ||||
|   password             = "Geheim1!" | ||||
|   email                = "test@user.dev" | ||||
|   must_change_password = false | ||||
|   admin                = true | ||||
| } | ||||
|  | ||||
| resource "gitea_token" "test_token" { | ||||
|   username = resource.gitea_user.test.username | ||||
|   name     = "test-token" | ||||
| } | ||||
|  | ||||
| output "token" { | ||||
|   value     = resource.gitea_token.test_token.token | ||||
|   sensitive = true | ||||
| } | ||||
							
								
								
									
										7
									
								
								examples/resources/gitea_user/resource.tf
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										7
									
								
								examples/resources/gitea_user/resource.tf
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,7 @@ | ||||
| resource "gitea_user" "test" { | ||||
|   username             = "test" | ||||
|   login_name           = "test" | ||||
|   password             = "Geheim1!" | ||||
|   email                = "test@user.dev" | ||||
|   must_change_password = false | ||||
| } | ||||
| @@ -9,3 +9,7 @@ variable "gitea_token" { | ||||
| variable "gitea_mirror_token" { | ||||
|    | ||||
| } | ||||
|  | ||||
| variable "gitea_password" { | ||||
|    | ||||
| } | ||||
| @@ -6,9 +6,10 @@ import ( | ||||
| 	"fmt" | ||||
| 	"io/ioutil" | ||||
| 	"net/http" | ||||
| 	"time" | ||||
|  | ||||
| 	"code.gitea.io/sdk/gitea" | ||||
| 	"github.com/hashicorp/terraform-plugin-sdk/helper/logging" | ||||
| 	"github.com/hashicorp/terraform-plugin-sdk/v2/helper/logging" | ||||
| ) | ||||
|  | ||||
| // Config is per-provider, specifies where to connect to gitea | ||||
| @@ -28,8 +29,7 @@ func (c *Config) Client() (interface{}, error) { | ||||
| 		return nil, fmt.Errorf("either a token or a username needs to be used") | ||||
| 	} | ||||
| 	// Configure TLS/SSL | ||||
| 	tlsConfig := &tls.Config{} | ||||
|  | ||||
| 	var tlsConfig tls.Config | ||||
| 	// If a CACertFile has been specified, use that for cert validation | ||||
| 	if c.CACertFile != "" { | ||||
| 		caCert, err := ioutil.ReadFile(c.CACertFile) | ||||
| @@ -43,13 +43,12 @@ func (c *Config) Client() (interface{}, error) { | ||||
| 	} | ||||
|  | ||||
| 	// If configured as insecure, turn off SSL verification | ||||
| 	if c.Insecure { | ||||
| 		tlsConfig.InsecureSkipVerify = true | ||||
| 	} | ||||
| 	tlsConfig.InsecureSkipVerify = c.Insecure | ||||
|  | ||||
| 	t := http.DefaultTransport.(*http.Transport).Clone() | ||||
| 	t.TLSClientConfig = tlsConfig | ||||
| 	t.TLSClientConfig = &tlsConfig | ||||
| 	t.MaxIdleConnsPerHost = 100 | ||||
| 	t.TLSHandshakeTimeout = 10 * time.Second | ||||
|  | ||||
| 	httpClient := &http.Client{ | ||||
| 		Transport: logging.NewTransport("Gitea", t), | ||||
| @@ -60,16 +59,23 @@ func (c *Config) Client() (interface{}, error) { | ||||
| 	} | ||||
|  | ||||
| 	var client *gitea.Client | ||||
| 	var err error | ||||
| 	if c.Token != "" { | ||||
| 		client, _ = gitea.NewClient(c.BaseURL, gitea.SetToken(c.Token), gitea.SetHTTPClient(httpClient)) | ||||
| 		client, err = gitea.NewClient(c.BaseURL, gitea.SetToken(c.Token), gitea.SetHTTPClient(httpClient)) | ||||
| 		if err != nil { | ||||
| 			return nil, err | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	if c.Username != "" { | ||||
| 		client, _ = gitea.NewClient(c.BaseURL, gitea.SetBasicAuth(c.Username, c.Password), gitea.SetHTTPClient(httpClient)) | ||||
| 		client, err = gitea.NewClient(c.BaseURL, gitea.SetBasicAuth(c.Username, c.Password), gitea.SetHTTPClient(httpClient)) | ||||
| 		if err != nil { | ||||
| 			return nil, err | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	// Test the credentials by checking we can get information about the authenticated user. | ||||
| 	_, _, err := client.GetMyUserInfo() | ||||
| 	_, _, err = client.GetMyUserInfo() | ||||
|  | ||||
| 	return client, err | ||||
| } | ||||
|   | ||||
| @@ -6,7 +6,7 @@ import ( | ||||
| 	"strings" | ||||
|  | ||||
| 	"code.gitea.io/sdk/gitea" | ||||
| 	"github.com/hashicorp/terraform-plugin-sdk/helper/schema" | ||||
| 	"github.com/hashicorp/terraform-plugin-sdk/v2/helper/schema" | ||||
| ) | ||||
|  | ||||
| func dataSourceGiteaOrg() *schema.Resource { | ||||
|   | ||||
| @@ -5,7 +5,7 @@ import ( | ||||
| 	"strings" | ||||
|  | ||||
| 	"code.gitea.io/sdk/gitea" | ||||
| 	"github.com/hashicorp/terraform-plugin-sdk/helper/schema" | ||||
| 	"github.com/hashicorp/terraform-plugin-sdk/v2/helper/schema" | ||||
| ) | ||||
|  | ||||
| func dataSourceGiteaRepo() *schema.Resource { | ||||
|   | ||||
| @@ -6,7 +6,7 @@ import ( | ||||
| 	"strings" | ||||
|  | ||||
| 	"code.gitea.io/sdk/gitea" | ||||
| 	"github.com/hashicorp/terraform-plugin-sdk/helper/schema" | ||||
| 	"github.com/hashicorp/terraform-plugin-sdk/v2/helper/schema" | ||||
| ) | ||||
|  | ||||
| func dataSourceGiteaUser() *schema.Resource { | ||||
|   | ||||
| @@ -4,8 +4,8 @@ import ( | ||||
| 	"fmt" | ||||
| 	"testing" | ||||
|  | ||||
| 	"github.com/hashicorp/terraform-plugin-sdk/helper/resource" | ||||
| 	"github.com/hashicorp/terraform-plugin-sdk/terraform" | ||||
| 	"github.com/hashicorp/terraform-plugin-sdk/v2/helper/resource" | ||||
| 	"github.com/hashicorp/terraform-plugin-sdk/v2/terraform" | ||||
| ) | ||||
|  | ||||
| func TestAccDataSourceGiteaUser_basic(t *testing.T) { | ||||
|   | ||||
| @@ -4,12 +4,11 @@ import ( | ||||
| 	"fmt" | ||||
| 	"strings" | ||||
|  | ||||
| 	"github.com/hashicorp/terraform-plugin-sdk/helper/schema" | ||||
| 	"github.com/hashicorp/terraform-plugin-sdk/terraform" | ||||
| 	"github.com/hashicorp/terraform-plugin-sdk/v2/helper/schema" | ||||
| ) | ||||
|  | ||||
| // Provider returns a terraform.ResourceProvider. | ||||
| func Provider() terraform.ResourceProvider { | ||||
| func Provider() *schema.Provider { | ||||
|  | ||||
| 	// The actual provider | ||||
| 	return &schema.Provider{ | ||||
| @@ -76,9 +75,14 @@ func Provider() terraform.ResourceProvider { | ||||
| 			"gitea_org": resourceGiteaOrg(), | ||||
| 			// "gitea_team":       resourceGiteaTeam(), | ||||
| 			// "gitea_repo":       resourceGiteaRepo(), | ||||
| 			// "gitea_user": resourceGiteaUser(), | ||||
| 			"gitea_user":       resourceGiteaUser(), | ||||
| 			"gitea_oauth2_app": resourceGiteaOauthApp(), | ||||
| 			"gitea_repository": resourceGiteaRepository(), | ||||
| 			"gitea_fork":       resourceGiteaFork(), | ||||
| 			"gitea_public_key": resourceGiteaPublicKey(), | ||||
| 			"gitea_team":       resourceGiteaTeam(), | ||||
| 			"gitea_git_hook":   resourceGiteaGitHook(), | ||||
| 			"gitea_token":      resourceGiteaToken(), | ||||
| 		}, | ||||
|  | ||||
| 		ConfigureFunc: providerConfigure, | ||||
| @@ -116,5 +120,8 @@ func validateAPIURLVersion(value interface{}, key string) (ws []string, es []err | ||||
| 	if strings.HasSuffix(v, "/api/v1") || strings.HasSuffix(v, "/api/v1/") { | ||||
| 		es = append(es, fmt.Errorf("terraform-gitea-provider base URL format is incorrect; Please leave out API Path %s", v)) | ||||
| 	} | ||||
| 	if strings.Contains(v, "localhost") && strings.Contains(v, ".") { | ||||
| 		es = append(es, fmt.Errorf("terraform-gitea-provider base URL violates RFC 2606; Please do not define a subdomain for localhost!")) | ||||
| 	} | ||||
| 	return | ||||
| } | ||||
|   | ||||
| @@ -4,30 +4,28 @@ import ( | ||||
| 	"os" | ||||
| 	"testing" | ||||
|  | ||||
| 	"github.com/hashicorp/terraform-plugin-sdk/helper/schema" | ||||
| 	"github.com/hashicorp/terraform-plugin-sdk/terraform" | ||||
| 	"github.com/hashicorp/terraform-plugin-sdk/v2/helper/schema" | ||||
| ) | ||||
|  | ||||
| var testAccProviders map[string]terraform.ResourceProvider | ||||
| var testAccProviders map[string]*schema.Provider | ||||
| var testAccProvider *schema.Provider | ||||
|  | ||||
| func init() { | ||||
| 	testAccProvider = Provider().(*schema.Provider) | ||||
| 	testAccProviders = map[string]terraform.ResourceProvider{ | ||||
| 	testAccProvider = Provider() | ||||
| 	testAccProviders = map[string]*schema.Provider{ | ||||
| 		"gitea": testAccProvider, | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestProvider(t *testing.T) { | ||||
| 	if err := Provider().(*schema.Provider).InternalValidate(); err != nil { | ||||
| 	if err := Provider().InternalValidate(); err != nil { | ||||
| 		t.Fatalf("err: %s", err) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestProvider_impl(t *testing.T) { | ||||
| 	var _ terraform.ResourceProvider = Provider() | ||||
| 	var _ *schema.Provider = Provider() | ||||
| } | ||||
|  | ||||
| func testAccPreCheck(t *testing.T) { | ||||
| 	if v := os.Getenv("GITEA_TOKEN"); v == "" { | ||||
| 		t.Fatal("GITEA_TOKEN must be set for acceptance tests") | ||||
|   | ||||
							
								
								
									
										127
									
								
								gitea/resource_gitea_fork.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										127
									
								
								gitea/resource_gitea_fork.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,127 @@ | ||||
| package gitea | ||||
|  | ||||
| import ( | ||||
| 	"fmt" | ||||
| 	"strconv" | ||||
|  | ||||
| 	"code.gitea.io/sdk/gitea" | ||||
| 	"github.com/hashicorp/terraform-plugin-sdk/v2/helper/schema" | ||||
| ) | ||||
|  | ||||
| const ( | ||||
| 	forkOwner        string = "owner" | ||||
| 	forkRepo         string = "repo" | ||||
| 	forkOrganization string = "organization" | ||||
| ) | ||||
|  | ||||
| func resourceForkCreate(d *schema.ResourceData, meta interface{}) (err error) { | ||||
| 	client := meta.(*gitea.Client) | ||||
|  | ||||
| 	var opts gitea.CreateForkOption | ||||
| 	var org string | ||||
| 	org = d.Get(forkOrganization).(string) | ||||
| 	if org != "" { | ||||
| 		opts.Organization = &org | ||||
| 	} | ||||
|  | ||||
| 	repo, _, err := client.CreateFork(d.Get(forkOwner).(string), | ||||
| 		d.Get(forkRepo).(string), | ||||
| 		opts) | ||||
| 	if err == nil { | ||||
| 		err = setForkResourceData(repo, d) | ||||
| 	} | ||||
| 	return err | ||||
| } | ||||
|  | ||||
| func resourceForkRead(d *schema.ResourceData, meta interface{}) (err error) { | ||||
| 	client := meta.(*gitea.Client) | ||||
|  | ||||
| 	id, err := strconv.ParseInt(d.Id(), 10, 64) | ||||
| 	var resp *gitea.Response | ||||
|  | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
|  | ||||
| 	repo, resp, err := client.GetRepoByID(id) | ||||
|  | ||||
| 	if err != nil { | ||||
| 		if resp.StatusCode == 404 { | ||||
| 			d.SetId("") | ||||
| 			return nil | ||||
| 		} else { | ||||
| 			return err | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	err = setForkResourceData(repo, d) | ||||
|  | ||||
| 	return | ||||
| } | ||||
|  | ||||
| func resourceForkDelete(d *schema.ResourceData, meta interface{}) (err error) { | ||||
| 	client := meta.(*gitea.Client) | ||||
|  | ||||
| 	id, err := strconv.ParseInt(d.Id(), 10, 64) | ||||
|  | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
|  | ||||
| 	repo, _, err := client.GetRepoByID(id) | ||||
| 	var resp *gitea.Response | ||||
|  | ||||
| 	resp, err = client.DeleteRepo(repo.Owner.UserName, repo.Name) | ||||
|  | ||||
| 	if err != nil { | ||||
| 		if resp.StatusCode == 404 { | ||||
| 			return | ||||
| 		} else { | ||||
| 			return err | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	return | ||||
| } | ||||
|  | ||||
| func setForkResourceData(repo *gitea.Repository, d *schema.ResourceData) (err error) { | ||||
|  | ||||
| 	d.SetId(fmt.Sprintf("%d", repo.ID)) | ||||
|  | ||||
| 	return | ||||
| } | ||||
|  | ||||
| func resourceGiteaFork() *schema.Resource { | ||||
| 	return &schema.Resource{ | ||||
| 		Read:   resourceForkRead, | ||||
| 		Create: resourceForkCreate, | ||||
| 		Delete: resourceForkDelete, | ||||
| 		Importer: &schema.ResourceImporter{ | ||||
| 			StateContext: schema.ImportStatePassthroughContext, | ||||
| 		}, | ||||
| 		Schema: map[string]*schema.Schema{ | ||||
| 			"owner": { | ||||
| 				Type:        schema.TypeString, | ||||
| 				Required:    true, | ||||
| 				ForceNew:    true, | ||||
| 				Description: "The owner or owning organization of the repository to fork", | ||||
| 			}, | ||||
| 			"repo": { | ||||
| 				Type:        schema.TypeString, | ||||
| 				Required:    true, | ||||
| 				ForceNew:    true, | ||||
| 				Description: "The name of the repository to fork", | ||||
| 			}, | ||||
| 			"organization": { | ||||
| 				Type:        schema.TypeString, | ||||
| 				Required:    false, | ||||
| 				Optional:    true, | ||||
| 				ForceNew:    true, | ||||
| 				Description: "The organization that owns the forked repo", | ||||
| 			}, | ||||
| 		}, | ||||
| 		Description: "`gitea_fork` manages repository fork to the current user or an organisation\n" + | ||||
| 			"Forking a repository to a dedicated user is currently unsupported\n" + | ||||
| 			"Creating a fork using this resource without an organisation will create the fork in the executors name", | ||||
| 	} | ||||
| } | ||||
							
								
								
									
										120
									
								
								gitea/resource_gitea_git_hook.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										120
									
								
								gitea/resource_gitea_git_hook.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,120 @@ | ||||
| package gitea | ||||
|  | ||||
| import ( | ||||
| 	"fmt" | ||||
|  | ||||
| 	"code.gitea.io/sdk/gitea" | ||||
| 	"github.com/hashicorp/terraform-plugin-sdk/v2/helper/schema" | ||||
| ) | ||||
|  | ||||
| const ( | ||||
| 	GitHookUser    string = "user" | ||||
| 	GitHookRepo    string = "repo" | ||||
| 	GitHookName    string = "name" | ||||
| 	GitHookContent string = "content" | ||||
| ) | ||||
|  | ||||
| func resourceGitHookRead(d *schema.ResourceData, meta interface{}) (err error) { | ||||
| 	client := meta.(*gitea.Client) | ||||
|  | ||||
| 	user := d.Get(GitHookUser).(string) | ||||
| 	repo := d.Get(GitHookRepo).(string) | ||||
| 	name := d.Get(GitHookName).(string) | ||||
|  | ||||
| 	gitHook, _, err := client.GetRepoGitHook(user, repo, name) | ||||
|  | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
|  | ||||
| 	err = setGitHookResourceData(user, repo, gitHook, d) | ||||
|  | ||||
| 	return | ||||
| } | ||||
|  | ||||
| func resourceGitHookUpdate(d *schema.ResourceData, meta interface{}) (err error) { | ||||
| 	client := meta.(*gitea.Client) | ||||
|  | ||||
| 	user := d.Get(GitHookUser).(string) | ||||
| 	repo := d.Get(GitHookRepo).(string) | ||||
| 	name := d.Get(GitHookName).(string) | ||||
|  | ||||
| 	opts := gitea.EditGitHookOption{ | ||||
| 		Content: d.Get(GitHookContent).(string), | ||||
| 	} | ||||
|  | ||||
| 	_, err = client.EditRepoGitHook(user, repo, name, opts) | ||||
|  | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
|  | ||||
| 	// Get gitHook ourselves, EditRepoGitHook does not return it | ||||
| 	gitHook, _, err := client.GetRepoGitHook(user, repo, name) | ||||
|  | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
|  | ||||
| 	err = setGitHookResourceData(user, repo, gitHook, d) | ||||
|  | ||||
| 	return | ||||
| } | ||||
|  | ||||
| func resourceGitHookDelete(d *schema.ResourceData, meta interface{}) (err error) { | ||||
| 	client := meta.(*gitea.Client) | ||||
|  | ||||
| 	user := d.Get(GitHookUser).(string) | ||||
| 	repo := d.Get(GitHookRepo).(string) | ||||
| 	name := d.Get(GitHookName).(string) | ||||
|  | ||||
| 	_, err = client.DeleteRepoGitHook(user, repo, name) | ||||
|  | ||||
| 	return | ||||
| } | ||||
|  | ||||
| func setGitHookResourceData(user string, repo string, gitHook *gitea.GitHook, d *schema.ResourceData) (err error) { | ||||
| 	d.SetId(fmt.Sprintf("%s/%s/%s", user, repo, gitHook.Name)) | ||||
| 	d.Set(GitHookUser, user) | ||||
| 	d.Set(GitHookRepo, repo) | ||||
| 	d.Set(GitHookName, gitHook.Name) | ||||
| 	d.Set(GitHookContent, gitHook.Content) | ||||
| 	return | ||||
| } | ||||
|  | ||||
| func resourceGiteaGitHook() *schema.Resource { | ||||
| 	return &schema.Resource{ | ||||
| 		Read:   resourceGitHookRead, | ||||
| 		Create: resourceGitHookUpdate, // All hooks already exist, just empty and disabled | ||||
| 		Update: resourceGitHookUpdate, | ||||
| 		Delete: resourceGitHookDelete, | ||||
| 		Schema: map[string]*schema.Schema{ | ||||
| 			"name": { | ||||
| 				Type:        schema.TypeString, | ||||
| 				Required:    true, | ||||
| 				Description: "Name of the git hook to configure", | ||||
| 			}, | ||||
| 			"repo": { | ||||
| 				Type:        schema.TypeString, | ||||
| 				Required:    true, | ||||
| 				Description: "The repository that this hook belongs too.", | ||||
| 			}, | ||||
| 			"user": { | ||||
| 				Type:        schema.TypeString, | ||||
| 				Required:    true, | ||||
| 				Description: "The user (or organisation) owning the repo this hook belongs too", | ||||
| 			}, | ||||
| 			"content": { | ||||
| 				Type:        schema.TypeString, | ||||
| 				Required:    true, | ||||
| 				Description: "Content of the git hook", | ||||
| 			}, | ||||
| 		}, | ||||
| 		Description: "`gitea_git_hook` manages git hooks on a repository.\n" + | ||||
| 			"import is currently not supported\n\n" + | ||||
| 			"WARNING: using this resource requires to enable server side hooks" + | ||||
| 			"which are known to cause [security issues](https://github.com/go-gitea/gitea/pull/13058)!\n\n" + | ||||
| 			"if you want to procede, you need to enable server side hooks as stated" + | ||||
| 			" [here](https://docs.gitea.io/en-us/config-cheat-sheet/#security-security)", | ||||
| 	} | ||||
| } | ||||
| @@ -4,7 +4,7 @@ import ( | ||||
| 	"fmt" | ||||
|  | ||||
| 	"code.gitea.io/sdk/gitea" | ||||
| 	"github.com/hashicorp/terraform-plugin-sdk/helper/schema" | ||||
| 	"github.com/hashicorp/terraform-plugin-sdk/v2/helper/schema" | ||||
| ) | ||||
|  | ||||
| const ( | ||||
|   | ||||
| @@ -2,9 +2,10 @@ package gitea | ||||
|  | ||||
| import ( | ||||
| 	"fmt" | ||||
| 	"strconv" | ||||
|  | ||||
| 	"code.gitea.io/sdk/gitea" | ||||
| 	"github.com/hashicorp/terraform-plugin-sdk/helper/schema" | ||||
| 	"github.com/hashicorp/terraform-plugin-sdk/v2/helper/schema" | ||||
| ) | ||||
|  | ||||
| const ( | ||||
| @@ -17,24 +18,48 @@ const ( | ||||
| 	RepoAdminChangeTeamAccess string = "repo_admin_change_team_access" | ||||
| ) | ||||
|  | ||||
| type VisibleType string | ||||
| // might come in handy if we want to stick to numeric IDs | ||||
| func searchOrgByClientId(c *gitea.Client, id int64) (res *gitea.Organization, err error) { | ||||
|  | ||||
| 	page := 1 | ||||
|  | ||||
| 	for { | ||||
| 		orgs, _, err := c.AdminListOrgs(gitea.AdminListOrgsOptions{ | ||||
| 			ListOptions: gitea.ListOptions{ | ||||
| 				Page:     page, | ||||
| 				PageSize: 50, | ||||
| 			}, | ||||
| 		}) | ||||
| 		if err != nil { | ||||
| 			return nil, err | ||||
| 		} | ||||
|  | ||||
| 		if len(orgs) == 0 { | ||||
| 			return nil, fmt.Errorf("Organisation with ID %d could not be found", id) | ||||
| 		} | ||||
|  | ||||
| 		for _, org := range orgs { | ||||
| 			if org.ID == id { | ||||
| 				return org, nil | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		page += 1 | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func resourceOrgRead(d *schema.ResourceData, meta interface{}) (err error) { | ||||
| 	client := meta.(*gitea.Client) | ||||
|  | ||||
| 	var org *gitea.Organization | ||||
| 	var resp *gitea.Response | ||||
|  | ||||
| 	org, resp, err = client.GetOrg(d.Get(orgName).(string)) | ||||
| 	id, err := strconv.ParseInt(d.Id(), 10, 64) | ||||
|  | ||||
| 	org, err = searchOrgByClientId(client, id) | ||||
|  | ||||
| 	if err != nil { | ||||
| 		if resp.StatusCode == 404 { | ||||
| 			d.SetId("") | ||||
| 			return nil | ||||
| 		} else { | ||||
| 		return err | ||||
| 	} | ||||
| 	} | ||||
|  | ||||
| 	err = setOrgResourceData(org, d) | ||||
|  | ||||
| @@ -97,7 +122,7 @@ func resourceOrgUpdate(d *schema.ResourceData, meta interface{}) (err error) { | ||||
| 	return | ||||
| } | ||||
|  | ||||
| func respurceOrgDelete(d *schema.ResourceData, meta interface{}) (err error) { | ||||
| func resourceOrgDelete(d *schema.ResourceData, meta interface{}) (err error) { | ||||
| 	client := meta.(*gitea.Client) | ||||
|  | ||||
| 	var resp *gitea.Response | ||||
| @@ -133,9 +158,9 @@ func resourceGiteaOrg() *schema.Resource { | ||||
| 		Read:   resourceOrgRead, | ||||
| 		Create: resourceOrgCreate, | ||||
| 		Update: resourceOrgUpdate, | ||||
| 		Delete: respurceOrgDelete, | ||||
| 		Delete: resourceOrgDelete, | ||||
| 		Importer: &schema.ResourceImporter{ | ||||
| 			State: schema.ImportStatePassthrough, | ||||
| 			StateContext: schema.ImportStatePassthroughContext, | ||||
| 		}, | ||||
| 		Schema: map[string]*schema.Schema{ | ||||
| 			"name": { | ||||
|   | ||||
							
								
								
									
										155
									
								
								gitea/resource_gitea_public_key.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										155
									
								
								gitea/resource_gitea_public_key.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,155 @@ | ||||
| package gitea | ||||
|  | ||||
| import ( | ||||
| 	"fmt" | ||||
| 	"strconv" | ||||
|  | ||||
| 	"code.gitea.io/sdk/gitea" | ||||
| 	"github.com/hashicorp/terraform-plugin-sdk/v2/helper/schema" | ||||
| ) | ||||
|  | ||||
| const ( | ||||
| 	PublicKeyUser         string = "username" | ||||
| 	PublicKey             string = "key" | ||||
| 	PublicKeyReadOnlyFlag string = "read_only" | ||||
| 	PublicKeyTitle        string = "title" | ||||
| 	PublicKeyId           string = "id" | ||||
| 	PublicKeyFingerprint  string = "fingerprint" | ||||
| 	PublicKeyCreated      string = "created" | ||||
| 	PublicKeyType         string = "type" | ||||
| ) | ||||
|  | ||||
| func resourcePublicKeyRead(d *schema.ResourceData, meta interface{}) (err error) { | ||||
| 	client := meta.(*gitea.Client) | ||||
|  | ||||
| 	id, err := strconv.ParseInt(d.Id(), 10, 64) | ||||
|  | ||||
| 	var resp *gitea.Response | ||||
| 	var pubKey *gitea.PublicKey | ||||
|  | ||||
| 	pubKey, resp, err = client.GetPublicKey(id) | ||||
|  | ||||
| 	if err != nil { | ||||
| 		if resp.StatusCode == 404 { | ||||
| 			d.SetId("") | ||||
| 			return nil | ||||
| 		} else { | ||||
| 			return err | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	err = setPublicKeyResourceData(pubKey, d) | ||||
|  | ||||
| 	return | ||||
| } | ||||
|  | ||||
| func resourcePublicKeyCreate(d *schema.ResourceData, meta interface{}) (err error) { | ||||
| 	client := meta.(*gitea.Client) | ||||
|  | ||||
| 	var pubKey *gitea.PublicKey | ||||
|  | ||||
| 	opts := gitea.CreateKeyOption{ | ||||
| 		Title:    d.Get(PublicKeyTitle).(string), | ||||
| 		Key:      d.Get(PublicKey).(string), | ||||
| 		ReadOnly: d.Get(PublicKeyReadOnlyFlag).(bool), | ||||
| 	} | ||||
|  | ||||
| 	pubKey, _, err = client.AdminCreateUserPublicKey(d.Get(PublicKeyUser).(string), opts) | ||||
|  | ||||
| 	err = setPublicKeyResourceData(pubKey, d) | ||||
|  | ||||
| 	return | ||||
| } | ||||
|  | ||||
| func resourcePublicKeyUpdate(d *schema.ResourceData, meta interface{}) (err error) { | ||||
| 	// update = recreate | ||||
| 	resourcePublicKeyDelete(d, meta) | ||||
| 	resourcePublicKeyCreate(d, meta) | ||||
| 	return | ||||
| } | ||||
|  | ||||
| func resourcePublicKeyDelete(d *schema.ResourceData, meta interface{}) (err error) { | ||||
| 	client := meta.(*gitea.Client) | ||||
|  | ||||
| 	id, err := strconv.ParseInt(d.Id(), 10, 64) | ||||
|  | ||||
| 	var resp *gitea.Response | ||||
|  | ||||
| 	resp, err = client.AdminDeleteUserPublicKey(d.Get(PublicKeyUser).(string), int(id)) | ||||
|  | ||||
| 	if err != nil { | ||||
| 		if resp.StatusCode == 404 { | ||||
| 			return | ||||
| 		} else { | ||||
| 			return err | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	return | ||||
| } | ||||
|  | ||||
| func setPublicKeyResourceData(pubKey *gitea.PublicKey, d *schema.ResourceData) (err error) { | ||||
| 	d.SetId(fmt.Sprintf("%d", pubKey.ID)) | ||||
| 	d.Set(PublicKeyUser, d.Get(PublicKeyUser).(string)) | ||||
| 	d.Set(PublicKey, d.Get(PublicKey).(string)) | ||||
| 	d.Set(PublicKeyTitle, pubKey.Title) | ||||
| 	d.Set(PublicKeyReadOnlyFlag, d.Get(PublicKeyReadOnlyFlag).(bool)) | ||||
| 	d.Set(PublicKeyCreated, pubKey.Created) | ||||
| 	d.Set(PublicKeyFingerprint, pubKey.Fingerprint) | ||||
| 	d.Set(PublicKeyType, pubKey.KeyType) | ||||
| 	return | ||||
| } | ||||
|  | ||||
| func resourceGiteaPublicKey() *schema.Resource { | ||||
| 	return &schema.Resource{ | ||||
| 		Read:   resourcePublicKeyRead, | ||||
| 		Create: resourcePublicKeyCreate, | ||||
| 		Update: resourcePublicKeyUpdate, | ||||
| 		Delete: resourcePublicKeyDelete, | ||||
| 		Importer: &schema.ResourceImporter{ | ||||
| 			State: schema.ImportStatePassthrough, | ||||
| 		}, | ||||
| 		Schema: map[string]*schema.Schema{ | ||||
| 			"title": { | ||||
| 				Type:        schema.TypeString, | ||||
| 				Required:    true, | ||||
| 				ForceNew:    true, | ||||
| 				Description: "Title of the key to add", | ||||
| 			}, | ||||
| 			"key": { | ||||
| 				Type:        schema.TypeString, | ||||
| 				Required:    true, | ||||
| 				ForceNew:    true, | ||||
| 				Sensitive:   true, | ||||
| 				Description: "An armored SSH key to add", | ||||
| 			}, | ||||
| 			"read_only": { | ||||
| 				Type:        schema.TypeBool, | ||||
| 				Required:    false, | ||||
| 				Optional:    true, | ||||
| 				Default:     false, | ||||
| 				Description: "Describe if the key has only read access or read/write", | ||||
| 			}, | ||||
| 			"username": { | ||||
| 				Type:        schema.TypeString, | ||||
| 				Required:    true, | ||||
| 				Optional:    false, | ||||
| 				ForceNew:    true, | ||||
| 				Description: "User to associate with the added key", | ||||
| 			}, | ||||
| 			"fingerprint": { | ||||
| 				Type:     schema.TypeString, | ||||
| 				Computed: true, | ||||
| 			}, | ||||
| 			"created": { | ||||
| 				Type:     schema.TypeString, | ||||
| 				Computed: true, | ||||
| 			}, | ||||
| 			"type": { | ||||
| 				Type:     schema.TypeString, | ||||
| 				Computed: true, | ||||
| 			}, | ||||
| 		}, | ||||
| 		Description: "`gitea_public_key` manages ssh key that are associated with users.", | ||||
| 	} | ||||
| } | ||||
| @@ -5,7 +5,7 @@ import ( | ||||
| 	"strconv" | ||||
|  | ||||
| 	"code.gitea.io/sdk/gitea" | ||||
| 	"github.com/hashicorp/terraform-plugin-sdk/helper/schema" | ||||
| 	"github.com/hashicorp/terraform-plugin-sdk/v2/helper/schema" | ||||
| ) | ||||
|  | ||||
| const ( | ||||
| @@ -34,7 +34,8 @@ const ( | ||||
| 	repoAllowManualMerge         string = "allow_manual_merge" | ||||
| 	repoAutodetectManualMerge    string = "autodetect_manual_merge" | ||||
| 	repoMirror                   string = "mirror" | ||||
| 	migrationCloneAddress        string = "migration_clone_addresse" | ||||
| 	migrationCloneAddresse       string = "migration_clone_addresse" | ||||
| 	migrationCloneAddress        string = "migration_clone_address" | ||||
| 	migrationService             string = "migration_service" | ||||
| 	migrationServiceAuthName     string = "migration_service_auth_username" | ||||
| 	migrationServiceAuthPassword string = "migration_service_auth_password" | ||||
| @@ -77,22 +78,20 @@ func resourceRepoCreate(d *schema.ResourceData, meta interface{}) (err error) { | ||||
| 	client := meta.(*gitea.Client) | ||||
|  | ||||
| 	var repo *gitea.Repository | ||||
| 	var resp *gitea.Response | ||||
| 	var orgRepo bool | ||||
|  | ||||
| 	_, resp, err = client.GetOrg(d.Get(repoOwner).(string)) | ||||
|  | ||||
| 	if resp.StatusCode == 404 { | ||||
| 		orgRepo = false | ||||
| 	} else { | ||||
| 		orgRepo = true | ||||
| 	} | ||||
|  | ||||
| 	if (d.Get(repoMirror)).(bool) { | ||||
|  | ||||
| 		var cloneAddr string | ||||
| 		if d.Get(migrationCloneAddresse).(string) != "" { | ||||
| 			cloneAddr = d.Get(migrationCloneAddresse).(string) | ||||
| 		} else { | ||||
| 			cloneAddr = d.Get(migrationCloneAddress).(string) | ||||
| 		} | ||||
|  | ||||
| 		opts := gitea.MigrateRepoOption{ | ||||
| 			RepoName:       d.Get(repoName).(string), | ||||
| 			RepoOwner:      d.Get(repoOwner).(string), | ||||
| 			CloneAddr:      d.Get(migrationCloneAddress).(string), | ||||
| 			CloneAddr:      cloneAddr, | ||||
| 			Service:        gitea.GitServiceType(d.Get(migrationService).(string)), | ||||
| 			Mirror:         d.Get(repoMirror).(bool), | ||||
| 			Private:        d.Get(repoPrivateFlag).(bool), | ||||
| @@ -135,15 +134,11 @@ func resourceRepoCreate(d *schema.ResourceData, meta interface{}) (err error) { | ||||
| 			TrustModel:    "default", | ||||
| 		} | ||||
|  | ||||
| 		if orgRepo { | ||||
| 		repo, _, err = client.CreateOrgRepo(d.Get(repoOwner).(string), opts) | ||||
| 		} else { | ||||
| 			repo, _, err = client.CreateRepo(opts) | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	if err != nil { | ||||
| 		return | ||||
| 		return err | ||||
| 	} | ||||
|  | ||||
| 	err = setRepoResourceData(repo, d) | ||||
| @@ -223,6 +218,7 @@ func respurceRepoDelete(d *schema.ResourceData, meta interface{}) (err error) { | ||||
|  | ||||
| func setRepoResourceData(repo *gitea.Repository, d *schema.ResourceData) (err error) { | ||||
| 	d.SetId(fmt.Sprintf("%d", repo.ID)) | ||||
| 	d.Set("username", repo.Owner.UserName) | ||||
| 	d.Set("name", repo.Name) | ||||
| 	d.Set("description", repo.Description) | ||||
| 	d.Set("full_name", repo.FullName) | ||||
| @@ -239,8 +235,8 @@ func setRepoResourceData(repo *gitea.Repository, d *schema.ResourceData) (err er | ||||
| 	d.Set("watchers", repo.Watchers) | ||||
| 	d.Set("open_issue_count", repo.OpenIssues) | ||||
| 	d.Set("default_branch", repo.DefaultBranch) | ||||
| 	d.Set("created", repo.Created) | ||||
| 	d.Set("updated", repo.Updated) | ||||
| 	d.Set("created", repo.Created.String()) | ||||
| 	d.Set("updated", repo.Updated.String()) | ||||
| 	d.Set("permission_admin", repo.Permissions.Admin) | ||||
| 	d.Set("permission_push", repo.Permissions.Push) | ||||
| 	d.Set("permission_pull", repo.Permissions.Pull) | ||||
| @@ -255,7 +251,7 @@ func resourceGiteaRepository() *schema.Resource { | ||||
| 		Update: resourceRepoUpdate, | ||||
| 		Delete: respurceRepoDelete, | ||||
| 		Importer: &schema.ResourceImporter{ | ||||
| 			State: schema.ImportStatePassthrough, | ||||
| 			StateContext: schema.ImportStatePassthroughContext, | ||||
| 		}, | ||||
| 		Schema: map[string]*schema.Schema{ | ||||
| 			"username": { | ||||
| @@ -449,6 +445,13 @@ func resourceGiteaRepository() *schema.Resource { | ||||
| 				Required:    false, | ||||
| 				Optional:    true, | ||||
| 				ForceNew:    true, | ||||
| 				Description: "DEPRECATED in favor of `migration_clone_address`", | ||||
| 			}, | ||||
| 			"migration_clone_address": { | ||||
| 				Type:     schema.TypeString, | ||||
| 				Required: false, | ||||
| 				Optional: true, | ||||
| 				ForceNew: true, | ||||
| 			}, | ||||
| 			"migration_service": { | ||||
| 				Type:        schema.TypeString, | ||||
| @@ -513,6 +516,18 @@ func resourceGiteaRepository() *schema.Resource { | ||||
| 				Optional: true, | ||||
| 				Default:  "", | ||||
| 			}, | ||||
| 			"clone_url": { | ||||
| 				Type:     schema.TypeString, | ||||
| 				Computed: true, | ||||
| 			}, | ||||
| 			"html_url": { | ||||
| 				Type:     schema.TypeString, | ||||
| 				Computed: true, | ||||
| 			}, | ||||
| 			"ssh_url": { | ||||
| 				Type:     schema.TypeString, | ||||
| 				Computed: true, | ||||
| 			}, | ||||
| 		}, | ||||
| 		Description: "`gitea_repository` manages a gitea repository.\n\n" + | ||||
| 			"Per default this repository will be initializiled with the provided configuration (gitignore, License etc.).\n" + | ||||
|   | ||||
							
								
								
									
										296
									
								
								gitea/resource_gitea_team.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										296
									
								
								gitea/resource_gitea_team.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,296 @@ | ||||
| package gitea | ||||
|  | ||||
| import ( | ||||
| 	"fmt" | ||||
| 	"strconv" | ||||
| 	"strings" | ||||
|  | ||||
| 	"code.gitea.io/sdk/gitea" | ||||
| 	"github.com/hashicorp/terraform-plugin-sdk/v2/helper/schema" | ||||
| ) | ||||
|  | ||||
| const ( | ||||
| 	TeamName                string = "name" | ||||
| 	TeamOrg                 string = "organisation" | ||||
| 	TeamDescription         string = "description" | ||||
| 	TeamPermissions         string = "permission" | ||||
| 	TeamCreateRepoFlag      string = "can_create_repos" | ||||
| 	TeamIncludeAllReposFlag string = "include_all_repositories" | ||||
| 	TeamUnits               string = "units" | ||||
| 	TeamMembers             string = "members" | ||||
| ) | ||||
|  | ||||
| func resourceTeamRead(d *schema.ResourceData, meta interface{}) (err error) { | ||||
| 	client := meta.(*gitea.Client) | ||||
|  | ||||
| 	id, err := strconv.ParseInt(d.Id(), 10, 64) | ||||
|  | ||||
| 	var resp *gitea.Response | ||||
| 	var team *gitea.Team | ||||
|  | ||||
| 	team, resp, err = client.GetTeam(id) | ||||
|  | ||||
| 	if err != nil { | ||||
| 		if resp.StatusCode == 404 { | ||||
| 			d.SetId("") | ||||
| 			return nil | ||||
| 		} else { | ||||
| 			return err | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	err = setTeamResourceData(team, d) | ||||
|  | ||||
| 	return | ||||
| } | ||||
|  | ||||
| func resourceTeamCreate(d *schema.ResourceData, meta interface{}) (err error) { | ||||
| 	client := meta.(*gitea.Client) | ||||
|  | ||||
| 	var team *gitea.Team | ||||
| 	var units []gitea.RepoUnitType | ||||
|  | ||||
| 	if strings.Contains(d.Get(TeamUnits).(string), "repo.code") { | ||||
| 		units = append(units, gitea.RepoUnitCode) | ||||
| 	} | ||||
| 	if strings.Contains(d.Get(TeamUnits).(string), "repo.issues") { | ||||
| 		units = append(units, gitea.RepoUnitIssues) | ||||
| 	} | ||||
| 	if strings.Contains(d.Get(TeamUnits).(string), "repo.ext_issues") { | ||||
| 		units = append(units, gitea.RepoUnitExtIssues) | ||||
| 	} | ||||
| 	if strings.Contains(d.Get(TeamUnits).(string), "repo.wiki") { | ||||
| 		units = append(units, gitea.RepoUnitWiki) | ||||
| 	} | ||||
| 	if strings.Contains(d.Get(TeamUnits).(string), "repo.pulls") { | ||||
| 		units = append(units, gitea.RepoUnitPulls) | ||||
| 	} | ||||
| 	if strings.Contains(d.Get(TeamUnits).(string), "repo.releases") { | ||||
| 		units = append(units, gitea.RepoUnitReleases) | ||||
| 	} | ||||
| 	if strings.Contains(d.Get(TeamUnits).(string), "repo.ext_wiki") { | ||||
| 		units = append(units, gitea.RepoUnitExtWiki) | ||||
| 	} | ||||
| 	if strings.Contains(d.Get(TeamUnits).(string), "repo.projects") { | ||||
| 		units = append(units, gitea.RepoUnitProjects) | ||||
| 	} | ||||
|  | ||||
| 	opts := gitea.CreateTeamOption{ | ||||
| 		Name:                    d.Get(TeamName).(string), | ||||
| 		Description:             d.Get(TeamDescription).(string), | ||||
| 		Permission:              gitea.AccessMode(d.Get(TeamPermissions).(string)), | ||||
| 		CanCreateOrgRepo:        d.Get(TeamCreateRepoFlag).(bool), | ||||
| 		IncludesAllRepositories: d.Get(TeamIncludeAllReposFlag).(bool), | ||||
| 		Units:                   units, | ||||
| 	} | ||||
|  | ||||
| 	team, _, err = client.CreateTeam(d.Get(TeamOrg).(string), opts) | ||||
|  | ||||
| 	if err != nil { | ||||
| 		return | ||||
| 	} | ||||
|  | ||||
| 	users := d.Get(TeamMembers).([]interface{}) | ||||
|  | ||||
| 	for _, user := range users { | ||||
| 		if user != "" { | ||||
| 			_, err = client.AddTeamMember(team.ID, user.(string)) | ||||
| 			if err != nil { | ||||
| 				return err | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	err = setTeamResourceData(team, d) | ||||
|  | ||||
| 	return | ||||
| } | ||||
|  | ||||
| func resourceTeamUpdate(d *schema.ResourceData, meta interface{}) (err error) { | ||||
| 	client := meta.(*gitea.Client) | ||||
|  | ||||
| 	id, err := strconv.ParseInt(d.Id(), 10, 64) | ||||
|  | ||||
| 	var resp *gitea.Response | ||||
| 	var team *gitea.Team | ||||
|  | ||||
| 	team, resp, err = client.GetTeam(id) | ||||
|  | ||||
| 	if err != nil { | ||||
| 		if resp.StatusCode == 404 { | ||||
| 			resourceTeamCreate(d, meta) | ||||
| 		} else { | ||||
| 			return err | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	description := d.Get(TeamDescription).(string) | ||||
| 	canCreateRepo := d.Get(TeamCreateRepoFlag).(bool) | ||||
| 	includeAllRepos := d.Get(TeamIncludeAllReposFlag).(bool) | ||||
|  | ||||
| 	var units []gitea.RepoUnitType | ||||
|  | ||||
| 	if strings.Contains(d.Get(TeamUnits).(string), "repo.code") { | ||||
| 		units = append(units, gitea.RepoUnitCode) | ||||
| 	} | ||||
| 	if strings.Contains(d.Get(TeamUnits).(string), "repo.issues") { | ||||
| 		units = append(units, gitea.RepoUnitIssues) | ||||
| 	} | ||||
| 	if strings.Contains(d.Get(TeamUnits).(string), "repo.ext_issues") { | ||||
| 		units = append(units, gitea.RepoUnitExtIssues) | ||||
| 	} | ||||
| 	if strings.Contains(d.Get(TeamUnits).(string), "repo.wiki") { | ||||
| 		units = append(units, gitea.RepoUnitWiki) | ||||
| 	} | ||||
| 	if strings.Contains(d.Get(TeamUnits).(string), "repo.pulls") { | ||||
| 		units = append(units, gitea.RepoUnitPulls) | ||||
| 	} | ||||
| 	if strings.Contains(d.Get(TeamUnits).(string), "repo.releases") { | ||||
| 		units = append(units, gitea.RepoUnitReleases) | ||||
| 	} | ||||
| 	if strings.Contains(d.Get(TeamUnits).(string), "repo.ext_wiki") { | ||||
| 		units = append(units, gitea.RepoUnitExtWiki) | ||||
| 	} | ||||
| 	if strings.Contains(d.Get(TeamUnits).(string), "repo.projects") { | ||||
| 		units = append(units, gitea.RepoUnitProjects) | ||||
| 	} | ||||
|  | ||||
| 	opts := gitea.EditTeamOption{ | ||||
| 		Name:                    d.Get(TeamName).(string), | ||||
| 		Description:             &description, | ||||
| 		Permission:              gitea.AccessMode(d.Get(TeamPermissions).(string)), | ||||
| 		CanCreateOrgRepo:        &canCreateRepo, | ||||
| 		IncludesAllRepositories: &includeAllRepos, | ||||
| 		Units:                   units, | ||||
| 	} | ||||
|  | ||||
| 	resp, err = client.EditTeam(id, opts) | ||||
|  | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
|  | ||||
| 	users := d.Get(TeamMembers).([]interface{}) | ||||
|  | ||||
| 	for _, user := range users { | ||||
| 		if user != "" { | ||||
| 			_, err = client.AddTeamMember(team.ID, user.(string)) | ||||
| 			if err != nil { | ||||
| 				return err | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	team, _, _ = client.GetTeam(id) | ||||
|  | ||||
| 	err = setTeamResourceData(team, d) | ||||
|  | ||||
| 	return | ||||
| } | ||||
|  | ||||
| func resourceTeamDelete(d *schema.ResourceData, meta interface{}) (err error) { | ||||
| 	client := meta.(*gitea.Client) | ||||
|  | ||||
| 	id, err := strconv.ParseInt(d.Id(), 10, 64) | ||||
|  | ||||
| 	var resp *gitea.Response | ||||
|  | ||||
| 	resp, err = client.DeleteTeam(id) | ||||
|  | ||||
| 	if err != nil { | ||||
| 		if resp.StatusCode == 404 { | ||||
| 			return | ||||
| 		} else { | ||||
| 			return err | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	return | ||||
| } | ||||
|  | ||||
| func setTeamResourceData(team *gitea.Team, d *schema.ResourceData) (err error) { | ||||
| 	d.SetId(fmt.Sprintf("%d", team.ID)) | ||||
| 	d.Set(TeamCreateRepoFlag, team.CanCreateOrgRepo) | ||||
| 	d.Set(TeamDescription, team.Description) | ||||
| 	d.Set(TeamName, team.Name) | ||||
| 	d.Set(TeamPermissions, string(team.Permission)) | ||||
| 	d.Set(TeamIncludeAllReposFlag, team.IncludesAllRepositories) | ||||
| 	d.Set(TeamUnits, d.Get(TeamUnits).(string)) | ||||
| 	d.Set(TeamOrg, d.Get(TeamOrg).(string)) | ||||
| 	d.Set(TeamMembers, d.Get(TeamMembers)) | ||||
| 	return | ||||
| } | ||||
|  | ||||
| func resourceGiteaTeam() *schema.Resource { | ||||
| 	return &schema.Resource{ | ||||
| 		Read:   resourceTeamRead, | ||||
| 		Create: resourceTeamCreate, | ||||
| 		Update: resourceTeamUpdate, | ||||
| 		Delete: resourceTeamDelete, | ||||
| 		Importer: &schema.ResourceImporter{ | ||||
| 			State: schema.ImportStatePassthrough, | ||||
| 		}, | ||||
| 		Schema: map[string]*schema.Schema{ | ||||
| 			"name": { | ||||
| 				Type:        schema.TypeString, | ||||
| 				Required:    true, | ||||
| 				ForceNew:    true, | ||||
| 				Description: "Name of the Team", | ||||
| 			}, | ||||
| 			"organisation": { | ||||
| 				Type:        schema.TypeString, | ||||
| 				Required:    true, | ||||
| 				ForceNew:    true, | ||||
| 				Description: "The organisation which this Team is part of.", | ||||
| 			}, | ||||
| 			"description": { | ||||
| 				Type:        schema.TypeString, | ||||
| 				Required:    false, | ||||
| 				Optional:    true, | ||||
| 				Default:     "", | ||||
| 				Description: "Description of the Team", | ||||
| 			}, | ||||
| 			"permission": { | ||||
| 				Type:     schema.TypeString, | ||||
| 				Required: false, | ||||
| 				Optional: true, | ||||
| 				Default:  "", | ||||
| 				Description: "Permissions associated with this Team\n" + | ||||
| 					"Can be `none`, `read`, `write`, `admin` or `owner`", | ||||
| 			}, | ||||
| 			"can_create_repos": { | ||||
| 				Type:        schema.TypeBool, | ||||
| 				Required:    false, | ||||
| 				Optional:    true, | ||||
| 				Default:     true, | ||||
| 				Description: "Flag if the Teams members should be able to create Rpositories in the Organisation", | ||||
| 			}, | ||||
| 			"include_all_repositories": { | ||||
| 				Type:        schema.TypeBool, | ||||
| 				Required:    false, | ||||
| 				Optional:    true, | ||||
| 				Default:     true, | ||||
| 				Description: "Flag if the Teams members should have access to all Repositories in the Organisation", | ||||
| 			}, | ||||
| 			"units": { | ||||
| 				Type:     schema.TypeString, | ||||
| 				Required: false, | ||||
| 				Optional: true, | ||||
| 				Default:  "[repo.code, repo.issues, repo.ext_issues, repo.wiki, repo.pulls, repo.releases, repo.projects, repo.ext_wiki]", | ||||
| 				Description: "List of types of Repositories that should be allowed to be created from Team members.\n" + | ||||
| 					"Can be `repo.code`, `repo.issues`, `repo.ext_issues`, `repo.wiki`, `repo.pulls`, `repo.releases`, `repo.projects` and/or `repo.ext_wiki`", | ||||
| 			}, | ||||
| 			"members": { | ||||
| 				Type: schema.TypeList, | ||||
| 				Elem: &schema.Schema{ | ||||
| 					Type: schema.TypeString, | ||||
| 				}, | ||||
| 				Optional:    true, | ||||
| 				Required:    false, | ||||
| 				Computed:    true, | ||||
| 				Description: "List of Users that should be part of this team", | ||||
| 			}, | ||||
| 		}, | ||||
| 		Description: "`gitea_team` manages Team that are part of an organisation.", | ||||
| 	} | ||||
| } | ||||
							
								
								
									
										151
									
								
								gitea/resource_gitea_token.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										151
									
								
								gitea/resource_gitea_token.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,151 @@ | ||||
| package gitea | ||||
|  | ||||
| import ( | ||||
| 	"fmt" | ||||
| 	"strconv" | ||||
|  | ||||
| 	"code.gitea.io/sdk/gitea" | ||||
| 	"github.com/hashicorp/terraform-plugin-sdk/v2/helper/schema" | ||||
| ) | ||||
|  | ||||
| const ( | ||||
| 	TokenUsername  string = "username" | ||||
| 	TokenName      string = "name" | ||||
| 	TokenHash      string = "token" | ||||
| 	TokenLastEight string = "last_eight" | ||||
| ) | ||||
|  | ||||
| func searchTokenById(c *gitea.Client, id int64) (res *gitea.AccessToken, err error) { | ||||
| 	page := 1 | ||||
|  | ||||
| 	for { | ||||
| 		tokens, _, err := c.ListAccessTokens(gitea.ListAccessTokensOptions{ | ||||
| 			ListOptions: gitea.ListOptions{ | ||||
| 				Page:     page, | ||||
| 				PageSize: 50, | ||||
| 			}, | ||||
| 		}) | ||||
| 		if err != nil { | ||||
| 			return nil, err | ||||
| 		} | ||||
|  | ||||
| 		if len(tokens) == 0 { | ||||
| 			return nil, fmt.Errorf("Token with ID %d could not be found", id) | ||||
| 		} | ||||
|  | ||||
| 		for _, token := range tokens { | ||||
| 			if token.ID == id { | ||||
| 				return token, nil | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		page += 1 | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func resourceTokenCreate(d *schema.ResourceData, meta interface{}) (err error) { | ||||
|  | ||||
| 	client := meta.(*gitea.Client) | ||||
|  | ||||
| 	var opt gitea.CreateAccessTokenOption | ||||
| 	opt.Name = d.Get(TokenName).(string) | ||||
|  | ||||
| 	token, _, err := client.CreateAccessToken(opt) | ||||
|  | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
|  | ||||
| 	err = setTokenResourceData(token, d) | ||||
|  | ||||
| 	return | ||||
| } | ||||
|  | ||||
| func resourceTokenRead(d *schema.ResourceData, meta interface{}) (err error) { | ||||
|  | ||||
| 	client := meta.(*gitea.Client) | ||||
|  | ||||
| 	var token *gitea.AccessToken | ||||
|  | ||||
| 	id, err := strconv.ParseInt(d.Id(), 10, 64) | ||||
|  | ||||
| 	token, err = searchTokenById(client, id) | ||||
|  | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
|  | ||||
| 	err = setTokenResourceData(token, d) | ||||
|  | ||||
| 	return | ||||
| } | ||||
|  | ||||
| func resourceTokenDelete(d *schema.ResourceData, meta interface{}) (err error) { | ||||
|  | ||||
| 	client := meta.(*gitea.Client) | ||||
| 	var resp *gitea.Response | ||||
|  | ||||
| 	resp, err = client.DeleteAccessToken(d.Get(TokenName).(string)) | ||||
|  | ||||
| 	if err != nil { | ||||
| 		if resp.StatusCode == 404 { | ||||
| 			return | ||||
| 		} else { | ||||
| 			return err | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	return | ||||
| } | ||||
|  | ||||
| func setTokenResourceData(token *gitea.AccessToken, d *schema.ResourceData) (err error) { | ||||
|  | ||||
| 	d.SetId(fmt.Sprintf("%d", token.ID)) | ||||
| 	d.Set(TokenName, token.Name) | ||||
| 	if token.Token != "" { | ||||
| 		d.Set(TokenHash, token.Token) | ||||
| 	} | ||||
| 	d.Set(TokenLastEight, token.TokenLastEight) | ||||
|  | ||||
| 	return | ||||
| } | ||||
|  | ||||
| func resourceGiteaToken() *schema.Resource { | ||||
| 	return &schema.Resource{ | ||||
| 		Read:   resourceTokenRead, | ||||
| 		Create: resourceTokenCreate, | ||||
| 		Delete: resourceTokenDelete, | ||||
| 		Importer: &schema.ResourceImporter{ | ||||
| 			StateContext: schema.ImportStatePassthroughContext, | ||||
| 		}, | ||||
| 		Schema: map[string]*schema.Schema{ | ||||
| 			"username": { | ||||
| 				Type:        schema.TypeString, | ||||
| 				Required:    true, | ||||
| 				ForceNew:    true, | ||||
| 				Description: "The owner of the Access Token", | ||||
| 			}, | ||||
| 			"name": { | ||||
| 				Type:        schema.TypeString, | ||||
| 				Required:    true, | ||||
| 				ForceNew:    true, | ||||
| 				Description: "The name of the Access Token", | ||||
| 			}, | ||||
| 			"token": { | ||||
| 				Type:        schema.TypeString, | ||||
| 				Computed:    true, | ||||
| 				Sensitive:   true, | ||||
| 				Description: "The actual Access Token", | ||||
| 			}, | ||||
| 			"last_eight": { | ||||
| 				Type:     schema.TypeString, | ||||
| 				Computed: true, | ||||
| 			}, | ||||
| 		}, | ||||
| 		Description: "`gitea_token` manages gitea Access Tokens.\n\n" + | ||||
| 			"Due to upstream limitations (see https://gitea.com/gitea/go-sdk/issues/610) this resource\n" + | ||||
| 			"can only be used with username/password provider configuration.\n\n" + | ||||
| 			"WARNING:\n" + | ||||
| 			"Tokens will be stored in the terraform state!", | ||||
| 	} | ||||
| } | ||||
							
								
								
									
										364
									
								
								gitea/resource_gitea_user.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										364
									
								
								gitea/resource_gitea_user.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,364 @@ | ||||
| package gitea | ||||
|  | ||||
| import ( | ||||
| 	"fmt" | ||||
| 	"strconv" | ||||
|  | ||||
| 	"code.gitea.io/sdk/gitea" | ||||
| 	"github.com/hashicorp/terraform-plugin-sdk/v2/helper/schema" | ||||
| ) | ||||
|  | ||||
| const ( | ||||
| 	userName                string = "username" | ||||
| 	userLoginName           string = "login_name" | ||||
| 	userEmail               string = "email" | ||||
| 	userFullName            string = "full_name" | ||||
| 	userPassword            string = "password" | ||||
| 	userMustChangePassword  string = "must_change_password" | ||||
| 	userSendNotification    string = "send_notification" | ||||
| 	userVisibility          string = "visibility" | ||||
| 	userDescription         string = "description" | ||||
| 	userLocation            string = "location" | ||||
| 	userActive              string = "active" | ||||
| 	userAdmin               string = "admin" | ||||
| 	userAllowGitHook        string = "allow_git_hook" | ||||
| 	userAllowLocalImport    string = "allow_import_local" | ||||
| 	userMaxRepoCreation     string = "max_repo_creation" | ||||
| 	userPhorbitLogin        string = "prohibit_login" | ||||
| 	userAllowCreateOrgs     string = "allow_create_organization" | ||||
| 	userRestricted          string = "restricted" | ||||
| 	userForcePasswordChange string = "force_password_change" | ||||
| ) | ||||
|  | ||||
| func resourceUserRead(d *schema.ResourceData, meta interface{}) (err error) { | ||||
| 	client := meta.(*gitea.Client) | ||||
|  | ||||
| 	id, err := strconv.ParseInt(d.Id(), 10, 64) | ||||
|  | ||||
| 	var resp *gitea.Response | ||||
| 	var user *gitea.User | ||||
|  | ||||
| 	user, resp, err = client.GetUserByID(id) | ||||
|  | ||||
| 	if err != nil { | ||||
| 		if resp.StatusCode == 404 { | ||||
| 			d.SetId("") | ||||
| 			return nil | ||||
| 		} else { | ||||
| 			return err | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	err = setUserResourceData(user, d) | ||||
|  | ||||
| 	return | ||||
| } | ||||
|  | ||||
| func resourceUserCreate(d *schema.ResourceData, meta interface{}) (err error) { | ||||
| 	client := meta.(*gitea.Client) | ||||
|  | ||||
| 	var user *gitea.User | ||||
| 	visibility := gitea.VisibleType(d.Get(userVisibility).(string)) | ||||
| 	changePassword := d.Get(userMustChangePassword).(bool) | ||||
|  | ||||
| 	opts := gitea.CreateUserOption{ | ||||
| 		SourceID:           0, | ||||
| 		LoginName:          d.Get(userLoginName).(string), | ||||
| 		Username:           d.Get(userName).(string), | ||||
| 		FullName:           d.Get(userFullName).(string), | ||||
| 		Email:              d.Get(userEmail).(string), | ||||
| 		Password:           d.Get(userPassword).(string), | ||||
| 		MustChangePassword: &changePassword, | ||||
| 		SendNotify:         d.Get(userSendNotification).(bool), | ||||
| 		Visibility:         &visibility, | ||||
| 	} | ||||
|  | ||||
| 	user, _, err = client.AdminCreateUser(opts) | ||||
| 	if err != nil { | ||||
| 		return | ||||
| 	} | ||||
|  | ||||
| 	d.SetId(fmt.Sprintf("%d", user.ID)) | ||||
|  | ||||
| 	err = resourceUserUpdate(d, meta) | ||||
|  | ||||
| 	return | ||||
| } | ||||
|  | ||||
| func resourceUserUpdate(d *schema.ResourceData, meta interface{}) (err error) { | ||||
| 	client := meta.(*gitea.Client) | ||||
|  | ||||
| 	id, err := strconv.ParseInt(d.Id(), 10, 64) | ||||
| 	var resp *gitea.Response | ||||
| 	var user *gitea.User | ||||
|  | ||||
| 	user, resp, err = client.GetUserByID(id) | ||||
|  | ||||
| 	if err != nil { | ||||
| 		if resp.StatusCode == 404 { | ||||
| 			resourceUserCreate(d, meta) | ||||
| 		} else { | ||||
| 			return err | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	mail := d.Get(userEmail).(string) | ||||
| 	fullName := d.Get(userFullName).(string) | ||||
| 	description := d.Get(userDescription).(string) | ||||
| 	changePassword := d.Get(userMustChangePassword).(bool) | ||||
| 	location := d.Get(userLocation).(string) | ||||
| 	active := d.Get(userActive).(bool) | ||||
| 	admin := d.Get(userAdmin).(bool) | ||||
| 	allowHook := d.Get(userAllowGitHook).(bool) | ||||
| 	allowImport := d.Get(userAllowLocalImport).(bool) | ||||
| 	maxRepoCreation := d.Get(userMaxRepoCreation).(int) | ||||
| 	accessDenied := d.Get(userPhorbitLogin).(bool) | ||||
| 	allowOrgs := d.Get(userAllowCreateOrgs).(bool) | ||||
| 	restricted := d.Get(userRestricted).(bool) | ||||
| 	visibility := gitea.VisibleType(d.Get(userVisibility).(string)) | ||||
|  | ||||
| 	if d.Get(userForcePasswordChange).(bool) { | ||||
| 		opts := gitea.EditUserOption{ | ||||
| 			SourceID:                0, | ||||
| 			LoginName:               d.Get(userLoginName).(string), | ||||
| 			Email:                   &mail, | ||||
| 			FullName:                &fullName, | ||||
| 			Password:                d.Get(userPassword).(string), | ||||
| 			Description:             &description, | ||||
| 			MustChangePassword:      &changePassword, | ||||
| 			Location:                &location, | ||||
| 			Active:                  &active, | ||||
| 			Admin:                   &admin, | ||||
| 			AllowGitHook:            &allowHook, | ||||
| 			AllowImportLocal:        &allowImport, | ||||
| 			MaxRepoCreation:         &maxRepoCreation, | ||||
| 			ProhibitLogin:           &accessDenied, | ||||
| 			AllowCreateOrganization: &allowOrgs, | ||||
| 			Restricted:              &restricted, | ||||
| 			Visibility:              &visibility, | ||||
| 		} | ||||
| 		_, err = client.AdminEditUser(d.Get(userName).(string), opts) | ||||
|  | ||||
| 		if err != nil { | ||||
| 			return err | ||||
| 		} | ||||
|  | ||||
| 	} else { | ||||
| 		opts := gitea.EditUserOption{ | ||||
| 			SourceID:                0, | ||||
| 			LoginName:               d.Get(userLoginName).(string), | ||||
| 			Email:                   &mail, | ||||
| 			FullName:                &fullName, | ||||
| 			Description:             &description, | ||||
| 			MustChangePassword:      &changePassword, | ||||
| 			Location:                &location, | ||||
| 			Active:                  &active, | ||||
| 			Admin:                   &admin, | ||||
| 			AllowGitHook:            &allowHook, | ||||
| 			AllowImportLocal:        &allowImport, | ||||
| 			MaxRepoCreation:         &maxRepoCreation, | ||||
| 			ProhibitLogin:           &accessDenied, | ||||
| 			AllowCreateOrganization: &allowOrgs, | ||||
| 			Restricted:              &restricted, | ||||
| 			Visibility:              &visibility, | ||||
| 		} | ||||
| 		_, err = client.AdminEditUser(d.Get(userName).(string), opts) | ||||
|  | ||||
| 		if err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	user, _, err = client.GetUserByID(id) | ||||
|  | ||||
| 	err = setUserResourceData(user, d) | ||||
|  | ||||
| 	return | ||||
| } | ||||
|  | ||||
| func resourceUserDelete(d *schema.ResourceData, meta interface{}) (err error) { | ||||
| 	client := meta.(*gitea.Client) | ||||
|  | ||||
| 	var resp *gitea.Response | ||||
|  | ||||
| 	resp, err = client.AdminDeleteUser(d.Get(userName).(string)) | ||||
|  | ||||
| 	if err != nil { | ||||
| 		if resp.StatusCode == 404 { | ||||
| 			return | ||||
| 		} else { | ||||
| 			return err | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	return | ||||
| } | ||||
|  | ||||
| func setUserResourceData(user *gitea.User, d *schema.ResourceData) (err error) { | ||||
| 	d.SetId(fmt.Sprintf("%d", user.ID)) | ||||
| 	d.Set(userName, user.UserName) | ||||
| 	d.Set(userEmail, user.Email) | ||||
| 	d.Set(userFullName, user.FullName) | ||||
| 	d.Set(userAdmin, user.IsAdmin) | ||||
| 	d.Set("created", user.Created) | ||||
| 	d.Set("avatar_url", user.AvatarURL) | ||||
| 	d.Set("last_login", user.LastLogin) | ||||
| 	d.Set("language", user.Language) | ||||
| 	d.Set(userLoginName, d.Get(userLoginName).(string)) | ||||
| 	d.Set(userMustChangePassword, d.Get(userMustChangePassword).(bool)) | ||||
| 	d.Set(userSendNotification, d.Get(userSendNotification).(bool)) | ||||
| 	d.Set(userVisibility, d.Get(userVisibility).(string)) | ||||
| 	d.Set(userDescription, d.Get(userDescription).(string)) | ||||
| 	d.Set(userLocation, d.Get(userLocation).(string)) | ||||
| 	d.Set(userActive, d.Get(userActive).(bool)) | ||||
| 	d.Set(userAllowGitHook, d.Get(userAllowGitHook).(bool)) | ||||
| 	d.Set(userAllowLocalImport, d.Get(userAllowLocalImport).(bool)) | ||||
| 	d.Set(userMaxRepoCreation, d.Get(userMaxRepoCreation).(int)) | ||||
| 	d.Set(userPhorbitLogin, d.Get(userPhorbitLogin).(bool)) | ||||
| 	d.Set(userAllowCreateOrgs, d.Get(userAllowCreateOrgs).(bool)) | ||||
| 	d.Set(userRestricted, d.Get(userRestricted).(bool)) | ||||
| 	d.Set(userForcePasswordChange, d.Get(userForcePasswordChange).(bool)) | ||||
|  | ||||
| 	return | ||||
| } | ||||
|  | ||||
| func resourceGiteaUser() *schema.Resource { | ||||
| 	return &schema.Resource{ | ||||
| 		Read:   resourceUserRead, | ||||
| 		Create: resourceUserCreate, | ||||
| 		Update: resourceUserUpdate, | ||||
| 		Delete: resourceUserDelete, | ||||
| 		Importer: &schema.ResourceImporter{ | ||||
| 			State: schema.ImportStatePassthrough, | ||||
| 		}, | ||||
| 		Schema: map[string]*schema.Schema{ | ||||
| 			"username": { | ||||
| 				Type:        schema.TypeString, | ||||
| 				Required:    true, | ||||
| 				ForceNew:    true, | ||||
| 				Description: "Username of the user to be created", | ||||
| 			}, | ||||
| 			"login_name": { | ||||
| 				Type:        schema.TypeString, | ||||
| 				Optional:    false, | ||||
| 				Required:    true, | ||||
| 				Description: "The login name can differ from the username", | ||||
| 			}, | ||||
| 			"email": { | ||||
| 				Type:        schema.TypeString, | ||||
| 				Optional:    false, | ||||
| 				Required:    true, | ||||
| 				Description: "E-Mail Address of the user", | ||||
| 			}, | ||||
| 			"full_name": { | ||||
| 				Type:        schema.TypeString, | ||||
| 				Computed:    true, | ||||
| 				Optional:    true, | ||||
| 				Required:    false, | ||||
| 				Description: "Full name of the user", | ||||
| 			}, | ||||
| 			"password": { | ||||
| 				Type:        schema.TypeString, | ||||
| 				Optional:    false, | ||||
| 				Required:    true, | ||||
| 				Sensitive:   true, | ||||
| 				Description: "Password to be set for the user", | ||||
| 			}, | ||||
| 			"must_change_password": { | ||||
| 				Type:        schema.TypeBool, | ||||
| 				Optional:    true, | ||||
| 				Required:    false, | ||||
| 				Default:     true, | ||||
| 				Description: "Flag if the user should change the password after first login", | ||||
| 			}, | ||||
| 			"send_notification": { | ||||
| 				Type:        schema.TypeBool, | ||||
| 				Optional:    true, | ||||
| 				Required:    false, | ||||
| 				Default:     true, | ||||
| 				Description: "Flag to send a notification about the user creation to the defined `email`", | ||||
| 			}, | ||||
| 			"visibility": { | ||||
| 				Type:        schema.TypeString, | ||||
| 				Optional:    true, | ||||
| 				Required:    false, | ||||
| 				Default:     "public", | ||||
| 				Description: "Visibility of the user. Can be `public`, `limited` or `private`", | ||||
| 			}, | ||||
| 			"description": { | ||||
| 				Type:        schema.TypeString, | ||||
| 				Optional:    true, | ||||
| 				Required:    false, | ||||
| 				Default:     "", | ||||
| 				Description: "A description of the user", | ||||
| 			}, | ||||
| 			"location": { | ||||
| 				Type:     schema.TypeString, | ||||
| 				Optional: true, | ||||
| 				Required: false, | ||||
| 				Default:  "", | ||||
| 			}, | ||||
| 			"active": { | ||||
| 				Type:        schema.TypeBool, | ||||
| 				Optional:    true, | ||||
| 				Required:    false, | ||||
| 				Default:     true, | ||||
| 				Description: "Flag if this user should be active or not", | ||||
| 			}, | ||||
| 			"admin": { | ||||
| 				Type:        schema.TypeBool, | ||||
| 				Optional:    true, | ||||
| 				Required:    false, | ||||
| 				Default:     false, | ||||
| 				Description: "Flag if this user should be an administrator or not", | ||||
| 			}, | ||||
| 			"allow_git_hook": { | ||||
| 				Type:     schema.TypeBool, | ||||
| 				Optional: true, | ||||
| 				Required: false, | ||||
| 				Default:  true, | ||||
| 			}, | ||||
| 			"allow_import_local": { | ||||
| 				Type:     schema.TypeBool, | ||||
| 				Optional: true, | ||||
| 				Required: false, | ||||
| 				Default:  true, | ||||
| 			}, | ||||
| 			"max_repo_creation": { | ||||
| 				Type:     schema.TypeInt, | ||||
| 				Optional: true, | ||||
| 				Required: false, | ||||
| 				Default:  -1, | ||||
| 			}, | ||||
| 			"prohibit_login": { | ||||
| 				Type:        schema.TypeBool, | ||||
| 				Optional:    true, | ||||
| 				Required:    false, | ||||
| 				Default:     false, | ||||
| 				Description: "Flag if the user should not be allowed to log in (bot user)", | ||||
| 			}, | ||||
| 			"allow_create_organization": { | ||||
| 				Type:     schema.TypeBool, | ||||
| 				Optional: true, | ||||
| 				Required: false, | ||||
| 				Default:  true, | ||||
| 			}, | ||||
| 			"restricted": { | ||||
| 				Type:     schema.TypeBool, | ||||
| 				Optional: true, | ||||
| 				Required: false, | ||||
| 				Default:  false, | ||||
| 			}, | ||||
| 			"force_password_change": { | ||||
| 				Type:        schema.TypeBool, | ||||
| 				Optional:    true, | ||||
| 				Required:    false, | ||||
| 				Default:     false, | ||||
| 				Description: "Flag if the user defined password should be overwritten or not", | ||||
| 			}, | ||||
| 		}, | ||||
| 		Description: "`gitea_user` manages a native gitea user.\n\n" + | ||||
| 			"If you are using OIDC or other kinds of authentication mechanisms you can still try to manage" + | ||||
| 			"ssh keys or other ressources this way", | ||||
| 	} | ||||
| } | ||||
							
								
								
									
										75
									
								
								go.mod
									
									
									
									
									
								
							
							
						
						
									
										75
									
								
								go.mod
									
									
									
									
									
								
							| @@ -5,79 +5,64 @@ go 1.18 | ||||
| require ( | ||||
| 	code.gitea.io/sdk/gitea v0.15.1 | ||||
| 	github.com/hashicorp/terraform-plugin-docs v0.7.0 | ||||
| 	github.com/hashicorp/terraform-plugin-sdk v1.13.0 | ||||
| 	github.com/hashicorp/terraform-plugin-sdk/v2 v2.20.0 | ||||
| ) | ||||
|  | ||||
| require ( | ||||
| 	cloud.google.com/go v0.45.1 // indirect | ||||
| 	github.com/Masterminds/goutils v1.1.0 // indirect | ||||
| 	github.com/Masterminds/semver v1.5.0 // indirect | ||||
| 	github.com/Masterminds/sprig v2.22.0+incompatible // indirect | ||||
| 	github.com/agext/levenshtein v1.2.2 // indirect | ||||
| 	github.com/apparentlymart/go-cidr v1.0.1 // indirect | ||||
| 	github.com/apparentlymart/go-textseg v1.0.0 // indirect | ||||
| 	github.com/apparentlymart/go-textseg/v13 v13.0.0 // indirect | ||||
| 	github.com/armon/go-radix v1.0.0 // indirect | ||||
| 	github.com/aws/aws-sdk-go v1.25.3 // indirect | ||||
| 	github.com/bgentry/go-netrc v0.0.0-20140422174119-9fd32a8b3d3d // indirect | ||||
| 	github.com/armon/go-radix v0.0.0-20180808171621-7fddfc383310 // indirect | ||||
| 	github.com/bgentry/speakeasy v0.1.0 // indirect | ||||
| 	github.com/davecgh/go-spew v1.1.1 // indirect | ||||
| 	github.com/fatih/color v1.7.0 // indirect | ||||
| 	github.com/golang/protobuf v1.3.4 // indirect | ||||
| 	github.com/google/go-cmp v0.5.7 // indirect | ||||
| 	github.com/fatih/color v1.13.0 // indirect | ||||
| 	github.com/golang/protobuf v1.5.2 // indirect | ||||
| 	github.com/google/go-cmp v0.5.8 // indirect | ||||
| 	github.com/google/uuid v1.1.2 // indirect | ||||
| 	github.com/googleapis/gax-go/v2 v2.0.5 // indirect | ||||
| 	github.com/hashicorp/errwrap v1.0.0 // indirect | ||||
| 	github.com/hashicorp/go-checkpoint v0.5.0 // indirect | ||||
| 	github.com/hashicorp/go-cleanhttp v0.5.2 // indirect | ||||
| 	github.com/hashicorp/go-getter v1.4.0 // indirect | ||||
| 	github.com/hashicorp/go-hclog v0.9.2 // indirect | ||||
| 	github.com/hashicorp/go-cty v1.4.1-0.20200414143053-d3edf31b6320 // indirect | ||||
| 	github.com/hashicorp/go-hclog v1.2.1 // indirect | ||||
| 	github.com/hashicorp/go-multierror v1.1.1 // indirect | ||||
| 	github.com/hashicorp/go-plugin v1.0.1 // indirect | ||||
| 	github.com/hashicorp/go-safetemp v1.0.0 // indirect | ||||
| 	github.com/hashicorp/go-uuid v1.0.1 // indirect | ||||
| 	github.com/hashicorp/go-version v1.4.0 // indirect | ||||
| 	github.com/hashicorp/golang-lru v0.5.1 // indirect | ||||
| 	github.com/hashicorp/hc-install v0.3.1 // indirect | ||||
| 	github.com/hashicorp/hcl v0.0.0-20170504190234-a4b07c25de5f // indirect | ||||
| 	github.com/hashicorp/hcl/v2 v2.0.0 // indirect | ||||
| 	github.com/hashicorp/go-plugin v1.4.4 // indirect | ||||
| 	github.com/hashicorp/go-uuid v1.0.3 // indirect | ||||
| 	github.com/hashicorp/go-version v1.6.0 // indirect | ||||
| 	github.com/hashicorp/hc-install v0.4.0 // indirect | ||||
| 	github.com/hashicorp/hcl/v2 v2.13.0 // indirect | ||||
| 	github.com/hashicorp/logutils v1.0.0 // indirect | ||||
| 	github.com/hashicorp/terraform-config-inspect v0.0.0-20191115094559-17f92b0546e8 // indirect | ||||
| 	github.com/hashicorp/terraform-exec v0.16.0 // indirect | ||||
| 	github.com/hashicorp/terraform-json v0.13.0 // indirect | ||||
| 	github.com/hashicorp/terraform-plugin-test v1.3.0 // indirect | ||||
| 	github.com/hashicorp/terraform-svchost v0.0.0-20191011084731-65d371908596 // indirect | ||||
| 	github.com/hashicorp/terraform-exec v0.17.2 // indirect | ||||
| 	github.com/hashicorp/terraform-json v0.14.0 // indirect | ||||
| 	github.com/hashicorp/terraform-plugin-go v0.12.0 // indirect | ||||
| 	github.com/hashicorp/terraform-plugin-log v0.7.0 // indirect | ||||
| 	github.com/hashicorp/terraform-registry-address v0.0.0-20220623143253-7d51757b572c // indirect | ||||
| 	github.com/hashicorp/terraform-svchost v0.0.0-20200729002733-f050f53b9734 // indirect | ||||
| 	github.com/hashicorp/yamux v0.0.0-20181012175058-2f1d1f20f75d // indirect | ||||
| 	github.com/huandu/xstrings v1.3.2 // indirect | ||||
| 	github.com/imdario/mergo v0.3.12 // indirect | ||||
| 	github.com/jmespath/go-jmespath v0.0.0-20180206201540-c2b33e8439af // indirect | ||||
| 	github.com/mattn/go-colorable v0.1.12 // indirect | ||||
| 	github.com/mattn/go-isatty v0.0.14 // indirect | ||||
| 	github.com/mitchellh/cli v1.1.2 // indirect | ||||
| 	github.com/mitchellh/colorstring v0.0.0-20190213212951-d06e56a500db // indirect | ||||
| 	github.com/mitchellh/copystructure v1.2.0 // indirect | ||||
| 	github.com/mitchellh/go-homedir v1.1.0 // indirect | ||||
| 	github.com/mitchellh/go-testing-interface v1.0.0 // indirect | ||||
| 	github.com/mitchellh/go-testing-interface v1.14.1 // indirect | ||||
| 	github.com/mitchellh/go-wordwrap v1.0.0 // indirect | ||||
| 	github.com/mitchellh/mapstructure v1.1.2 // indirect | ||||
| 	github.com/mitchellh/mapstructure v1.5.0 // indirect | ||||
| 	github.com/mitchellh/reflectwalk v1.0.2 // indirect | ||||
| 	github.com/oklog/run v1.0.0 // indirect | ||||
| 	github.com/posener/complete v1.2.1 // indirect | ||||
| 	github.com/posener/complete v1.1.1 // indirect | ||||
| 	github.com/russross/blackfriday v1.6.0 // indirect | ||||
| 	github.com/spf13/afero v1.2.2 // indirect | ||||
| 	github.com/ulikunitz/xz v0.5.5 // indirect | ||||
| 	github.com/vmihailenco/msgpack v4.0.4+incompatible // indirect | ||||
| 	github.com/vmihailenco/msgpack/v4 v4.3.12 // indirect | ||||
| 	github.com/vmihailenco/tagparser v0.1.1 // indirect | ||||
| 	github.com/zclconf/go-cty v1.10.0 // indirect | ||||
| 	github.com/zclconf/go-cty-yaml v1.0.1 // indirect | ||||
| 	go.opencensus.io v0.22.0 // indirect | ||||
| 	golang.org/x/crypto v0.0.0-20210616213533-5ff15b29337e // indirect | ||||
| 	golang.org/x/net v0.0.0-20210326060303-6b1517762897 // indirect | ||||
| 	golang.org/x/oauth2 v0.0.0-20190604053449-0f29369cfe45 // indirect | ||||
| 	golang.org/x/sys v0.0.0-20210927094055-39ccf1dd6fa6 // indirect | ||||
| 	golang.org/x/text v0.3.5 // indirect | ||||
| 	google.golang.org/api v0.9.0 // indirect | ||||
| 	google.golang.org/appengine v1.6.5 // indirect | ||||
| 	google.golang.org/genproto v0.0.0-20200310143817-43be25429f5a // indirect | ||||
| 	google.golang.org/grpc v1.27.1 // indirect | ||||
| 	golang.org/x/crypto v0.0.0-20220517005047-85d78b3ac167 // indirect | ||||
| 	golang.org/x/net v0.0.0-20211112202133-69e39bad7dc2 // indirect | ||||
| 	golang.org/x/sys v0.0.0-20220503163025-988cb79eb6c6 // indirect | ||||
| 	golang.org/x/text v0.3.7 // indirect | ||||
| 	google.golang.org/appengine v1.6.6 // indirect | ||||
| 	google.golang.org/genproto v0.0.0-20200711021454-869866162049 // indirect | ||||
| 	google.golang.org/grpc v1.48.0 // indirect | ||||
| 	google.golang.org/protobuf v1.28.0 // indirect | ||||
| ) | ||||
|   | ||||
							
								
								
									
										311
									
								
								go.sum
									
									
									
									
									
								
							
							
						
						
									
										311
									
								
								go.sum
									
									
									
									
									
								
							| @@ -1,17 +1,9 @@ | ||||
| cloud.google.com/go v0.26.0/go.mod h1:aQUYkXzVsufM+DwF1aE+0xfcU+56JwCaLick0ClmMTw= | ||||
| cloud.google.com/go v0.34.0/go.mod h1:aQUYkXzVsufM+DwF1aE+0xfcU+56JwCaLick0ClmMTw= | ||||
| cloud.google.com/go v0.38.0/go.mod h1:990N+gfupTy94rShfmMCWGDn0LpTmnzTp2qbd1dvSRU= | ||||
| cloud.google.com/go v0.44.1/go.mod h1:iSa0KzasP4Uvy3f1mN/7PiObzGgflwredwwASm/v6AU= | ||||
| cloud.google.com/go v0.44.2/go.mod h1:60680Gw3Yr4ikxnPRS/oxxkBccT6SA1yMk63TGekxKY= | ||||
| cloud.google.com/go v0.45.1 h1:lRi0CHyU+ytlvylOlFKKq0af6JncuyoRh1J+QJBqQx0= | ||||
| cloud.google.com/go v0.45.1/go.mod h1:RpBamKRgapWJb87xiFSdk4g1CME7QZg3uwTez+TSTjc= | ||||
| cloud.google.com/go/bigquery v1.0.1/go.mod h1:i/xbL2UlR5RvWAURpBYZTtm/cXjCha9lbfbpx4poX+o= | ||||
| cloud.google.com/go/datastore v1.0.0/go.mod h1:LXYbyblFSglQ5pkeyhO+Qmw7ukd3C+pD7TKLgZqpHYE= | ||||
| code.gitea.io/gitea-vet v0.2.1/go.mod h1:zcNbT/aJEmivCAhfmkHOlT645KNOf9W2KnkLgFjGGfE= | ||||
| code.gitea.io/sdk/gitea v0.15.1 h1:WJreC7YYuxbn0UDaPuWIe/mtiNKTvLN8MLkaw71yx/M= | ||||
| code.gitea.io/sdk/gitea v0.15.1/go.mod h1:klY2LVI3s3NChzIk/MzMn7G1FHrfU7qd63iSMVoHRBA= | ||||
| github.com/BurntSushi/toml v0.3.1/go.mod h1:xHWCNGjB5oqiDr8zfno3MHue2Ht5sIBksp03qcyfWMU= | ||||
| github.com/BurntSushi/xgb v0.0.0-20160522181843-27f122750802/go.mod h1:IVnqGOEym/WlBOVXweHU+Q+/VP0lqqI8lqeDx9IjBqo= | ||||
| github.com/Masterminds/goutils v1.1.0 h1:zukEsf/1JZwCMgHiK3GZftabmxiCw4apj3a28RPBiVg= | ||||
| github.com/Masterminds/goutils v1.1.0/go.mod h1:8cTjp+g8YejhMuvIA5y2vz3BpJxksy863GQaJW2MFNU= | ||||
| github.com/Masterminds/semver v1.5.0 h1:H65muMkzWKEuNDnfl9d70GUjFniHKHRbFPGBuZ3QEww= | ||||
| @@ -25,46 +17,46 @@ github.com/ProtonMail/go-crypto v0.0.0-20210428141323-04723f9f07d7 h1:YoJbenK9C6 | ||||
| github.com/ProtonMail/go-crypto v0.0.0-20210428141323-04723f9f07d7/go.mod h1:z4/9nQmJSSwwds7ejkxaJwO37dru3geImFUdJlaLzQo= | ||||
| github.com/acomagu/bufpipe v1.0.3 h1:fxAGrHZTgQ9w5QqVItgzwj235/uYZYgbXitB+dLupOk= | ||||
| github.com/acomagu/bufpipe v1.0.3/go.mod h1:mxdxdup/WdsKVreO5GpW4+M/1CE2sMG4jeGJ2sYmHc4= | ||||
| github.com/agext/levenshtein v1.2.1/go.mod h1:JEDfjyjHDjOF/1e4FlBE/PkbqA9OfWu2ki2W0IB5558= | ||||
| github.com/agext/levenshtein v1.2.2 h1:0S/Yg6LYmFJ5stwQeRp6EeOcCbj7xiqQSdNelsXvaqE= | ||||
| github.com/agext/levenshtein v1.2.2/go.mod h1:JEDfjyjHDjOF/1e4FlBE/PkbqA9OfWu2ki2W0IB5558= | ||||
| github.com/agl/ed25519 v0.0.0-20170116200512-5312a6153412/go.mod h1:WPjqKcmVOxf0XSf3YxCJs6N6AOSrOx3obionmG7T0y0= | ||||
| github.com/andybalholm/crlf v0.0.0-20171020200849-670099aa064f/go.mod h1:k8feO4+kXDxro6ErPXBRTJ/ro2mf0SsFG8s7doP9kJE= | ||||
| github.com/anmitsu/go-shlex v0.0.0-20161002113705-648efa622239/go.mod h1:2FmKhYUyUczH0OGQWaF5ceTx0UBShxjsH6f8oGKYe2c= | ||||
| github.com/apparentlymart/go-cidr v1.0.1 h1:NmIwLZ/KdsjIUlhf+/Np40atNXm/+lZ5txfTJ/SpF+U= | ||||
| github.com/apparentlymart/go-cidr v1.0.1/go.mod h1:EBcsNrHc3zQeuaeCeCtQruQm+n9/YjEn/vI25Lg7Gwc= | ||||
| github.com/apparentlymart/go-dump v0.0.0-20180507223929-23540a00eaa3/go.mod h1:oL81AME2rN47vu18xqj1S1jPIPuN7afo62yKTNn3XMM= | ||||
| github.com/antihax/optional v1.0.0/go.mod h1:uupD/76wgC+ih3iEmQUL+0Ugr19nfwCT1kdvxnR2qWY= | ||||
| github.com/apparentlymart/go-dump v0.0.0-20190214190832-042adf3cf4a0 h1:MzVXffFUye+ZcSR6opIgz9Co7WcDx6ZcY+RjfFHoA0I= | ||||
| github.com/apparentlymart/go-dump v0.0.0-20190214190832-042adf3cf4a0/go.mod h1:oL81AME2rN47vu18xqj1S1jPIPuN7afo62yKTNn3XMM= | ||||
| github.com/apparentlymart/go-textseg v1.0.0 h1:rRmlIsPEEhUTIKQb7T++Nz/A5Q6C9IuX2wFoYVvnCs0= | ||||
| github.com/apparentlymart/go-textseg v1.0.0/go.mod h1:z96Txxhf3xSFMPmb5X/1W05FF/Nj9VFpLOpjS5yuumk= | ||||
| github.com/apparentlymart/go-textseg/v12 v12.0.0/go.mod h1:S/4uRK2UtaQttw1GenVJEynmyUenKwP++x/+DdGV/Ec= | ||||
| github.com/apparentlymart/go-textseg/v13 v13.0.0 h1:Y+KvPE1NYz0xl601PVImeQfFyEy6iT90AvPUL1NNfNw= | ||||
| github.com/apparentlymart/go-textseg/v13 v13.0.0/go.mod h1:ZK2fH7c4NqDTLtiYLvIkEghdlcqw7yxLeM89kiTRPUo= | ||||
| github.com/armon/go-radix v0.0.0-20180808171621-7fddfc383310 h1:BUAU3CGlLvorLI26FmByPp2eC2qla6E1Tw+scpcg/to= | ||||
| github.com/armon/go-radix v0.0.0-20180808171621-7fddfc383310/go.mod h1:ufUuZ+zHj4x4TnLV4JWEpy2hxWSpsRywHrMgIH9cCH8= | ||||
| github.com/armon/go-radix v1.0.0 h1:F4z6KzEeeQIMeLFa97iZU6vupzoecKdU5TX24SNppXI= | ||||
| github.com/armon/go-radix v1.0.0/go.mod h1:ufUuZ+zHj4x4TnLV4JWEpy2hxWSpsRywHrMgIH9cCH8= | ||||
| github.com/armon/go-socks5 v0.0.0-20160902184237-e75332964ef5/go.mod h1:wHh0iHkYZB8zMSxRWpUBQtwG5a7fFgvEO+odwuTv2gs= | ||||
| github.com/aws/aws-sdk-go v1.15.78/go.mod h1:E3/ieXAlvM0XWO57iftYVDLLvQ824smPP3ATZkfNZeM= | ||||
| github.com/aws/aws-sdk-go v1.25.3 h1:uM16hIw9BotjZKMZlX05SN2EFtaWfi/NonPKIARiBLQ= | ||||
| github.com/aws/aws-sdk-go v1.25.3/go.mod h1:KmX6BPdI08NWTb3/sm4ZGu5ShLoqVDhKgpiN924inxo= | ||||
| github.com/bgentry/go-netrc v0.0.0-20140422174119-9fd32a8b3d3d h1:xDfNPAt8lFiC1UJrqV3uuy861HCTo708pDMbjHHdCas= | ||||
| github.com/bgentry/go-netrc v0.0.0-20140422174119-9fd32a8b3d3d/go.mod h1:6QX/PXZ00z/TKoufEY6K/a0k6AhaJrQKdFe6OfVXsa4= | ||||
| github.com/bgentry/speakeasy v0.1.0 h1:ByYyxL9InA1OWqxJqqp2A5pYHUrCiAL6K3J+LKSsQkY= | ||||
| github.com/bgentry/speakeasy v0.1.0/go.mod h1:+zsyZBPWlz7T6j88CTgSN5bM796AkVf0kBD4zp0CCIs= | ||||
| github.com/census-instrumentation/opencensus-proto v0.2.1/go.mod h1:f6KPmirojxKA12rnyqOA5BBL4O983OfeGPqjHWSTneU= | ||||
| github.com/cheggaaa/pb v1.0.27/go.mod h1:pQciLPpbU0oxA0h+VJYYLxO+XeDQb5pZijXscXHm81s= | ||||
| github.com/cespare/xxhash/v2 v2.1.1/go.mod h1:VGX0DQ3Q6kWi7AoAeZDth3/j3BFtOZR5XLFGgcrjCOs= | ||||
| github.com/client9/misspell v0.3.4/go.mod h1:qj6jICC3Q7zFZvVWo7KLAzC3yx5G7kyvSDkc90ppPyw= | ||||
| github.com/cncf/udpa/go v0.0.0-20191209042840-269d4d468f6f/go.mod h1:M8M6+tZqaGXZJjfX53e64911xZQV5JYwmTeXPW+k8Sc= | ||||
| github.com/cncf/udpa/go v0.0.0-20201120205902-5459f2c99403/go.mod h1:WmhPx2Nbnhtbo57+VJT5O0JRkEi1Wbu0z5j0R8u5Hbk= | ||||
| github.com/cncf/udpa/go v0.0.0-20210930031921-04548b0d99d4/go.mod h1:6pvJx4me5XPnfI9Z40ddWsdw2W/uZgQLFXToKeRcDiI= | ||||
| github.com/cncf/xds/go v0.0.0-20210922020428-25de7278fc84/go.mod h1:eXthEFrGJvWHgFFCl3hGmgk+/aYT6PnTQLykKQRLhEs= | ||||
| github.com/cncf/xds/go v0.0.0-20211001041855-01bcc9b48dfe/go.mod h1:eXthEFrGJvWHgFFCl3hGmgk+/aYT6PnTQLykKQRLhEs= | ||||
| github.com/cncf/xds/go v0.0.0-20211011173535-cb28da3451f1/go.mod h1:eXthEFrGJvWHgFFCl3hGmgk+/aYT6PnTQLykKQRLhEs= | ||||
| github.com/creack/pty v1.1.9/go.mod h1:oKZEueFk5CKHvIhNR5MUki03XCEU+Q6VDXinZuGJ33E= | ||||
| github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= | ||||
| github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c= | ||||
| github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= | ||||
| github.com/emirpasic/gods v1.12.0 h1:QAUIPSaCu4G+POclxeqb3F+WPpdKqFGlw36+yOzGlrg= | ||||
| github.com/emirpasic/gods v1.12.0/go.mod h1:YfzfFFoVP/catgzJb4IKIqXjX78Ha8FMSDh3ymbK86o= | ||||
| github.com/envoyproxy/go-control-plane v0.9.0/go.mod h1:YTl/9mNaCwkRvm6d1a2C3ymFceY/DCBVvsKhRF0iEA4= | ||||
| github.com/envoyproxy/go-control-plane v0.9.1-0.20191026205805-5f8ba28d4473/go.mod h1:YTl/9mNaCwkRvm6d1a2C3ymFceY/DCBVvsKhRF0iEA4= | ||||
| github.com/envoyproxy/go-control-plane v0.9.4/go.mod h1:6rpuAdCZL397s3pYoYcLgu1mIlRU8Am5FuJP05cCM98= | ||||
| github.com/envoyproxy/go-control-plane v0.9.9-0.20201210154907-fd9021fe5dad/go.mod h1:cXg6YxExXjJnVBQHBLXeUAgxn2UodCpnH306RInaBQk= | ||||
| github.com/envoyproxy/go-control-plane v0.10.2-0.20220325020618-49ff273808a1/go.mod h1:KJwIaB5Mv44NWtYuAOFCVOjcI94vtpEz2JU/D2v6IjE= | ||||
| github.com/envoyproxy/protoc-gen-validate v0.1.0/go.mod h1:iSmxcyjqTsJpI2R4NaDN7+kN2VEUnK/pcBlmesArF7c= | ||||
| github.com/fatih/color v1.7.0 h1:DkWD4oS2D8LGGgTQ6IvwJJXSL5Vp2ffcQg58nFV38Ys= | ||||
| github.com/fatih/color v1.7.0/go.mod h1:Zm6kSWBoL9eyXnKyktHP6abPY2pDugNf5KwzbycvMj4= | ||||
| github.com/fatih/color v1.13.0 h1:8LOYc1KYPPmyKMuN8QV2DNRWNbLo6LZ0iLs8+mlH53w= | ||||
| github.com/fatih/color v1.13.0/go.mod h1:kLAiJbzzSOZDVNGyDpeOxJ47H46qBXwg5ILebYFFOfk= | ||||
| github.com/flynn/go-shlex v0.0.0-20150515145356-3f9db97f8568/go.mod h1:xEzjJPgXI435gkrCt3MPfRiAkVrwSbHsst4LCFVfpJc= | ||||
| github.com/ghodss/yaml v1.0.0/go.mod h1:4dBDuWmgqj2HViK6kFavaiC9ZROes6MMH2rRYeMEF04= | ||||
| github.com/gliderlabs/ssh v0.2.2/go.mod h1:U7qILu1NlMHj9FlMhZLlkCdDnU1DBEAqr0aevW3Awn0= | ||||
| github.com/go-git/gcfg v1.5.0 h1:Q5ViNfGF8zFgyJWPqYwA7qGFoMTEiBmdlkcfRmpIMa4= | ||||
| github.com/go-git/gcfg v1.5.0/go.mod h1:5m20vg6GwYabIxaOonVkTdrILxQMpEShl1xiMF4ua+E= | ||||
| @@ -75,39 +67,37 @@ github.com/go-git/go-git-fixtures/v4 v4.2.1/go.mod h1:K8zd3kDUAykwTdDCr+I0per6Y6 | ||||
| github.com/go-git/go-git/v5 v5.4.2 h1:BXyZu9t0VkbiHtqrsvdq39UDhGJTl1h55VW6CSC4aY4= | ||||
| github.com/go-git/go-git/v5 v5.4.2/go.mod h1:gQ1kArt6d+n+BGd+/B/I74HwRTLhth2+zti4ihgckDc= | ||||
| github.com/go-test/deep v1.0.3 h1:ZrJSEWsXzPOxaZnFteGEfooLba+ju3FYIbOrS+rQd68= | ||||
| github.com/go-test/deep v1.0.3/go.mod h1:wGDj63lr65AM2AQyKZd/NYHGb0R+1RLqB8NKt3aSFNA= | ||||
| github.com/golang/glog v0.0.0-20160126235308-23def4e6c14b h1:VKtxabqXZkF25pY9ekfRL6a582T4P37/31XEstQ5p58= | ||||
| github.com/golang/glog v0.0.0-20160126235308-23def4e6c14b/go.mod h1:SBH7ygxi8pfUlaOkMMuAQtPIUF8ecWP5IEl/CR7VP2Q= | ||||
| 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.3.1 h1:qGJ6qTW+x6xX/my+8YUVl4WNpX9B7+/l2tRsHGZ7f2s= | ||||
| github.com/golang/mock v1.3.1/go.mod h1:sBzyDLLjw3U8JLTeZvSv8jJB+tU5PVekmnlKIyFUx0Y= | ||||
| github.com/golang/protobuf v1.1.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.2/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U= | ||||
| github.com/golang/protobuf v1.3.3/go.mod h1:vzj43D7+SQXF/4pzW/hwtAqwc6iTitCiVSaWz5lYuqw= | ||||
| github.com/golang/protobuf v1.3.4 h1:87PNWwrRvUSnqS4dlcBU/ftvOIBep4sYuBLlh6rX2wk= | ||||
| github.com/golang/protobuf v1.3.4/go.mod h1:vzj43D7+SQXF/4pzW/hwtAqwc6iTitCiVSaWz5lYuqw= | ||||
| github.com/golang/snappy v0.0.1/go.mod h1:/XxbfmMg8lxefKM7IXC3fBNl/7bRcc72aCRzEWrmP2Q= | ||||
| github.com/google/btree v0.0.0-20180813153112-4030bb1f1f0c/go.mod h1:lNA+9X1NB3Zf8V7Ke586lFgjr2dZNuvo3lPJSGZ5JPQ= | ||||
| github.com/google/btree v1.0.0/go.mod h1:lNA+9X1NB3Zf8V7Ke586lFgjr2dZNuvo3lPJSGZ5JPQ= | ||||
| github.com/golang/protobuf v1.4.0-rc.1/go.mod h1:ceaxUfeHdC40wWswd/P6IGgMaK3YpKi5j83Wpe3EHw8= | ||||
| github.com/golang/protobuf v1.4.0-rc.1.0.20200221234624-67d41d38c208/go.mod h1:xKAWHe0F5eneWXFV3EuXVDTCmh+JuBKY0li0aMyXATA= | ||||
| github.com/golang/protobuf v1.4.0-rc.2/go.mod h1:LlEzMj4AhA7rCAGe4KMBDvJI+AwstrUpVNzEA03Pprs= | ||||
| github.com/golang/protobuf v1.4.0-rc.4.0.20200313231945-b860323f09d0/go.mod h1:WU3c8KckQ9AFe+yFwt9sWVRKCVIyN9cPHBJSNnbL67w= | ||||
| github.com/golang/protobuf v1.4.0/go.mod h1:jodUvKwWbYaEsadDk5Fwe5c77LiNKVO9IDvqG2KuDX0= | ||||
| github.com/golang/protobuf v1.4.1/go.mod h1:U8fpvMrcmy5pZrNK1lt4xCsGvpyWQ/VVv6QDs8UjoX8= | ||||
| github.com/golang/protobuf v1.4.2/go.mod h1:oDoupMAO8OvCJWAcko0GGGIgR6R6ocIYbsSw735rRwI= | ||||
| github.com/golang/protobuf v1.4.3/go.mod h1:oDoupMAO8OvCJWAcko0GGGIgR6R6ocIYbsSw735rRwI= | ||||
| github.com/golang/protobuf v1.5.0/go.mod h1:FsONVRAS9T7sI+LIUmWTfcYkHO4aIWwzhcaSAoJOfIk= | ||||
| github.com/golang/protobuf v1.5.2 h1:ROPKBNFfQgOUMifHyP+KYbvpjbdoFNs+aK7DXlji0Tw= | ||||
| github.com/golang/protobuf v1.5.2/go.mod h1:XVQd3VNwM+JqD3oG2Ue2ip4fOMUkwXdXDdiuN0vRsmY= | ||||
| github.com/google/go-cmp v0.2.0/go.mod h1:oXzfMopK8JAjlY9xF4vHSVASa0yLyX7SntLO5aqRK0M= | ||||
| github.com/google/go-cmp v0.3.0/go.mod h1:8QqcDgzrUqlUb/G2PQTWiueGozuR1884gddMywk6iLU= | ||||
| github.com/google/go-cmp v0.3.1/go.mod h1:8QqcDgzrUqlUb/G2PQTWiueGozuR1884gddMywk6iLU= | ||||
| github.com/google/go-cmp v0.4.0/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= | ||||
| github.com/google/go-cmp v0.5.0/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= | ||||
| github.com/google/go-cmp v0.5.5/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= | ||||
| github.com/google/go-cmp v0.5.6/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= | ||||
| github.com/google/go-cmp v0.5.7 h1:81/ik6ipDQS2aGcBfIN5dHDB36BwrStyeAQquSYCV4o= | ||||
| github.com/google/go-cmp v0.5.7/go.mod h1:n+brtR0CgQNWTVd5ZUFpTBC8YFBDLK/h/bpaJ8/DtOE= | ||||
| github.com/google/martian v2.1.0+incompatible h1:/CP5g8u/VJHijgedC/Legn3BAbAaWPgecwXBIDzw5no= | ||||
| github.com/google/martian v2.1.0+incompatible/go.mod h1:9I4somxYTbIHy5NJKHRl3wXiIaQGbYVAs8BPL6v8lEs= | ||||
| github.com/google/pprof v0.0.0-20181206194817-3ea8567a2e57/go.mod h1:zfwlbNMJ+OItoe0UupaVj+oy1omPYYDuagoSzA8v9mc= | ||||
| github.com/google/pprof v0.0.0-20190515194954-54271f7e092f/go.mod h1:zfwlbNMJ+OItoe0UupaVj+oy1omPYYDuagoSzA8v9mc= | ||||
| github.com/google/uuid v1.1.1/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo= | ||||
| github.com/google/go-cmp v0.5.8 h1:e6P7q2lk1O+qJJb4BtCQXlK8vWEO8V1ZeuEdJNOqZyg= | ||||
| github.com/google/go-cmp v0.5.8/go.mod h1:17dUlkBOakJ0+DkrSSNjCkIjxS6bF9zb3elmeNGIjoY= | ||||
| github.com/google/uuid v1.1.2 h1:EVhdT+1Kseyi1/pUmXKaFxYsDNy9RQYkMWRH68J/W7Y= | ||||
| github.com/google/uuid v1.1.2/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo= | ||||
| github.com/googleapis/gax-go/v2 v2.0.4/go.mod h1:0Wqv26UfaUD9n4G6kQubkQ+KchISgw+vpHVxEJEs9eg= | ||||
| github.com/googleapis/gax-go/v2 v2.0.5 h1:sjZBwGj9Jlw33ImPtvFviGYvseOtDM7hkSKB7+Tv3SM= | ||||
| github.com/googleapis/gax-go/v2 v2.0.5/go.mod h1:DWXyrwAJ9X0FpwwEdw+IPEYBICEFu5mhpdKc/us6bOk= | ||||
| github.com/grpc-ecosystem/grpc-gateway v1.16.0/go.mod h1:BDjrQk3hbvj6Nolgz8mAMFbcEtjT1g+wF4CSlocrBnw= | ||||
| github.com/hashicorp/errwrap v1.0.0 h1:hLrqtEDnRye3+sgx6z4qVLNuviH3MR5aQ0ykNJa/UYA= | ||||
| github.com/hashicorp/errwrap v1.0.0/go.mod h1:YH+1FKiLXxHSkmPseP+kNlulaMuP3n2brvKWEqk/Jc4= | ||||
| github.com/hashicorp/go-checkpoint v0.5.0 h1:MFYpPZCnQqQTE18jFwSII6eUQrD/oxMFp3mlgcqk5mU= | ||||
| @@ -116,54 +106,45 @@ github.com/hashicorp/go-cleanhttp v0.5.0/go.mod h1:JpRdi6/HCYpAwUzNwuwqhbovhLtng | ||||
| github.com/hashicorp/go-cleanhttp v0.5.1/go.mod h1:JpRdi6/HCYpAwUzNwuwqhbovhLtngrth3wmdIIUrZ80= | ||||
| github.com/hashicorp/go-cleanhttp v0.5.2 h1:035FKYIWjmULyFRBKPs8TBQoi0x6d9G4xc9neXJWAZQ= | ||||
| github.com/hashicorp/go-cleanhttp v0.5.2/go.mod h1:kO/YDlP8L1346E6Sodw+PrpBSV4/SoxCXGY6BqNFT48= | ||||
| github.com/hashicorp/go-getter v1.4.0 h1:ENHNi8494porjD0ZhIrjlAHnveSFhY7hvOJrV/fsKkw= | ||||
| github.com/hashicorp/go-getter v1.4.0/go.mod h1:7qxyCd8rBfcShwsvxgIguu4KbS3l8bUCwg2Umn7RjeY= | ||||
| github.com/hashicorp/go-hclog v0.0.0-20180709165350-ff2cf002a8dd/go.mod h1:9bjs9uLqI8l75knNv3lV1kA55veR+WUPSiKIWcQHudI= | ||||
| github.com/hashicorp/go-hclog v0.9.2 h1:CG6TE5H9/JXsFWJCfoIVpKFIkFe6ysEuHirp4DxCsHI= | ||||
| github.com/hashicorp/go-hclog v0.9.2/go.mod h1:5CU+agLiy3J7N7QjHK5d05KxGsuXiQLrjA0H7acj2lQ= | ||||
| github.com/hashicorp/go-cty v1.4.1-0.20200414143053-d3edf31b6320 h1:1/D3zfFHttUKaCaGKZ/dR2roBXv0vKbSCnssIldfQdI= | ||||
| github.com/hashicorp/go-cty v1.4.1-0.20200414143053-d3edf31b6320/go.mod h1:EiZBMaudVLy8fmjf9Npq1dq9RalhveqZG5w/yz3mHWs= | ||||
| github.com/hashicorp/go-hclog v1.2.1 h1:YQsLlGDJgwhXFpucSPyVbCBviQtjlHv3jLTlp8YmtEw= | ||||
| github.com/hashicorp/go-hclog v1.2.1/go.mod h1:W4Qnvbt70Wk/zYJryRzDRU/4r0kIg0PVHBcfoyhpF5M= | ||||
| github.com/hashicorp/go-multierror v1.0.0/go.mod h1:dHtQlpGsu+cZNNAkkCN/P3hoUDHhCYQXV3UM06sGGrk= | ||||
| github.com/hashicorp/go-multierror v1.1.1 h1:H5DkEtf6CXdFp0N0Em5UCwQpXMWke8IA0+lD48awMYo= | ||||
| github.com/hashicorp/go-multierror v1.1.1/go.mod h1:iw975J/qwKPdAO1clOe2L8331t/9/fmwbPZ6JB6eMoM= | ||||
| github.com/hashicorp/go-plugin v1.0.1 h1:4OtAfUGbnKC6yS48p0CtMX2oFYtzFZVv6rok3cRWgnE= | ||||
| github.com/hashicorp/go-plugin v1.0.1/go.mod h1:++UyYGoz3o5w9ZzAdZxtQKrWWP+iqPBn3cQptSMzBuY= | ||||
| github.com/hashicorp/go-safetemp v1.0.0 h1:2HR189eFNrjHQyENnQMMpCiBAsRxzbTMIgBhEyExpmo= | ||||
| github.com/hashicorp/go-safetemp v1.0.0/go.mod h1:oaerMy3BhqiTbVye6QuFhFtIceqFoDHxNAB65b+Rj1I= | ||||
| github.com/hashicorp/go-plugin v1.4.4 h1:NVdrSdFRt3SkZtNckJ6tog7gbpRrcbOjQi/rgF7JYWQ= | ||||
| github.com/hashicorp/go-plugin v1.4.4/go.mod h1:viDMjcLJuDui6pXb8U4HVfb8AamCWhHGUjr2IrTF67s= | ||||
| github.com/hashicorp/go-uuid v1.0.0/go.mod h1:6SBZvOh/SIDV7/2o3Jml5SYk/TvGqwFJ/bN7x4byOro= | ||||
| github.com/hashicorp/go-uuid v1.0.1 h1:fv1ep09latC32wFoVwnqcnKJGnMSdBanPczbHAYm1BE= | ||||
| github.com/hashicorp/go-uuid v1.0.1/go.mod h1:6SBZvOh/SIDV7/2o3Jml5SYk/TvGqwFJ/bN7x4byOro= | ||||
| github.com/hashicorp/go-version v1.1.0/go.mod h1:fltr4n8CU8Ke44wwGCBoEymUuxUHl09ZGVZPK5anwXA= | ||||
| github.com/hashicorp/go-uuid v1.0.3 h1:2gKiV6YVmrJ1i2CKKa9obLvRieoRGviZFL26PcT/Co8= | ||||
| github.com/hashicorp/go-uuid v1.0.3/go.mod h1:6SBZvOh/SIDV7/2o3Jml5SYk/TvGqwFJ/bN7x4byOro= | ||||
| github.com/hashicorp/go-version v1.2.0/go.mod h1:fltr4n8CU8Ke44wwGCBoEymUuxUHl09ZGVZPK5anwXA= | ||||
| github.com/hashicorp/go-version v1.2.1/go.mod h1:fltr4n8CU8Ke44wwGCBoEymUuxUHl09ZGVZPK5anwXA= | ||||
| github.com/hashicorp/go-version v1.3.0/go.mod h1:fltr4n8CU8Ke44wwGCBoEymUuxUHl09ZGVZPK5anwXA= | ||||
| github.com/hashicorp/go-version v1.4.0 h1:aAQzgqIrRKRa7w75CKpbBxYsmUoPjzVm1W59ca1L0J4= | ||||
| github.com/hashicorp/go-version v1.4.0/go.mod h1:fltr4n8CU8Ke44wwGCBoEymUuxUHl09ZGVZPK5anwXA= | ||||
| github.com/hashicorp/golang-lru v0.5.0/go.mod h1:/m3WP610KZHVQ1SGc6re/UDhFvYD7pJ4Ao+sR/qLZy8= | ||||
| github.com/hashicorp/golang-lru v0.5.1 h1:0hERBMJE1eitiLkihrMvRVBYAkpHzc/J3QdDN+dAcgU= | ||||
| github.com/hashicorp/golang-lru v0.5.1/go.mod h1:/m3WP610KZHVQ1SGc6re/UDhFvYD7pJ4Ao+sR/qLZy8= | ||||
| github.com/hashicorp/hc-install v0.3.1 h1:VIjllE6KyAI1A244G8kTaHXy+TL5/XYzvrtFi8po/Yk= | ||||
| github.com/hashicorp/hc-install v0.3.1/go.mod h1:3LCdWcCDS1gaHC9mhHCGbkYfoY6vdsKohGjugbZdZak= | ||||
| github.com/hashicorp/hcl v0.0.0-20170504190234-a4b07c25de5f h1:UdxlrJz4JOnY8W+DbLISwf2B8WXEolNRA8BGCwI9jws= | ||||
| github.com/hashicorp/hcl v0.0.0-20170504190234-a4b07c25de5f/go.mod h1:oZtUIOe8dh44I2q6ScRibXws4Ajl+d+nod3AaR9vL5w= | ||||
| github.com/hashicorp/hcl/v2 v2.0.0 h1:efQznTz+ydmQXq3BOnRa3AXzvCeTq1P4dKj/z5GLlY8= | ||||
| github.com/hashicorp/hcl/v2 v2.0.0/go.mod h1:oVVDG71tEinNGYCxinCYadcmKU9bglqW9pV3txagJ90= | ||||
| github.com/hashicorp/go-version v1.5.0/go.mod h1:fltr4n8CU8Ke44wwGCBoEymUuxUHl09ZGVZPK5anwXA= | ||||
| github.com/hashicorp/go-version v1.6.0 h1:feTTfFNnjP967rlCxM/I9g701jU+RN74YKx2mOkIeek= | ||||
| github.com/hashicorp/go-version v1.6.0/go.mod h1:fltr4n8CU8Ke44wwGCBoEymUuxUHl09ZGVZPK5anwXA= | ||||
| github.com/hashicorp/hc-install v0.4.0 h1:cZkRFr1WVa0Ty6x5fTvL1TuO1flul231rWkGH92oYYk= | ||||
| github.com/hashicorp/hc-install v0.4.0/go.mod h1:5d155H8EC5ewegao9A4PUTMNPZaq+TbOzkJJZ4vrXeI= | ||||
| github.com/hashicorp/hcl/v2 v2.13.0 h1:0Apadu1w6M11dyGFxWnmhhcMjkbAiKCv7G1r/2QgCNc= | ||||
| github.com/hashicorp/hcl/v2 v2.13.0/go.mod h1:e4z5nxYlWNPdDSNYX+ph14EvWYMFm3eP0zIUqPc2jr0= | ||||
| github.com/hashicorp/logutils v1.0.0 h1:dLEQVugN8vlakKOUE3ihGLTZJRB4j+M2cdTm/ORI65Y= | ||||
| github.com/hashicorp/logutils v1.0.0/go.mod h1:QIAnNjmIWmVIIkWDTG1z5v++HQmx9WQRO+LraFDTW64= | ||||
| github.com/hashicorp/terraform-config-inspect v0.0.0-20191115094559-17f92b0546e8 h1:+RyjwU+Gnd/aTJBPZVDNm903eXVjjqhbaR4Ypx3xYyY= | ||||
| github.com/hashicorp/terraform-config-inspect v0.0.0-20191115094559-17f92b0546e8/go.mod h1:p+ivJws3dpqbp1iP84+npOyAmTTOLMgCzrXd3GSdn/A= | ||||
| github.com/hashicorp/terraform-exec v0.16.0 h1:XUh9pJPcbfZsuhReVvmRarQTaiiCnYogFCCjOvEYuug= | ||||
| github.com/hashicorp/terraform-exec v0.16.0/go.mod h1:wB5JHmjxZ/YVNZuv9npAXKmz5pGyxy8PSi0GRR0+YjA= | ||||
| github.com/hashicorp/terraform-json v0.4.0/go.mod h1:eAbqb4w0pSlRmdvl8fOyHAi/+8jnkVYN28gJkSJrLhU= | ||||
| github.com/hashicorp/terraform-json v0.13.0 h1:Li9L+lKD1FO5RVFRM1mMMIBDoUHslOniyEi5CM+FWGY= | ||||
| github.com/hashicorp/terraform-json v0.13.0/go.mod h1:y5OdLBCT+rxbwnpxZs9kGL7R9ExU76+cpdY8zHwoazk= | ||||
| github.com/hashicorp/terraform-exec v0.17.2 h1:EU7i3Fh7vDUI9nNRdMATCEfnm9axzTnad8zszYZ73Go= | ||||
| github.com/hashicorp/terraform-exec v0.17.2/go.mod h1:tuIbsL2l4MlwwIZx9HPM+LOV9vVyEfBYu2GsO1uH3/8= | ||||
| github.com/hashicorp/terraform-json v0.14.0 h1:sh9iZ1Y8IFJLx+xQiKHGud6/TSUCM0N8e17dKDpqV7s= | ||||
| github.com/hashicorp/terraform-json v0.14.0/go.mod h1:5A9HIWPkk4e5aeeXIBbkcOvaZbIYnAIkEyqP2pNSckM= | ||||
| github.com/hashicorp/terraform-plugin-docs v0.7.0 h1:7XKAOYHAxghe7q4/vx468X43X9GikdQ2dxtmcu2gQv0= | ||||
| github.com/hashicorp/terraform-plugin-docs v0.7.0/go.mod h1:57CICKfW7/KbW4lPhKOledyT6vu1LeAOzuvWXsVaxUE= | ||||
| github.com/hashicorp/terraform-plugin-sdk v1.13.0 h1:8v2/ZNiI12OHxEn8pzJ3noCHyRc0biKbKj+iFv5ZWKw= | ||||
| github.com/hashicorp/terraform-plugin-sdk v1.13.0/go.mod h1:HiWIPD/T9HixIhQUwaSoDQxo4BLFdmiBi/Qz5gjB8Q0= | ||||
| github.com/hashicorp/terraform-plugin-test v1.3.0 h1:hU5LoxrOn9qvOo+LTKN6mSav2J+dAMprbdxJPEQvp4U= | ||||
| github.com/hashicorp/terraform-plugin-test v1.3.0/go.mod h1:QIJHYz8j+xJtdtLrFTlzQVC0ocr3rf/OjIpgZLK56Hs= | ||||
| github.com/hashicorp/terraform-svchost v0.0.0-20191011084731-65d371908596 h1:hjyO2JsNZUKT1ym+FAdlBEkGPevazYsmVgIMw7dVELg= | ||||
| github.com/hashicorp/terraform-svchost v0.0.0-20191011084731-65d371908596/go.mod h1:kNDNcF7sN4DocDLBkQYz73HGKwN1ANB1blq4lIYLYvg= | ||||
| github.com/hashicorp/yamux v0.0.0-20180604194846-3520598351bb/go.mod h1:+NfK9FKeTrX5uv1uIXGdwYDTeHna2qgaIlx54MXqjAM= | ||||
| github.com/hashicorp/terraform-plugin-go v0.12.0 h1:6wW9mT1dSs0Xq4LR6HXj1heQ5ovr5GxXNJwkErZzpJw= | ||||
| github.com/hashicorp/terraform-plugin-go v0.12.0/go.mod h1:kwhmaWHNDvT1B3QiSJdAtrB/D4RaKSY/v3r2BuoWK4M= | ||||
| github.com/hashicorp/terraform-plugin-log v0.7.0 h1:SDxJUyT8TwN4l5b5/VkiTIaQgY6R+Y2BQ0sRZftGKQs= | ||||
| github.com/hashicorp/terraform-plugin-log v0.7.0/go.mod h1:p4R1jWBXRTvL4odmEkFfDdhUjHf9zcs/BCoNHAc7IK4= | ||||
| github.com/hashicorp/terraform-plugin-sdk/v2 v2.20.0 h1:+KxZULPsbjpAVoP0WNj/8aVW6EqpcX5JcUcQ5wl7Da4= | ||||
| github.com/hashicorp/terraform-plugin-sdk/v2 v2.20.0/go.mod h1:DwGJG3KNxIPluVk6hexvDfYR/MS/eKGpiztJoT3Bbbw= | ||||
| github.com/hashicorp/terraform-registry-address v0.0.0-20220623143253-7d51757b572c h1:D8aRO6+mTqHfLsK/BC3j5OAoogv1WLRWzY1AaTo3rBg= | ||||
| github.com/hashicorp/terraform-registry-address v0.0.0-20220623143253-7d51757b572c/go.mod h1:Wn3Na71knbXc1G8Lh+yu/dQWWJeFQEpDeJMtWMtlmNI= | ||||
| github.com/hashicorp/terraform-svchost v0.0.0-20200729002733-f050f53b9734 h1:HKLsbzeOsfXmKNpr3GiT18XAblV0BjCbzL8KQAMZGa0= | ||||
| github.com/hashicorp/terraform-svchost v0.0.0-20200729002733-f050f53b9734/go.mod h1:kNDNcF7sN4DocDLBkQYz73HGKwN1ANB1blq4lIYLYvg= | ||||
| github.com/hashicorp/yamux v0.0.0-20181012175058-2f1d1f20f75d h1:kJCB4vdITiW1eC1vq2e6IsrXKrZit1bv/TDYFGMp4BQ= | ||||
| github.com/hashicorp/yamux v0.0.0-20181012175058-2f1d1f20f75d/go.mod h1:+NfK9FKeTrX5uv1uIXGdwYDTeHna2qgaIlx54MXqjAM= | ||||
| github.com/huandu/xstrings v1.3.2 h1:L18LIDzqlW6xN2rEkpdV8+oL/IXWJ1APd+vsdYy4Wdw= | ||||
| @@ -174,13 +155,9 @@ github.com/imdario/mergo v0.3.12/go.mod h1:jmQim1M+e3UYxmgPu/WyfjB3N3VflVyUjjjwH | ||||
| github.com/jbenet/go-context v0.0.0-20150711004518-d14ea06fba99 h1:BQSFePA1RWJOlocH6Fxy8MmwDt+yVQYULKfN0RoTN8A= | ||||
| github.com/jbenet/go-context v0.0.0-20150711004518-d14ea06fba99/go.mod h1:1lJo3i6rXxKeerYnT8Nvf0QmHCRC1n8sfWVwXF2Frvo= | ||||
| github.com/jessevdk/go-flags v1.5.0/go.mod h1:Fw0T6WPc1dYxT4mKEZRfG5kJhaTDP9pj1c2EWnYs/m4= | ||||
| github.com/jmespath/go-jmespath v0.0.0-20160202185014-0b12d6b521d8/go.mod h1:Nht3zPeWKUH0NzdCt2Blrr5ys8VGpn0CEB0cQHVjt7k= | ||||
| github.com/jmespath/go-jmespath v0.0.0-20180206201540-c2b33e8439af h1:pmfjZENx5imkbgOkpRUYLnmbU7UEFbjtDA2hxJ1ichM= | ||||
| github.com/jmespath/go-jmespath v0.0.0-20180206201540-c2b33e8439af/go.mod h1:Nht3zPeWKUH0NzdCt2Blrr5ys8VGpn0CEB0cQHVjt7k= | ||||
| github.com/jstemmer/go-junit-report v0.0.0-20190106144839-af01ea7f8024/go.mod h1:6v2b51hI/fHJwM22ozAgKL4VKDeJcHhJFhtBdhmNjmU= | ||||
| github.com/jhump/protoreflect v1.6.0 h1:h5jfMVslIg6l29nsMs0D8Wj17RDVdNYti0vDN/PZZoE= | ||||
| github.com/kevinburke/ssh_config v0.0.0-20201106050909-4977a11b4351 h1:DowS9hvgyYSX4TO5NpyC606/Z4SxnNYbT+WX27or6Ck= | ||||
| github.com/kevinburke/ssh_config v0.0.0-20201106050909-4977a11b4351/go.mod h1:CT57kijsi8u/K/BOFA39wgDQJ9CxiF4nAY/ojJ6r6mM= | ||||
| github.com/keybase/go-crypto v0.0.0-20161004153544-93f5b35093ba/go.mod h1:ghbZscTyKdM07+Fw3KSi0hcJm+AlEUWj8QLlPtijN/M= | ||||
| github.com/konsorten/go-windows-terminal-sequences v1.0.1/go.mod h1:T0+1ngSBFLxvqU3pZ+m/2kptfBszLMUkC4ZK/EgS/cQ= | ||||
| github.com/kr/pretty v0.1.0/go.mod h1:dAy3ld7l9f0ibDNOQOHHMYYIIbhfbHSm3C4ZsoJORNo= | ||||
| github.com/kr/pretty v0.2.1 h1:Fmg33tUaq4/8ym9TJN1x7sLJnHVwhP33CNkpYV/7rwI= | ||||
| @@ -191,77 +168,62 @@ github.com/kr/text v0.2.0 h1:5Nx0Ya0ZqY2ygV366QzturHI13Jq95ApcVaJBhpS+AY= | ||||
| github.com/kr/text v0.2.0/go.mod h1:eLer722TekiGuMkidMxC/pM04lWEeraHUUmBw8l2grE= | ||||
| github.com/kylelemons/godebug v0.0.0-20170820004349-d65d576e9348/go.mod h1:B69LEHPfb2qLo0BaaOLcbitczOKLWTsrBG9LczfCD4k= | ||||
| github.com/kylelemons/godebug v1.1.0 h1:RPNrshWIDI6G2gRW9EHilWtl7Z6Sb1BR0xunSBf0SNc= | ||||
| github.com/kylelemons/godebug v1.1.0/go.mod h1:9/0rRGxNHcop5bhtWyNeEfOS8JIWk580+fNqagV/RAw= | ||||
| github.com/matryer/is v1.2.0/go.mod h1:2fLPjFQM9rhQ15aVEtbuwhJinnOqrmgXPNdZsdwlWXA= | ||||
| github.com/mattn/go-colorable v0.0.9/go.mod h1:9vuHe8Xs5qXnSaW/c/ABM9alt+Vo+STaOChaDxuIBZU= | ||||
| github.com/mattn/go-colorable v0.1.1/go.mod h1:FuOcm+DKB9mbwrcAfNl7/TZVBZ6rcnceauSikq3lYCQ= | ||||
| github.com/mattn/go-colorable v0.1.9/go.mod h1:u6P/XSegPjTcexA+o6vUJrdnUu04hMope9wVRipJSqc= | ||||
| github.com/mattn/go-colorable v0.1.12 h1:jF+Du6AlPIjs2BiUiQlKOX0rt3SujHxPnksPKZbaA40= | ||||
| github.com/mattn/go-colorable v0.1.12/go.mod h1:u5H1YNBxpqRaxsYJYSkiCWKzEfiAb1Gb520KVy5xxl4= | ||||
| github.com/mattn/go-isatty v0.0.3/go.mod h1:M+lRXTBqGeGNdLjl/ufCoiOlB5xdOkqRJdNxMWT7Zi4= | ||||
| github.com/mattn/go-isatty v0.0.4/go.mod h1:M+lRXTBqGeGNdLjl/ufCoiOlB5xdOkqRJdNxMWT7Zi4= | ||||
| github.com/mattn/go-isatty v0.0.5/go.mod h1:Iq45c/XA43vh69/j3iqttzPXn0bhXyGjM0Hdxcsrc5s= | ||||
| github.com/mattn/go-isatty v0.0.12/go.mod h1:cbi8OIDigv2wuxKPP5vlRcQ1OAZbq2CE4Kysco4FUpU= | ||||
| github.com/mattn/go-isatty v0.0.14 h1:yVuAays6BHfxijgZPzw+3Zlu5yQgKGP2/hcQbHb7S9Y= | ||||
| github.com/mattn/go-isatty v0.0.14/go.mod h1:7GGIvUiUoEMVVmxf/4nioHXj79iQHKdU27kJ6hsGG94= | ||||
| github.com/mattn/go-runewidth v0.0.4/go.mod h1:LwmH8dsx7+W8Uxz3IHJYH5QSwggIsqBzpuz5H//U1FU= | ||||
| github.com/mitchellh/cli v1.0.0/go.mod h1:hNIlj7HEI86fIcpObd7a0FcrxTWetlwJDGcceTlRvqc= | ||||
| github.com/mitchellh/cli v1.1.2 h1:PvH+lL2B7IQ101xQL63Of8yFS2y+aDlsFcsqNc+u/Kw= | ||||
| github.com/mitchellh/cli v1.1.2/go.mod h1:6iaV0fGdElS6dPBx0EApTxHrcWvmJphyh2n8YBLPPZ4= | ||||
| github.com/mitchellh/colorstring v0.0.0-20190213212951-d06e56a500db h1:62I3jR2EmQ4l5rM/4FEfDWcRD+abF5XlKShorW5LRoQ= | ||||
| github.com/mitchellh/colorstring v0.0.0-20190213212951-d06e56a500db/go.mod h1:l0dey0ia/Uv7NcFFVbCLtqEBQbrT4OCwCSKTEv6enCw= | ||||
| github.com/mitchellh/copystructure v1.0.0/go.mod h1:SNtv71yrdKgLRyLFxmLdkAbkKEFWgYaq1OVrnRcwhnw= | ||||
| github.com/mitchellh/copystructure v1.2.0 h1:vpKXTN4ewci03Vljg/q9QvCGUDttBOGBIa15WveJJGw= | ||||
| github.com/mitchellh/copystructure v1.2.0/go.mod h1:qLl+cE2AmVv+CoeAwDPye/v+N2HKCj9FbZEVFJRxO9s= | ||||
| github.com/mitchellh/go-homedir v1.0.0/go.mod h1:SfyaCUpYCn1Vlf4IUYiD9fPX4A5wJrkLzIz1N1q0pr0= | ||||
| github.com/mitchellh/go-homedir v1.1.0 h1:lukF9ziXFxDFPkA1vsr5zpc1XuPDn/wFntq5mG+4E0Y= | ||||
| github.com/mitchellh/go-homedir v1.1.0/go.mod h1:SfyaCUpYCn1Vlf4IUYiD9fPX4A5wJrkLzIz1N1q0pr0= | ||||
| github.com/mitchellh/go-testing-interface v0.0.0-20171004221916-a61a99592b77/go.mod h1:kRemZodwjscx+RGhAo8eIhFbs2+BFgRtFPeD/KE+zxI= | ||||
| github.com/mitchellh/go-testing-interface v1.0.0 h1:fzU/JVNcaqHQEcVFAKeR41fkiLdIPrefOvVG1VZ96U0= | ||||
| github.com/mitchellh/go-testing-interface v1.0.0/go.mod h1:kRemZodwjscx+RGhAo8eIhFbs2+BFgRtFPeD/KE+zxI= | ||||
| github.com/mitchellh/go-wordwrap v0.0.0-20150314170334-ad45545899c7/go.mod h1:ZXFpozHsX6DPmq2I0TCekCxypsnAUbP2oI0UX1GXzOo= | ||||
| github.com/mitchellh/go-testing-interface v1.14.1 h1:jrgshOhYAUVNMAJiKbEu7EqAwgJJ2JqpQmpLJOu07cU= | ||||
| github.com/mitchellh/go-testing-interface v1.14.1/go.mod h1:gfgS7OtZj6MA4U1UrDRp04twqAjfvlZyCfX3sDjEym8= | ||||
| github.com/mitchellh/go-wordwrap v1.0.0 h1:6GlHJ/LTGMrIJbwgdqdl2eEH8o+Exx/0m8ir9Gns0u4= | ||||
| github.com/mitchellh/go-wordwrap v1.0.0/go.mod h1:ZXFpozHsX6DPmq2I0TCekCxypsnAUbP2oI0UX1GXzOo= | ||||
| github.com/mitchellh/mapstructure v1.1.2 h1:fmNYVwqnSfB9mZU6OS2O6GsXM+wcskZDuKQzvN1EDeE= | ||||
| github.com/mitchellh/mapstructure v1.1.2/go.mod h1:FVVH3fgwuzCH5S8UJGiWEs2h04kUh9fWfEaFds41c1Y= | ||||
| github.com/mitchellh/mapstructure v1.5.0 h1:jeMsZIYE/09sWLaz43PL7Gy6RuMjD2eJVyuac5Z2hdY= | ||||
| github.com/mitchellh/mapstructure v1.5.0/go.mod h1:bFUtVrKA4DC2yAKiSyO/QUcy7e+RRV2QTWOzhPopBRo= | ||||
| github.com/mitchellh/reflectwalk v1.0.0/go.mod h1:mSTlrgnPZtwu0c4WaC2kGObEpuNDbx0jmZXqmk4esnw= | ||||
| github.com/mitchellh/reflectwalk v1.0.1/go.mod h1:mSTlrgnPZtwu0c4WaC2kGObEpuNDbx0jmZXqmk4esnw= | ||||
| github.com/mitchellh/reflectwalk v1.0.2 h1:G2LzWKi524PWgd3mLHV8Y5k7s6XUvT0Gef6zxSIeXaQ= | ||||
| github.com/mitchellh/reflectwalk v1.0.2/go.mod h1:mSTlrgnPZtwu0c4WaC2kGObEpuNDbx0jmZXqmk4esnw= | ||||
| github.com/niemeyer/pretty v0.0.0-20200227124842-a10e7caefd8e/go.mod h1:zD1mROLANZcx1PVRCS0qkT7pwLkGfwJo4zjcN/Tysno= | ||||
| github.com/nsf/jsondiff v0.0.0-20200515183724-f29ed568f4ce h1:RPclfga2SEJmgMmz2k+Mg7cowZ8yv4Trqw9UsJby758= | ||||
| github.com/oklog/run v1.0.0 h1:Ru7dDtJNOyC66gQ5dQmaCa0qIsAUFY3sFpK1Xk8igrw= | ||||
| github.com/oklog/run v1.0.0/go.mod h1:dlhp/R75TPv97u0XWUtDeV/lRKWPKSdTuV0TZvrmrQA= | ||||
| github.com/pierrec/lz4 v2.0.5+incompatible/go.mod h1:pdkljMzZIN41W+lC3N2tnIh5sFi+IEE17M5jbnwPHcY= | ||||
| github.com/pkg/errors v0.8.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0= | ||||
| github.com/pkg/errors v0.9.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0= | ||||
| github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM= | ||||
| github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4= | ||||
| github.com/posener/complete v1.1.1 h1:ccV59UEOTzVDnDUEFdT95ZzHVZ+5+158q8+SJb2QV5w= | ||||
| github.com/posener/complete v1.1.1/go.mod h1:em0nMJCgc9GFtwrmVmEMR/ZL6WyhyjMBndrE9hABlRI= | ||||
| github.com/posener/complete v1.2.1 h1:LrvDIY//XNo65Lq84G/akBuMGlawHvGBABv8f/ZN6DI= | ||||
| github.com/posener/complete v1.2.1/go.mod h1:6gapUrK/U1TAN7ciCoNRIdVC5sbdBTUh1DKN0g6uH7E= | ||||
| github.com/prometheus/client_model v0.0.0-20190812154241-14fe0d1b01d4/go.mod h1:xMI15A0UPsDsEKsMN9yxemIoYk6Tm2C1GtYGdfGttqA= | ||||
| github.com/rogpeppe/fastuuid v1.2.0/go.mod h1:jVj6XXZzXRy/MSR5jhDC/2q6DgLz+nrA6LYCDYWNEvQ= | ||||
| github.com/russross/blackfriday v1.6.0 h1:KqfZb0pUVN2lYqZUYRddxF4OR8ZMURnJIG5Y3VRLtww= | ||||
| github.com/russross/blackfriday v1.6.0/go.mod h1:ti0ldHuxg49ri4ksnFxlkCfN+hvslNlmVHqNRXXJNAY= | ||||
| github.com/sebdah/goldie v1.0.0/go.mod h1:jXP4hmWywNEwZzhMuv2ccnqTSFpuq8iyQhtQdkkZBH4= | ||||
| github.com/sergi/go-diff v1.0.0/go.mod h1:0CfEIISq7TuYL3j771MWULgwwjU+GofnZX9QAmXWZgo= | ||||
| github.com/sergi/go-diff v1.1.0/go.mod h1:STckp+ISIX8hZLjrqAeVduY0gWCT9IjLuqbuNXdaHfM= | ||||
| github.com/sergi/go-diff v1.2.0 h1:XU+rvMAioB0UC3q1MFrIQy4Vo5/4VsRDQQXHsEya6xQ= | ||||
| github.com/sergi/go-diff v1.2.0/go.mod h1:STckp+ISIX8hZLjrqAeVduY0gWCT9IjLuqbuNXdaHfM= | ||||
| github.com/sirupsen/logrus v1.4.1/go.mod h1:ni0Sbl8bgC9z8RoU9G6nDWqqs/fq4eDPysMBDgk/93Q= | ||||
| github.com/spf13/afero v1.2.2 h1:5jhuqJyZCZf2JRofRvN/nIFgIWNzPa3/Vz8mYylgbWc= | ||||
| github.com/spf13/afero v1.2.2/go.mod h1:9ZxEEn6pIJ8Rxe320qSDBk6AsU0r9pR7Q4OcevTdifk= | ||||
| github.com/spf13/pflag v1.0.2/go.mod h1:DYY7MBk1bdzusC3SYhjObp+wFpr4gzcvqqNjLnInEg4= | ||||
| github.com/spf13/pflag v1.0.3/go.mod h1:DYY7MBk1bdzusC3SYhjObp+wFpr4gzcvqqNjLnInEg4= | ||||
| github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME= | ||||
| github.com/stretchr/objx v0.1.1/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME= | ||||
| github.com/stretchr/testify v1.2.2/go.mod h1:a8OnRcib4nhh0OaRAV+Yts87kKdq0PP7pXfy6kDkUVs= | ||||
| github.com/stretchr/testify v1.3.0/go.mod h1:M5WIy9Dh21IEIfnGCwXGc5bZfKNJtfHm1UVUgZn+9EI= | ||||
| github.com/stretchr/testify v1.4.0/go.mod h1:j7eGeouHqKxXV5pUuKE4zz7dFj8WfuZ+81PSLYec5m4= | ||||
| github.com/stretchr/testify v1.5.1/go.mod h1:5W2xD1RspED5o8YsWQXVCued0rvSQ+mT+I5cxcmMvtA= | ||||
| github.com/stretchr/testify v1.6.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= | ||||
| github.com/stretchr/testify v1.7.0 h1:nwc3DEeHmmLAfoZucVR881uASk0Mfjw8xYJ99tb5CcY= | ||||
| github.com/stretchr/testify v1.7.0/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= | ||||
| github.com/ulikunitz/xz v0.5.5 h1:pFrO0lVpTBXLpYw+pnLj6TbvHuyjXMfjGeCwSqCVwok= | ||||
| github.com/ulikunitz/xz v0.5.5/go.mod h1:2bypXElzHzzJZwzH67Y6wb67pO62Rzfn7BSiF4ABRW8= | ||||
| github.com/stretchr/testify v1.7.2 h1:4jaiDzPyXQvSd7D0EjG45355tLlV3VOECpq10pLC+8s= | ||||
| github.com/stretchr/testify v1.7.2/go.mod h1:R6va5+xMeoiuVRoj+gSkQ7d3FALtqAAGI1FQKckRals= | ||||
| github.com/vmihailenco/msgpack v3.3.3+incompatible/go.mod h1:fy3FlTQTDXWkZ7Bh6AcGMlsjHatGryHQYUTf1ShIgkk= | ||||
| github.com/vmihailenco/msgpack v4.0.1+incompatible/go.mod h1:fy3FlTQTDXWkZ7Bh6AcGMlsjHatGryHQYUTf1ShIgkk= | ||||
| github.com/vmihailenco/msgpack v4.0.4+incompatible h1:dSLoQfGFAo3F6OoNhwUmLwVgaUXK79GlxNBwueZn0xI= | ||||
| github.com/vmihailenco/msgpack v4.0.4+incompatible/go.mod h1:fy3FlTQTDXWkZ7Bh6AcGMlsjHatGryHQYUTf1ShIgkk= | ||||
| github.com/vmihailenco/msgpack/v4 v4.3.12 h1:07s4sz9IReOgdikxLTKNbBdqDMLsjPKXwvCazn8G65U= | ||||
| github.com/vmihailenco/msgpack/v4 v4.3.12/go.mod h1:gborTTJjAo/GWTqqRjrLCn9pgNN+NXzzngzBKDPIqw4= | ||||
| github.com/vmihailenco/tagparser v0.1.1 h1:quXMXlA39OCbd2wAdTsGDlK9RkOk6Wuw+x37wVyIuWY= | ||||
| @@ -269,39 +231,26 @@ github.com/vmihailenco/tagparser v0.1.1/go.mod h1:OeAg3pn3UbLjkWt+rN9oFYB6u/cQgq | ||||
| github.com/xanzy/ssh-agent v0.3.0 h1:wUMzuKtKilRgBAD1sUb8gOwwRr2FGoBVumcjoOACClI= | ||||
| github.com/xanzy/ssh-agent v0.3.0/go.mod h1:3s9xbODqPuuhK9JV1R321M/FlMZSBvE5aY6eAcqrDh0= | ||||
| github.com/yuin/goldmark v1.1.25/go.mod h1:3hX8gzYuyVAZsxl0MRgGTJEmQBFcNTphYh9decYSb74= | ||||
| github.com/zclconf/go-cty v1.0.0/go.mod h1:xnAOWiHeOqg2nWS62VtQ7pbOu17FtxJNW8RLEih+O3s= | ||||
| github.com/zclconf/go-cty v1.1.0/go.mod h1:xnAOWiHeOqg2nWS62VtQ7pbOu17FtxJNW8RLEih+O3s= | ||||
| github.com/zclconf/go-cty v1.2.0/go.mod h1:hOPWgoHbaTUnI5k4D2ld+GRpFJSCe6bCM7m1q/N4PQ8= | ||||
| github.com/zclconf/go-cty v1.2.1/go.mod h1:hOPWgoHbaTUnI5k4D2ld+GRpFJSCe6bCM7m1q/N4PQ8= | ||||
| github.com/zclconf/go-cty v1.9.1/go.mod h1:vVKLxnk3puL4qRAv72AO+W99LUD4da90g3uUAzyuvAk= | ||||
| github.com/zclconf/go-cty v1.10.0 h1:mp9ZXQeIcN8kAwuqorjH+Q+njbJKjLrvB2yIh4q7U+0= | ||||
| github.com/zclconf/go-cty v1.10.0/go.mod h1:vVKLxnk3puL4qRAv72AO+W99LUD4da90g3uUAzyuvAk= | ||||
| github.com/zclconf/go-cty-debug v0.0.0-20191215020915-b22d67c1ba0b/go.mod h1:ZRKQfBXbGkpdV6QMzT3rU1kSTAnfu1dO8dPKjYprgj8= | ||||
| github.com/zclconf/go-cty-yaml v1.0.1 h1:up11wlgAaDvlAGENcFDnZgkn0qUJurso7k6EpURKNF8= | ||||
| github.com/zclconf/go-cty-yaml v1.0.1/go.mod h1:IP3Ylp0wQpYm50IHK8OZWKMu6sPJIUgKa8XhiVHura0= | ||||
| go.opencensus.io v0.21.0/go.mod h1:mSImk1erAIZhrmZN+AvHh14ztQfjbGwt4TtuofqLduU= | ||||
| go.opencensus.io v0.22.0 h1:C9hSCOW830chIVkdja34wa6Ky+IzWllkUinR+BtRZd4= | ||||
| go.opencensus.io v0.22.0/go.mod h1:+kGneAE2xo2IficOXnaByMWTGM9T73dGwxeWcUqIpI8= | ||||
| go.opentelemetry.io/proto/otlp v0.7.0/go.mod h1:PqfVotwruBrMGOCsRd/89rSnXhoiJIqeYNgFYFoEGnI= | ||||
| golang.org/x/crypto v0.0.0-20190219172222-a4c6cb3142f2/go.mod h1:6SG95UA2DQfeDnfUPMdvaQW0Q7yPrPDi9nlGo2tz2b4= | ||||
| golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w= | ||||
| golang.org/x/crypto v0.0.0-20190426145343-a29dc8fdc734/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI= | ||||
| golang.org/x/crypto v0.0.0-20190605123033-f99c8df09eb5/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI= | ||||
| golang.org/x/crypto v0.0.0-20190820162420-60c769a6c586/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI= | ||||
| golang.org/x/crypto v0.0.0-20191011191535-87dc89f01550/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI= | ||||
| golang.org/x/crypto v0.0.0-20200622213623-75b288015ac9/go.mod h1:LzIPMQfyMNhhGPhUkYOs5KpL4U8rLKemX1yGLhDgUto= | ||||
| golang.org/x/crypto v0.0.0-20200820211705-5c72a883971a/go.mod h1:LzIPMQfyMNhhGPhUkYOs5KpL4U8rLKemX1yGLhDgUto= | ||||
| golang.org/x/crypto v0.0.0-20210322153248-0c34fe9e7dc2/go.mod h1:T9bdIzuCu7OtxOm1hfPfRQxPLYneinmdGuTeoZ9dtd4= | ||||
| golang.org/x/crypto v0.0.0-20210421170649-83a5a9bb288b/go.mod h1:T9bdIzuCu7OtxOm1hfPfRQxPLYneinmdGuTeoZ9dtd4= | ||||
| golang.org/x/crypto v0.0.0-20210616213533-5ff15b29337e h1:gsTQYXdTw2Gq7RBsWvlQ91b+aEQ6bXFUngBGuR8sPpI= | ||||
| golang.org/x/crypto v0.0.0-20210616213533-5ff15b29337e/go.mod h1:GvvjBRRGRdwPK5ydBHafDWAxML/pGHZbMvKqRZ5+Abc= | ||||
| golang.org/x/crypto v0.0.0-20220517005047-85d78b3ac167 h1:O8uGbHCqlTp2P6QJSLmCojM4mN6UemYv8K+dCnmHmu0= | ||||
| golang.org/x/crypto v0.0.0-20220517005047-85d78b3ac167/go.mod h1:IxCIyHEi3zRg3s0A5j5BB6A9Jmi73HwBIUl50j+osU4= | ||||
| golang.org/x/exp v0.0.0-20190121172915-509febef88a4/go.mod h1:CJ0aWSM057203Lf6IL+f9T1iT9GByDxfZKAQTCR3kQA= | ||||
| golang.org/x/exp v0.0.0-20190510132918-efd6b22b2522/go.mod h1:ZjyILWgesfNpC6sMxTJOJm9Kp84zZh5NQWvqDGG3Qr8= | ||||
| golang.org/x/image v0.0.0-20190227222117-0694c2d4d067/go.mod h1:kZ7UVZpmo3dzQBMxlp+ypCbDeSB+sBbTgSJuh5dn5js= | ||||
| 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-20190301231843-5614ed5bae6f/go.mod h1:UVdnD1Gm6xHRNCYTkRU2/jEulfH38KcIWyp/GAMgvoE= | ||||
| golang.org/x/lint v0.0.0-20190313153728-d0100b6bd8b3/go.mod h1:6SW0HCj/g11FgYtHlgUYUwCkIfeOF89ocIRzGO/8vkc= | ||||
| golang.org/x/lint v0.0.0-20190409202823-959b441ac422/go.mod h1:6SW0HCj/g11FgYtHlgUYUwCkIfeOF89ocIRzGO/8vkc= | ||||
| golang.org/x/mobile v0.0.0-20190312151609-d3739f865fa6/go.mod h1:z+o9i4GpDbdi3rU15maQ/Ox0txvL9dWGYEHz965HBQE= | ||||
| golang.org/x/mod v0.2.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-20180811021610-c39426892332/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= | ||||
| @@ -310,120 +259,114 @@ golang.org/x/net v0.0.0-20190108225652-1e06a53dbb7e/go.mod h1:mL1N/T3taQHkDXs73r | ||||
| golang.org/x/net v0.0.0-20190213061140-3a22650c66bd/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= | ||||
| 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-20190501004415-9ce7a6920f09/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg= | ||||
| golang.org/x/net v0.0.0-20190503192946-f4e77d36d62c/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg= | ||||
| golang.org/x/net v0.0.0-20190603091049-60506f45cf65/go.mod h1:HSz+uSET+XFnRR8LxR5pz3Of3rY3CfYBVs4xY44aLks= | ||||
| golang.org/x/net v0.0.0-20190620200207-3b0461eec859/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= | ||||
| golang.org/x/net v0.0.0-20191009170851-d66e71096ffb/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= | ||||
| golang.org/x/net v0.0.0-20200226121028-0de0cce0169b/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= | ||||
| golang.org/x/net v0.0.0-20200301022130-244492dfa37a/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= | ||||
| golang.org/x/net v0.0.0-20200822124328-c89045814202/go.mod h1:/O7V0waA8r7cgGh81Ro3o1hOxt32SMVPicZroKQ2sZA= | ||||
| golang.org/x/net v0.0.0-20201021035429-f5854403a974/go.mod h1:sp8m0HH+o8qH0wwXwYZr8TS3Oi6o0r6Gce1SSxlDquU= | ||||
| golang.org/x/net v0.0.0-20210119194325-5f4716e94777/go.mod h1:m0MpNAwzfU5UDzcl9v0D8zg8gWTRqZa9RBIspLL5mdg= | ||||
| golang.org/x/net v0.0.0-20210226172049-e18ecbb05110/go.mod h1:m0MpNAwzfU5UDzcl9v0D8zg8gWTRqZa9RBIspLL5mdg= | ||||
| golang.org/x/net v0.0.0-20210326060303-6b1517762897 h1:KrsHThm5nFk34YtATK1LsThyGhGbGe1olrte/HInHvs= | ||||
| golang.org/x/net v0.0.0-20210326060303-6b1517762897/go.mod h1:uSPa2vr4CLtc/ILN5odXGNXS6mhrKVzTaCXzk9m6W3k= | ||||
| golang.org/x/net v0.0.0-20211112202133-69e39bad7dc2 h1:CIJ76btIcR3eFI5EgSo6k1qKw9KJexJuRLI9G7Hp5wE= | ||||
| golang.org/x/net v0.0.0-20211112202133-69e39bad7dc2/go.mod h1:9nx3DQGgdP8bBQD5qxJ1jj9UTztislL4KSBs9R2vV5Y= | ||||
| golang.org/x/oauth2 v0.0.0-20180821212333-d2e6202438be/go.mod h1:N/0e6XlmueqKjAGxoOufVs8QHGRruUQn6yWY3a++T0U= | ||||
| golang.org/x/oauth2 v0.0.0-20190226205417-e64efc72b421/go.mod h1:gOpvHmFTYa4IltrdGE7lF6nIHvwfUNPOp7c8zoXwtLw= | ||||
| golang.org/x/oauth2 v0.0.0-20190604053449-0f29369cfe45 h1:SVwTIAaPC2U/AvvLNZ2a7OVsmBpC8L5BlwK1whH3hm0= | ||||
| golang.org/x/oauth2 v0.0.0-20190604053449-0f29369cfe45/go.mod h1:gOpvHmFTYa4IltrdGE7lF6nIHvwfUNPOp7c8zoXwtLw= | ||||
| golang.org/x/oauth2 v0.0.0-20200107190931-bf48bf16ab8d/go.mod h1:gOpvHmFTYa4IltrdGE7lF6nIHvwfUNPOp7c8zoXwtLw= | ||||
| golang.org/x/sync v0.0.0-20180314180146-1d60e4601c6f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= | ||||
| golang.org/x/sync v0.0.0-20181108010431-42b317875d0f/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-20190423024810-112230192c58/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-20180823144017-11551d06cbcc/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-20180905080454-ebe1bf3edb33/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= | ||||
| golang.org/x/sys v0.0.0-20190129075346-302c3dd5f1cc/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= | ||||
| golang.org/x/sys v0.0.0-20190215142949-d0b11bdaac8a/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= | ||||
| golang.org/x/sys v0.0.0-20190222072716-a9d3bda3a223/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= | ||||
| golang.org/x/sys v0.0.0-20190312061237-fead79001313/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20190412213103-97732733099d/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20190502145724-3ef323f4f1fd/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20190502175342-a43fa875dd82/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20190507160741-ecd444e8653b/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20190606165138-5da285871e9c/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20190624142023-c5567b49c5d0/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20190916202348-b4ddaad3f8a3/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20200116001909-b77594299b42/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20200223170610-d5e6a3e2c0ae/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20200302150141-5c8b2ff67527/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20200323222414-85ca7c5b95cd/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20200930185726-fdedc70b468f/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/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20210320140829-1e4c9ba3b0c4/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20210324051608-47abb6519492/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20210502180810-71e4cd670f79/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20210615035016-665e8c7367d1/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | ||||
| golang.org/x/sys v0.0.0-20210630005230-0f9fa26af87c/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | ||||
| golang.org/x/sys v0.0.0-20210927094055-39ccf1dd6fa6 h1:foEbQz/B0Oz6YIqu/69kfXPYeFQAuuMYFkjaqXzl5Wo= | ||||
| golang.org/x/sys v0.0.0-20210927094055-39ccf1dd6fa6/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | ||||
| golang.org/x/sys v0.0.0-20220503163025-988cb79eb6c6 h1:nonptSpoQ4vQjyraW20DXPAglgQfVnM9ZC6MmNLMR60= | ||||
| golang.org/x/sys v0.0.0-20220503163025-988cb79eb6c6/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.1-0.20180807135948-17ff2d5776d2/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ= | ||||
| golang.org/x/text v0.3.2/go.mod h1:bEr9sfX3Q8Zfm5fL9x+3itogRgK3+ptLWKqgva+5dAk= | ||||
| golang.org/x/text v0.3.3/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= | ||||
| golang.org/x/text v0.3.5 h1:i6eZZ+zk0SOf0xgBpEpPD18qWcJda6q1sxt3S0kzyUQ= | ||||
| golang.org/x/text v0.3.5/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= | ||||
| golang.org/x/time v0.0.0-20181108054448-85acf8d2951c/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ= | ||||
| golang.org/x/time v0.0.0-20190308202827-9d24e82272b4/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ= | ||||
| golang.org/x/text v0.3.7 h1:olpwvP2KacW1ZWvsR7uQhoyTYvKAupfQrRGBFM352Gk= | ||||
| golang.org/x/text v0.3.7/go.mod h1:u+2+/6zg+i71rQMx5EYifcz6MCKuco9NR6JIITiCfzQ= | ||||
| golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= | ||||
| golang.org/x/tools v0.0.0-20190114222345-bf090417da8b/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= | ||||
| golang.org/x/tools v0.0.0-20190226205152-f727befe758c/go.mod h1:9Yl7xja0Znq3iFh3HoIrodX9oNMXvdceNzlUR8zjMvY= | ||||
| golang.org/x/tools v0.0.0-20190311212946-11955173bddd/go.mod h1:LCzVGOaR6xXOjkQ3onu1FJEFr0SW1gC7cKk1uF8kGRs= | ||||
| golang.org/x/tools v0.0.0-20190312151545-0bb0c0a6e846/go.mod h1:LCzVGOaR6xXOjkQ3onu1FJEFr0SW1gC7cKk1uF8kGRs= | ||||
| golang.org/x/tools v0.0.0-20190312170243-e65039ee4138/go.mod h1:LCzVGOaR6xXOjkQ3onu1FJEFr0SW1gC7cKk1uF8kGRs= | ||||
| golang.org/x/tools v0.0.0-20190425150028-36563e24a262/go.mod h1:RgjU9mgBXZiqYHBnxXauZ1Gv1EHHAz9KjViQ78xBX0Q= | ||||
| golang.org/x/tools v0.0.0-20190506145303-2d16b83fe98c/go.mod h1:RgjU9mgBXZiqYHBnxXauZ1Gv1EHHAz9KjViQ78xBX0Q= | ||||
| golang.org/x/tools v0.0.0-20190524140312-2c0ae7006135/go.mod h1:RgjU9mgBXZiqYHBnxXauZ1Gv1EHHAz9KjViQ78xBX0Q= | ||||
| golang.org/x/tools v0.0.0-20190606124116-d0a3d012864b/go.mod h1:/rFqwRUd4F7ZHNgwSSTFct+R/Kf4OFW1sUzUTQQTgfc= | ||||
| golang.org/x/tools v0.0.0-20190628153133-6cdbf07be9d0/go.mod h1:/rFqwRUd4F7ZHNgwSSTFct+R/Kf4OFW1sUzUTQQTgfc= | ||||
| golang.org/x/tools v0.0.0-20191119224855-298f0cb1881e/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= | ||||
| golang.org/x/tools v0.0.0-20200325010219-a49f79bcc224/go.mod h1:Sl4aGygMT6LrqrWclx+PTx3U+LnKx/seiNR+3G19Ar8= | ||||
| 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-20191204190536-9bdfabe68543 h1:E7g+9GITq07hpfrRu66IVDexMakfv52eLZ2CXBWiKr4= | ||||
| golang.org/x/xerrors v0.0.0-20191204190536-9bdfabe68543/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= | ||||
| google.golang.org/api v0.4.0/go.mod h1:8k5glujaEP+g9n7WNsDg8QP6cUVNI86fCNMcbazEtwE= | ||||
| google.golang.org/api v0.7.0/go.mod h1:WtwebWUNSVBH/HAw79HIFXZNqEvBhG+Ra+ax0hx3E3M= | ||||
| google.golang.org/api v0.8.0/go.mod h1:o4eAsZoiT+ibD93RtjEohWalFOjRDx6CVaqeizhEnKg= | ||||
| google.golang.org/api v0.9.0 h1:jbyannxz0XFD3zdjgrSUsaJbgpH4eTrkdhRChkHPfO8= | ||||
| google.golang.org/api v0.9.0/go.mod h1:o4eAsZoiT+ibD93RtjEohWalFOjRDx6CVaqeizhEnKg= | ||||
| golang.org/x/xerrors v0.0.0-20200804184101-5ec99f83aff1/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= | ||||
| google.golang.org/appengine v1.1.0/go.mod h1:EbEs0AVv82hx2wNQdGPgUI5lhzA/G0D9YwlJXL52JkM= | ||||
| google.golang.org/appengine v1.4.0/go.mod h1:xpcJRLb0r/rnEns0DIKYYv+WjYCduHsrkT7/EB5XEv4= | ||||
| google.golang.org/appengine v1.5.0/go.mod h1:xpcJRLb0r/rnEns0DIKYYv+WjYCduHsrkT7/EB5XEv4= | ||||
| google.golang.org/appengine v1.6.1/go.mod h1:i06prIuMbXzDqacNJfV5OdTW448YApPu5ww/cMBSeb0= | ||||
| google.golang.org/appengine v1.6.5 h1:tycE03LOZYQNhDpS27tcQdAzLCVMaj7QT2SXxebnpCM= | ||||
| google.golang.org/appengine v1.6.5/go.mod h1:8WjMMxjGQR8xUklV/ARdw2HLXBOI7O7uCIDZVag1xfc= | ||||
| google.golang.org/appengine v1.6.6 h1:lMO5rYAqUxkmaj76jAkRUvt5JZgFymx/+Q5Mzfivuhc= | ||||
| google.golang.org/appengine v1.6.6/go.mod h1:8WjMMxjGQR8xUklV/ARdw2HLXBOI7O7uCIDZVag1xfc= | ||||
| google.golang.org/genproto v0.0.0-20180817151627-c66870c02cf8/go.mod h1:JiN7NxoALGmiZfu7CAH4rXhgtRTLTxftemlI0sWmxmc= | ||||
| google.golang.org/genproto v0.0.0-20190307195333-5fe7a883aa19/go.mod h1:VzzqZJRnGkLBvHegQrXjBqPurQTc5/KpmUdxsrq26oE= | ||||
| google.golang.org/genproto v0.0.0-20190418145605-e7d98fc518a7/go.mod h1:VzzqZJRnGkLBvHegQrXjBqPurQTc5/KpmUdxsrq26oE= | ||||
| google.golang.org/genproto v0.0.0-20190425155659-357c62f0e4bb/go.mod h1:VzzqZJRnGkLBvHegQrXjBqPurQTc5/KpmUdxsrq26oE= | ||||
| google.golang.org/genproto v0.0.0-20190502173448-54afdca5d873/go.mod h1:VzzqZJRnGkLBvHegQrXjBqPurQTc5/KpmUdxsrq26oE= | ||||
| google.golang.org/genproto v0.0.0-20190801165951-fa694d86fc64/go.mod h1:DMBHOl98Agz4BDEuKkezgsaosCRResVns1a3J2ZsMNc= | ||||
| google.golang.org/genproto v0.0.0-20190819201941-24fa4b261c55/go.mod h1:DMBHOl98Agz4BDEuKkezgsaosCRResVns1a3J2ZsMNc= | ||||
| google.golang.org/genproto v0.0.0-20200310143817-43be25429f5a h1:lRlI5zu6AFy3iU/F8YWyNrAmn/tPCnhiTxfwhWb76eU= | ||||
| google.golang.org/genproto v0.0.0-20200310143817-43be25429f5a/go.mod h1:55QSHmfGQM9UVYDPBsyGGes0y52j32PQ3BqQfXhyH3c= | ||||
| google.golang.org/grpc v1.14.0/go.mod h1:yo6s7OP7yaDglbqo1J04qKzAhqBH6lvTonzMVmEdcZw= | ||||
| google.golang.org/genproto v0.0.0-20200513103714-09dca8ec2884/go.mod h1:55QSHmfGQM9UVYDPBsyGGes0y52j32PQ3BqQfXhyH3c= | ||||
| google.golang.org/genproto v0.0.0-20200526211855-cb27e3aa2013/go.mod h1:NbSheEEYHJ7i3ixzK3sjbqSGDJWnxyFXZblF3eUsNvo= | ||||
| google.golang.org/genproto v0.0.0-20200711021454-869866162049 h1:YFTFpQhgvrLrmxtiIncJxFXeCyq84ixuKWVCaCAi9Oc= | ||||
| google.golang.org/genproto v0.0.0-20200711021454-869866162049/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no= | ||||
| google.golang.org/grpc v1.19.0/go.mod h1:mqu4LbDTu4XGKhr4mRzUsmM4RtVoemTSY81AxZiDr8c= | ||||
| google.golang.org/grpc v1.20.1/go.mod h1:10oTOabMzJvdu6/UiuZezV6QK5dSlG84ov/aaiqXj38= | ||||
| google.golang.org/grpc v1.21.1/go.mod h1:oYelfM1adQP15Ek0mdvEgi9Df8B9CZIaU1084ijfRaM= | ||||
| google.golang.org/grpc v1.23.0/go.mod h1:Y5yQAOtifL1yxbo5wqy6BxZv8vAUGQwXBOALyacEbxg= | ||||
| google.golang.org/grpc v1.25.1/go.mod h1:c3i+UQWmh7LiEpx4sFZnkU36qjEYZ0imhYfXVyQciAY= | ||||
| google.golang.org/grpc v1.27.0/go.mod h1:qbnxyOmOxrQa7FizSgH+ReBfzJrCY1pSN7KXBS8abTk= | ||||
| google.golang.org/grpc v1.27.1 h1:zvIju4sqAGvwKspUQOhwnpcqSbzi7/H6QomNNjTL4sk= | ||||
| google.golang.org/grpc v1.27.1/go.mod h1:qbnxyOmOxrQa7FizSgH+ReBfzJrCY1pSN7KXBS8abTk= | ||||
| google.golang.org/grpc v1.33.1/go.mod h1:fr5YgcSWrqhRRxogOsw7RzIpsmvOZ6IcH4kBYTpR3n0= | ||||
| google.golang.org/grpc v1.36.0/go.mod h1:qjiiYl8FncCW8feJPdyg3v6XW24KsRHe+dy9BAGRRjU= | ||||
| google.golang.org/grpc v1.48.0 h1:rQOsyJ/8+ufEDJd/Gdsz7HG220Mh9HAhFHRGnIjda0w= | ||||
| google.golang.org/grpc v1.48.0/go.mod h1:vN9eftEi1UMyUsIF80+uQXhHjbXYbm0uXoFCACuMGWk= | ||||
| google.golang.org/protobuf v0.0.0-20200109180630-ec00e32a8dfd/go.mod h1:DFci5gLYBciE7Vtevhsrf46CRTquxDuWsQurQQe4oz8= | ||||
| google.golang.org/protobuf v0.0.0-20200221191635-4d8936d0db64/go.mod h1:kwYJMbMJ01Woi6D6+Kah6886xMZcty6N08ah7+eCXa0= | ||||
| google.golang.org/protobuf v0.0.0-20200228230310-ab0ca4ff8a60/go.mod h1:cfTl7dwQJ+fmap5saPgwCLgHXTUD7jkjRqWcaiX5VyM= | ||||
| google.golang.org/protobuf v1.20.1-0.20200309200217-e05f789c0967/go.mod h1:A+miEFZTKqfCUM6K7xSMQL9OKL/b6hQv+e19PK+JZNE= | ||||
| google.golang.org/protobuf v1.21.0/go.mod h1:47Nbq4nVaFHyn7ilMalzfO3qCViNmqZ2kzikPIcrTAo= | ||||
| google.golang.org/protobuf v1.22.0/go.mod h1:EGpADcykh3NcUnDUJcl1+ZksZNG86OlYog2l/sGQquU= | ||||
| google.golang.org/protobuf v1.23.0/go.mod h1:EGpADcykh3NcUnDUJcl1+ZksZNG86OlYog2l/sGQquU= | ||||
| google.golang.org/protobuf v1.23.1-0.20200526195155-81db48ad09cc/go.mod h1:EGpADcykh3NcUnDUJcl1+ZksZNG86OlYog2l/sGQquU= | ||||
| google.golang.org/protobuf v1.24.0/go.mod h1:r/3tXBNzIEhYS9I1OUVjXDlt8tc493IdKGjtUeSXeh4= | ||||
| google.golang.org/protobuf v1.25.0/go.mod h1:9JNX74DMeImyA3h4bdi1ymwjUzf21/xIlbajtzgsN7c= | ||||
| google.golang.org/protobuf v1.26.0-rc.1/go.mod h1:jlhhOSvTdKEhbULTjvd4ARK9grFBp09yW+WbY/TyQbw= | ||||
| google.golang.org/protobuf v1.26.0/go.mod h1:9q0QmTI4eRPtz6boOQmLYwt+qCgq0jsYwAQnmE0givc= | ||||
| google.golang.org/protobuf v1.27.1/go.mod h1:9q0QmTI4eRPtz6boOQmLYwt+qCgq0jsYwAQnmE0givc= | ||||
| google.golang.org/protobuf v1.28.0 h1:w43yiav+6bVFTBQFZX0r7ipe9JQ1QsbMgHwbBziscLw= | ||||
| google.golang.org/protobuf v1.28.0/go.mod h1:HV8QOd/L58Z+nl8r43ehVNZIU/HEI6OcFqwMG9pJV4I= | ||||
| gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= | ||||
| gopkg.in/check.v1 v1.0.0-20180628173108-788fd7840127/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= | ||||
| gopkg.in/check.v1 v1.0.0-20190902080502-41f04d3bba15/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= | ||||
| gopkg.in/check.v1 v1.0.0-20200227125254-8fa46927fb4f/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= | ||||
| gopkg.in/check.v1 v1.0.0-20201130134442-10cb98267c6c h1:Hei/4ADfdWqJk1ZMxUNpqntNwaWcugrBjAiHlqqRiVk= | ||||
| gopkg.in/check.v1 v1.0.0-20201130134442-10cb98267c6c/go.mod h1:JHkPIbrfpd72SG/EVd6muEfDQjcINNoR0C8j2r3qZ4Q= | ||||
| gopkg.in/cheggaaa/pb.v1 v1.0.27/go.mod h1:V/YB90LKu/1FcN3WVnfiiE5oMCibMjukxqG/qStrOgw= | ||||
| gopkg.in/warnings.v0 v0.1.2 h1:wFXVbFY8DY5/xOe1ECiWdKCzZlxgshcYVNkBHstARME= | ||||
| gopkg.in/warnings.v0 v0.1.2/go.mod h1:jksf8JmL6Qr/oQM2OXTHunEvvTAsrWBLb6OOjuVWRNI= | ||||
| gopkg.in/yaml.v2 v2.2.2/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= | ||||
| gopkg.in/yaml.v2 v2.2.3/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= | ||||
| gopkg.in/yaml.v2 v2.2.4/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= | ||||
| gopkg.in/yaml.v2 v2.3.0 h1:clyUAQHOM3G0M3f5vQj7LuJrETvjVot3Z5el9nffUtU= | ||||
| gopkg.in/yaml.v2 v2.3.0/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= | ||||
| gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c h1:dUUwHk2QECo/6vqA44rthZ8ie2QXMNeKRTHCNY2nXvo= | ||||
| gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= | ||||
| gopkg.in/yaml.v3 v3.0.1 h1:fxVm/GzAzEWqLHuvctI91KS9hhNmmWOoWu0XTYJS7CA= | ||||
| gopkg.in/yaml.v3 v3.0.1/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= | ||||
| honnef.co/go/tools v0.0.0-20190102054323-c2f93a96b099/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4= | ||||
| honnef.co/go/tools v0.0.0-20190106161140-3f1c8253044a/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4= | ||||
| honnef.co/go/tools v0.0.0-20190418001031-e561f6794a2a/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4= | ||||
| honnef.co/go/tools v0.0.0-20190523083050-ea95bdfd59fc/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4= | ||||
| rsc.io/binaryregexp v0.2.0/go.mod h1:qTv7/COck+e2FymRvadv62gMdZztPaShugOCi3I+8D8= | ||||
|   | ||||
							
								
								
									
										2
									
								
								main.go
									
									
									
									
									
								
							
							
						
						
									
										2
									
								
								main.go
									
									
									
									
									
								
							| @@ -2,7 +2,7 @@ package main // import "src.techknowlogick.com/terraform-provider-gitea" | ||||
|  | ||||
| import ( | ||||
| 	"git.uploadfilter24.eu/terraform-provider-gitea/gitea" | ||||
| 	"github.com/hashicorp/terraform-plugin-sdk/plugin" | ||||
| 	"github.com/hashicorp/terraform-plugin-sdk/v2/plugin" | ||||
| ) | ||||
|  | ||||
| var Version = "development" | ||||
|   | ||||
| @@ -6,11 +6,12 @@ networks: | ||||
|  | ||||
| services: | ||||
|   server: | ||||
|     image: gitea/gitea:1.16.8 | ||||
|     image: gitea/gitea:1.17.1 | ||||
|     container_name: gitea | ||||
|     environment: | ||||
|       - USER_UID=1000 | ||||
|       - USER_GID=1000 | ||||
|       - DISABLE_GIT_HOOKS=false | ||||
|     restart: always | ||||
|     networks: | ||||
|       - gitea | ||||
|   | ||||
							
								
								
									
										202
									
								
								vendor/cloud.google.com/go/LICENSE
									
									
									
										generated
									
									
										vendored
									
									
								
							
							
						
						
									
										202
									
								
								vendor/cloud.google.com/go/LICENSE
									
									
									
										generated
									
									
										vendored
									
									
								
							| @@ -1,202 +0,0 @@ | ||||
|  | ||||
|                                  Apache License | ||||
|                            Version 2.0, January 2004 | ||||
|                         http://www.apache.org/licenses/ | ||||
|  | ||||
|    TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION | ||||
|  | ||||
|    1. Definitions. | ||||
|  | ||||
|       "License" shall mean the terms and conditions for use, reproduction, | ||||
|       and distribution as defined by Sections 1 through 9 of this document. | ||||
|  | ||||
|       "Licensor" shall mean the copyright owner or entity authorized by | ||||
|       the copyright owner that is granting the License. | ||||
|  | ||||
|       "Legal Entity" shall mean the union of the acting entity and all | ||||
|       other entities that control, are controlled by, or are under common | ||||
|       control with that entity. For the purposes of this definition, | ||||
|       "control" means (i) the power, direct or indirect, to cause the | ||||
|       direction or management of such entity, whether by contract or | ||||
|       otherwise, or (ii) ownership of fifty percent (50%) or more of the | ||||
|       outstanding shares, or (iii) beneficial ownership of such entity. | ||||
|  | ||||
|       "You" (or "Your") shall mean an individual or Legal Entity | ||||
|       exercising permissions granted by this License. | ||||
|  | ||||
|       "Source" form shall mean the preferred form for making modifications, | ||||
|       including but not limited to software source code, documentation | ||||
|       source, and configuration files. | ||||
|  | ||||
|       "Object" form shall mean any form resulting from mechanical | ||||
|       transformation or translation of a Source form, including but | ||||
|       not limited to compiled object code, generated documentation, | ||||
|       and conversions to other media types. | ||||
|  | ||||
|       "Work" shall mean the work of authorship, whether in Source or | ||||
|       Object form, made available under the License, as indicated by a | ||||
|       copyright notice that is included in or attached to the work | ||||
|       (an example is provided in the Appendix below). | ||||
|  | ||||
|       "Derivative Works" shall mean any work, whether in Source or Object | ||||
|       form, that is based on (or derived from) the Work and for which the | ||||
|       editorial revisions, annotations, elaborations, or other modifications | ||||
|       represent, as a whole, an original work of authorship. For the purposes | ||||
|       of this License, Derivative Works shall not include works that remain | ||||
|       separable from, or merely link (or bind by name) to the interfaces of, | ||||
|       the Work and Derivative Works thereof. | ||||
|  | ||||
|       "Contribution" shall mean any work of authorship, including | ||||
|       the original version of the Work and any modifications or additions | ||||
|       to that Work or Derivative Works thereof, that is intentionally | ||||
|       submitted to Licensor for inclusion in the Work by the copyright owner | ||||
|       or by an individual or Legal Entity authorized to submit on behalf of | ||||
|       the copyright owner. For the purposes of this definition, "submitted" | ||||
|       means any form of electronic, verbal, or written communication sent | ||||
|       to the Licensor or its representatives, including but not limited to | ||||
|       communication on electronic mailing lists, source code control systems, | ||||
|       and issue tracking systems that are managed by, or on behalf of, the | ||||
|       Licensor for the purpose of discussing and improving the Work, but | ||||
|       excluding communication that is conspicuously marked or otherwise | ||||
|       designated in writing by the copyright owner as "Not a Contribution." | ||||
|  | ||||
|       "Contributor" shall mean Licensor and any individual or Legal Entity | ||||
|       on behalf of whom a Contribution has been received by Licensor and | ||||
|       subsequently incorporated within the Work. | ||||
|  | ||||
|    2. Grant of Copyright License. Subject to the terms and conditions of | ||||
|       this License, each Contributor hereby grants to You a perpetual, | ||||
|       worldwide, non-exclusive, no-charge, royalty-free, irrevocable | ||||
|       copyright license to reproduce, prepare Derivative Works of, | ||||
|       publicly display, publicly perform, sublicense, and distribute the | ||||
|       Work and such Derivative Works in Source or Object form. | ||||
|  | ||||
|    3. Grant of Patent License. Subject to the terms and conditions of | ||||
|       this License, each Contributor hereby grants to You a perpetual, | ||||
|       worldwide, non-exclusive, no-charge, royalty-free, irrevocable | ||||
|       (except as stated in this section) patent license to make, have made, | ||||
|       use, offer to sell, sell, import, and otherwise transfer the Work, | ||||
|       where such license applies only to those patent claims licensable | ||||
|       by such Contributor that are necessarily infringed by their | ||||
|       Contribution(s) alone or by combination of their Contribution(s) | ||||
|       with the Work to which such Contribution(s) was submitted. If You | ||||
|       institute patent litigation against any entity (including a | ||||
|       cross-claim or counterclaim in a lawsuit) alleging that the Work | ||||
|       or a Contribution incorporated within the Work constitutes direct | ||||
|       or contributory patent infringement, then any patent licenses | ||||
|       granted to You under this License for that Work shall terminate | ||||
|       as of the date such litigation is filed. | ||||
|  | ||||
|    4. Redistribution. You may reproduce and distribute copies of the | ||||
|       Work or Derivative Works thereof in any medium, with or without | ||||
|       modifications, and in Source or Object form, provided that You | ||||
|       meet the following conditions: | ||||
|  | ||||
|       (a) You must give any other recipients of the Work or | ||||
|           Derivative Works a copy of this License; and | ||||
|  | ||||
|       (b) You must cause any modified files to carry prominent notices | ||||
|           stating that You changed the files; and | ||||
|  | ||||
|       (c) You must retain, in the Source form of any Derivative Works | ||||
|           that You distribute, all copyright, patent, trademark, and | ||||
|           attribution notices from the Source form of the Work, | ||||
|           excluding those notices that do not pertain to any part of | ||||
|           the Derivative Works; and | ||||
|  | ||||
|       (d) If the Work includes a "NOTICE" text file as part of its | ||||
|           distribution, then any Derivative Works that You distribute must | ||||
|           include a readable copy of the attribution notices contained | ||||
|           within such NOTICE file, excluding those notices that do not | ||||
|           pertain to any part of the Derivative Works, in at least one | ||||
|           of the following places: within a NOTICE text file distributed | ||||
|           as part of the Derivative Works; within the Source form or | ||||
|           documentation, if provided along with the Derivative Works; or, | ||||
|           within a display generated by the Derivative Works, if and | ||||
|           wherever such third-party notices normally appear. The contents | ||||
|           of the NOTICE file are for informational purposes only and | ||||
|           do not modify the License. You may add Your own attribution | ||||
|           notices within Derivative Works that You distribute, alongside | ||||
|           or as an addendum to the NOTICE text from the Work, provided | ||||
|           that such additional attribution notices cannot be construed | ||||
|           as modifying the License. | ||||
|  | ||||
|       You may add Your own copyright statement to Your modifications and | ||||
|       may provide additional or different license terms and conditions | ||||
|       for use, reproduction, or distribution of Your modifications, or | ||||
|       for any such Derivative Works as a whole, provided Your use, | ||||
|       reproduction, and distribution of the Work otherwise complies with | ||||
|       the conditions stated in this License. | ||||
|  | ||||
|    5. Submission of Contributions. Unless You explicitly state otherwise, | ||||
|       any Contribution intentionally submitted for inclusion in the Work | ||||
|       by You to the Licensor shall be under the terms and conditions of | ||||
|       this License, without any additional terms or conditions. | ||||
|       Notwithstanding the above, nothing herein shall supersede or modify | ||||
|       the terms of any separate license agreement you may have executed | ||||
|       with Licensor regarding such Contributions. | ||||
|  | ||||
|    6. Trademarks. This License does not grant permission to use the trade | ||||
|       names, trademarks, service marks, or product names of the Licensor, | ||||
|       except as required for reasonable and customary use in describing the | ||||
|       origin of the Work and reproducing the content of the NOTICE file. | ||||
|  | ||||
|    7. Disclaimer of Warranty. Unless required by applicable law or | ||||
|       agreed to in writing, Licensor provides the Work (and each | ||||
|       Contributor provides its Contributions) on an "AS IS" BASIS, | ||||
|       WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or | ||||
|       implied, including, without limitation, any warranties or conditions | ||||
|       of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A | ||||
|       PARTICULAR PURPOSE. You are solely responsible for determining the | ||||
|       appropriateness of using or redistributing the Work and assume any | ||||
|       risks associated with Your exercise of permissions under this License. | ||||
|  | ||||
|    8. Limitation of Liability. In no event and under no legal theory, | ||||
|       whether in tort (including negligence), contract, or otherwise, | ||||
|       unless required by applicable law (such as deliberate and grossly | ||||
|       negligent acts) or agreed to in writing, shall any Contributor be | ||||
|       liable to You for damages, including any direct, indirect, special, | ||||
|       incidental, or consequential damages of any character arising as a | ||||
|       result of this License or out of the use or inability to use the | ||||
|       Work (including but not limited to damages for loss of goodwill, | ||||
|       work stoppage, computer failure or malfunction, or any and all | ||||
|       other commercial damages or losses), even if such Contributor | ||||
|       has been advised of the possibility of such damages. | ||||
|  | ||||
|    9. Accepting Warranty or Additional Liability. While redistributing | ||||
|       the Work or Derivative Works thereof, You may choose to offer, | ||||
|       and charge a fee for, acceptance of support, warranty, indemnity, | ||||
|       or other liability obligations and/or rights consistent with this | ||||
|       License. However, in accepting such obligations, You may act only | ||||
|       on Your own behalf and on Your sole responsibility, not on behalf | ||||
|       of any other Contributor, and only if You agree to indemnify, | ||||
|       defend, and hold each Contributor harmless for any liability | ||||
|       incurred by, or claims asserted against, such Contributor by reason | ||||
|       of your accepting any such warranty or additional liability. | ||||
|  | ||||
|    END OF TERMS AND CONDITIONS | ||||
|  | ||||
|    APPENDIX: How to apply the Apache License to your work. | ||||
|  | ||||
|       To apply the Apache License to your work, attach the following | ||||
|       boilerplate notice, with the fields enclosed by brackets "[]" | ||||
|       replaced with your own identifying information. (Don't include | ||||
|       the brackets!)  The text should be enclosed in the appropriate | ||||
|       comment syntax for the file format. We also recommend that a | ||||
|       file or class name and description of purpose be included on the | ||||
|       same "printed page" as the copyright notice for easier | ||||
|       identification within third-party archives. | ||||
|  | ||||
|    Copyright [yyyy] [name of copyright owner] | ||||
|  | ||||
|    Licensed under the Apache License, Version 2.0 (the "License"); | ||||
|    you may not use this file except in compliance with the License. | ||||
|    You may obtain a copy of the License at | ||||
|  | ||||
|        http://www.apache.org/licenses/LICENSE-2.0 | ||||
|  | ||||
|    Unless required by applicable law or agreed to in writing, software | ||||
|    distributed under the License is distributed on an "AS IS" BASIS, | ||||
|    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||||
|    See the License for the specific language governing permissions and | ||||
|    limitations under the License. | ||||
							
								
								
									
										526
									
								
								vendor/cloud.google.com/go/compute/metadata/metadata.go
									
									
									
										generated
									
									
										vendored
									
									
								
							
							
						
						
									
										526
									
								
								vendor/cloud.google.com/go/compute/metadata/metadata.go
									
									
									
										generated
									
									
										vendored
									
									
								
							| @@ -1,526 +0,0 @@ | ||||
| // Copyright 2014 Google LLC | ||||
| // | ||||
| // Licensed under the Apache License, Version 2.0 (the "License"); | ||||
| // you may not use this file except in compliance with the License. | ||||
| // You may obtain a copy of the License at | ||||
| // | ||||
| //      http://www.apache.org/licenses/LICENSE-2.0 | ||||
| // | ||||
| // Unless required by applicable law or agreed to in writing, software | ||||
| // distributed under the License is distributed on an "AS IS" BASIS, | ||||
| // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||||
| // See the License for the specific language governing permissions and | ||||
| // limitations under the License. | ||||
|  | ||||
| // Package metadata provides access to Google Compute Engine (GCE) | ||||
| // metadata and API service accounts. | ||||
| // | ||||
| // This package is a wrapper around the GCE metadata service, | ||||
| // as documented at https://developers.google.com/compute/docs/metadata. | ||||
| package metadata // import "cloud.google.com/go/compute/metadata" | ||||
|  | ||||
| import ( | ||||
| 	"context" | ||||
| 	"encoding/json" | ||||
| 	"fmt" | ||||
| 	"io/ioutil" | ||||
| 	"net" | ||||
| 	"net/http" | ||||
| 	"net/url" | ||||
| 	"os" | ||||
| 	"runtime" | ||||
| 	"strings" | ||||
| 	"sync" | ||||
| 	"time" | ||||
| ) | ||||
|  | ||||
| const ( | ||||
| 	// metadataIP is the documented metadata server IP address. | ||||
| 	metadataIP = "169.254.169.254" | ||||
|  | ||||
| 	// metadataHostEnv is the environment variable specifying the | ||||
| 	// GCE metadata hostname.  If empty, the default value of | ||||
| 	// metadataIP ("169.254.169.254") is used instead. | ||||
| 	// This is variable name is not defined by any spec, as far as | ||||
| 	// I know; it was made up for the Go package. | ||||
| 	metadataHostEnv = "GCE_METADATA_HOST" | ||||
|  | ||||
| 	userAgent = "gcloud-golang/0.1" | ||||
| ) | ||||
|  | ||||
| type cachedValue struct { | ||||
| 	k    string | ||||
| 	trim bool | ||||
| 	mu   sync.Mutex | ||||
| 	v    string | ||||
| } | ||||
|  | ||||
| var ( | ||||
| 	projID  = &cachedValue{k: "project/project-id", trim: true} | ||||
| 	projNum = &cachedValue{k: "project/numeric-project-id", trim: true} | ||||
| 	instID  = &cachedValue{k: "instance/id", trim: true} | ||||
| ) | ||||
|  | ||||
| var ( | ||||
| 	defaultClient = &Client{hc: &http.Client{ | ||||
| 		Transport: &http.Transport{ | ||||
| 			Dial: (&net.Dialer{ | ||||
| 				Timeout:   2 * time.Second, | ||||
| 				KeepAlive: 30 * time.Second, | ||||
| 			}).Dial, | ||||
| 			ResponseHeaderTimeout: 2 * time.Second, | ||||
| 		}, | ||||
| 	}} | ||||
| 	subscribeClient = &Client{hc: &http.Client{ | ||||
| 		Transport: &http.Transport{ | ||||
| 			Dial: (&net.Dialer{ | ||||
| 				Timeout:   2 * time.Second, | ||||
| 				KeepAlive: 30 * time.Second, | ||||
| 			}).Dial, | ||||
| 		}, | ||||
| 	}} | ||||
| ) | ||||
|  | ||||
| // NotDefinedError is returned when requested metadata is not defined. | ||||
| // | ||||
| // The underlying string is the suffix after "/computeMetadata/v1/". | ||||
| // | ||||
| // This error is not returned if the value is defined to be the empty | ||||
| // string. | ||||
| type NotDefinedError string | ||||
|  | ||||
| func (suffix NotDefinedError) Error() string { | ||||
| 	return fmt.Sprintf("metadata: GCE metadata %q not defined", string(suffix)) | ||||
| } | ||||
|  | ||||
| func (c *cachedValue) get(cl *Client) (v string, err error) { | ||||
| 	defer c.mu.Unlock() | ||||
| 	c.mu.Lock() | ||||
| 	if c.v != "" { | ||||
| 		return c.v, nil | ||||
| 	} | ||||
| 	if c.trim { | ||||
| 		v, err = cl.getTrimmed(c.k) | ||||
| 	} else { | ||||
| 		v, err = cl.Get(c.k) | ||||
| 	} | ||||
| 	if err == nil { | ||||
| 		c.v = v | ||||
| 	} | ||||
| 	return | ||||
| } | ||||
|  | ||||
| var ( | ||||
| 	onGCEOnce sync.Once | ||||
| 	onGCE     bool | ||||
| ) | ||||
|  | ||||
| // OnGCE reports whether this process is running on Google Compute Engine. | ||||
| func OnGCE() bool { | ||||
| 	onGCEOnce.Do(initOnGCE) | ||||
| 	return onGCE | ||||
| } | ||||
|  | ||||
| func initOnGCE() { | ||||
| 	onGCE = testOnGCE() | ||||
| } | ||||
|  | ||||
| func testOnGCE() bool { | ||||
| 	// The user explicitly said they're on GCE, so trust them. | ||||
| 	if os.Getenv(metadataHostEnv) != "" { | ||||
| 		return true | ||||
| 	} | ||||
|  | ||||
| 	ctx, cancel := context.WithCancel(context.Background()) | ||||
| 	defer cancel() | ||||
|  | ||||
| 	resc := make(chan bool, 2) | ||||
|  | ||||
| 	// Try two strategies in parallel. | ||||
| 	// See https://github.com/googleapis/google-cloud-go/issues/194 | ||||
| 	go func() { | ||||
| 		req, _ := http.NewRequest("GET", "http://"+metadataIP, nil) | ||||
| 		req.Header.Set("User-Agent", userAgent) | ||||
| 		res, err := defaultClient.hc.Do(req.WithContext(ctx)) | ||||
| 		if err != nil { | ||||
| 			resc <- false | ||||
| 			return | ||||
| 		} | ||||
| 		defer res.Body.Close() | ||||
| 		resc <- res.Header.Get("Metadata-Flavor") == "Google" | ||||
| 	}() | ||||
|  | ||||
| 	go func() { | ||||
| 		addrs, err := net.LookupHost("metadata.google.internal") | ||||
| 		if err != nil || len(addrs) == 0 { | ||||
| 			resc <- false | ||||
| 			return | ||||
| 		} | ||||
| 		resc <- strsContains(addrs, metadataIP) | ||||
| 	}() | ||||
|  | ||||
| 	tryHarder := systemInfoSuggestsGCE() | ||||
| 	if tryHarder { | ||||
| 		res := <-resc | ||||
| 		if res { | ||||
| 			// The first strategy succeeded, so let's use it. | ||||
| 			return true | ||||
| 		} | ||||
| 		// Wait for either the DNS or metadata server probe to | ||||
| 		// contradict the other one and say we are running on | ||||
| 		// GCE. Give it a lot of time to do so, since the system | ||||
| 		// info already suggests we're running on a GCE BIOS. | ||||
| 		timer := time.NewTimer(5 * time.Second) | ||||
| 		defer timer.Stop() | ||||
| 		select { | ||||
| 		case res = <-resc: | ||||
| 			return res | ||||
| 		case <-timer.C: | ||||
| 			// Too slow. Who knows what this system is. | ||||
| 			return false | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	// There's no hint from the system info that we're running on | ||||
| 	// GCE, so use the first probe's result as truth, whether it's | ||||
| 	// true or false. The goal here is to optimize for speed for | ||||
| 	// users who are NOT running on GCE. We can't assume that | ||||
| 	// either a DNS lookup or an HTTP request to a blackholed IP | ||||
| 	// address is fast. Worst case this should return when the | ||||
| 	// metaClient's Transport.ResponseHeaderTimeout or | ||||
| 	// Transport.Dial.Timeout fires (in two seconds). | ||||
| 	return <-resc | ||||
| } | ||||
|  | ||||
| // systemInfoSuggestsGCE reports whether the local system (without | ||||
| // doing network requests) suggests that we're running on GCE. If this | ||||
| // returns true, testOnGCE tries a bit harder to reach its metadata | ||||
| // server. | ||||
| func systemInfoSuggestsGCE() bool { | ||||
| 	if runtime.GOOS != "linux" { | ||||
| 		// We don't have any non-Linux clues available, at least yet. | ||||
| 		return false | ||||
| 	} | ||||
| 	slurp, _ := ioutil.ReadFile("/sys/class/dmi/id/product_name") | ||||
| 	name := strings.TrimSpace(string(slurp)) | ||||
| 	return name == "Google" || name == "Google Compute Engine" | ||||
| } | ||||
|  | ||||
| // Subscribe calls Client.Subscribe on a client designed for subscribing (one with no | ||||
| // ResponseHeaderTimeout). | ||||
| func Subscribe(suffix string, fn func(v string, ok bool) error) error { | ||||
| 	return subscribeClient.Subscribe(suffix, fn) | ||||
| } | ||||
|  | ||||
| // Get calls Client.Get on the default client. | ||||
| func Get(suffix string) (string, error) { return defaultClient.Get(suffix) } | ||||
|  | ||||
| // ProjectID returns the current instance's project ID string. | ||||
| func ProjectID() (string, error) { return defaultClient.ProjectID() } | ||||
|  | ||||
| // NumericProjectID returns the current instance's numeric project ID. | ||||
| func NumericProjectID() (string, error) { return defaultClient.NumericProjectID() } | ||||
|  | ||||
| // InternalIP returns the instance's primary internal IP address. | ||||
| func InternalIP() (string, error) { return defaultClient.InternalIP() } | ||||
|  | ||||
| // ExternalIP returns the instance's primary external (public) IP address. | ||||
| func ExternalIP() (string, error) { return defaultClient.ExternalIP() } | ||||
|  | ||||
| // Email calls Client.Email on the default client. | ||||
| func Email(serviceAccount string) (string, error) { return defaultClient.Email(serviceAccount) } | ||||
|  | ||||
| // Hostname returns the instance's hostname. This will be of the form | ||||
| // "<instanceID>.c.<projID>.internal". | ||||
| func Hostname() (string, error) { return defaultClient.Hostname() } | ||||
|  | ||||
| // InstanceTags returns the list of user-defined instance tags, | ||||
| // assigned when initially creating a GCE instance. | ||||
| func InstanceTags() ([]string, error) { return defaultClient.InstanceTags() } | ||||
|  | ||||
| // InstanceID returns the current VM's numeric instance ID. | ||||
| func InstanceID() (string, error) { return defaultClient.InstanceID() } | ||||
|  | ||||
| // InstanceName returns the current VM's instance ID string. | ||||
| func InstanceName() (string, error) { return defaultClient.InstanceName() } | ||||
|  | ||||
| // Zone returns the current VM's zone, such as "us-central1-b". | ||||
| func Zone() (string, error) { return defaultClient.Zone() } | ||||
|  | ||||
| // InstanceAttributes calls Client.InstanceAttributes on the default client. | ||||
| func InstanceAttributes() ([]string, error) { return defaultClient.InstanceAttributes() } | ||||
|  | ||||
| // ProjectAttributes calls Client.ProjectAttributes on the default client. | ||||
| func ProjectAttributes() ([]string, error) { return defaultClient.ProjectAttributes() } | ||||
|  | ||||
| // InstanceAttributeValue calls Client.InstanceAttributeValue on the default client. | ||||
| func InstanceAttributeValue(attr string) (string, error) { | ||||
| 	return defaultClient.InstanceAttributeValue(attr) | ||||
| } | ||||
|  | ||||
| // ProjectAttributeValue calls Client.ProjectAttributeValue on the default client. | ||||
| func ProjectAttributeValue(attr string) (string, error) { | ||||
| 	return defaultClient.ProjectAttributeValue(attr) | ||||
| } | ||||
|  | ||||
| // Scopes calls Client.Scopes on the default client. | ||||
| func Scopes(serviceAccount string) ([]string, error) { return defaultClient.Scopes(serviceAccount) } | ||||
|  | ||||
| func strsContains(ss []string, s string) bool { | ||||
| 	for _, v := range ss { | ||||
| 		if v == s { | ||||
| 			return true | ||||
| 		} | ||||
| 	} | ||||
| 	return false | ||||
| } | ||||
|  | ||||
| // A Client provides metadata. | ||||
| type Client struct { | ||||
| 	hc *http.Client | ||||
| } | ||||
|  | ||||
| // NewClient returns a Client that can be used to fetch metadata. All HTTP requests | ||||
| // will use the given http.Client instead of the default client. | ||||
| func NewClient(c *http.Client) *Client { | ||||
| 	return &Client{hc: c} | ||||
| } | ||||
|  | ||||
| // getETag returns a value from the metadata service as well as the associated ETag. | ||||
| // This func is otherwise equivalent to Get. | ||||
| func (c *Client) getETag(suffix string) (value, etag string, err error) { | ||||
| 	// Using a fixed IP makes it very difficult to spoof the metadata service in | ||||
| 	// a container, which is an important use-case for local testing of cloud | ||||
| 	// deployments. To enable spoofing of the metadata service, the environment | ||||
| 	// variable GCE_METADATA_HOST is first inspected to decide where metadata | ||||
| 	// requests shall go. | ||||
| 	host := os.Getenv(metadataHostEnv) | ||||
| 	if host == "" { | ||||
| 		// Using 169.254.169.254 instead of "metadata" here because Go | ||||
| 		// binaries built with the "netgo" tag and without cgo won't | ||||
| 		// know the search suffix for "metadata" is | ||||
| 		// ".google.internal", and this IP address is documented as | ||||
| 		// being stable anyway. | ||||
| 		host = metadataIP | ||||
| 	} | ||||
| 	u := "http://" + host + "/computeMetadata/v1/" + suffix | ||||
| 	req, _ := http.NewRequest("GET", u, nil) | ||||
| 	req.Header.Set("Metadata-Flavor", "Google") | ||||
| 	req.Header.Set("User-Agent", userAgent) | ||||
| 	res, err := c.hc.Do(req) | ||||
| 	if err != nil { | ||||
| 		return "", "", err | ||||
| 	} | ||||
| 	defer res.Body.Close() | ||||
| 	if res.StatusCode == http.StatusNotFound { | ||||
| 		return "", "", NotDefinedError(suffix) | ||||
| 	} | ||||
| 	all, err := ioutil.ReadAll(res.Body) | ||||
| 	if err != nil { | ||||
| 		return "", "", err | ||||
| 	} | ||||
| 	if res.StatusCode != 200 { | ||||
| 		return "", "", &Error{Code: res.StatusCode, Message: string(all)} | ||||
| 	} | ||||
| 	return string(all), res.Header.Get("Etag"), nil | ||||
| } | ||||
|  | ||||
| // Get returns a value from the metadata service. | ||||
| // The suffix is appended to "http://${GCE_METADATA_HOST}/computeMetadata/v1/". | ||||
| // | ||||
| // If the GCE_METADATA_HOST environment variable is not defined, a default of | ||||
| // 169.254.169.254 will be used instead. | ||||
| // | ||||
| // If the requested metadata is not defined, the returned error will | ||||
| // be of type NotDefinedError. | ||||
| func (c *Client) Get(suffix string) (string, error) { | ||||
| 	val, _, err := c.getETag(suffix) | ||||
| 	return val, err | ||||
| } | ||||
|  | ||||
| func (c *Client) getTrimmed(suffix string) (s string, err error) { | ||||
| 	s, err = c.Get(suffix) | ||||
| 	s = strings.TrimSpace(s) | ||||
| 	return | ||||
| } | ||||
|  | ||||
| func (c *Client) lines(suffix string) ([]string, error) { | ||||
| 	j, err := c.Get(suffix) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	s := strings.Split(strings.TrimSpace(j), "\n") | ||||
| 	for i := range s { | ||||
| 		s[i] = strings.TrimSpace(s[i]) | ||||
| 	} | ||||
| 	return s, nil | ||||
| } | ||||
|  | ||||
| // ProjectID returns the current instance's project ID string. | ||||
| func (c *Client) ProjectID() (string, error) { return projID.get(c) } | ||||
|  | ||||
| // NumericProjectID returns the current instance's numeric project ID. | ||||
| func (c *Client) NumericProjectID() (string, error) { return projNum.get(c) } | ||||
|  | ||||
| // InstanceID returns the current VM's numeric instance ID. | ||||
| func (c *Client) InstanceID() (string, error) { return instID.get(c) } | ||||
|  | ||||
| // InternalIP returns the instance's primary internal IP address. | ||||
| func (c *Client) InternalIP() (string, error) { | ||||
| 	return c.getTrimmed("instance/network-interfaces/0/ip") | ||||
| } | ||||
|  | ||||
| // Email returns the email address associated with the service account. | ||||
| // The account may be empty or the string "default" to use the instance's | ||||
| // main account. | ||||
| func (c *Client) Email(serviceAccount string) (string, error) { | ||||
| 	if serviceAccount == "" { | ||||
| 		serviceAccount = "default" | ||||
| 	} | ||||
| 	return c.getTrimmed("instance/service-accounts/" + serviceAccount + "/email") | ||||
| } | ||||
|  | ||||
| // ExternalIP returns the instance's primary external (public) IP address. | ||||
| func (c *Client) ExternalIP() (string, error) { | ||||
| 	return c.getTrimmed("instance/network-interfaces/0/access-configs/0/external-ip") | ||||
| } | ||||
|  | ||||
| // Hostname returns the instance's hostname. This will be of the form | ||||
| // "<instanceID>.c.<projID>.internal". | ||||
| func (c *Client) Hostname() (string, error) { | ||||
| 	return c.getTrimmed("instance/hostname") | ||||
| } | ||||
|  | ||||
| // InstanceTags returns the list of user-defined instance tags, | ||||
| // assigned when initially creating a GCE instance. | ||||
| func (c *Client) InstanceTags() ([]string, error) { | ||||
| 	var s []string | ||||
| 	j, err := c.Get("instance/tags") | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	if err := json.NewDecoder(strings.NewReader(j)).Decode(&s); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	return s, nil | ||||
| } | ||||
|  | ||||
| // InstanceName returns the current VM's instance ID string. | ||||
| func (c *Client) InstanceName() (string, error) { | ||||
| 	host, err := c.Hostname() | ||||
| 	if err != nil { | ||||
| 		return "", err | ||||
| 	} | ||||
| 	return strings.Split(host, ".")[0], nil | ||||
| } | ||||
|  | ||||
| // Zone returns the current VM's zone, such as "us-central1-b". | ||||
| func (c *Client) Zone() (string, error) { | ||||
| 	zone, err := c.getTrimmed("instance/zone") | ||||
| 	// zone is of the form "projects/<projNum>/zones/<zoneName>". | ||||
| 	if err != nil { | ||||
| 		return "", err | ||||
| 	} | ||||
| 	return zone[strings.LastIndex(zone, "/")+1:], nil | ||||
| } | ||||
|  | ||||
| // InstanceAttributes returns the list of user-defined attributes, | ||||
| // assigned when initially creating a GCE VM instance. The value of an | ||||
| // attribute can be obtained with InstanceAttributeValue. | ||||
| func (c *Client) InstanceAttributes() ([]string, error) { return c.lines("instance/attributes/") } | ||||
|  | ||||
| // ProjectAttributes returns the list of user-defined attributes | ||||
| // applying to the project as a whole, not just this VM.  The value of | ||||
| // an attribute can be obtained with ProjectAttributeValue. | ||||
| func (c *Client) ProjectAttributes() ([]string, error) { return c.lines("project/attributes/") } | ||||
|  | ||||
| // InstanceAttributeValue returns the value of the provided VM | ||||
| // instance attribute. | ||||
| // | ||||
| // If the requested attribute is not defined, the returned error will | ||||
| // be of type NotDefinedError. | ||||
| // | ||||
| // InstanceAttributeValue may return ("", nil) if the attribute was | ||||
| // defined to be the empty string. | ||||
| func (c *Client) InstanceAttributeValue(attr string) (string, error) { | ||||
| 	return c.Get("instance/attributes/" + attr) | ||||
| } | ||||
|  | ||||
| // ProjectAttributeValue returns the value of the provided | ||||
| // project attribute. | ||||
| // | ||||
| // If the requested attribute is not defined, the returned error will | ||||
| // be of type NotDefinedError. | ||||
| // | ||||
| // ProjectAttributeValue may return ("", nil) if the attribute was | ||||
| // defined to be the empty string. | ||||
| func (c *Client) ProjectAttributeValue(attr string) (string, error) { | ||||
| 	return c.Get("project/attributes/" + attr) | ||||
| } | ||||
|  | ||||
| // Scopes returns the service account scopes for the given account. | ||||
| // The account may be empty or the string "default" to use the instance's | ||||
| // main account. | ||||
| func (c *Client) Scopes(serviceAccount string) ([]string, error) { | ||||
| 	if serviceAccount == "" { | ||||
| 		serviceAccount = "default" | ||||
| 	} | ||||
| 	return c.lines("instance/service-accounts/" + serviceAccount + "/scopes") | ||||
| } | ||||
|  | ||||
| // Subscribe subscribes to a value from the metadata service. | ||||
| // The suffix is appended to "http://${GCE_METADATA_HOST}/computeMetadata/v1/". | ||||
| // The suffix may contain query parameters. | ||||
| // | ||||
| // Subscribe calls fn with the latest metadata value indicated by the provided | ||||
| // suffix. If the metadata value is deleted, fn is called with the empty string | ||||
| // and ok false. Subscribe blocks until fn returns a non-nil error or the value | ||||
| // is deleted. Subscribe returns the error value returned from the last call to | ||||
| // fn, which may be nil when ok == false. | ||||
| func (c *Client) Subscribe(suffix string, fn func(v string, ok bool) error) error { | ||||
| 	const failedSubscribeSleep = time.Second * 5 | ||||
|  | ||||
| 	// First check to see if the metadata value exists at all. | ||||
| 	val, lastETag, err := c.getETag(suffix) | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
|  | ||||
| 	if err := fn(val, true); err != nil { | ||||
| 		return err | ||||
| 	} | ||||
|  | ||||
| 	ok := true | ||||
| 	if strings.ContainsRune(suffix, '?') { | ||||
| 		suffix += "&wait_for_change=true&last_etag=" | ||||
| 	} else { | ||||
| 		suffix += "?wait_for_change=true&last_etag=" | ||||
| 	} | ||||
| 	for { | ||||
| 		val, etag, err := c.getETag(suffix + url.QueryEscape(lastETag)) | ||||
| 		if err != nil { | ||||
| 			if _, deleted := err.(NotDefinedError); !deleted { | ||||
| 				time.Sleep(failedSubscribeSleep) | ||||
| 				continue // Retry on other errors. | ||||
| 			} | ||||
| 			ok = false | ||||
| 		} | ||||
| 		lastETag = etag | ||||
|  | ||||
| 		if err := fn(val, ok); err != nil || !ok { | ||||
| 			return err | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // Error contains an error response from the server. | ||||
| type Error struct { | ||||
| 	// Code is the HTTP response status code. | ||||
| 	Code int | ||||
| 	// Message is the server response message. | ||||
| 	Message string | ||||
| } | ||||
|  | ||||
| func (e *Error) Error() string { | ||||
| 	return fmt.Sprintf("compute: Received %d `%s`", e.Code, e.Message) | ||||
| } | ||||
							
								
								
									
										315
									
								
								vendor/cloud.google.com/go/iam/iam.go
									
									
									
										generated
									
									
										vendored
									
									
								
							
							
						
						
									
										315
									
								
								vendor/cloud.google.com/go/iam/iam.go
									
									
									
										generated
									
									
										vendored
									
									
								
							| @@ -1,315 +0,0 @@ | ||||
| // Copyright 2016 Google LLC | ||||
| // | ||||
| // Licensed under the Apache License, Version 2.0 (the "License"); | ||||
| // you may not use this file except in compliance with the License. | ||||
| // You may obtain a copy of the License at | ||||
| // | ||||
| //      http://www.apache.org/licenses/LICENSE-2.0 | ||||
| // | ||||
| // Unless required by applicable law or agreed to in writing, software | ||||
| // distributed under the License is distributed on an "AS IS" BASIS, | ||||
| // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||||
| // See the License for the specific language governing permissions and | ||||
| // limitations under the License. | ||||
|  | ||||
| // Package iam supports the resource-specific operations of Google Cloud | ||||
| // IAM (Identity and Access Management) for the Google Cloud Libraries. | ||||
| // See https://cloud.google.com/iam for more about IAM. | ||||
| // | ||||
| // Users of the Google Cloud Libraries will typically not use this package | ||||
| // directly. Instead they will begin with some resource that supports IAM, like | ||||
| // a pubsub topic, and call its IAM method to get a Handle for that resource. | ||||
| package iam | ||||
|  | ||||
| import ( | ||||
| 	"context" | ||||
| 	"fmt" | ||||
| 	"time" | ||||
|  | ||||
| 	gax "github.com/googleapis/gax-go/v2" | ||||
| 	pb "google.golang.org/genproto/googleapis/iam/v1" | ||||
| 	"google.golang.org/grpc" | ||||
| 	"google.golang.org/grpc/codes" | ||||
| 	"google.golang.org/grpc/metadata" | ||||
| ) | ||||
|  | ||||
| // client abstracts the IAMPolicy API to allow multiple implementations. | ||||
| type client interface { | ||||
| 	Get(ctx context.Context, resource string) (*pb.Policy, error) | ||||
| 	Set(ctx context.Context, resource string, p *pb.Policy) error | ||||
| 	Test(ctx context.Context, resource string, perms []string) ([]string, error) | ||||
| } | ||||
|  | ||||
| // grpcClient implements client for the standard gRPC-based IAMPolicy service. | ||||
| type grpcClient struct { | ||||
| 	c pb.IAMPolicyClient | ||||
| } | ||||
|  | ||||
| var withRetry = gax.WithRetry(func() gax.Retryer { | ||||
| 	return gax.OnCodes([]codes.Code{ | ||||
| 		codes.DeadlineExceeded, | ||||
| 		codes.Unavailable, | ||||
| 	}, gax.Backoff{ | ||||
| 		Initial:    100 * time.Millisecond, | ||||
| 		Max:        60 * time.Second, | ||||
| 		Multiplier: 1.3, | ||||
| 	}) | ||||
| }) | ||||
|  | ||||
| func (g *grpcClient) Get(ctx context.Context, resource string) (*pb.Policy, error) { | ||||
| 	var proto *pb.Policy | ||||
| 	md := metadata.Pairs("x-goog-request-params", fmt.Sprintf("%s=%v", "resource", resource)) | ||||
| 	ctx = insertMetadata(ctx, md) | ||||
|  | ||||
| 	err := gax.Invoke(ctx, func(ctx context.Context, _ gax.CallSettings) error { | ||||
| 		var err error | ||||
| 		proto, err = g.c.GetIamPolicy(ctx, &pb.GetIamPolicyRequest{Resource: resource}) | ||||
| 		return err | ||||
| 	}, withRetry) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	return proto, nil | ||||
| } | ||||
|  | ||||
| func (g *grpcClient) Set(ctx context.Context, resource string, p *pb.Policy) error { | ||||
| 	md := metadata.Pairs("x-goog-request-params", fmt.Sprintf("%s=%v", "resource", resource)) | ||||
| 	ctx = insertMetadata(ctx, md) | ||||
|  | ||||
| 	return gax.Invoke(ctx, func(ctx context.Context, _ gax.CallSettings) error { | ||||
| 		_, err := g.c.SetIamPolicy(ctx, &pb.SetIamPolicyRequest{ | ||||
| 			Resource: resource, | ||||
| 			Policy:   p, | ||||
| 		}) | ||||
| 		return err | ||||
| 	}, withRetry) | ||||
| } | ||||
|  | ||||
| func (g *grpcClient) Test(ctx context.Context, resource string, perms []string) ([]string, error) { | ||||
| 	var res *pb.TestIamPermissionsResponse | ||||
| 	md := metadata.Pairs("x-goog-request-params", fmt.Sprintf("%s=%v", "resource", resource)) | ||||
| 	ctx = insertMetadata(ctx, md) | ||||
|  | ||||
| 	err := gax.Invoke(ctx, func(ctx context.Context, _ gax.CallSettings) error { | ||||
| 		var err error | ||||
| 		res, err = g.c.TestIamPermissions(ctx, &pb.TestIamPermissionsRequest{ | ||||
| 			Resource:    resource, | ||||
| 			Permissions: perms, | ||||
| 		}) | ||||
| 		return err | ||||
| 	}, withRetry) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	return res.Permissions, nil | ||||
| } | ||||
|  | ||||
| // A Handle provides IAM operations for a resource. | ||||
| type Handle struct { | ||||
| 	c        client | ||||
| 	resource string | ||||
| } | ||||
|  | ||||
| // InternalNewHandle is for use by the Google Cloud Libraries only. | ||||
| // | ||||
| // InternalNewHandle returns a Handle for resource. | ||||
| // The conn parameter refers to a server that must support the IAMPolicy service. | ||||
| func InternalNewHandle(conn *grpc.ClientConn, resource string) *Handle { | ||||
| 	return InternalNewHandleGRPCClient(pb.NewIAMPolicyClient(conn), resource) | ||||
| } | ||||
|  | ||||
| // InternalNewHandleGRPCClient is for use by the Google Cloud Libraries only. | ||||
| // | ||||
| // InternalNewHandleClient returns a Handle for resource using the given | ||||
| // grpc service that implements IAM as a mixin | ||||
| func InternalNewHandleGRPCClient(c pb.IAMPolicyClient, resource string) *Handle { | ||||
| 	return InternalNewHandleClient(&grpcClient{c: c}, resource) | ||||
| } | ||||
|  | ||||
| // InternalNewHandleClient is for use by the Google Cloud Libraries only. | ||||
| // | ||||
| // InternalNewHandleClient returns a Handle for resource using the given | ||||
| // client implementation. | ||||
| func InternalNewHandleClient(c client, resource string) *Handle { | ||||
| 	return &Handle{ | ||||
| 		c:        c, | ||||
| 		resource: resource, | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // Policy retrieves the IAM policy for the resource. | ||||
| func (h *Handle) Policy(ctx context.Context) (*Policy, error) { | ||||
| 	proto, err := h.c.Get(ctx, h.resource) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	return &Policy{InternalProto: proto}, nil | ||||
| } | ||||
|  | ||||
| // SetPolicy replaces the resource's current policy with the supplied Policy. | ||||
| // | ||||
| // If policy was created from a prior call to Get, then the modification will | ||||
| // only succeed if the policy has not changed since the Get. | ||||
| func (h *Handle) SetPolicy(ctx context.Context, policy *Policy) error { | ||||
| 	return h.c.Set(ctx, h.resource, policy.InternalProto) | ||||
| } | ||||
|  | ||||
| // TestPermissions returns the subset of permissions that the caller has on the resource. | ||||
| func (h *Handle) TestPermissions(ctx context.Context, permissions []string) ([]string, error) { | ||||
| 	return h.c.Test(ctx, h.resource, permissions) | ||||
| } | ||||
|  | ||||
| // A RoleName is a name representing a collection of permissions. | ||||
| type RoleName string | ||||
|  | ||||
| // Common role names. | ||||
| const ( | ||||
| 	Owner  RoleName = "roles/owner" | ||||
| 	Editor RoleName = "roles/editor" | ||||
| 	Viewer RoleName = "roles/viewer" | ||||
| ) | ||||
|  | ||||
| const ( | ||||
| 	// AllUsers is a special member that denotes all users, even unauthenticated ones. | ||||
| 	AllUsers = "allUsers" | ||||
|  | ||||
| 	// AllAuthenticatedUsers is a special member that denotes all authenticated users. | ||||
| 	AllAuthenticatedUsers = "allAuthenticatedUsers" | ||||
| ) | ||||
|  | ||||
| // A Policy is a list of Bindings representing roles | ||||
| // granted to members. | ||||
| // | ||||
| // The zero Policy is a valid policy with no bindings. | ||||
| type Policy struct { | ||||
| 	// TODO(jba): when type aliases are available, put Policy into an internal package | ||||
| 	// and provide an exported alias here. | ||||
|  | ||||
| 	// This field is exported for use by the Google Cloud Libraries only. | ||||
| 	// It may become unexported in a future release. | ||||
| 	InternalProto *pb.Policy | ||||
| } | ||||
|  | ||||
| // Members returns the list of members with the supplied role. | ||||
| // The return value should not be modified. Use Add and Remove | ||||
| // to modify the members of a role. | ||||
| func (p *Policy) Members(r RoleName) []string { | ||||
| 	b := p.binding(r) | ||||
| 	if b == nil { | ||||
| 		return nil | ||||
| 	} | ||||
| 	return b.Members | ||||
| } | ||||
|  | ||||
| // HasRole reports whether member has role r. | ||||
| func (p *Policy) HasRole(member string, r RoleName) bool { | ||||
| 	return memberIndex(member, p.binding(r)) >= 0 | ||||
| } | ||||
|  | ||||
| // Add adds member member to role r if it is not already present. | ||||
| // A new binding is created if there is no binding for the role. | ||||
| func (p *Policy) Add(member string, r RoleName) { | ||||
| 	b := p.binding(r) | ||||
| 	if b == nil { | ||||
| 		if p.InternalProto == nil { | ||||
| 			p.InternalProto = &pb.Policy{} | ||||
| 		} | ||||
| 		p.InternalProto.Bindings = append(p.InternalProto.Bindings, &pb.Binding{ | ||||
| 			Role:    string(r), | ||||
| 			Members: []string{member}, | ||||
| 		}) | ||||
| 		return | ||||
| 	} | ||||
| 	if memberIndex(member, b) < 0 { | ||||
| 		b.Members = append(b.Members, member) | ||||
| 		return | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // Remove removes member from role r if it is present. | ||||
| func (p *Policy) Remove(member string, r RoleName) { | ||||
| 	bi := p.bindingIndex(r) | ||||
| 	if bi < 0 { | ||||
| 		return | ||||
| 	} | ||||
| 	bindings := p.InternalProto.Bindings | ||||
| 	b := bindings[bi] | ||||
| 	mi := memberIndex(member, b) | ||||
| 	if mi < 0 { | ||||
| 		return | ||||
| 	} | ||||
| 	// Order doesn't matter for bindings or members, so to remove, move the last item | ||||
| 	// into the removed spot and shrink the slice. | ||||
| 	if len(b.Members) == 1 { | ||||
| 		// Remove binding. | ||||
| 		last := len(bindings) - 1 | ||||
| 		bindings[bi] = bindings[last] | ||||
| 		bindings[last] = nil | ||||
| 		p.InternalProto.Bindings = bindings[:last] | ||||
| 		return | ||||
| 	} | ||||
| 	// Remove member. | ||||
| 	// TODO(jba): worry about multiple copies of m? | ||||
| 	last := len(b.Members) - 1 | ||||
| 	b.Members[mi] = b.Members[last] | ||||
| 	b.Members[last] = "" | ||||
| 	b.Members = b.Members[:last] | ||||
| } | ||||
|  | ||||
| // Roles returns the names of all the roles that appear in the Policy. | ||||
| func (p *Policy) Roles() []RoleName { | ||||
| 	if p.InternalProto == nil { | ||||
| 		return nil | ||||
| 	} | ||||
| 	var rns []RoleName | ||||
| 	for _, b := range p.InternalProto.Bindings { | ||||
| 		rns = append(rns, RoleName(b.Role)) | ||||
| 	} | ||||
| 	return rns | ||||
| } | ||||
|  | ||||
| // binding returns the Binding for the suppied role, or nil if there isn't one. | ||||
| func (p *Policy) binding(r RoleName) *pb.Binding { | ||||
| 	i := p.bindingIndex(r) | ||||
| 	if i < 0 { | ||||
| 		return nil | ||||
| 	} | ||||
| 	return p.InternalProto.Bindings[i] | ||||
| } | ||||
|  | ||||
| func (p *Policy) bindingIndex(r RoleName) int { | ||||
| 	if p.InternalProto == nil { | ||||
| 		return -1 | ||||
| 	} | ||||
| 	for i, b := range p.InternalProto.Bindings { | ||||
| 		if b.Role == string(r) { | ||||
| 			return i | ||||
| 		} | ||||
| 	} | ||||
| 	return -1 | ||||
| } | ||||
|  | ||||
| // memberIndex returns the index of m in b's Members, or -1 if not found. | ||||
| func memberIndex(m string, b *pb.Binding) int { | ||||
| 	if b == nil { | ||||
| 		return -1 | ||||
| 	} | ||||
| 	for i, mm := range b.Members { | ||||
| 		if mm == m { | ||||
| 			return i | ||||
| 		} | ||||
| 	} | ||||
| 	return -1 | ||||
| } | ||||
|  | ||||
| // insertMetadata inserts metadata into the given context | ||||
| func insertMetadata(ctx context.Context, mds ...metadata.MD) context.Context { | ||||
| 	out, _ := metadata.FromOutgoingContext(ctx) | ||||
| 	out = out.Copy() | ||||
| 	for _, md := range mds { | ||||
| 		for k, v := range md { | ||||
| 			out[k] = append(out[k], v...) | ||||
| 		} | ||||
| 	} | ||||
| 	return metadata.NewOutgoingContext(ctx, out) | ||||
| } | ||||
							
								
								
									
										54
									
								
								vendor/cloud.google.com/go/internal/annotate.go
									
									
									
										generated
									
									
										vendored
									
									
								
							
							
						
						
									
										54
									
								
								vendor/cloud.google.com/go/internal/annotate.go
									
									
									
										generated
									
									
										vendored
									
									
								
							| @@ -1,54 +0,0 @@ | ||||
| // Copyright 2017 Google LLC | ||||
| // | ||||
| // Licensed under the Apache License, Version 2.0 (the "License"); | ||||
| // you may not use this file except in compliance with the License. | ||||
| // You may obtain a copy of the License at | ||||
| // | ||||
| //      http://www.apache.org/licenses/LICENSE-2.0 | ||||
| // | ||||
| // Unless required by applicable law or agreed to in writing, software | ||||
| // distributed under the License is distributed on an "AS IS" BASIS, | ||||
| // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||||
| // See the License for the specific language governing permissions and | ||||
| // limitations under the License. | ||||
|  | ||||
| package internal | ||||
|  | ||||
| import ( | ||||
| 	"fmt" | ||||
|  | ||||
| 	"google.golang.org/api/googleapi" | ||||
| 	"google.golang.org/grpc/status" | ||||
| ) | ||||
|  | ||||
| // Annotate prepends msg to the error message in err, attempting | ||||
| // to preserve other information in err, like an error code. | ||||
| // | ||||
| // Annotate panics if err is nil. | ||||
| // | ||||
| // Annotate knows about these error types: | ||||
| // - "google.golang.org/grpc/status".Status | ||||
| // - "google.golang.org/api/googleapi".Error | ||||
| // If the error is not one of these types, Annotate behaves | ||||
| // like | ||||
| //   fmt.Errorf("%s: %v", msg, err) | ||||
| func Annotate(err error, msg string) error { | ||||
| 	if err == nil { | ||||
| 		panic("Annotate called with nil") | ||||
| 	} | ||||
| 	if s, ok := status.FromError(err); ok { | ||||
| 		p := s.Proto() | ||||
| 		p.Message = msg + ": " + p.Message | ||||
| 		return status.ErrorProto(p) | ||||
| 	} | ||||
| 	if g, ok := err.(*googleapi.Error); ok { | ||||
| 		g.Message = msg + ": " + g.Message | ||||
| 		return g | ||||
| 	} | ||||
| 	return fmt.Errorf("%s: %v", msg, err) | ||||
| } | ||||
|  | ||||
| // Annotatef uses format and args to format a string, then calls Annotate. | ||||
| func Annotatef(err error, format string, args ...interface{}) error { | ||||
| 	return Annotate(err, fmt.Sprintf(format, args...)) | ||||
| } | ||||
							
								
								
									
										108
									
								
								vendor/cloud.google.com/go/internal/optional/optional.go
									
									
									
										generated
									
									
										vendored
									
									
								
							
							
						
						
									
										108
									
								
								vendor/cloud.google.com/go/internal/optional/optional.go
									
									
									
										generated
									
									
										vendored
									
									
								
							| @@ -1,108 +0,0 @@ | ||||
| // Copyright 2016 Google LLC | ||||
| // | ||||
| // Licensed under the Apache License, Version 2.0 (the "License"); | ||||
| // you may not use this file except in compliance with the License. | ||||
| // You may obtain a copy of the License at | ||||
| // | ||||
| //      http://www.apache.org/licenses/LICENSE-2.0 | ||||
| // | ||||
| // Unless required by applicable law or agreed to in writing, software | ||||
| // distributed under the License is distributed on an "AS IS" BASIS, | ||||
| // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||||
| // See the License for the specific language governing permissions and | ||||
| // limitations under the License. | ||||
|  | ||||
| // Package optional provides versions of primitive types that can | ||||
| // be nil. These are useful in methods that update some of an API object's | ||||
| // fields. | ||||
| package optional | ||||
|  | ||||
| import ( | ||||
| 	"fmt" | ||||
| 	"strings" | ||||
| 	"time" | ||||
| ) | ||||
|  | ||||
| type ( | ||||
| 	// Bool is either a bool or nil. | ||||
| 	Bool interface{} | ||||
|  | ||||
| 	// String is either a string or nil. | ||||
| 	String interface{} | ||||
|  | ||||
| 	// Int is either an int or nil. | ||||
| 	Int interface{} | ||||
|  | ||||
| 	// Uint is either a uint or nil. | ||||
| 	Uint interface{} | ||||
|  | ||||
| 	// Float64 is either a float64 or nil. | ||||
| 	Float64 interface{} | ||||
|  | ||||
| 	// Duration is either a time.Duration or nil. | ||||
| 	Duration interface{} | ||||
| ) | ||||
|  | ||||
| // ToBool returns its argument as a bool. | ||||
| // It panics if its argument is nil or not a bool. | ||||
| func ToBool(v Bool) bool { | ||||
| 	x, ok := v.(bool) | ||||
| 	if !ok { | ||||
| 		doPanic("Bool", v) | ||||
| 	} | ||||
| 	return x | ||||
| } | ||||
|  | ||||
| // ToString returns its argument as a string. | ||||
| // It panics if its argument is nil or not a string. | ||||
| func ToString(v String) string { | ||||
| 	x, ok := v.(string) | ||||
| 	if !ok { | ||||
| 		doPanic("String", v) | ||||
| 	} | ||||
| 	return x | ||||
| } | ||||
|  | ||||
| // ToInt returns its argument as an int. | ||||
| // It panics if its argument is nil or not an int. | ||||
| func ToInt(v Int) int { | ||||
| 	x, ok := v.(int) | ||||
| 	if !ok { | ||||
| 		doPanic("Int", v) | ||||
| 	} | ||||
| 	return x | ||||
| } | ||||
|  | ||||
| // ToUint returns its argument as a uint. | ||||
| // It panics if its argument is nil or not a uint. | ||||
| func ToUint(v Uint) uint { | ||||
| 	x, ok := v.(uint) | ||||
| 	if !ok { | ||||
| 		doPanic("Uint", v) | ||||
| 	} | ||||
| 	return x | ||||
| } | ||||
|  | ||||
| // ToFloat64 returns its argument as a float64. | ||||
| // It panics if its argument is nil or not a float64. | ||||
| func ToFloat64(v Float64) float64 { | ||||
| 	x, ok := v.(float64) | ||||
| 	if !ok { | ||||
| 		doPanic("Float64", v) | ||||
| 	} | ||||
| 	return x | ||||
| } | ||||
|  | ||||
| // ToDuration returns its argument as a time.Duration. | ||||
| // It panics if its argument is nil or not a time.Duration. | ||||
| func ToDuration(v Duration) time.Duration { | ||||
| 	x, ok := v.(time.Duration) | ||||
| 	if !ok { | ||||
| 		doPanic("Duration", v) | ||||
| 	} | ||||
| 	return x | ||||
| } | ||||
|  | ||||
| func doPanic(capType string, v interface{}) { | ||||
| 	panic(fmt.Sprintf("optional.%s value should be %s, got %T", capType, strings.ToLower(capType), v)) | ||||
| } | ||||
							
								
								
									
										54
									
								
								vendor/cloud.google.com/go/internal/retry.go
									
									
									
										generated
									
									
										vendored
									
									
								
							
							
						
						
									
										54
									
								
								vendor/cloud.google.com/go/internal/retry.go
									
									
									
										generated
									
									
										vendored
									
									
								
							| @@ -1,54 +0,0 @@ | ||||
| // Copyright 2016 Google LLC | ||||
| // | ||||
| // Licensed under the Apache License, Version 2.0 (the "License"); | ||||
| // you may not use this file except in compliance with the License. | ||||
| // You may obtain a copy of the License at | ||||
| // | ||||
| //      http://www.apache.org/licenses/LICENSE-2.0 | ||||
| // | ||||
| // Unless required by applicable law or agreed to in writing, software | ||||
| // distributed under the License is distributed on an "AS IS" BASIS, | ||||
| // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||||
| // See the License for the specific language governing permissions and | ||||
| // limitations under the License. | ||||
|  | ||||
| package internal | ||||
|  | ||||
| import ( | ||||
| 	"context" | ||||
| 	"time" | ||||
|  | ||||
| 	gax "github.com/googleapis/gax-go/v2" | ||||
| ) | ||||
|  | ||||
| // Retry calls the supplied function f repeatedly according to the provided | ||||
| // backoff parameters. It returns when one of the following occurs: | ||||
| // When f's first return value is true, Retry immediately returns with f's second | ||||
| // return value. | ||||
| // When the provided context is done, Retry returns with an error that | ||||
| // includes both ctx.Error() and the last error returned by f. | ||||
| func Retry(ctx context.Context, bo gax.Backoff, f func() (stop bool, err error)) error { | ||||
| 	return retry(ctx, bo, f, gax.Sleep) | ||||
| } | ||||
|  | ||||
| func retry(ctx context.Context, bo gax.Backoff, f func() (stop bool, err error), | ||||
| 	sleep func(context.Context, time.Duration) error) error { | ||||
| 	var lastErr error | ||||
| 	for { | ||||
| 		stop, err := f() | ||||
| 		if stop { | ||||
| 			return err | ||||
| 		} | ||||
| 		// Remember the last "real" error from f. | ||||
| 		if err != nil && err != context.Canceled && err != context.DeadlineExceeded { | ||||
| 			lastErr = err | ||||
| 		} | ||||
| 		p := bo.Pause() | ||||
| 		if cerr := sleep(ctx, p); cerr != nil { | ||||
| 			if lastErr != nil { | ||||
| 				return Annotatef(lastErr, "retry failed with %v; last error", cerr) | ||||
| 			} | ||||
| 			return cerr | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
							
								
								
									
										109
									
								
								vendor/cloud.google.com/go/internal/trace/trace.go
									
									
									
										generated
									
									
										vendored
									
									
								
							
							
						
						
									
										109
									
								
								vendor/cloud.google.com/go/internal/trace/trace.go
									
									
									
										generated
									
									
										vendored
									
									
								
							| @@ -1,109 +0,0 @@ | ||||
| // Copyright 2018 Google LLC | ||||
| // | ||||
| // Licensed under the Apache License, Version 2.0 (the "License"); | ||||
| // you may not use this file except in compliance with the License. | ||||
| // You may obtain a copy of the License at | ||||
| // | ||||
| //      http://www.apache.org/licenses/LICENSE-2.0 | ||||
| // | ||||
| // Unless required by applicable law or agreed to in writing, software | ||||
| // distributed under the License is distributed on an "AS IS" BASIS, | ||||
| // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||||
| // See the License for the specific language governing permissions and | ||||
| // limitations under the License. | ||||
|  | ||||
| package trace | ||||
|  | ||||
| import ( | ||||
| 	"context" | ||||
| 	"fmt" | ||||
|  | ||||
| 	"go.opencensus.io/trace" | ||||
| 	"google.golang.org/api/googleapi" | ||||
| 	"google.golang.org/genproto/googleapis/rpc/code" | ||||
| 	"google.golang.org/grpc/status" | ||||
| ) | ||||
|  | ||||
| // StartSpan adds a span to the trace with the given name. | ||||
| func StartSpan(ctx context.Context, name string) context.Context { | ||||
| 	ctx, _ = trace.StartSpan(ctx, name) | ||||
| 	return ctx | ||||
| } | ||||
|  | ||||
| // EndSpan ends a span with the given error. | ||||
| func EndSpan(ctx context.Context, err error) { | ||||
| 	span := trace.FromContext(ctx) | ||||
| 	if err != nil { | ||||
| 		span.SetStatus(toStatus(err)) | ||||
| 	} | ||||
| 	span.End() | ||||
| } | ||||
|  | ||||
| // toStatus interrogates an error and converts it to an appropriate | ||||
| // OpenCensus status. | ||||
| func toStatus(err error) trace.Status { | ||||
| 	if err2, ok := err.(*googleapi.Error); ok { | ||||
| 		return trace.Status{Code: httpStatusCodeToOCCode(err2.Code), Message: err2.Message} | ||||
| 	} else if s, ok := status.FromError(err); ok { | ||||
| 		return trace.Status{Code: int32(s.Code()), Message: s.Message()} | ||||
| 	} else { | ||||
| 		return trace.Status{Code: int32(code.Code_UNKNOWN), Message: err.Error()} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // TODO(deklerk): switch to using OpenCensus function when it becomes available. | ||||
| // Reference: https://github.com/googleapis/googleapis/blob/26b634d2724ac5dd30ae0b0cbfb01f07f2e4050e/google/rpc/code.proto | ||||
| func httpStatusCodeToOCCode(httpStatusCode int) int32 { | ||||
| 	switch httpStatusCode { | ||||
| 	case 200: | ||||
| 		return int32(code.Code_OK) | ||||
| 	case 499: | ||||
| 		return int32(code.Code_CANCELLED) | ||||
| 	case 500: | ||||
| 		return int32(code.Code_UNKNOWN) // Could also be Code_INTERNAL, Code_DATA_LOSS | ||||
| 	case 400: | ||||
| 		return int32(code.Code_INVALID_ARGUMENT) // Could also be Code_OUT_OF_RANGE | ||||
| 	case 504: | ||||
| 		return int32(code.Code_DEADLINE_EXCEEDED) | ||||
| 	case 404: | ||||
| 		return int32(code.Code_NOT_FOUND) | ||||
| 	case 409: | ||||
| 		return int32(code.Code_ALREADY_EXISTS) // Could also be Code_ABORTED | ||||
| 	case 403: | ||||
| 		return int32(code.Code_PERMISSION_DENIED) | ||||
| 	case 401: | ||||
| 		return int32(code.Code_UNAUTHENTICATED) | ||||
| 	case 429: | ||||
| 		return int32(code.Code_RESOURCE_EXHAUSTED) | ||||
| 	case 501: | ||||
| 		return int32(code.Code_UNIMPLEMENTED) | ||||
| 	case 503: | ||||
| 		return int32(code.Code_UNAVAILABLE) | ||||
| 	default: | ||||
| 		return int32(code.Code_UNKNOWN) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // TODO: (odeke-em): perhaps just pass around spans due to the cost | ||||
| // incurred from using trace.FromContext(ctx) yet we could avoid | ||||
| // throwing away the work done by ctx, span := trace.StartSpan. | ||||
| func TracePrintf(ctx context.Context, attrMap map[string]interface{}, format string, args ...interface{}) { | ||||
| 	var attrs []trace.Attribute | ||||
| 	for k, v := range attrMap { | ||||
| 		var a trace.Attribute | ||||
| 		switch v := v.(type) { | ||||
| 		case string: | ||||
| 			a = trace.StringAttribute(k, v) | ||||
| 		case bool: | ||||
| 			a = trace.BoolAttribute(k, v) | ||||
| 		case int: | ||||
| 			a = trace.Int64Attribute(k, int64(v)) | ||||
| 		case int64: | ||||
| 			a = trace.Int64Attribute(k, v) | ||||
| 		default: | ||||
| 			a = trace.StringAttribute(k, fmt.Sprintf("%#v", v)) | ||||
| 		} | ||||
| 		attrs = append(attrs, a) | ||||
| 	} | ||||
| 	trace.FromContext(ctx).Annotatef(attrs, format, args...) | ||||
| } | ||||
							
								
								
									
										19
									
								
								vendor/cloud.google.com/go/internal/version/update_version.sh
									
									
									
										generated
									
									
										vendored
									
									
								
							
							
						
						
									
										19
									
								
								vendor/cloud.google.com/go/internal/version/update_version.sh
									
									
									
										generated
									
									
										vendored
									
									
								
							| @@ -1,19 +0,0 @@ | ||||
| #!/bin/bash | ||||
| # Copyright 2019 Google LLC | ||||
| # | ||||
| # Licensed under the Apache License, Version 2.0 (the "License"); | ||||
| # you may not use this file except in compliance with the License. | ||||
| # You may obtain a copy of the License at | ||||
| # | ||||
| #      http://www.apache.org/licenses/LICENSE-2.0 | ||||
| # | ||||
| # Unless required by applicable law or agreed to in writing, software | ||||
| # distributed under the License is distributed on an "AS IS" BASIS, | ||||
| # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||||
| # See the License for the specific language governing permissions and | ||||
| # limitations under the License. | ||||
|  | ||||
| today=$(date +%Y%m%d) | ||||
|  | ||||
| sed -i -r -e 's/const Repo = "([0-9]{8})"/const Repo = "'$today'"/' $GOFILE | ||||
|  | ||||
							
								
								
									
										71
									
								
								vendor/cloud.google.com/go/internal/version/version.go
									
									
									
										generated
									
									
										vendored
									
									
								
							
							
						
						
									
										71
									
								
								vendor/cloud.google.com/go/internal/version/version.go
									
									
									
										generated
									
									
										vendored
									
									
								
							| @@ -1,71 +0,0 @@ | ||||
| // Copyright 2016 Google LLC | ||||
| // | ||||
| // Licensed under the Apache License, Version 2.0 (the "License"); | ||||
| // you may not use this file except in compliance with the License. | ||||
| // You may obtain a copy of the License at | ||||
| // | ||||
| //      http://www.apache.org/licenses/LICENSE-2.0 | ||||
| // | ||||
| // Unless required by applicable law or agreed to in writing, software | ||||
| // distributed under the License is distributed on an "AS IS" BASIS, | ||||
| // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||||
| // See the License for the specific language governing permissions and | ||||
| // limitations under the License. | ||||
|  | ||||
| //go:generate ./update_version.sh | ||||
|  | ||||
| // Package version contains version information for Google Cloud Client | ||||
| // Libraries for Go, as reported in request headers. | ||||
| package version | ||||
|  | ||||
| import ( | ||||
| 	"runtime" | ||||
| 	"strings" | ||||
| 	"unicode" | ||||
| ) | ||||
|  | ||||
| // Repo is the current version of the client libraries in this | ||||
| // repo. It should be a date in YYYYMMDD format. | ||||
| const Repo = "20190802" | ||||
|  | ||||
| // Go returns the Go runtime version. The returned string | ||||
| // has no whitespace. | ||||
| func Go() string { | ||||
| 	return goVersion | ||||
| } | ||||
|  | ||||
| var goVersion = goVer(runtime.Version()) | ||||
|  | ||||
| const develPrefix = "devel +" | ||||
|  | ||||
| func goVer(s string) string { | ||||
| 	if strings.HasPrefix(s, develPrefix) { | ||||
| 		s = s[len(develPrefix):] | ||||
| 		if p := strings.IndexFunc(s, unicode.IsSpace); p >= 0 { | ||||
| 			s = s[:p] | ||||
| 		} | ||||
| 		return s | ||||
| 	} | ||||
|  | ||||
| 	if strings.HasPrefix(s, "go1") { | ||||
| 		s = s[2:] | ||||
| 		var prerelease string | ||||
| 		if p := strings.IndexFunc(s, notSemverRune); p >= 0 { | ||||
| 			s, prerelease = s[:p], s[p:] | ||||
| 		} | ||||
| 		if strings.HasSuffix(s, ".") { | ||||
| 			s += "0" | ||||
| 		} else if strings.Count(s, ".") < 2 { | ||||
| 			s += ".0" | ||||
| 		} | ||||
| 		if prerelease != "" { | ||||
| 			s += "-" + prerelease | ||||
| 		} | ||||
| 		return s | ||||
| 	} | ||||
| 	return "" | ||||
| } | ||||
|  | ||||
| func notSemverRune(r rune) bool { | ||||
| 	return !strings.ContainsRune("0123456789.", r) | ||||
| } | ||||
							
								
								
									
										32
									
								
								vendor/cloud.google.com/go/storage/README.md
									
									
									
										generated
									
									
										vendored
									
									
								
							
							
						
						
									
										32
									
								
								vendor/cloud.google.com/go/storage/README.md
									
									
									
										generated
									
									
										vendored
									
									
								
							| @@ -1,32 +0,0 @@ | ||||
| ## Cloud Storage [](https://godoc.org/cloud.google.com/go/storage) | ||||
|  | ||||
| - [About Cloud Storage](https://cloud.google.com/storage/) | ||||
| - [API documentation](https://cloud.google.com/storage/docs) | ||||
| - [Go client documentation](https://godoc.org/cloud.google.com/go/storage) | ||||
| - [Complete sample programs](https://github.com/GoogleCloudPlatform/golang-samples/tree/master/storage) | ||||
|  | ||||
| ### Example Usage | ||||
|  | ||||
| First create a `storage.Client` to use throughout your application: | ||||
|  | ||||
| [snip]:# (storage-1) | ||||
| ```go | ||||
| client, err := storage.NewClient(ctx) | ||||
| if err != nil { | ||||
| 	log.Fatal(err) | ||||
| } | ||||
| ``` | ||||
|  | ||||
| [snip]:# (storage-2) | ||||
| ```go | ||||
| // Read the object1 from bucket. | ||||
| rc, err := client.Bucket("bucket").Object("object1").NewReader(ctx) | ||||
| if err != nil { | ||||
| 	log.Fatal(err) | ||||
| } | ||||
| defer rc.Close() | ||||
| body, err := ioutil.ReadAll(rc) | ||||
| if err != nil { | ||||
| 	log.Fatal(err) | ||||
| } | ||||
| ``` | ||||
							
								
								
									
										335
									
								
								vendor/cloud.google.com/go/storage/acl.go
									
									
									
										generated
									
									
										vendored
									
									
								
							
							
						
						
									
										335
									
								
								vendor/cloud.google.com/go/storage/acl.go
									
									
									
										generated
									
									
										vendored
									
									
								
							| @@ -1,335 +0,0 @@ | ||||
| // Copyright 2014 Google LLC | ||||
| // | ||||
| // Licensed under the Apache License, Version 2.0 (the "License"); | ||||
| // you may not use this file except in compliance with the License. | ||||
| // You may obtain a copy of the License at | ||||
| // | ||||
| //      http://www.apache.org/licenses/LICENSE-2.0 | ||||
| // | ||||
| // Unless required by applicable law or agreed to in writing, software | ||||
| // distributed under the License is distributed on an "AS IS" BASIS, | ||||
| // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||||
| // See the License for the specific language governing permissions and | ||||
| // limitations under the License. | ||||
|  | ||||
| package storage | ||||
|  | ||||
| import ( | ||||
| 	"context" | ||||
| 	"net/http" | ||||
| 	"reflect" | ||||
|  | ||||
| 	"cloud.google.com/go/internal/trace" | ||||
| 	"google.golang.org/api/googleapi" | ||||
| 	raw "google.golang.org/api/storage/v1" | ||||
| ) | ||||
|  | ||||
| // ACLRole is the level of access to grant. | ||||
| type ACLRole string | ||||
|  | ||||
| const ( | ||||
| 	RoleOwner  ACLRole = "OWNER" | ||||
| 	RoleReader ACLRole = "READER" | ||||
| 	RoleWriter ACLRole = "WRITER" | ||||
| ) | ||||
|  | ||||
| // ACLEntity refers to a user or group. | ||||
| // They are sometimes referred to as grantees. | ||||
| // | ||||
| // It could be in the form of: | ||||
| // "user-<userId>", "user-<email>", "group-<groupId>", "group-<email>", | ||||
| // "domain-<domain>" and "project-team-<projectId>". | ||||
| // | ||||
| // Or one of the predefined constants: AllUsers, AllAuthenticatedUsers. | ||||
| type ACLEntity string | ||||
|  | ||||
| const ( | ||||
| 	AllUsers              ACLEntity = "allUsers" | ||||
| 	AllAuthenticatedUsers ACLEntity = "allAuthenticatedUsers" | ||||
| ) | ||||
|  | ||||
| // ACLRule represents a grant for a role to an entity (user, group or team) for a | ||||
| // Google Cloud Storage object or bucket. | ||||
| type ACLRule struct { | ||||
| 	Entity      ACLEntity | ||||
| 	EntityID    string | ||||
| 	Role        ACLRole | ||||
| 	Domain      string | ||||
| 	Email       string | ||||
| 	ProjectTeam *ProjectTeam | ||||
| } | ||||
|  | ||||
| // ProjectTeam is the project team associated with the entity, if any. | ||||
| type ProjectTeam struct { | ||||
| 	ProjectNumber string | ||||
| 	Team          string | ||||
| } | ||||
|  | ||||
| // ACLHandle provides operations on an access control list for a Google Cloud Storage bucket or object. | ||||
| type ACLHandle struct { | ||||
| 	c           *Client | ||||
| 	bucket      string | ||||
| 	object      string | ||||
| 	isDefault   bool | ||||
| 	userProject string // for requester-pays buckets | ||||
| } | ||||
|  | ||||
| // Delete permanently deletes the ACL entry for the given entity. | ||||
| func (a *ACLHandle) Delete(ctx context.Context, entity ACLEntity) (err error) { | ||||
| 	ctx = trace.StartSpan(ctx, "cloud.google.com/go/storage.ACL.Delete") | ||||
| 	defer func() { trace.EndSpan(ctx, err) }() | ||||
|  | ||||
| 	if a.object != "" { | ||||
| 		return a.objectDelete(ctx, entity) | ||||
| 	} | ||||
| 	if a.isDefault { | ||||
| 		return a.bucketDefaultDelete(ctx, entity) | ||||
| 	} | ||||
| 	return a.bucketDelete(ctx, entity) | ||||
| } | ||||
|  | ||||
| // Set sets the role for the given entity. | ||||
| func (a *ACLHandle) Set(ctx context.Context, entity ACLEntity, role ACLRole) (err error) { | ||||
| 	ctx = trace.StartSpan(ctx, "cloud.google.com/go/storage.ACL.Set") | ||||
| 	defer func() { trace.EndSpan(ctx, err) }() | ||||
|  | ||||
| 	if a.object != "" { | ||||
| 		return a.objectSet(ctx, entity, role, false) | ||||
| 	} | ||||
| 	if a.isDefault { | ||||
| 		return a.objectSet(ctx, entity, role, true) | ||||
| 	} | ||||
| 	return a.bucketSet(ctx, entity, role) | ||||
| } | ||||
|  | ||||
| // List retrieves ACL entries. | ||||
| func (a *ACLHandle) List(ctx context.Context) (rules []ACLRule, err error) { | ||||
| 	ctx = trace.StartSpan(ctx, "cloud.google.com/go/storage.ACL.List") | ||||
| 	defer func() { trace.EndSpan(ctx, err) }() | ||||
|  | ||||
| 	if a.object != "" { | ||||
| 		return a.objectList(ctx) | ||||
| 	} | ||||
| 	if a.isDefault { | ||||
| 		return a.bucketDefaultList(ctx) | ||||
| 	} | ||||
| 	return a.bucketList(ctx) | ||||
| } | ||||
|  | ||||
| func (a *ACLHandle) bucketDefaultList(ctx context.Context) ([]ACLRule, error) { | ||||
| 	var acls *raw.ObjectAccessControls | ||||
| 	var err error | ||||
| 	err = runWithRetry(ctx, func() error { | ||||
| 		req := a.c.raw.DefaultObjectAccessControls.List(a.bucket) | ||||
| 		a.configureCall(ctx, req) | ||||
| 		acls, err = req.Do() | ||||
| 		return err | ||||
| 	}) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	return toObjectACLRules(acls.Items), nil | ||||
| } | ||||
|  | ||||
| func (a *ACLHandle) bucketDefaultDelete(ctx context.Context, entity ACLEntity) error { | ||||
| 	return runWithRetry(ctx, func() error { | ||||
| 		req := a.c.raw.DefaultObjectAccessControls.Delete(a.bucket, string(entity)) | ||||
| 		a.configureCall(ctx, req) | ||||
| 		return req.Do() | ||||
| 	}) | ||||
| } | ||||
|  | ||||
| func (a *ACLHandle) bucketList(ctx context.Context) ([]ACLRule, error) { | ||||
| 	var acls *raw.BucketAccessControls | ||||
| 	var err error | ||||
| 	err = runWithRetry(ctx, func() error { | ||||
| 		req := a.c.raw.BucketAccessControls.List(a.bucket) | ||||
| 		a.configureCall(ctx, req) | ||||
| 		acls, err = req.Do() | ||||
| 		return err | ||||
| 	}) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	return toBucketACLRules(acls.Items), nil | ||||
| } | ||||
|  | ||||
| func (a *ACLHandle) bucketSet(ctx context.Context, entity ACLEntity, role ACLRole) error { | ||||
| 	acl := &raw.BucketAccessControl{ | ||||
| 		Bucket: a.bucket, | ||||
| 		Entity: string(entity), | ||||
| 		Role:   string(role), | ||||
| 	} | ||||
| 	err := runWithRetry(ctx, func() error { | ||||
| 		req := a.c.raw.BucketAccessControls.Update(a.bucket, string(entity), acl) | ||||
| 		a.configureCall(ctx, req) | ||||
| 		_, err := req.Do() | ||||
| 		return err | ||||
| 	}) | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (a *ACLHandle) bucketDelete(ctx context.Context, entity ACLEntity) error { | ||||
| 	return runWithRetry(ctx, func() error { | ||||
| 		req := a.c.raw.BucketAccessControls.Delete(a.bucket, string(entity)) | ||||
| 		a.configureCall(ctx, req) | ||||
| 		return req.Do() | ||||
| 	}) | ||||
| } | ||||
|  | ||||
| func (a *ACLHandle) objectList(ctx context.Context) ([]ACLRule, error) { | ||||
| 	var acls *raw.ObjectAccessControls | ||||
| 	var err error | ||||
| 	err = runWithRetry(ctx, func() error { | ||||
| 		req := a.c.raw.ObjectAccessControls.List(a.bucket, a.object) | ||||
| 		a.configureCall(ctx, req) | ||||
| 		acls, err = req.Do() | ||||
| 		return err | ||||
| 	}) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	return toObjectACLRules(acls.Items), nil | ||||
| } | ||||
|  | ||||
| func (a *ACLHandle) objectSet(ctx context.Context, entity ACLEntity, role ACLRole, isBucketDefault bool) error { | ||||
| 	type setRequest interface { | ||||
| 		Do(opts ...googleapi.CallOption) (*raw.ObjectAccessControl, error) | ||||
| 		Header() http.Header | ||||
| 	} | ||||
|  | ||||
| 	acl := &raw.ObjectAccessControl{ | ||||
| 		Bucket: a.bucket, | ||||
| 		Entity: string(entity), | ||||
| 		Role:   string(role), | ||||
| 	} | ||||
| 	var req setRequest | ||||
| 	if isBucketDefault { | ||||
| 		req = a.c.raw.DefaultObjectAccessControls.Update(a.bucket, string(entity), acl) | ||||
| 	} else { | ||||
| 		req = a.c.raw.ObjectAccessControls.Update(a.bucket, a.object, string(entity), acl) | ||||
| 	} | ||||
| 	a.configureCall(ctx, req) | ||||
| 	return runWithRetry(ctx, func() error { | ||||
| 		_, err := req.Do() | ||||
| 		return err | ||||
| 	}) | ||||
| } | ||||
|  | ||||
| func (a *ACLHandle) objectDelete(ctx context.Context, entity ACLEntity) error { | ||||
| 	return runWithRetry(ctx, func() error { | ||||
| 		req := a.c.raw.ObjectAccessControls.Delete(a.bucket, a.object, string(entity)) | ||||
| 		a.configureCall(ctx, req) | ||||
| 		return req.Do() | ||||
| 	}) | ||||
| } | ||||
|  | ||||
| func (a *ACLHandle) configureCall(ctx context.Context, call interface{ Header() http.Header }) { | ||||
| 	vc := reflect.ValueOf(call) | ||||
| 	vc.MethodByName("Context").Call([]reflect.Value{reflect.ValueOf(ctx)}) | ||||
| 	if a.userProject != "" { | ||||
| 		vc.MethodByName("UserProject").Call([]reflect.Value{reflect.ValueOf(a.userProject)}) | ||||
| 	} | ||||
| 	setClientHeader(call.Header()) | ||||
| } | ||||
|  | ||||
| func toObjectACLRules(items []*raw.ObjectAccessControl) []ACLRule { | ||||
| 	var rs []ACLRule | ||||
| 	for _, item := range items { | ||||
| 		rs = append(rs, toObjectACLRule(item)) | ||||
| 	} | ||||
| 	return rs | ||||
| } | ||||
|  | ||||
| func toBucketACLRules(items []*raw.BucketAccessControl) []ACLRule { | ||||
| 	var rs []ACLRule | ||||
| 	for _, item := range items { | ||||
| 		rs = append(rs, toBucketACLRule(item)) | ||||
| 	} | ||||
| 	return rs | ||||
| } | ||||
|  | ||||
| func toObjectACLRule(a *raw.ObjectAccessControl) ACLRule { | ||||
| 	return ACLRule{ | ||||
| 		Entity:      ACLEntity(a.Entity), | ||||
| 		EntityID:    a.EntityId, | ||||
| 		Role:        ACLRole(a.Role), | ||||
| 		Domain:      a.Domain, | ||||
| 		Email:       a.Email, | ||||
| 		ProjectTeam: toObjectProjectTeam(a.ProjectTeam), | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func toBucketACLRule(a *raw.BucketAccessControl) ACLRule { | ||||
| 	return ACLRule{ | ||||
| 		Entity:      ACLEntity(a.Entity), | ||||
| 		EntityID:    a.EntityId, | ||||
| 		Role:        ACLRole(a.Role), | ||||
| 		Domain:      a.Domain, | ||||
| 		Email:       a.Email, | ||||
| 		ProjectTeam: toBucketProjectTeam(a.ProjectTeam), | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func toRawObjectACL(rules []ACLRule) []*raw.ObjectAccessControl { | ||||
| 	if len(rules) == 0 { | ||||
| 		return nil | ||||
| 	} | ||||
| 	r := make([]*raw.ObjectAccessControl, 0, len(rules)) | ||||
| 	for _, rule := range rules { | ||||
| 		r = append(r, rule.toRawObjectAccessControl("")) // bucket name unnecessary | ||||
| 	} | ||||
| 	return r | ||||
| } | ||||
|  | ||||
| func toRawBucketACL(rules []ACLRule) []*raw.BucketAccessControl { | ||||
| 	if len(rules) == 0 { | ||||
| 		return nil | ||||
| 	} | ||||
| 	r := make([]*raw.BucketAccessControl, 0, len(rules)) | ||||
| 	for _, rule := range rules { | ||||
| 		r = append(r, rule.toRawBucketAccessControl("")) // bucket name unnecessary | ||||
| 	} | ||||
| 	return r | ||||
| } | ||||
|  | ||||
| func (r ACLRule) toRawBucketAccessControl(bucket string) *raw.BucketAccessControl { | ||||
| 	return &raw.BucketAccessControl{ | ||||
| 		Bucket: bucket, | ||||
| 		Entity: string(r.Entity), | ||||
| 		Role:   string(r.Role), | ||||
| 		// The other fields are not settable. | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (r ACLRule) toRawObjectAccessControl(bucket string) *raw.ObjectAccessControl { | ||||
| 	return &raw.ObjectAccessControl{ | ||||
| 		Bucket: bucket, | ||||
| 		Entity: string(r.Entity), | ||||
| 		Role:   string(r.Role), | ||||
| 		// The other fields are not settable. | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func toBucketProjectTeam(p *raw.BucketAccessControlProjectTeam) *ProjectTeam { | ||||
| 	if p == nil { | ||||
| 		return nil | ||||
| 	} | ||||
| 	return &ProjectTeam{ | ||||
| 		ProjectNumber: p.ProjectNumber, | ||||
| 		Team:          p.Team, | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func toObjectProjectTeam(p *raw.ObjectAccessControlProjectTeam) *ProjectTeam { | ||||
| 	if p == nil { | ||||
| 		return nil | ||||
| 	} | ||||
| 	return &ProjectTeam{ | ||||
| 		ProjectNumber: p.ProjectNumber, | ||||
| 		Team:          p.Team, | ||||
| 	} | ||||
| } | ||||
							
								
								
									
										1195
									
								
								vendor/cloud.google.com/go/storage/bucket.go
									
									
									
										generated
									
									
										vendored
									
									
								
							
							
						
						
									
										1195
									
								
								vendor/cloud.google.com/go/storage/bucket.go
									
									
									
										generated
									
									
										vendored
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										228
									
								
								vendor/cloud.google.com/go/storage/copy.go
									
									
									
										generated
									
									
										vendored
									
									
								
							
							
						
						
									
										228
									
								
								vendor/cloud.google.com/go/storage/copy.go
									
									
									
										generated
									
									
										vendored
									
									
								
							| @@ -1,228 +0,0 @@ | ||||
| // Copyright 2016 Google LLC | ||||
| // | ||||
| // Licensed under the Apache License, Version 2.0 (the "License"); | ||||
| // you may not use this file except in compliance with the License. | ||||
| // You may obtain a copy of the License at | ||||
| // | ||||
| //      http://www.apache.org/licenses/LICENSE-2.0 | ||||
| // | ||||
| // Unless required by applicable law or agreed to in writing, software | ||||
| // distributed under the License is distributed on an "AS IS" BASIS, | ||||
| // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||||
| // See the License for the specific language governing permissions and | ||||
| // limitations under the License. | ||||
|  | ||||
| package storage | ||||
|  | ||||
| import ( | ||||
| 	"context" | ||||
| 	"errors" | ||||
| 	"fmt" | ||||
|  | ||||
| 	"cloud.google.com/go/internal/trace" | ||||
| 	raw "google.golang.org/api/storage/v1" | ||||
| ) | ||||
|  | ||||
| // CopierFrom creates a Copier that can copy src to dst. | ||||
| // You can immediately call Run on the returned Copier, or | ||||
| // you can configure it first. | ||||
| // | ||||
| // For Requester Pays buckets, the user project of dst is billed, unless it is empty, | ||||
| // in which case the user project of src is billed. | ||||
| func (dst *ObjectHandle) CopierFrom(src *ObjectHandle) *Copier { | ||||
| 	return &Copier{dst: dst, src: src} | ||||
| } | ||||
|  | ||||
| // A Copier copies a source object to a destination. | ||||
| type Copier struct { | ||||
| 	// ObjectAttrs are optional attributes to set on the destination object. | ||||
| 	// Any attributes must be initialized before any calls on the Copier. Nil | ||||
| 	// or zero-valued attributes are ignored. | ||||
| 	ObjectAttrs | ||||
|  | ||||
| 	// RewriteToken can be set before calling Run to resume a copy | ||||
| 	// operation. After Run returns a non-nil error, RewriteToken will | ||||
| 	// have been updated to contain the value needed to resume the copy. | ||||
| 	RewriteToken string | ||||
|  | ||||
| 	// ProgressFunc can be used to monitor the progress of a multi-RPC copy | ||||
| 	// operation. If ProgressFunc is not nil and copying requires multiple | ||||
| 	// calls to the underlying service (see | ||||
| 	// https://cloud.google.com/storage/docs/json_api/v1/objects/rewrite), then | ||||
| 	// ProgressFunc will be invoked after each call with the number of bytes of | ||||
| 	// content copied so far and the total size in bytes of the source object. | ||||
| 	// | ||||
| 	// ProgressFunc is intended to make upload progress available to the | ||||
| 	// application. For example, the implementation of ProgressFunc may update | ||||
| 	// a progress bar in the application's UI, or log the result of | ||||
| 	// float64(copiedBytes)/float64(totalBytes). | ||||
| 	// | ||||
| 	// ProgressFunc should return quickly without blocking. | ||||
| 	ProgressFunc func(copiedBytes, totalBytes uint64) | ||||
|  | ||||
| 	// The Cloud KMS key, in the form projects/P/locations/L/keyRings/R/cryptoKeys/K, | ||||
| 	// that will be used to encrypt the object. Overrides the object's KMSKeyName, if | ||||
| 	// any. | ||||
| 	// | ||||
| 	// Providing both a DestinationKMSKeyName and a customer-supplied encryption key | ||||
| 	// (via ObjectHandle.Key) on the destination object will result in an error when | ||||
| 	// Run is called. | ||||
| 	DestinationKMSKeyName string | ||||
|  | ||||
| 	dst, src *ObjectHandle | ||||
| } | ||||
|  | ||||
| // Run performs the copy. | ||||
| func (c *Copier) Run(ctx context.Context) (attrs *ObjectAttrs, err error) { | ||||
| 	ctx = trace.StartSpan(ctx, "cloud.google.com/go/storage.Copier.Run") | ||||
| 	defer func() { trace.EndSpan(ctx, err) }() | ||||
|  | ||||
| 	if err := c.src.validate(); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	if err := c.dst.validate(); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	if c.DestinationKMSKeyName != "" && c.dst.encryptionKey != nil { | ||||
| 		return nil, errors.New("storage: cannot use DestinationKMSKeyName with a customer-supplied encryption key") | ||||
| 	} | ||||
| 	// Convert destination attributes to raw form, omitting the bucket. | ||||
| 	// If the bucket is included but name or content-type aren't, the service | ||||
| 	// returns a 400 with "Required" as the only message. Omitting the bucket | ||||
| 	// does not cause any problems. | ||||
| 	rawObject := c.ObjectAttrs.toRawObject("") | ||||
| 	for { | ||||
| 		res, err := c.callRewrite(ctx, rawObject) | ||||
| 		if err != nil { | ||||
| 			return nil, err | ||||
| 		} | ||||
| 		if c.ProgressFunc != nil { | ||||
| 			c.ProgressFunc(uint64(res.TotalBytesRewritten), uint64(res.ObjectSize)) | ||||
| 		} | ||||
| 		if res.Done { // Finished successfully. | ||||
| 			return newObject(res.Resource), nil | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (c *Copier) callRewrite(ctx context.Context, rawObj *raw.Object) (*raw.RewriteResponse, error) { | ||||
| 	call := c.dst.c.raw.Objects.Rewrite(c.src.bucket, c.src.object, c.dst.bucket, c.dst.object, rawObj) | ||||
|  | ||||
| 	call.Context(ctx).Projection("full") | ||||
| 	if c.RewriteToken != "" { | ||||
| 		call.RewriteToken(c.RewriteToken) | ||||
| 	} | ||||
| 	if c.DestinationKMSKeyName != "" { | ||||
| 		call.DestinationKmsKeyName(c.DestinationKMSKeyName) | ||||
| 	} | ||||
| 	if c.PredefinedACL != "" { | ||||
| 		call.DestinationPredefinedAcl(c.PredefinedACL) | ||||
| 	} | ||||
| 	if err := applyConds("Copy destination", c.dst.gen, c.dst.conds, call); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	if c.dst.userProject != "" { | ||||
| 		call.UserProject(c.dst.userProject) | ||||
| 	} else if c.src.userProject != "" { | ||||
| 		call.UserProject(c.src.userProject) | ||||
| 	} | ||||
| 	if err := applySourceConds(c.src.gen, c.src.conds, call); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	if err := setEncryptionHeaders(call.Header(), c.dst.encryptionKey, false); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	if err := setEncryptionHeaders(call.Header(), c.src.encryptionKey, true); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	var res *raw.RewriteResponse | ||||
| 	var err error | ||||
| 	setClientHeader(call.Header()) | ||||
| 	err = runWithRetry(ctx, func() error { res, err = call.Do(); return err }) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	c.RewriteToken = res.RewriteToken | ||||
| 	return res, nil | ||||
| } | ||||
|  | ||||
| // ComposerFrom creates a Composer that can compose srcs into dst. | ||||
| // You can immediately call Run on the returned Composer, or you can | ||||
| // configure it first. | ||||
| // | ||||
| // The encryption key for the destination object will be used to decrypt all | ||||
| // source objects and encrypt the destination object. It is an error | ||||
| // to specify an encryption key for any of the source objects. | ||||
| func (dst *ObjectHandle) ComposerFrom(srcs ...*ObjectHandle) *Composer { | ||||
| 	return &Composer{dst: dst, srcs: srcs} | ||||
| } | ||||
|  | ||||
| // A Composer composes source objects into a destination object. | ||||
| // | ||||
| // For Requester Pays buckets, the user project of dst is billed. | ||||
| type Composer struct { | ||||
| 	// ObjectAttrs are optional attributes to set on the destination object. | ||||
| 	// Any attributes must be initialized before any calls on the Composer. Nil | ||||
| 	// or zero-valued attributes are ignored. | ||||
| 	ObjectAttrs | ||||
|  | ||||
| 	dst  *ObjectHandle | ||||
| 	srcs []*ObjectHandle | ||||
| } | ||||
|  | ||||
| // Run performs the compose operation. | ||||
| func (c *Composer) Run(ctx context.Context) (attrs *ObjectAttrs, err error) { | ||||
| 	ctx = trace.StartSpan(ctx, "cloud.google.com/go/storage.Composer.Run") | ||||
| 	defer func() { trace.EndSpan(ctx, err) }() | ||||
|  | ||||
| 	if err := c.dst.validate(); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	if len(c.srcs) == 0 { | ||||
| 		return nil, errors.New("storage: at least one source object must be specified") | ||||
| 	} | ||||
|  | ||||
| 	req := &raw.ComposeRequest{} | ||||
| 	// Compose requires a non-empty Destination, so we always set it, | ||||
| 	// even if the caller-provided ObjectAttrs is the zero value. | ||||
| 	req.Destination = c.ObjectAttrs.toRawObject(c.dst.bucket) | ||||
| 	for _, src := range c.srcs { | ||||
| 		if err := src.validate(); err != nil { | ||||
| 			return nil, err | ||||
| 		} | ||||
| 		if src.bucket != c.dst.bucket { | ||||
| 			return nil, fmt.Errorf("storage: all source objects must be in bucket %q, found %q", c.dst.bucket, src.bucket) | ||||
| 		} | ||||
| 		if src.encryptionKey != nil { | ||||
| 			return nil, fmt.Errorf("storage: compose source %s.%s must not have encryption key", src.bucket, src.object) | ||||
| 		} | ||||
| 		srcObj := &raw.ComposeRequestSourceObjects{ | ||||
| 			Name: src.object, | ||||
| 		} | ||||
| 		if err := applyConds("ComposeFrom source", src.gen, src.conds, composeSourceObj{srcObj}); err != nil { | ||||
| 			return nil, err | ||||
| 		} | ||||
| 		req.SourceObjects = append(req.SourceObjects, srcObj) | ||||
| 	} | ||||
|  | ||||
| 	call := c.dst.c.raw.Objects.Compose(c.dst.bucket, c.dst.object, req).Context(ctx) | ||||
| 	if err := applyConds("ComposeFrom destination", c.dst.gen, c.dst.conds, call); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	if c.dst.userProject != "" { | ||||
| 		call.UserProject(c.dst.userProject) | ||||
| 	} | ||||
| 	if c.PredefinedACL != "" { | ||||
| 		call.DestinationPredefinedAcl(c.PredefinedACL) | ||||
| 	} | ||||
| 	if err := setEncryptionHeaders(call.Header(), c.dst.encryptionKey, false); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	var obj *raw.Object | ||||
| 	setClientHeader(call.Header()) | ||||
| 	err = runWithRetry(ctx, func() error { obj, err = call.Do(); return err }) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	return newObject(obj), nil | ||||
| } | ||||
							
								
								
									
										176
									
								
								vendor/cloud.google.com/go/storage/doc.go
									
									
									
										generated
									
									
										vendored
									
									
								
							
							
						
						
									
										176
									
								
								vendor/cloud.google.com/go/storage/doc.go
									
									
									
										generated
									
									
										vendored
									
									
								
							| @@ -1,176 +0,0 @@ | ||||
| // Copyright 2016 Google LLC | ||||
| // | ||||
| // Licensed under the Apache License, Version 2.0 (the "License"); | ||||
| // you may not use this file except in compliance with the License. | ||||
| // You may obtain a copy of the License at | ||||
| // | ||||
| //      http://www.apache.org/licenses/LICENSE-2.0 | ||||
| // | ||||
| // Unless required by applicable law or agreed to in writing, software | ||||
| // distributed under the License is distributed on an "AS IS" BASIS, | ||||
| // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||||
| // See the License for the specific language governing permissions and | ||||
| // limitations under the License. | ||||
|  | ||||
| /* | ||||
| Package storage provides an easy way to work with Google Cloud Storage. | ||||
| Google Cloud Storage stores data in named objects, which are grouped into buckets. | ||||
|  | ||||
| More information about Google Cloud Storage is available at | ||||
| https://cloud.google.com/storage/docs. | ||||
|  | ||||
| See https://godoc.org/cloud.google.com/go for authentication, timeouts, | ||||
| connection pooling and similar aspects of this package. | ||||
|  | ||||
| All of the methods of this package use exponential backoff to retry calls that fail | ||||
| with certain errors, as described in | ||||
| https://cloud.google.com/storage/docs/exponential-backoff. Retrying continues | ||||
| indefinitely unless the controlling context is canceled or the client is closed. See | ||||
| context.WithTimeout and context.WithCancel. | ||||
|  | ||||
|  | ||||
| Creating a Client | ||||
|  | ||||
| To start working with this package, create a client: | ||||
|  | ||||
|     ctx := context.Background() | ||||
|     client, err := storage.NewClient(ctx) | ||||
|     if err != nil { | ||||
|         // TODO: Handle error. | ||||
|     } | ||||
|  | ||||
| The client will use your default application credentials. | ||||
|  | ||||
| If you only wish to access public data, you can create | ||||
| an unauthenticated client with | ||||
|  | ||||
|     client, err := storage.NewClient(ctx, option.WithoutAuthentication()) | ||||
|  | ||||
| Buckets | ||||
|  | ||||
| A Google Cloud Storage bucket is a collection of objects. To work with a | ||||
| bucket, make a bucket handle: | ||||
|  | ||||
|     bkt := client.Bucket(bucketName) | ||||
|  | ||||
| A handle is a reference to a bucket. You can have a handle even if the | ||||
| bucket doesn't exist yet. To create a bucket in Google Cloud Storage, | ||||
| call Create on the handle: | ||||
|  | ||||
|     if err := bkt.Create(ctx, projectID, nil); err != nil { | ||||
|         // TODO: Handle error. | ||||
|     } | ||||
|  | ||||
| Note that although buckets are associated with projects, bucket names are | ||||
| global across all projects. | ||||
|  | ||||
| Each bucket has associated metadata, represented in this package by | ||||
| BucketAttrs. The third argument to BucketHandle.Create allows you to set | ||||
| the initial BucketAttrs of a bucket. To retrieve a bucket's attributes, use | ||||
| Attrs: | ||||
|  | ||||
|     attrs, err := bkt.Attrs(ctx) | ||||
|     if err != nil { | ||||
|         // TODO: Handle error. | ||||
|     } | ||||
|     fmt.Printf("bucket %s, created at %s, is located in %s with storage class %s\n", | ||||
|         attrs.Name, attrs.Created, attrs.Location, attrs.StorageClass) | ||||
|  | ||||
| Objects | ||||
|  | ||||
| An object holds arbitrary data as a sequence of bytes, like a file. You | ||||
| refer to objects using a handle, just as with buckets, but unlike buckets | ||||
| you don't explicitly create an object. Instead, the first time you write | ||||
| to an object it will be created. You can use the standard Go io.Reader | ||||
| and io.Writer interfaces to read and write object data: | ||||
|  | ||||
|     obj := bkt.Object("data") | ||||
|     // Write something to obj. | ||||
|     // w implements io.Writer. | ||||
|     w := obj.NewWriter(ctx) | ||||
|     // Write some text to obj. This will either create the object or overwrite whatever is there already. | ||||
|     if _, err := fmt.Fprintf(w, "This object contains text.\n"); err != nil { | ||||
|         // TODO: Handle error. | ||||
|     } | ||||
|     // Close, just like writing a file. | ||||
|     if err := w.Close(); err != nil { | ||||
|         // TODO: Handle error. | ||||
|     } | ||||
|  | ||||
|     // Read it back. | ||||
|     r, err := obj.NewReader(ctx) | ||||
|     if err != nil { | ||||
|         // TODO: Handle error. | ||||
|     } | ||||
|     defer r.Close() | ||||
|     if _, err := io.Copy(os.Stdout, r); err != nil { | ||||
|         // TODO: Handle error. | ||||
|     } | ||||
|     // Prints "This object contains text." | ||||
|  | ||||
| Objects also have attributes, which you can fetch with Attrs: | ||||
|  | ||||
|     objAttrs, err := obj.Attrs(ctx) | ||||
|     if err != nil { | ||||
|         // TODO: Handle error. | ||||
|     } | ||||
|     fmt.Printf("object %s has size %d and can be read using %s\n", | ||||
|         objAttrs.Name, objAttrs.Size, objAttrs.MediaLink) | ||||
|  | ||||
| ACLs | ||||
|  | ||||
| Both objects and buckets have ACLs (Access Control Lists). An ACL is a list of | ||||
| ACLRules, each of which specifies the role of a user, group or project. ACLs | ||||
| are suitable for fine-grained control, but you may prefer using IAM to control | ||||
| access at the project level (see | ||||
| https://cloud.google.com/storage/docs/access-control/iam). | ||||
|  | ||||
| To list the ACLs of a bucket or object, obtain an ACLHandle and call its List method: | ||||
|  | ||||
|     acls, err := obj.ACL().List(ctx) | ||||
|     if err != nil { | ||||
|         // TODO: Handle error. | ||||
|     } | ||||
|     for _, rule := range acls { | ||||
|         fmt.Printf("%s has role %s\n", rule.Entity, rule.Role) | ||||
|     } | ||||
|  | ||||
| You can also set and delete ACLs. | ||||
|  | ||||
| Conditions | ||||
|  | ||||
| Every object has a generation and a metageneration. The generation changes | ||||
| whenever the content changes, and the metageneration changes whenever the | ||||
| metadata changes. Conditions let you check these values before an operation; | ||||
| the operation only executes if the conditions match. You can use conditions to | ||||
| prevent race conditions in read-modify-write operations. | ||||
|  | ||||
| For example, say you've read an object's metadata into objAttrs. Now | ||||
| you want to write to that object, but only if its contents haven't changed | ||||
| since you read it. Here is how to express that: | ||||
|  | ||||
|     w = obj.If(storage.Conditions{GenerationMatch: objAttrs.Generation}).NewWriter(ctx) | ||||
|     // Proceed with writing as above. | ||||
|  | ||||
| Signed URLs | ||||
|  | ||||
| You can obtain a URL that lets anyone read or write an object for a limited time. | ||||
| You don't need to create a client to do this. See the documentation of | ||||
| SignedURL for details. | ||||
|  | ||||
|     url, err := storage.SignedURL(bucketName, "shared-object", opts) | ||||
|     if err != nil { | ||||
|         // TODO: Handle error. | ||||
|     } | ||||
|     fmt.Println(url) | ||||
|  | ||||
| Errors | ||||
|  | ||||
| Errors returned by this client are often of the type [`googleapi.Error`](https://godoc.org/google.golang.org/api/googleapi#Error). | ||||
| These errors can be introspected for more information by type asserting to the richer `googleapi.Error` type. For example: | ||||
|  | ||||
| 	if e, ok := err.(*googleapi.Error); ok { | ||||
| 		  if e.Code == 409 { ... } | ||||
| 	} | ||||
| */ | ||||
| package storage // import "cloud.google.com/go/storage" | ||||
							
								
								
									
										32
									
								
								vendor/cloud.google.com/go/storage/go110.go
									
									
									
										generated
									
									
										vendored
									
									
								
							
							
						
						
									
										32
									
								
								vendor/cloud.google.com/go/storage/go110.go
									
									
									
										generated
									
									
										vendored
									
									
								
							| @@ -1,32 +0,0 @@ | ||||
| // Copyright 2017 Google LLC | ||||
| // | ||||
| // Licensed under the Apache License, Version 2.0 (the "License"); | ||||
| // you may not use this file except in compliance with the License. | ||||
| // You may obtain a copy of the License at | ||||
| // | ||||
| //      http://www.apache.org/licenses/LICENSE-2.0 | ||||
| // | ||||
| // Unless required by applicable law or agreed to in writing, software | ||||
| // distributed under the License is distributed on an "AS IS" BASIS, | ||||
| // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||||
| // See the License for the specific language governing permissions and | ||||
| // limitations under the License. | ||||
|  | ||||
| // +build go1.10 | ||||
|  | ||||
| package storage | ||||
|  | ||||
| import "google.golang.org/api/googleapi" | ||||
|  | ||||
| func shouldRetry(err error) bool { | ||||
| 	switch e := err.(type) { | ||||
| 	case *googleapi.Error: | ||||
| 		// Retry on 429 and 5xx, according to | ||||
| 		// https://cloud.google.com/storage/docs/exponential-backoff. | ||||
| 		return e.Code == 429 || (e.Code >= 500 && e.Code < 600) | ||||
| 	case interface{ Temporary() bool }: | ||||
| 		return e.Temporary() | ||||
| 	default: | ||||
| 		return false | ||||
| 	} | ||||
| } | ||||
							
								
								
									
										330
									
								
								vendor/cloud.google.com/go/storage/hmac.go
									
									
									
										generated
									
									
										vendored
									
									
								
							
							
						
						
									
										330
									
								
								vendor/cloud.google.com/go/storage/hmac.go
									
									
									
										generated
									
									
										vendored
									
									
								
							| @@ -1,330 +0,0 @@ | ||||
| // Copyright 2019 Google LLC | ||||
| // | ||||
| // Licensed under the Apache License, Version 2.0 (the "License"); | ||||
| // you may not use this file except in compliance with the License. | ||||
| // You may obtain a copy of the License at | ||||
| // | ||||
| //      http://www.apache.org/licenses/LICENSE-2.0 | ||||
| // | ||||
| // Unless required by applicable law or agreed to in writing, software | ||||
| // distributed under the License is distributed on an "AS IS" BASIS, | ||||
| // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||||
| // See the License for the specific language governing permissions and | ||||
| // limitations under the License. | ||||
|  | ||||
| package storage | ||||
|  | ||||
| import ( | ||||
| 	"context" | ||||
| 	"errors" | ||||
| 	"fmt" | ||||
| 	"time" | ||||
|  | ||||
| 	"google.golang.org/api/iterator" | ||||
| 	raw "google.golang.org/api/storage/v1" | ||||
| ) | ||||
|  | ||||
| // HMACState is the state of the HMAC key. | ||||
| type HMACState string | ||||
|  | ||||
| const ( | ||||
| 	// Active is the status for an active key that can be used to sign | ||||
| 	// requests. | ||||
| 	Active HMACState = "ACTIVE" | ||||
|  | ||||
| 	// Inactive is the status for an inactive key thus requests signed by | ||||
| 	// this key will be denied. | ||||
| 	Inactive HMACState = "INACTIVE" | ||||
|  | ||||
| 	// Deleted is the status for a key that is deleted. | ||||
| 	// Once in this state the key cannot key cannot be recovered | ||||
| 	// and does not count towards key limits. Deleted keys will be cleaned | ||||
| 	// up later. | ||||
| 	Deleted HMACState = "DELETED" | ||||
| ) | ||||
|  | ||||
| // HMACKey is the representation of a Google Cloud Storage HMAC key. | ||||
| // | ||||
| // HMAC keys are used to authenticate signed access to objects. To enable HMAC key | ||||
| // authentication, please visit https://cloud.google.com/storage/docs/migrating. | ||||
| // | ||||
| // This type is EXPERIMENTAL and subject to change or removal without notice. | ||||
| type HMACKey struct { | ||||
| 	// The HMAC's secret key. | ||||
| 	Secret string | ||||
|  | ||||
| 	// AccessID is the ID of the HMAC key. | ||||
| 	AccessID string | ||||
|  | ||||
| 	// Etag is the HTTP/1.1 Entity tag. | ||||
| 	Etag string | ||||
|  | ||||
| 	// ID is the ID of the HMAC key, including the ProjectID and AccessID. | ||||
| 	ID string | ||||
|  | ||||
| 	// ProjectID is the ID of the project that owns the | ||||
| 	// service account to which the key authenticates. | ||||
| 	ProjectID string | ||||
|  | ||||
| 	// ServiceAccountEmail is the email address | ||||
| 	// of the key's associated service account. | ||||
| 	ServiceAccountEmail string | ||||
|  | ||||
| 	// CreatedTime is the creation time of the HMAC key. | ||||
| 	CreatedTime time.Time | ||||
|  | ||||
| 	// UpdatedTime is the last modification time of the HMAC key metadata. | ||||
| 	UpdatedTime time.Time | ||||
|  | ||||
| 	// State is the state of the HMAC key. | ||||
| 	// It can be one of StateActive, StateInactive or StateDeleted. | ||||
| 	State HMACState | ||||
| } | ||||
|  | ||||
| // HMACKeyHandle helps provide access and management for HMAC keys. | ||||
| // | ||||
| // This type is EXPERIMENTAL and subject to change or removal without notice. | ||||
| type HMACKeyHandle struct { | ||||
| 	projectID string | ||||
| 	accessID  string | ||||
|  | ||||
| 	raw *raw.ProjectsHmacKeysService | ||||
| } | ||||
|  | ||||
| // HMACKeyHandle creates a handle that will be used for HMACKey operations. | ||||
| // | ||||
| // This method is EXPERIMENTAL and subject to change or removal without notice. | ||||
| func (c *Client) HMACKeyHandle(projectID, accessID string) *HMACKeyHandle { | ||||
| 	return &HMACKeyHandle{ | ||||
| 		projectID: projectID, | ||||
| 		accessID:  accessID, | ||||
| 		raw:       raw.NewProjectsHmacKeysService(c.raw), | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // Get invokes an RPC to retrieve the HMAC key referenced by the | ||||
| // HMACKeyHandle's accessID. | ||||
| // | ||||
| // This method is EXPERIMENTAL and subject to change or removal without notice. | ||||
| func (hkh *HMACKeyHandle) Get(ctx context.Context) (*HMACKey, error) { | ||||
| 	call := hkh.raw.Get(hkh.projectID, hkh.accessID) | ||||
| 	setClientHeader(call.Header()) | ||||
|  | ||||
| 	var metadata *raw.HmacKeyMetadata | ||||
| 	var err error | ||||
| 	err = runWithRetry(ctx, func() error { | ||||
| 		metadata, err = call.Context(ctx).Do() | ||||
| 		return err | ||||
| 	}) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
|  | ||||
| 	hkPb := &raw.HmacKey{ | ||||
| 		Metadata: metadata, | ||||
| 	} | ||||
| 	return pbHmacKeyToHMACKey(hkPb, false) | ||||
| } | ||||
|  | ||||
| // Delete invokes an RPC to delete the key referenced by accessID, on Google Cloud Storage. | ||||
| // Only inactive HMAC keys can be deleted. | ||||
| // After deletion, a key cannot be used to authenticate requests. | ||||
| // | ||||
| // This method is EXPERIMENTAL and subject to change or removal without notice. | ||||
| func (hkh *HMACKeyHandle) Delete(ctx context.Context) error { | ||||
| 	delCall := hkh.raw.Delete(hkh.projectID, hkh.accessID) | ||||
| 	setClientHeader(delCall.Header()) | ||||
|  | ||||
| 	return runWithRetry(ctx, func() error { | ||||
| 		return delCall.Context(ctx).Do() | ||||
| 	}) | ||||
| } | ||||
|  | ||||
| func pbHmacKeyToHMACKey(pb *raw.HmacKey, updatedTimeCanBeNil bool) (*HMACKey, error) { | ||||
| 	pbmd := pb.Metadata | ||||
| 	if pbmd == nil { | ||||
| 		return nil, errors.New("field Metadata cannot be nil") | ||||
| 	} | ||||
| 	createdTime, err := time.Parse(time.RFC3339, pbmd.TimeCreated) | ||||
| 	if err != nil { | ||||
| 		return nil, fmt.Errorf("field CreatedTime: %v", err) | ||||
| 	} | ||||
| 	updatedTime, err := time.Parse(time.RFC3339, pbmd.Updated) | ||||
| 	if err != nil && !updatedTimeCanBeNil { | ||||
| 		return nil, fmt.Errorf("field UpdatedTime: %v", err) | ||||
| 	} | ||||
|  | ||||
| 	hmk := &HMACKey{ | ||||
| 		AccessID:    pbmd.AccessId, | ||||
| 		Secret:      pb.Secret, | ||||
| 		Etag:        pbmd.Etag, | ||||
| 		ID:          pbmd.Id, | ||||
| 		State:       HMACState(pbmd.State), | ||||
| 		ProjectID:   pbmd.ProjectId, | ||||
| 		CreatedTime: createdTime, | ||||
| 		UpdatedTime: updatedTime, | ||||
|  | ||||
| 		ServiceAccountEmail: pbmd.ServiceAccountEmail, | ||||
| 	} | ||||
|  | ||||
| 	return hmk, nil | ||||
| } | ||||
|  | ||||
| // CreateHMACKey invokes an RPC for Google Cloud Storage to create a new HMACKey. | ||||
| // | ||||
| // This method is EXPERIMENTAL and subject to change or removal without notice. | ||||
| func (c *Client) CreateHMACKey(ctx context.Context, projectID, serviceAccountEmail string) (*HMACKey, error) { | ||||
| 	if projectID == "" { | ||||
| 		return nil, errors.New("storage: expecting a non-blank projectID") | ||||
| 	} | ||||
| 	if serviceAccountEmail == "" { | ||||
| 		return nil, errors.New("storage: expecting a non-blank service account email") | ||||
| 	} | ||||
|  | ||||
| 	svc := raw.NewProjectsHmacKeysService(c.raw) | ||||
| 	call := svc.Create(projectID, serviceAccountEmail) | ||||
| 	setClientHeader(call.Header()) | ||||
|  | ||||
| 	var hkPb *raw.HmacKey | ||||
| 	var err error | ||||
| 	err = runWithRetry(ctx, func() error { | ||||
| 		hkPb, err = call.Context(ctx).Do() | ||||
| 		return err | ||||
| 	}) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
|  | ||||
| 	return pbHmacKeyToHMACKey(hkPb, true) | ||||
| } | ||||
|  | ||||
| // HMACKeyAttrsToUpdate defines the attributes of an HMACKey that will be updated. | ||||
| // | ||||
| // This type is EXPERIMENTAL and subject to change or removal without notice. | ||||
| type HMACKeyAttrsToUpdate struct { | ||||
| 	// State is required and must be either StateActive or StateInactive. | ||||
| 	State HMACState | ||||
|  | ||||
| 	// Etag is an optional field and it is the HTTP/1.1 Entity tag. | ||||
| 	Etag string | ||||
| } | ||||
|  | ||||
| // Update mutates the HMACKey referred to by accessID. | ||||
| // | ||||
| // This method is EXPERIMENTAL and subject to change or removal without notice. | ||||
| func (h *HMACKeyHandle) Update(ctx context.Context, au HMACKeyAttrsToUpdate) (*HMACKey, error) { | ||||
| 	if au.State != Active && au.State != Inactive { | ||||
| 		return nil, fmt.Errorf("storage: invalid state %q for update, must be either %q or %q", au.State, Active, Inactive) | ||||
| 	} | ||||
|  | ||||
| 	call := h.raw.Update(h.projectID, h.accessID, &raw.HmacKeyMetadata{ | ||||
| 		Etag:  au.Etag, | ||||
| 		State: string(au.State), | ||||
| 	}) | ||||
| 	setClientHeader(call.Header()) | ||||
|  | ||||
| 	var metadata *raw.HmacKeyMetadata | ||||
| 	var err error | ||||
| 	err = runWithRetry(ctx, func() error { | ||||
| 		metadata, err = call.Context(ctx).Do() | ||||
| 		return err | ||||
| 	}) | ||||
|  | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	hkPb := &raw.HmacKey{ | ||||
| 		Metadata: metadata, | ||||
| 	} | ||||
| 	return pbHmacKeyToHMACKey(hkPb, false) | ||||
| } | ||||
|  | ||||
| // An HMACKeysIterator is an iterator over HMACKeys. | ||||
| // | ||||
| // This type is EXPERIMENTAL and subject to change or removal without notice. | ||||
| type HMACKeysIterator struct { | ||||
| 	ctx       context.Context | ||||
| 	raw       *raw.ProjectsHmacKeysService | ||||
| 	projectID string | ||||
| 	hmacKeys  []*HMACKey | ||||
| 	pageInfo  *iterator.PageInfo | ||||
| 	nextFunc  func() error | ||||
| 	index     int | ||||
| } | ||||
|  | ||||
| // ListHMACKeys returns an iterator for listing HMACKeys. | ||||
| // | ||||
| // This method is EXPERIMENTAL and subject to change or removal without notice. | ||||
| func (c *Client) ListHMACKeys(ctx context.Context, projectID string) *HMACKeysIterator { | ||||
| 	it := &HMACKeysIterator{ | ||||
| 		ctx:       ctx, | ||||
| 		raw:       raw.NewProjectsHmacKeysService(c.raw), | ||||
| 		projectID: projectID, | ||||
| 	} | ||||
|  | ||||
| 	it.pageInfo, it.nextFunc = iterator.NewPageInfo( | ||||
| 		it.fetch, | ||||
| 		func() int { return len(it.hmacKeys) - it.index }, | ||||
| 		func() interface{} { | ||||
| 			prev := it.hmacKeys | ||||
| 			it.hmacKeys = it.hmacKeys[:0] | ||||
| 			it.index = 0 | ||||
| 			return prev | ||||
| 		}) | ||||
| 	return it | ||||
| } | ||||
|  | ||||
| // Next returns the next result. Its second return value is iterator.Done if | ||||
| // there are no more results. Once Next returns iterator.Done, all subsequent | ||||
| // calls will return iterator.Done. | ||||
| // | ||||
| // This method is EXPERIMENTAL and subject to change or removal without notice. | ||||
| func (it *HMACKeysIterator) Next() (*HMACKey, error) { | ||||
| 	if err := it.nextFunc(); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
|  | ||||
| 	key := it.hmacKeys[it.index] | ||||
| 	it.index++ | ||||
|  | ||||
| 	return key, nil | ||||
| } | ||||
|  | ||||
| // PageInfo supports pagination. See the google.golang.org/api/iterator package for details. | ||||
| // | ||||
| // This method is EXPERIMENTAL and subject to change or removal without notice. | ||||
| func (it *HMACKeysIterator) PageInfo() *iterator.PageInfo { return it.pageInfo } | ||||
|  | ||||
| func (it *HMACKeysIterator) fetch(pageSize int, pageToken string) (token string, err error) { | ||||
| 	call := it.raw.List(it.projectID) | ||||
| 	setClientHeader(call.Header()) | ||||
| 	call = call.PageToken(pageToken) | ||||
| 	// By default we'll also show deleted keys and then | ||||
| 	// let users filter on their own. | ||||
| 	call = call.ShowDeletedKeys(true) | ||||
| 	if pageSize > 0 { | ||||
| 		call = call.MaxResults(int64(pageSize)) | ||||
| 	} | ||||
|  | ||||
| 	ctx := it.ctx | ||||
| 	var resp *raw.HmacKeysMetadata | ||||
| 	err = runWithRetry(it.ctx, func() error { | ||||
| 		resp, err = call.Context(ctx).Do() | ||||
| 		return err | ||||
| 	}) | ||||
| 	if err != nil { | ||||
| 		return "", err | ||||
| 	} | ||||
|  | ||||
| 	for _, metadata := range resp.Items { | ||||
| 		hkPb := &raw.HmacKey{ | ||||
| 			Metadata: metadata, | ||||
| 		} | ||||
| 		hkey, err := pbHmacKeyToHMACKey(hkPb, true) | ||||
| 		if err != nil { | ||||
| 			return "", err | ||||
| 		} | ||||
| 		it.hmacKeys = append(it.hmacKeys, hkey) | ||||
| 	} | ||||
| 	return resp.NextPageToken, nil | ||||
| } | ||||
							
								
								
									
										130
									
								
								vendor/cloud.google.com/go/storage/iam.go
									
									
									
										generated
									
									
										vendored
									
									
								
							
							
						
						
									
										130
									
								
								vendor/cloud.google.com/go/storage/iam.go
									
									
									
										generated
									
									
										vendored
									
									
								
							| @@ -1,130 +0,0 @@ | ||||
| // Copyright 2017 Google LLC | ||||
| // | ||||
| // Licensed under the Apache License, Version 2.0 (the "License"); | ||||
| // you may not use this file except in compliance with the License. | ||||
| // You may obtain a copy of the License at | ||||
| // | ||||
| //      http://www.apache.org/licenses/LICENSE-2.0 | ||||
| // | ||||
| // Unless required by applicable law or agreed to in writing, software | ||||
| // distributed under the License is distributed on an "AS IS" BASIS, | ||||
| // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||||
| // See the License for the specific language governing permissions and | ||||
| // limitations under the License. | ||||
|  | ||||
| package storage | ||||
|  | ||||
| import ( | ||||
| 	"context" | ||||
|  | ||||
| 	"cloud.google.com/go/iam" | ||||
| 	"cloud.google.com/go/internal/trace" | ||||
| 	raw "google.golang.org/api/storage/v1" | ||||
| 	iampb "google.golang.org/genproto/googleapis/iam/v1" | ||||
| ) | ||||
|  | ||||
| // IAM provides access to IAM access control for the bucket. | ||||
| func (b *BucketHandle) IAM() *iam.Handle { | ||||
| 	return iam.InternalNewHandleClient(&iamClient{ | ||||
| 		raw:         b.c.raw, | ||||
| 		userProject: b.userProject, | ||||
| 	}, b.name) | ||||
| } | ||||
|  | ||||
| // iamClient implements the iam.client interface. | ||||
| type iamClient struct { | ||||
| 	raw         *raw.Service | ||||
| 	userProject string | ||||
| } | ||||
|  | ||||
| func (c *iamClient) Get(ctx context.Context, resource string) (p *iampb.Policy, err error) { | ||||
| 	ctx = trace.StartSpan(ctx, "cloud.google.com/go/storage.IAM.Get") | ||||
| 	defer func() { trace.EndSpan(ctx, err) }() | ||||
|  | ||||
| 	call := c.raw.Buckets.GetIamPolicy(resource) | ||||
| 	setClientHeader(call.Header()) | ||||
| 	if c.userProject != "" { | ||||
| 		call.UserProject(c.userProject) | ||||
| 	} | ||||
| 	var rp *raw.Policy | ||||
| 	err = runWithRetry(ctx, func() error { | ||||
| 		rp, err = call.Context(ctx).Do() | ||||
| 		return err | ||||
| 	}) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	return iamFromStoragePolicy(rp), nil | ||||
| } | ||||
|  | ||||
| func (c *iamClient) Set(ctx context.Context, resource string, p *iampb.Policy) (err error) { | ||||
| 	ctx = trace.StartSpan(ctx, "cloud.google.com/go/storage.IAM.Set") | ||||
| 	defer func() { trace.EndSpan(ctx, err) }() | ||||
|  | ||||
| 	rp := iamToStoragePolicy(p) | ||||
| 	call := c.raw.Buckets.SetIamPolicy(resource, rp) | ||||
| 	setClientHeader(call.Header()) | ||||
| 	if c.userProject != "" { | ||||
| 		call.UserProject(c.userProject) | ||||
| 	} | ||||
| 	return runWithRetry(ctx, func() error { | ||||
| 		_, err := call.Context(ctx).Do() | ||||
| 		return err | ||||
| 	}) | ||||
| } | ||||
|  | ||||
| func (c *iamClient) Test(ctx context.Context, resource string, perms []string) (permissions []string, err error) { | ||||
| 	ctx = trace.StartSpan(ctx, "cloud.google.com/go/storage.IAM.Test") | ||||
| 	defer func() { trace.EndSpan(ctx, err) }() | ||||
|  | ||||
| 	call := c.raw.Buckets.TestIamPermissions(resource, perms) | ||||
| 	setClientHeader(call.Header()) | ||||
| 	if c.userProject != "" { | ||||
| 		call.UserProject(c.userProject) | ||||
| 	} | ||||
| 	var res *raw.TestIamPermissionsResponse | ||||
| 	err = runWithRetry(ctx, func() error { | ||||
| 		res, err = call.Context(ctx).Do() | ||||
| 		return err | ||||
| 	}) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	return res.Permissions, nil | ||||
| } | ||||
|  | ||||
| func iamToStoragePolicy(ip *iampb.Policy) *raw.Policy { | ||||
| 	return &raw.Policy{ | ||||
| 		Bindings: iamToStorageBindings(ip.Bindings), | ||||
| 		Etag:     string(ip.Etag), | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func iamToStorageBindings(ibs []*iampb.Binding) []*raw.PolicyBindings { | ||||
| 	var rbs []*raw.PolicyBindings | ||||
| 	for _, ib := range ibs { | ||||
| 		rbs = append(rbs, &raw.PolicyBindings{ | ||||
| 			Role:    ib.Role, | ||||
| 			Members: ib.Members, | ||||
| 		}) | ||||
| 	} | ||||
| 	return rbs | ||||
| } | ||||
|  | ||||
| func iamFromStoragePolicy(rp *raw.Policy) *iampb.Policy { | ||||
| 	return &iampb.Policy{ | ||||
| 		Bindings: iamFromStorageBindings(rp.Bindings), | ||||
| 		Etag:     []byte(rp.Etag), | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func iamFromStorageBindings(rbs []*raw.PolicyBindings) []*iampb.Binding { | ||||
| 	var ibs []*iampb.Binding | ||||
| 	for _, rb := range rbs { | ||||
| 		ibs = append(ibs, &iampb.Binding{ | ||||
| 			Role:    rb.Role, | ||||
| 			Members: rb.Members, | ||||
| 		}) | ||||
| 	} | ||||
| 	return ibs | ||||
| } | ||||
							
								
								
									
										37
									
								
								vendor/cloud.google.com/go/storage/invoke.go
									
									
									
										generated
									
									
										vendored
									
									
								
							
							
						
						
									
										37
									
								
								vendor/cloud.google.com/go/storage/invoke.go
									
									
									
										generated
									
									
										vendored
									
									
								
							| @@ -1,37 +0,0 @@ | ||||
| // Copyright 2014 Google LLC | ||||
| // | ||||
| // Licensed under the Apache License, Version 2.0 (the "License"); | ||||
| // you may not use this file except in compliance with the License. | ||||
| // You may obtain a copy of the License at | ||||
| // | ||||
| //      http://www.apache.org/licenses/LICENSE-2.0 | ||||
| // | ||||
| // Unless required by applicable law or agreed to in writing, software | ||||
| // distributed under the License is distributed on an "AS IS" BASIS, | ||||
| // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||||
| // See the License for the specific language governing permissions and | ||||
| // limitations under the License. | ||||
|  | ||||
| package storage | ||||
|  | ||||
| import ( | ||||
| 	"context" | ||||
|  | ||||
| 	"cloud.google.com/go/internal" | ||||
| 	gax "github.com/googleapis/gax-go/v2" | ||||
| ) | ||||
|  | ||||
| // runWithRetry calls the function until it returns nil or a non-retryable error, or | ||||
| // the context is done. | ||||
| func runWithRetry(ctx context.Context, call func() error) error { | ||||
| 	return internal.Retry(ctx, gax.Backoff{}, func() (stop bool, err error) { | ||||
| 		err = call() | ||||
| 		if err == nil { | ||||
| 			return true, nil | ||||
| 		} | ||||
| 		if shouldRetry(err) { | ||||
| 			return false, nil | ||||
| 		} | ||||
| 		return true, err | ||||
| 	}) | ||||
| } | ||||
							
								
								
									
										42
									
								
								vendor/cloud.google.com/go/storage/not_go110.go
									
									
									
										generated
									
									
										vendored
									
									
								
							
							
						
						
									
										42
									
								
								vendor/cloud.google.com/go/storage/not_go110.go
									
									
									
										generated
									
									
										vendored
									
									
								
							| @@ -1,42 +0,0 @@ | ||||
| // Copyright 2017 Google LLC | ||||
| // | ||||
| // Licensed under the Apache License, Version 2.0 (the "License"); | ||||
| // you may not use this file except in compliance with the License. | ||||
| // You may obtain a copy of the License at | ||||
| // | ||||
| //      http://www.apache.org/licenses/LICENSE-2.0 | ||||
| // | ||||
| // Unless required by applicable law or agreed to in writing, software | ||||
| // distributed under the License is distributed on an "AS IS" BASIS, | ||||
| // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||||
| // See the License for the specific language governing permissions and | ||||
| // limitations under the License. | ||||
|  | ||||
| // +build !go1.10 | ||||
|  | ||||
| package storage | ||||
|  | ||||
| import ( | ||||
| 	"net/url" | ||||
| 	"strings" | ||||
|  | ||||
| 	"google.golang.org/api/googleapi" | ||||
| ) | ||||
|  | ||||
| func shouldRetry(err error) bool { | ||||
| 	switch e := err.(type) { | ||||
| 	case *googleapi.Error: | ||||
| 		// Retry on 429 and 5xx, according to | ||||
| 		// https://cloud.google.com/storage/docs/exponential-backoff. | ||||
| 		return e.Code == 429 || (e.Code >= 500 && e.Code < 600) | ||||
| 	case *url.Error: | ||||
| 		// Retry on REFUSED_STREAM. | ||||
| 		// Unfortunately the error type is unexported, so we resort to string | ||||
| 		// matching. | ||||
| 		return strings.Contains(e.Error(), "REFUSED_STREAM") | ||||
| 	case interface{ Temporary() bool }: | ||||
| 		return e.Temporary() | ||||
| 	default: | ||||
| 		return false | ||||
| 	} | ||||
| } | ||||
							
								
								
									
										188
									
								
								vendor/cloud.google.com/go/storage/notifications.go
									
									
									
										generated
									
									
										vendored
									
									
								
							
							
						
						
									
										188
									
								
								vendor/cloud.google.com/go/storage/notifications.go
									
									
									
										generated
									
									
										vendored
									
									
								
							| @@ -1,188 +0,0 @@ | ||||
| // Copyright 2017 Google LLC | ||||
| // | ||||
| // Licensed under the Apache License, Version 2.0 (the "License"); | ||||
| // you may not use this file except in compliance with the License. | ||||
| // You may obtain a copy of the License at | ||||
| // | ||||
| //      http://www.apache.org/licenses/LICENSE-2.0 | ||||
| // | ||||
| // Unless required by applicable law or agreed to in writing, software | ||||
| // distributed under the License is distributed on an "AS IS" BASIS, | ||||
| // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||||
| // See the License for the specific language governing permissions and | ||||
| // limitations under the License. | ||||
|  | ||||
| package storage | ||||
|  | ||||
| import ( | ||||
| 	"context" | ||||
| 	"errors" | ||||
| 	"fmt" | ||||
| 	"regexp" | ||||
|  | ||||
| 	"cloud.google.com/go/internal/trace" | ||||
| 	raw "google.golang.org/api/storage/v1" | ||||
| ) | ||||
|  | ||||
| // A Notification describes how to send Cloud PubSub messages when certain | ||||
| // events occur in a bucket. | ||||
| type Notification struct { | ||||
| 	//The ID of the notification. | ||||
| 	ID string | ||||
|  | ||||
| 	// The ID of the topic to which this subscription publishes. | ||||
| 	TopicID string | ||||
|  | ||||
| 	// The ID of the project to which the topic belongs. | ||||
| 	TopicProjectID string | ||||
|  | ||||
| 	// Only send notifications about listed event types. If empty, send notifications | ||||
| 	// for all event types. | ||||
| 	// See https://cloud.google.com/storage/docs/pubsub-notifications#events. | ||||
| 	EventTypes []string | ||||
|  | ||||
| 	// If present, only apply this notification configuration to object names that | ||||
| 	// begin with this prefix. | ||||
| 	ObjectNamePrefix string | ||||
|  | ||||
| 	// An optional list of additional attributes to attach to each Cloud PubSub | ||||
| 	// message published for this notification subscription. | ||||
| 	CustomAttributes map[string]string | ||||
|  | ||||
| 	// The contents of the message payload. | ||||
| 	// See https://cloud.google.com/storage/docs/pubsub-notifications#payload. | ||||
| 	PayloadFormat string | ||||
| } | ||||
|  | ||||
| // Values for Notification.PayloadFormat. | ||||
| const ( | ||||
| 	// Send no payload with notification messages. | ||||
| 	NoPayload = "NONE" | ||||
|  | ||||
| 	// Send object metadata as JSON with notification messages. | ||||
| 	JSONPayload = "JSON_API_V1" | ||||
| ) | ||||
|  | ||||
| // Values for Notification.EventTypes. | ||||
| const ( | ||||
| 	// Event that occurs when an object is successfully created. | ||||
| 	ObjectFinalizeEvent = "OBJECT_FINALIZE" | ||||
|  | ||||
| 	// Event that occurs when the metadata of an existing object changes. | ||||
| 	ObjectMetadataUpdateEvent = "OBJECT_METADATA_UPDATE" | ||||
|  | ||||
| 	// Event that occurs when an object is permanently deleted. | ||||
| 	ObjectDeleteEvent = "OBJECT_DELETE" | ||||
|  | ||||
| 	// Event that occurs when the live version of an object becomes an | ||||
| 	// archived version. | ||||
| 	ObjectArchiveEvent = "OBJECT_ARCHIVE" | ||||
| ) | ||||
|  | ||||
| func toNotification(rn *raw.Notification) *Notification { | ||||
| 	n := &Notification{ | ||||
| 		ID:               rn.Id, | ||||
| 		EventTypes:       rn.EventTypes, | ||||
| 		ObjectNamePrefix: rn.ObjectNamePrefix, | ||||
| 		CustomAttributes: rn.CustomAttributes, | ||||
| 		PayloadFormat:    rn.PayloadFormat, | ||||
| 	} | ||||
| 	n.TopicProjectID, n.TopicID = parseNotificationTopic(rn.Topic) | ||||
| 	return n | ||||
| } | ||||
|  | ||||
| var topicRE = regexp.MustCompile("^//pubsub.googleapis.com/projects/([^/]+)/topics/([^/]+)") | ||||
|  | ||||
| // parseNotificationTopic extracts the project and topic IDs from from the full | ||||
| // resource name returned by the service. If the name is malformed, it returns | ||||
| // "?" for both IDs. | ||||
| func parseNotificationTopic(nt string) (projectID, topicID string) { | ||||
| 	matches := topicRE.FindStringSubmatch(nt) | ||||
| 	if matches == nil { | ||||
| 		return "?", "?" | ||||
| 	} | ||||
| 	return matches[1], matches[2] | ||||
| } | ||||
|  | ||||
| func toRawNotification(n *Notification) *raw.Notification { | ||||
| 	return &raw.Notification{ | ||||
| 		Id: n.ID, | ||||
| 		Topic: fmt.Sprintf("//pubsub.googleapis.com/projects/%s/topics/%s", | ||||
| 			n.TopicProjectID, n.TopicID), | ||||
| 		EventTypes:       n.EventTypes, | ||||
| 		ObjectNamePrefix: n.ObjectNamePrefix, | ||||
| 		CustomAttributes: n.CustomAttributes, | ||||
| 		PayloadFormat:    string(n.PayloadFormat), | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // AddNotification adds a notification to b. You must set n's TopicProjectID, TopicID | ||||
| // and PayloadFormat, and must not set its ID. The other fields are all optional. The | ||||
| // returned Notification's ID can be used to refer to it. | ||||
| func (b *BucketHandle) AddNotification(ctx context.Context, n *Notification) (ret *Notification, err error) { | ||||
| 	ctx = trace.StartSpan(ctx, "cloud.google.com/go/storage.Bucket.AddNotification") | ||||
| 	defer func() { trace.EndSpan(ctx, err) }() | ||||
|  | ||||
| 	if n.ID != "" { | ||||
| 		return nil, errors.New("storage: AddNotification: ID must not be set") | ||||
| 	} | ||||
| 	if n.TopicProjectID == "" { | ||||
| 		return nil, errors.New("storage: AddNotification: missing TopicProjectID") | ||||
| 	} | ||||
| 	if n.TopicID == "" { | ||||
| 		return nil, errors.New("storage: AddNotification: missing TopicID") | ||||
| 	} | ||||
| 	call := b.c.raw.Notifications.Insert(b.name, toRawNotification(n)) | ||||
| 	setClientHeader(call.Header()) | ||||
| 	if b.userProject != "" { | ||||
| 		call.UserProject(b.userProject) | ||||
| 	} | ||||
| 	rn, err := call.Context(ctx).Do() | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	return toNotification(rn), nil | ||||
| } | ||||
|  | ||||
| // Notifications returns all the Notifications configured for this bucket, as a map | ||||
| // indexed by notification ID. | ||||
| func (b *BucketHandle) Notifications(ctx context.Context) (n map[string]*Notification, err error) { | ||||
| 	ctx = trace.StartSpan(ctx, "cloud.google.com/go/storage.Bucket.Notifications") | ||||
| 	defer func() { trace.EndSpan(ctx, err) }() | ||||
|  | ||||
| 	call := b.c.raw.Notifications.List(b.name) | ||||
| 	setClientHeader(call.Header()) | ||||
| 	if b.userProject != "" { | ||||
| 		call.UserProject(b.userProject) | ||||
| 	} | ||||
| 	var res *raw.Notifications | ||||
| 	err = runWithRetry(ctx, func() error { | ||||
| 		res, err = call.Context(ctx).Do() | ||||
| 		return err | ||||
| 	}) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	return notificationsToMap(res.Items), nil | ||||
| } | ||||
|  | ||||
| func notificationsToMap(rns []*raw.Notification) map[string]*Notification { | ||||
| 	m := map[string]*Notification{} | ||||
| 	for _, rn := range rns { | ||||
| 		m[rn.Id] = toNotification(rn) | ||||
| 	} | ||||
| 	return m | ||||
| } | ||||
|  | ||||
| // DeleteNotification deletes the notification with the given ID. | ||||
| func (b *BucketHandle) DeleteNotification(ctx context.Context, id string) (err error) { | ||||
| 	ctx = trace.StartSpan(ctx, "cloud.google.com/go/storage.Bucket.DeleteNotification") | ||||
| 	defer func() { trace.EndSpan(ctx, err) }() | ||||
|  | ||||
| 	call := b.c.raw.Notifications.Delete(b.name, id) | ||||
| 	setClientHeader(call.Header()) | ||||
| 	if b.userProject != "" { | ||||
| 		call.UserProject(b.userProject) | ||||
| 	} | ||||
| 	return call.Context(ctx).Do() | ||||
| } | ||||
							
								
								
									
										403
									
								
								vendor/cloud.google.com/go/storage/reader.go
									
									
									
										generated
									
									
										vendored
									
									
								
							
							
						
						
									
										403
									
								
								vendor/cloud.google.com/go/storage/reader.go
									
									
									
										generated
									
									
										vendored
									
									
								
							| @@ -1,403 +0,0 @@ | ||||
| // Copyright 2016 Google LLC | ||||
| // | ||||
| // Licensed under the Apache License, Version 2.0 (the "License"); | ||||
| // you may not use this file except in compliance with the License. | ||||
| // You may obtain a copy of the License at | ||||
| // | ||||
| //      http://www.apache.org/licenses/LICENSE-2.0 | ||||
| // | ||||
| // Unless required by applicable law or agreed to in writing, software | ||||
| // distributed under the License is distributed on an "AS IS" BASIS, | ||||
| // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||||
| // See the License for the specific language governing permissions and | ||||
| // limitations under the License. | ||||
|  | ||||
| package storage | ||||
|  | ||||
| import ( | ||||
| 	"context" | ||||
| 	"errors" | ||||
| 	"fmt" | ||||
| 	"hash/crc32" | ||||
| 	"io" | ||||
| 	"io/ioutil" | ||||
| 	"net/http" | ||||
| 	"net/url" | ||||
| 	"reflect" | ||||
| 	"strconv" | ||||
| 	"strings" | ||||
| 	"time" | ||||
|  | ||||
| 	"cloud.google.com/go/internal/trace" | ||||
| 	"google.golang.org/api/googleapi" | ||||
| ) | ||||
|  | ||||
| var crc32cTable = crc32.MakeTable(crc32.Castagnoli) | ||||
|  | ||||
| // ReaderObjectAttrs are attributes about the object being read. These are populated | ||||
| // during the New call. This struct only holds a subset of object attributes: to | ||||
| // get the full set of attributes, use ObjectHandle.Attrs. | ||||
| // | ||||
| // Each field is read-only. | ||||
| type ReaderObjectAttrs struct { | ||||
| 	// Size is the length of the object's content. | ||||
| 	Size int64 | ||||
|  | ||||
| 	// StartOffset is the byte offset within the object | ||||
| 	// from which reading begins. | ||||
| 	// This value is only non-zero for range requests. | ||||
| 	StartOffset int64 | ||||
|  | ||||
| 	// ContentType is the MIME type of the object's content. | ||||
| 	ContentType string | ||||
|  | ||||
| 	// ContentEncoding is the encoding of the object's content. | ||||
| 	ContentEncoding string | ||||
|  | ||||
| 	// CacheControl specifies whether and for how long browser and Internet | ||||
| 	// caches are allowed to cache your objects. | ||||
| 	CacheControl string | ||||
|  | ||||
| 	// LastModified is the time that the object was last modified. | ||||
| 	LastModified time.Time | ||||
|  | ||||
| 	// Generation is the generation number of the object's content. | ||||
| 	Generation int64 | ||||
|  | ||||
| 	// Metageneration is the version of the metadata for this object at | ||||
| 	// this generation. This field is used for preconditions and for | ||||
| 	// detecting changes in metadata. A metageneration number is only | ||||
| 	// meaningful in the context of a particular generation of a | ||||
| 	// particular object. | ||||
| 	Metageneration int64 | ||||
| } | ||||
|  | ||||
| // NewReader creates a new Reader to read the contents of the | ||||
| // object. | ||||
| // ErrObjectNotExist will be returned if the object is not found. | ||||
| // | ||||
| // The caller must call Close on the returned Reader when done reading. | ||||
| func (o *ObjectHandle) NewReader(ctx context.Context) (*Reader, error) { | ||||
| 	return o.NewRangeReader(ctx, 0, -1) | ||||
| } | ||||
|  | ||||
| // NewRangeReader reads part of an object, reading at most length bytes | ||||
| // starting at the given offset. If length is negative, the object is read | ||||
| // until the end. If offset is negative, the object is read abs(offset) bytes | ||||
| // from the end, and length must also be negative to indicate all remaining | ||||
| // bytes will be read. | ||||
| func (o *ObjectHandle) NewRangeReader(ctx context.Context, offset, length int64) (r *Reader, err error) { | ||||
| 	ctx = trace.StartSpan(ctx, "cloud.google.com/go/storage.Object.NewRangeReader") | ||||
| 	defer func() { trace.EndSpan(ctx, err) }() | ||||
|  | ||||
| 	if err := o.validate(); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	if offset < 0 && length >= 0 { | ||||
| 		return nil, fmt.Errorf("storage: invalid offset %d < 0 requires negative length", offset) | ||||
| 	} | ||||
| 	if o.conds != nil { | ||||
| 		if err := o.conds.validate("NewRangeReader"); err != nil { | ||||
| 			return nil, err | ||||
| 		} | ||||
| 	} | ||||
| 	u := &url.URL{ | ||||
| 		Scheme: o.c.scheme, | ||||
| 		Host:   o.c.readHost, | ||||
| 		Path:   fmt.Sprintf("/%s/%s", o.bucket, o.object), | ||||
| 	} | ||||
| 	verb := "GET" | ||||
| 	if length == 0 { | ||||
| 		verb = "HEAD" | ||||
| 	} | ||||
| 	req, err := http.NewRequest(verb, u.String(), nil) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	req = req.WithContext(ctx) | ||||
| 	if o.userProject != "" { | ||||
| 		req.Header.Set("X-Goog-User-Project", o.userProject) | ||||
| 	} | ||||
| 	if o.readCompressed { | ||||
| 		req.Header.Set("Accept-Encoding", "gzip") | ||||
| 	} | ||||
| 	if err := setEncryptionHeaders(req.Header, o.encryptionKey, false); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
|  | ||||
| 	gen := o.gen | ||||
|  | ||||
| 	// Define a function that initiates a Read with offset and length, assuming we | ||||
| 	// have already read seen bytes. | ||||
| 	reopen := func(seen int64) (*http.Response, error) { | ||||
| 		start := offset + seen | ||||
| 		if length < 0 && start < 0 { | ||||
| 			req.Header.Set("Range", fmt.Sprintf("bytes=%d", start)) | ||||
| 		} else if length < 0 && start > 0 { | ||||
| 			req.Header.Set("Range", fmt.Sprintf("bytes=%d-", start)) | ||||
| 		} else if length > 0 { | ||||
| 			// The end character isn't affected by how many bytes we've seen. | ||||
| 			req.Header.Set("Range", fmt.Sprintf("bytes=%d-%d", start, offset+length-1)) | ||||
| 		} | ||||
| 		// We wait to assign conditions here because the generation number can change in between reopen() runs. | ||||
| 		req.URL.RawQuery = conditionsQuery(gen, o.conds) | ||||
| 		var res *http.Response | ||||
| 		err = runWithRetry(ctx, func() error { | ||||
| 			res, err = o.c.hc.Do(req) | ||||
| 			if err != nil { | ||||
| 				return err | ||||
| 			} | ||||
| 			if res.StatusCode == http.StatusNotFound { | ||||
| 				res.Body.Close() | ||||
| 				return ErrObjectNotExist | ||||
| 			} | ||||
| 			if res.StatusCode < 200 || res.StatusCode > 299 { | ||||
| 				body, _ := ioutil.ReadAll(res.Body) | ||||
| 				res.Body.Close() | ||||
| 				return &googleapi.Error{ | ||||
| 					Code:   res.StatusCode, | ||||
| 					Header: res.Header, | ||||
| 					Body:   string(body), | ||||
| 				} | ||||
| 			} | ||||
| 			if start > 0 && length != 0 && res.StatusCode != http.StatusPartialContent { | ||||
| 				res.Body.Close() | ||||
| 				return errors.New("storage: partial request not satisfied") | ||||
| 			} | ||||
| 			// If a generation hasn't been specified, and this is the first response we get, let's record the | ||||
| 			// generation. In future requests we'll use this generation as a precondition to avoid data races. | ||||
| 			if gen < 0 && res.Header.Get("X-Goog-Generation") != "" { | ||||
| 				gen64, err := strconv.ParseInt(res.Header.Get("X-Goog-Generation"), 10, 64) | ||||
| 				if err != nil { | ||||
| 					return err | ||||
| 				} | ||||
| 				gen = gen64 | ||||
| 			} | ||||
| 			return nil | ||||
| 		}) | ||||
| 		if err != nil { | ||||
| 			return nil, err | ||||
| 		} | ||||
| 		return res, nil | ||||
| 	} | ||||
|  | ||||
| 	res, err := reopen(0) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	var ( | ||||
| 		size        int64 // total size of object, even if a range was requested. | ||||
| 		checkCRC    bool | ||||
| 		crc         uint32 | ||||
| 		startOffset int64 // non-zero if range request. | ||||
| 	) | ||||
| 	if res.StatusCode == http.StatusPartialContent { | ||||
| 		cr := strings.TrimSpace(res.Header.Get("Content-Range")) | ||||
| 		if !strings.HasPrefix(cr, "bytes ") || !strings.Contains(cr, "/") { | ||||
| 			return nil, fmt.Errorf("storage: invalid Content-Range %q", cr) | ||||
| 		} | ||||
| 		size, err = strconv.ParseInt(cr[strings.LastIndex(cr, "/")+1:], 10, 64) | ||||
| 		if err != nil { | ||||
| 			return nil, fmt.Errorf("storage: invalid Content-Range %q", cr) | ||||
| 		} | ||||
|  | ||||
| 		dashIndex := strings.Index(cr, "-") | ||||
| 		if dashIndex >= 0 { | ||||
| 			startOffset, err = strconv.ParseInt(cr[len("bytes="):dashIndex], 10, 64) | ||||
| 			if err != nil { | ||||
| 				return nil, fmt.Errorf("storage: invalid Content-Range %q: %v", cr, err) | ||||
| 			} | ||||
| 		} | ||||
| 	} else { | ||||
| 		size = res.ContentLength | ||||
| 		// Check the CRC iff all of the following hold: | ||||
| 		// - We asked for content (length != 0). | ||||
| 		// - We got all the content (status != PartialContent). | ||||
| 		// - The server sent a CRC header. | ||||
| 		// - The Go http stack did not uncompress the file. | ||||
| 		// - We were not served compressed data that was uncompressed on download. | ||||
| 		// The problem with the last two cases is that the CRC will not match -- GCS | ||||
| 		// computes it on the compressed contents, but we compute it on the | ||||
| 		// uncompressed contents. | ||||
| 		if length != 0 && !res.Uncompressed && !uncompressedByServer(res) { | ||||
| 			crc, checkCRC = parseCRC32c(res) | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	remain := res.ContentLength | ||||
| 	body := res.Body | ||||
| 	if length == 0 { | ||||
| 		remain = 0 | ||||
| 		body.Close() | ||||
| 		body = emptyBody | ||||
| 	} | ||||
| 	var metaGen int64 | ||||
| 	if res.Header.Get("X-Goog-Generation") != "" { | ||||
| 		metaGen, err = strconv.ParseInt(res.Header.Get("X-Goog-Metageneration"), 10, 64) | ||||
| 		if err != nil { | ||||
| 			return nil, err | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	var lm time.Time | ||||
| 	if res.Header.Get("Last-Modified") != "" { | ||||
| 		lm, err = http.ParseTime(res.Header.Get("Last-Modified")) | ||||
| 		if err != nil { | ||||
| 			return nil, err | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	attrs := ReaderObjectAttrs{ | ||||
| 		Size:            size, | ||||
| 		ContentType:     res.Header.Get("Content-Type"), | ||||
| 		ContentEncoding: res.Header.Get("Content-Encoding"), | ||||
| 		CacheControl:    res.Header.Get("Cache-Control"), | ||||
| 		LastModified:    lm, | ||||
| 		StartOffset:     startOffset, | ||||
| 		Generation:      gen, | ||||
| 		Metageneration:  metaGen, | ||||
| 	} | ||||
| 	return &Reader{ | ||||
| 		Attrs:    attrs, | ||||
| 		body:     body, | ||||
| 		size:     size, | ||||
| 		remain:   remain, | ||||
| 		wantCRC:  crc, | ||||
| 		checkCRC: checkCRC, | ||||
| 		reopen:   reopen, | ||||
| 	}, nil | ||||
| } | ||||
|  | ||||
| func uncompressedByServer(res *http.Response) bool { | ||||
| 	// If the data is stored as gzip but is not encoded as gzip, then it | ||||
| 	// was uncompressed by the server. | ||||
| 	return res.Header.Get("X-Goog-Stored-Content-Encoding") == "gzip" && | ||||
| 		res.Header.Get("Content-Encoding") != "gzip" | ||||
| } | ||||
|  | ||||
| func parseCRC32c(res *http.Response) (uint32, bool) { | ||||
| 	const prefix = "crc32c=" | ||||
| 	for _, spec := range res.Header["X-Goog-Hash"] { | ||||
| 		if strings.HasPrefix(spec, prefix) { | ||||
| 			c, err := decodeUint32(spec[len(prefix):]) | ||||
| 			if err == nil { | ||||
| 				return c, true | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| 	return 0, false | ||||
| } | ||||
|  | ||||
| var emptyBody = ioutil.NopCloser(strings.NewReader("")) | ||||
|  | ||||
| // Reader reads a Cloud Storage object. | ||||
| // It implements io.Reader. | ||||
| // | ||||
| // Typically, a Reader computes the CRC of the downloaded content and compares it to | ||||
| // the stored CRC, returning an error from Read if there is a mismatch. This integrity check | ||||
| // is skipped if transcoding occurs. See https://cloud.google.com/storage/docs/transcoding. | ||||
| type Reader struct { | ||||
| 	Attrs              ReaderObjectAttrs | ||||
| 	body               io.ReadCloser | ||||
| 	seen, remain, size int64 | ||||
| 	checkCRC           bool   // should we check the CRC? | ||||
| 	wantCRC            uint32 // the CRC32c value the server sent in the header | ||||
| 	gotCRC             uint32 // running crc | ||||
| 	reopen             func(seen int64) (*http.Response, error) | ||||
| } | ||||
|  | ||||
| // Close closes the Reader. It must be called when done reading. | ||||
| func (r *Reader) Close() error { | ||||
| 	return r.body.Close() | ||||
| } | ||||
|  | ||||
| func (r *Reader) Read(p []byte) (int, error) { | ||||
| 	n, err := r.readWithRetry(p) | ||||
| 	if r.remain != -1 { | ||||
| 		r.remain -= int64(n) | ||||
| 	} | ||||
| 	if r.checkCRC { | ||||
| 		r.gotCRC = crc32.Update(r.gotCRC, crc32cTable, p[:n]) | ||||
| 		// Check CRC here. It would be natural to check it in Close, but | ||||
| 		// everybody defers Close on the assumption that it doesn't return | ||||
| 		// anything worth looking at. | ||||
| 		if err == io.EOF { | ||||
| 			if r.gotCRC != r.wantCRC { | ||||
| 				return n, fmt.Errorf("storage: bad CRC on read: got %d, want %d", | ||||
| 					r.gotCRC, r.wantCRC) | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| 	return n, err | ||||
| } | ||||
|  | ||||
| func (r *Reader) readWithRetry(p []byte) (int, error) { | ||||
| 	n := 0 | ||||
| 	for len(p[n:]) > 0 { | ||||
| 		m, err := r.body.Read(p[n:]) | ||||
| 		n += m | ||||
| 		r.seen += int64(m) | ||||
| 		if !shouldRetryRead(err) { | ||||
| 			return n, err | ||||
| 		} | ||||
| 		// Read failed, but we will try again. Send a ranged read request that takes | ||||
| 		// into account the number of bytes we've already seen. | ||||
| 		res, err := r.reopen(r.seen) | ||||
| 		if err != nil { | ||||
| 			// reopen already retries | ||||
| 			return n, err | ||||
| 		} | ||||
| 		r.body.Close() | ||||
| 		r.body = res.Body | ||||
| 	} | ||||
| 	return n, nil | ||||
| } | ||||
|  | ||||
| func shouldRetryRead(err error) bool { | ||||
| 	if err == nil { | ||||
| 		return false | ||||
| 	} | ||||
| 	return strings.HasSuffix(err.Error(), "INTERNAL_ERROR") && strings.Contains(reflect.TypeOf(err).String(), "http2") | ||||
| } | ||||
|  | ||||
| // Size returns the size of the object in bytes. | ||||
| // The returned value is always the same and is not affected by | ||||
| // calls to Read or Close. | ||||
| // | ||||
| // Deprecated: use Reader.Attrs.Size. | ||||
| func (r *Reader) Size() int64 { | ||||
| 	return r.Attrs.Size | ||||
| } | ||||
|  | ||||
| // Remain returns the number of bytes left to read, or -1 if unknown. | ||||
| func (r *Reader) Remain() int64 { | ||||
| 	return r.remain | ||||
| } | ||||
|  | ||||
| // ContentType returns the content type of the object. | ||||
| // | ||||
| // Deprecated: use Reader.Attrs.ContentType. | ||||
| func (r *Reader) ContentType() string { | ||||
| 	return r.Attrs.ContentType | ||||
| } | ||||
|  | ||||
| // ContentEncoding returns the content encoding of the object. | ||||
| // | ||||
| // Deprecated: use Reader.Attrs.ContentEncoding. | ||||
| func (r *Reader) ContentEncoding() string { | ||||
| 	return r.Attrs.ContentEncoding | ||||
| } | ||||
|  | ||||
| // CacheControl returns the cache control of the object. | ||||
| // | ||||
| // Deprecated: use Reader.Attrs.CacheControl. | ||||
| func (r *Reader) CacheControl() string { | ||||
| 	return r.Attrs.CacheControl | ||||
| } | ||||
|  | ||||
| // LastModified returns the value of the Last-Modified header. | ||||
| // | ||||
| // Deprecated: use Reader.Attrs.LastModified. | ||||
| func (r *Reader) LastModified() (time.Time, error) { | ||||
| 	return r.Attrs.LastModified, nil | ||||
| } | ||||
							
								
								
									
										1369
									
								
								vendor/cloud.google.com/go/storage/storage.go
									
									
									
										generated
									
									
										vendored
									
									
								
							
							
						
						
									
										1369
									
								
								vendor/cloud.google.com/go/storage/storage.go
									
									
									
										generated
									
									
										vendored
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										30067
									
								
								vendor/cloud.google.com/go/storage/storage.replay
									
									
									
										generated
									
									
										vendored
									
									
								
							
							
						
						
									
										30067
									
								
								vendor/cloud.google.com/go/storage/storage.replay
									
									
									
										generated
									
									
										vendored
									
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
							
								
								
									
										260
									
								
								vendor/cloud.google.com/go/storage/writer.go
									
									
									
										generated
									
									
										vendored
									
									
								
							
							
						
						
									
										260
									
								
								vendor/cloud.google.com/go/storage/writer.go
									
									
									
										generated
									
									
										vendored
									
									
								
							| @@ -1,260 +0,0 @@ | ||||
| // Copyright 2014 Google LLC | ||||
| // | ||||
| // Licensed under the Apache License, Version 2.0 (the "License"); | ||||
| // you may not use this file except in compliance with the License. | ||||
| // You may obtain a copy of the License at | ||||
| // | ||||
| //      http://www.apache.org/licenses/LICENSE-2.0 | ||||
| // | ||||
| // Unless required by applicable law or agreed to in writing, software | ||||
| // distributed under the License is distributed on an "AS IS" BASIS, | ||||
| // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||||
| // See the License for the specific language governing permissions and | ||||
| // limitations under the License. | ||||
|  | ||||
| package storage | ||||
|  | ||||
| import ( | ||||
| 	"context" | ||||
| 	"encoding/base64" | ||||
| 	"errors" | ||||
| 	"fmt" | ||||
| 	"io" | ||||
| 	"sync" | ||||
| 	"unicode/utf8" | ||||
|  | ||||
| 	"google.golang.org/api/googleapi" | ||||
| 	raw "google.golang.org/api/storage/v1" | ||||
| ) | ||||
|  | ||||
| // A Writer writes a Cloud Storage object. | ||||
| type Writer struct { | ||||
| 	// ObjectAttrs are optional attributes to set on the object. Any attributes | ||||
| 	// must be initialized before the first Write call. Nil or zero-valued | ||||
| 	// attributes are ignored. | ||||
| 	ObjectAttrs | ||||
|  | ||||
| 	// SendCRC specifies whether to transmit a CRC32C field. It should be set | ||||
| 	// to true in addition to setting the Writer's CRC32C field, because zero | ||||
| 	// is a valid CRC and normally a zero would not be transmitted. | ||||
| 	// If a CRC32C is sent, and the data written does not match the checksum, | ||||
| 	// the write will be rejected. | ||||
| 	SendCRC32C bool | ||||
|  | ||||
| 	// ChunkSize controls the maximum number of bytes of the object that the | ||||
| 	// Writer will attempt to send to the server in a single request. Objects | ||||
| 	// smaller than the size will be sent in a single request, while larger | ||||
| 	// objects will be split over multiple requests. The size will be rounded up | ||||
| 	// to the nearest multiple of 256K. If zero, chunking will be disabled and | ||||
| 	// the object will be uploaded in a single request. | ||||
| 	// | ||||
| 	// ChunkSize will default to a reasonable value. If you perform many concurrent | ||||
| 	// writes of small objects, you may wish set ChunkSize to a value that matches | ||||
| 	// your objects' sizes to avoid consuming large amounts of memory. | ||||
| 	// | ||||
| 	// ChunkSize must be set before the first Write call. | ||||
| 	ChunkSize int | ||||
|  | ||||
| 	// ProgressFunc can be used to monitor the progress of a large write. | ||||
| 	// operation. If ProgressFunc is not nil and writing requires multiple | ||||
| 	// calls to the underlying service (see | ||||
| 	// https://cloud.google.com/storage/docs/json_api/v1/how-tos/resumable-upload), | ||||
| 	// then ProgressFunc will be invoked after each call with the number of bytes of | ||||
| 	// content copied so far. | ||||
| 	// | ||||
| 	// ProgressFunc should return quickly without blocking. | ||||
| 	ProgressFunc func(int64) | ||||
|  | ||||
| 	ctx context.Context | ||||
| 	o   *ObjectHandle | ||||
|  | ||||
| 	opened bool | ||||
| 	pw     *io.PipeWriter | ||||
|  | ||||
| 	donec chan struct{} // closed after err and obj are set. | ||||
| 	obj   *ObjectAttrs | ||||
|  | ||||
| 	mu  sync.Mutex | ||||
| 	err error | ||||
| } | ||||
|  | ||||
| func (w *Writer) open() error { | ||||
| 	attrs := w.ObjectAttrs | ||||
| 	// Check the developer didn't change the object Name (this is unfortunate, but | ||||
| 	// we don't want to store an object under the wrong name). | ||||
| 	if attrs.Name != w.o.object { | ||||
| 		return fmt.Errorf("storage: Writer.Name %q does not match object name %q", attrs.Name, w.o.object) | ||||
| 	} | ||||
| 	if !utf8.ValidString(attrs.Name) { | ||||
| 		return fmt.Errorf("storage: object name %q is not valid UTF-8", attrs.Name) | ||||
| 	} | ||||
| 	if attrs.KMSKeyName != "" && w.o.encryptionKey != nil { | ||||
| 		return errors.New("storage: cannot use KMSKeyName with a customer-supplied encryption key") | ||||
| 	} | ||||
| 	pr, pw := io.Pipe() | ||||
| 	w.pw = pw | ||||
| 	w.opened = true | ||||
|  | ||||
| 	go w.monitorCancel() | ||||
|  | ||||
| 	if w.ChunkSize < 0 { | ||||
| 		return errors.New("storage: Writer.ChunkSize must be non-negative") | ||||
| 	} | ||||
| 	mediaOpts := []googleapi.MediaOption{ | ||||
| 		googleapi.ChunkSize(w.ChunkSize), | ||||
| 	} | ||||
| 	if c := attrs.ContentType; c != "" { | ||||
| 		mediaOpts = append(mediaOpts, googleapi.ContentType(c)) | ||||
| 	} | ||||
|  | ||||
| 	go func() { | ||||
| 		defer close(w.donec) | ||||
|  | ||||
| 		rawObj := attrs.toRawObject(w.o.bucket) | ||||
| 		if w.SendCRC32C { | ||||
| 			rawObj.Crc32c = encodeUint32(attrs.CRC32C) | ||||
| 		} | ||||
| 		if w.MD5 != nil { | ||||
| 			rawObj.Md5Hash = base64.StdEncoding.EncodeToString(w.MD5) | ||||
| 		} | ||||
| 		if w.o.c.envHost != "" { | ||||
| 			w.o.c.raw.BasePath = fmt.Sprintf("%s://%s", w.o.c.scheme, w.o.c.envHost) | ||||
| 		} | ||||
| 		call := w.o.c.raw.Objects.Insert(w.o.bucket, rawObj). | ||||
| 			Media(pr, mediaOpts...). | ||||
| 			Projection("full"). | ||||
| 			Context(w.ctx) | ||||
|  | ||||
| 		if w.ProgressFunc != nil { | ||||
| 			call.ProgressUpdater(func(n, _ int64) { w.ProgressFunc(n) }) | ||||
| 		} | ||||
| 		if attrs.KMSKeyName != "" { | ||||
| 			call.KmsKeyName(attrs.KMSKeyName) | ||||
| 		} | ||||
| 		if attrs.PredefinedACL != "" { | ||||
| 			call.PredefinedAcl(attrs.PredefinedACL) | ||||
| 		} | ||||
| 		if err := setEncryptionHeaders(call.Header(), w.o.encryptionKey, false); err != nil { | ||||
| 			w.mu.Lock() | ||||
| 			w.err = err | ||||
| 			w.mu.Unlock() | ||||
| 			pr.CloseWithError(err) | ||||
| 			return | ||||
| 		} | ||||
| 		var resp *raw.Object | ||||
| 		err := applyConds("NewWriter", w.o.gen, w.o.conds, call) | ||||
| 		if err == nil { | ||||
| 			if w.o.userProject != "" { | ||||
| 				call.UserProject(w.o.userProject) | ||||
| 			} | ||||
| 			setClientHeader(call.Header()) | ||||
|  | ||||
| 			// The internals that perform call.Do automatically retry | ||||
| 			// uploading chunks, hence no need to add retries here. | ||||
| 			// See issue https://github.com/googleapis/google-cloud-go/issues/1507. | ||||
| 			// | ||||
| 			// However, since this whole call's internals involve making the initial | ||||
| 			// resumable upload session, the first HTTP request is not retried. | ||||
| 			// TODO: Follow-up with google.golang.org/gensupport to solve | ||||
| 			// https://github.com/googleapis/google-api-go-client/issues/392. | ||||
| 			resp, err = call.Do() | ||||
| 		} | ||||
| 		if err != nil { | ||||
| 			w.mu.Lock() | ||||
| 			w.err = err | ||||
| 			w.mu.Unlock() | ||||
| 			pr.CloseWithError(err) | ||||
| 			return | ||||
| 		} | ||||
| 		w.obj = newObject(resp) | ||||
| 	}() | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| // Write appends to w. It implements the io.Writer interface. | ||||
| // | ||||
| // Since writes happen asynchronously, Write may return a nil | ||||
| // error even though the write failed (or will fail). Always | ||||
| // use the error returned from Writer.Close to determine if | ||||
| // the upload was successful. | ||||
| func (w *Writer) Write(p []byte) (n int, err error) { | ||||
| 	w.mu.Lock() | ||||
| 	werr := w.err | ||||
| 	w.mu.Unlock() | ||||
| 	if werr != nil { | ||||
| 		return 0, werr | ||||
| 	} | ||||
| 	if !w.opened { | ||||
| 		if err := w.open(); err != nil { | ||||
| 			return 0, err | ||||
| 		} | ||||
| 	} | ||||
| 	n, err = w.pw.Write(p) | ||||
| 	if err != nil { | ||||
| 		w.mu.Lock() | ||||
| 		werr := w.err | ||||
| 		w.mu.Unlock() | ||||
| 		// Preserve existing functionality that when context is canceled, Write will return | ||||
| 		// context.Canceled instead of "io: read/write on closed pipe". This hides the | ||||
| 		// pipe implementation detail from users and makes Write seem as though it's an RPC. | ||||
| 		if werr == context.Canceled || werr == context.DeadlineExceeded { | ||||
| 			return n, werr | ||||
| 		} | ||||
| 	} | ||||
| 	return n, err | ||||
| } | ||||
|  | ||||
| // Close completes the write operation and flushes any buffered data. | ||||
| // If Close doesn't return an error, metadata about the written object | ||||
| // can be retrieved by calling Attrs. | ||||
| func (w *Writer) Close() error { | ||||
| 	if !w.opened { | ||||
| 		if err := w.open(); err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	// Closing either the read or write causes the entire pipe to close. | ||||
| 	if err := w.pw.Close(); err != nil { | ||||
| 		return err | ||||
| 	} | ||||
|  | ||||
| 	<-w.donec | ||||
| 	w.mu.Lock() | ||||
| 	defer w.mu.Unlock() | ||||
| 	return w.err | ||||
| } | ||||
|  | ||||
| // monitorCancel is intended to be used as a background goroutine. It monitors the | ||||
| // context, and when it observes that the context has been canceled, it manually | ||||
| // closes things that do not take a context. | ||||
| func (w *Writer) monitorCancel() { | ||||
| 	select { | ||||
| 	case <-w.ctx.Done(): | ||||
| 		w.mu.Lock() | ||||
| 		werr := w.ctx.Err() | ||||
| 		w.err = werr | ||||
| 		w.mu.Unlock() | ||||
|  | ||||
| 		// Closing either the read or write causes the entire pipe to close. | ||||
| 		w.CloseWithError(werr) | ||||
| 	case <-w.donec: | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // CloseWithError aborts the write operation with the provided error. | ||||
| // CloseWithError always returns nil. | ||||
| // | ||||
| // Deprecated: cancel the context passed to NewWriter instead. | ||||
| func (w *Writer) CloseWithError(err error) error { | ||||
| 	if !w.opened { | ||||
| 		return nil | ||||
| 	} | ||||
| 	return w.pw.CloseWithError(err) | ||||
| } | ||||
|  | ||||
| // Attrs returns metadata about a successfully-written object. | ||||
| // It's only valid to call it after Close returns nil. | ||||
| func (w *Writer) Attrs() *ObjectAttrs { | ||||
| 	return w.obj | ||||
| } | ||||
							
								
								
									
										218
									
								
								vendor/github.com/apparentlymart/go-cidr/cidr/cidr.go
									
									
									
										generated
									
									
										vendored
									
									
								
							
							
						
						
									
										218
									
								
								vendor/github.com/apparentlymart/go-cidr/cidr/cidr.go
									
									
									
										generated
									
									
										vendored
									
									
								
							| @@ -1,218 +0,0 @@ | ||||
| // Package cidr is a collection of assorted utilities for computing | ||||
| // network and host addresses within network ranges. | ||||
| // | ||||
| // It expects a CIDR-type address structure where addresses are divided into | ||||
| // some number of prefix bits representing the network and then the remaining | ||||
| // suffix bits represent the host. | ||||
| // | ||||
| // For example, it can help to calculate addresses for sub-networks of a | ||||
| // parent network, or to calculate host addresses within a particular prefix. | ||||
| // | ||||
| // At present this package is prioritizing simplicity of implementation and | ||||
| // de-prioritizing speed and memory usage. Thus caution is advised before | ||||
| // using this package in performance-critical applications or hot codepaths. | ||||
| // Patches to improve the speed and memory usage may be accepted as long as | ||||
| // they do not result in a significant increase in code complexity. | ||||
| package cidr | ||||
|  | ||||
| import ( | ||||
| 	"fmt" | ||||
| 	"math/big" | ||||
| 	"net" | ||||
| ) | ||||
|  | ||||
| // Subnet takes a parent CIDR range and creates a subnet within it | ||||
| // with the given number of additional prefix bits and the given | ||||
| // network number. | ||||
| // | ||||
| // For example, 10.3.0.0/16, extended by 8 bits, with a network number | ||||
| // of 5, becomes 10.3.5.0/24 . | ||||
| func Subnet(base *net.IPNet, newBits int, num int) (*net.IPNet, error) { | ||||
| 	ip := base.IP | ||||
| 	mask := base.Mask | ||||
|  | ||||
| 	parentLen, addrLen := mask.Size() | ||||
| 	newPrefixLen := parentLen + newBits | ||||
|  | ||||
| 	if newPrefixLen > addrLen { | ||||
| 		return nil, fmt.Errorf("insufficient address space to extend prefix of %d by %d", parentLen, newBits) | ||||
| 	} | ||||
|  | ||||
| 	maxNetNum := uint64(1<<uint64(newBits)) - 1 | ||||
| 	if uint64(num) > maxNetNum { | ||||
| 		return nil, fmt.Errorf("prefix extension of %d does not accommodate a subnet numbered %d", newBits, num) | ||||
| 	} | ||||
|  | ||||
| 	return &net.IPNet{ | ||||
| 		IP:   insertNumIntoIP(ip, big.NewInt(int64(num)), newPrefixLen), | ||||
| 		Mask: net.CIDRMask(newPrefixLen, addrLen), | ||||
| 	}, nil | ||||
| } | ||||
|  | ||||
| // Host takes a parent CIDR range and turns it into a host IP address with | ||||
| // the given host number. | ||||
| // | ||||
| // For example, 10.3.0.0/16 with a host number of 2 gives 10.3.0.2. | ||||
| func Host(base *net.IPNet, num int) (net.IP, error) { | ||||
| 	ip := base.IP | ||||
| 	mask := base.Mask | ||||
| 	bigNum := big.NewInt(int64(num)) | ||||
|  | ||||
| 	parentLen, addrLen := mask.Size() | ||||
| 	hostLen := addrLen - parentLen | ||||
|  | ||||
| 	maxHostNum := big.NewInt(int64(1)) | ||||
| 	maxHostNum.Lsh(maxHostNum, uint(hostLen)) | ||||
| 	maxHostNum.Sub(maxHostNum, big.NewInt(1)) | ||||
|  | ||||
| 	numUint64 := big.NewInt(int64(bigNum.Uint64())) | ||||
| 	if bigNum.Cmp(big.NewInt(0)) == -1 { | ||||
| 		numUint64.Neg(bigNum) | ||||
| 		numUint64.Sub(numUint64, big.NewInt(int64(1))) | ||||
| 		bigNum.Sub(maxHostNum, numUint64) | ||||
| 	} | ||||
|  | ||||
| 	if numUint64.Cmp(maxHostNum) == 1 { | ||||
| 		return nil, fmt.Errorf("prefix of %d does not accommodate a host numbered %d", parentLen, num) | ||||
| 	} | ||||
| 	var bitlength int | ||||
| 	if ip.To4() != nil { | ||||
| 		bitlength = 32 | ||||
| 	} else { | ||||
| 		bitlength = 128 | ||||
| 	} | ||||
| 	return insertNumIntoIP(ip, bigNum, bitlength), nil | ||||
| } | ||||
|  | ||||
| // AddressRange returns the first and last addresses in the given CIDR range. | ||||
| func AddressRange(network *net.IPNet) (net.IP, net.IP) { | ||||
| 	// the first IP is easy | ||||
| 	firstIP := network.IP | ||||
|  | ||||
| 	// the last IP is the network address OR NOT the mask address | ||||
| 	prefixLen, bits := network.Mask.Size() | ||||
| 	if prefixLen == bits { | ||||
| 		// Easy! | ||||
| 		// But make sure that our two slices are distinct, since they | ||||
| 		// would be in all other cases. | ||||
| 		lastIP := make([]byte, len(firstIP)) | ||||
| 		copy(lastIP, firstIP) | ||||
| 		return firstIP, lastIP | ||||
| 	} | ||||
|  | ||||
| 	firstIPInt, bits := ipToInt(firstIP) | ||||
| 	hostLen := uint(bits) - uint(prefixLen) | ||||
| 	lastIPInt := big.NewInt(1) | ||||
| 	lastIPInt.Lsh(lastIPInt, hostLen) | ||||
| 	lastIPInt.Sub(lastIPInt, big.NewInt(1)) | ||||
| 	lastIPInt.Or(lastIPInt, firstIPInt) | ||||
|  | ||||
| 	return firstIP, intToIP(lastIPInt, bits) | ||||
| } | ||||
|  | ||||
| // AddressCount returns the number of distinct host addresses within the given | ||||
| // CIDR range. | ||||
| // | ||||
| // Since the result is a uint64, this function returns meaningful information | ||||
| // only for IPv4 ranges and IPv6 ranges with a prefix size of at least 65. | ||||
| func AddressCount(network *net.IPNet) uint64 { | ||||
| 	prefixLen, bits := network.Mask.Size() | ||||
| 	return 1 << (uint64(bits) - uint64(prefixLen)) | ||||
| } | ||||
|  | ||||
| //VerifyNoOverlap takes a list subnets and supernet (CIDRBlock) and verifies | ||||
| //none of the subnets overlap and all subnets are in the supernet | ||||
| //it returns an error if any of those conditions are not satisfied | ||||
| func VerifyNoOverlap(subnets []*net.IPNet, CIDRBlock *net.IPNet) error { | ||||
| 	firstLastIP := make([][]net.IP, len(subnets)) | ||||
| 	for i, s := range subnets { | ||||
| 		first, last := AddressRange(s) | ||||
| 		firstLastIP[i] = []net.IP{first, last} | ||||
| 	} | ||||
| 	for i, s := range subnets { | ||||
| 		if !CIDRBlock.Contains(firstLastIP[i][0]) || !CIDRBlock.Contains(firstLastIP[i][1]) { | ||||
| 			return fmt.Errorf("%s does not fully contain %s", CIDRBlock.String(), s.String()) | ||||
| 		} | ||||
| 		for j := 0; j < len(subnets); j++ { | ||||
| 			if i == j { | ||||
| 				continue | ||||
| 			} | ||||
|  | ||||
| 			first := firstLastIP[j][0] | ||||
| 			last := firstLastIP[j][1] | ||||
| 			if s.Contains(first) || s.Contains(last) { | ||||
| 				return fmt.Errorf("%s overlaps with %s", subnets[j].String(), s.String()) | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| // PreviousSubnet returns the subnet of the desired mask in the IP space | ||||
| // just lower than the start of IPNet provided. If the IP space rolls over | ||||
| // then the second return value is true | ||||
| func PreviousSubnet(network *net.IPNet, prefixLen int) (*net.IPNet, bool) { | ||||
| 	startIP := checkIPv4(network.IP) | ||||
| 	previousIP := make(net.IP, len(startIP)) | ||||
| 	copy(previousIP, startIP) | ||||
| 	cMask := net.CIDRMask(prefixLen, 8*len(previousIP)) | ||||
| 	previousIP = Dec(previousIP) | ||||
| 	previous := &net.IPNet{IP: previousIP.Mask(cMask), Mask: cMask} | ||||
| 	if startIP.Equal(net.IPv4zero) || startIP.Equal(net.IPv6zero) { | ||||
| 		return previous, true | ||||
| 	} | ||||
| 	return previous, false | ||||
| } | ||||
|  | ||||
| // NextSubnet returns the next available subnet of the desired mask size | ||||
| // starting for the maximum IP of the offset subnet | ||||
| // If the IP exceeds the maxium IP then the second return value is true | ||||
| func NextSubnet(network *net.IPNet, prefixLen int) (*net.IPNet, bool) { | ||||
| 	_, currentLast := AddressRange(network) | ||||
| 	mask := net.CIDRMask(prefixLen, 8*len(currentLast)) | ||||
| 	currentSubnet := &net.IPNet{IP: currentLast.Mask(mask), Mask: mask} | ||||
| 	_, last := AddressRange(currentSubnet) | ||||
| 	last = Inc(last) | ||||
| 	next := &net.IPNet{IP: last.Mask(mask), Mask: mask} | ||||
| 	if last.Equal(net.IPv4zero) || last.Equal(net.IPv6zero) { | ||||
| 		return next, true | ||||
| 	} | ||||
| 	return next, false | ||||
| } | ||||
|  | ||||
| //Inc increases the IP by one this returns a new []byte for the IP | ||||
| func Inc(IP net.IP) net.IP { | ||||
| 	IP = checkIPv4(IP) | ||||
| 	incIP := make([]byte, len(IP)) | ||||
| 	copy(incIP, IP) | ||||
| 	for j := len(incIP) - 1; j >= 0; j-- { | ||||
| 		incIP[j]++ | ||||
| 		if incIP[j] > 0 { | ||||
| 			break | ||||
| 		} | ||||
| 	} | ||||
| 	return incIP | ||||
| } | ||||
|  | ||||
| //Dec decreases the IP by one this returns a new []byte for the IP | ||||
| func Dec(IP net.IP) net.IP { | ||||
| 	IP = checkIPv4(IP) | ||||
| 	decIP := make([]byte, len(IP)) | ||||
| 	copy(decIP, IP) | ||||
| 	decIP = checkIPv4(decIP) | ||||
| 	for j := len(decIP) - 1; j >= 0; j-- { | ||||
| 		decIP[j]-- | ||||
| 		if decIP[j] < 255 { | ||||
| 			break | ||||
| 		} | ||||
| 	} | ||||
| 	return decIP | ||||
| } | ||||
|  | ||||
| func checkIPv4(ip net.IP) net.IP { | ||||
| 	// Go for some reason allocs IPv6len for IPv4 so we have to correct it | ||||
| 	if v4 := ip.To4(); v4 != nil { | ||||
| 		return v4 | ||||
| 	} | ||||
| 	return ip | ||||
| } | ||||
							
								
								
									
										37
									
								
								vendor/github.com/apparentlymart/go-cidr/cidr/wrangling.go
									
									
									
										generated
									
									
										vendored
									
									
								
							
							
						
						
									
										37
									
								
								vendor/github.com/apparentlymart/go-cidr/cidr/wrangling.go
									
									
									
										generated
									
									
										vendored
									
									
								
							| @@ -1,37 +0,0 @@ | ||||
| package cidr | ||||
|  | ||||
| import ( | ||||
| 	"fmt" | ||||
| 	"math/big" | ||||
| 	"net" | ||||
| ) | ||||
|  | ||||
| func ipToInt(ip net.IP) (*big.Int, int) { | ||||
| 	val := &big.Int{} | ||||
| 	val.SetBytes([]byte(ip)) | ||||
| 	if len(ip) == net.IPv4len { | ||||
| 		return val, 32 | ||||
| 	} else if len(ip) == net.IPv6len { | ||||
| 		return val, 128 | ||||
| 	} else { | ||||
| 		panic(fmt.Errorf("Unsupported address length %d", len(ip))) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func intToIP(ipInt *big.Int, bits int) net.IP { | ||||
| 	ipBytes := ipInt.Bytes() | ||||
| 	ret := make([]byte, bits/8) | ||||
| 	// Pack our IP bytes into the end of the return array, | ||||
| 	// since big.Int.Bytes() removes front zero padding. | ||||
| 	for i := 1; i <= len(ipBytes); i++ { | ||||
| 		ret[len(ret)-i] = ipBytes[len(ipBytes)-i] | ||||
| 	} | ||||
| 	return net.IP(ret) | ||||
| } | ||||
|  | ||||
| func insertNumIntoIP(ip net.IP, bigNum *big.Int, prefixLen int) net.IP { | ||||
| 	ipInt, totalBits := ipToInt(ip) | ||||
| 	bigNum.Lsh(bigNum, uint(totalBits-prefixLen)) | ||||
| 	ipInt.Or(ipInt, bigNum) | ||||
| 	return intToIP(ipInt, totalBits) | ||||
| } | ||||
							
								
								
									
										95
									
								
								vendor/github.com/apparentlymart/go-textseg/LICENSE
									
									
									
										generated
									
									
										vendored
									
									
								
							
							
						
						
									
										95
									
								
								vendor/github.com/apparentlymart/go-textseg/LICENSE
									
									
									
										generated
									
									
										vendored
									
									
								
							| @@ -1,95 +0,0 @@ | ||||
| Copyright (c) 2017 Martin Atkins | ||||
|  | ||||
| Permission is hereby granted, free of charge, to any person obtaining a copy | ||||
| of this software and associated documentation files (the "Software"), to deal | ||||
| in the Software without restriction, including without limitation the rights | ||||
| to use, copy, modify, merge, publish, distribute, sublicense, and/or sell | ||||
| copies of the Software, and to permit persons to whom the Software is | ||||
| furnished to do so, subject to the following conditions: | ||||
|  | ||||
| The above copyright notice and this permission notice shall be included in all | ||||
| copies or substantial portions of the Software. | ||||
|  | ||||
| THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR | ||||
| IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, | ||||
| FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE | ||||
| AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER | ||||
| LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, | ||||
| OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE | ||||
| SOFTWARE. | ||||
|  | ||||
| --------- | ||||
|  | ||||
| Unicode table generation programs are under a separate copyright and license: | ||||
|  | ||||
| Copyright (c) 2014 Couchbase, Inc. | ||||
| Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file | ||||
| except in compliance with the License. You may obtain a copy of the License at | ||||
|  | ||||
|     http://www.apache.org/licenses/LICENSE-2.0 | ||||
|  | ||||
| Unless required by applicable law or agreed to in writing, software distributed under the | ||||
| License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, | ||||
| either express or implied. See the License for the specific language governing permissions | ||||
| and limitations under the License. | ||||
|  | ||||
| --------- | ||||
|  | ||||
| Grapheme break data is provided as part of the Unicode character database, | ||||
| copright 2016 Unicode, Inc, which is provided with the following license: | ||||
|  | ||||
| Unicode Data Files include all data files under the directories | ||||
| http://www.unicode.org/Public/, http://www.unicode.org/reports/, | ||||
| http://www.unicode.org/cldr/data/, http://source.icu-project.org/repos/icu/, and | ||||
| http://www.unicode.org/utility/trac/browser/. | ||||
|  | ||||
| Unicode Data Files do not include PDF online code charts under the | ||||
| directory http://www.unicode.org/Public/. | ||||
|  | ||||
| Software includes any source code published in the Unicode Standard | ||||
| or under the directories | ||||
| http://www.unicode.org/Public/, http://www.unicode.org/reports/, | ||||
| http://www.unicode.org/cldr/data/, http://source.icu-project.org/repos/icu/, and | ||||
| http://www.unicode.org/utility/trac/browser/. | ||||
|  | ||||
| NOTICE TO USER: Carefully read the following legal agreement. | ||||
| BY DOWNLOADING, INSTALLING, COPYING OR OTHERWISE USING UNICODE INC.'S | ||||
| DATA FILES ("DATA FILES"), AND/OR SOFTWARE ("SOFTWARE"), | ||||
| YOU UNEQUIVOCALLY ACCEPT, AND AGREE TO BE BOUND BY, ALL OF THE | ||||
| TERMS AND CONDITIONS OF THIS AGREEMENT. | ||||
| IF YOU DO NOT AGREE, DO NOT DOWNLOAD, INSTALL, COPY, DISTRIBUTE OR USE | ||||
| THE DATA FILES OR SOFTWARE. | ||||
|  | ||||
| COPYRIGHT AND PERMISSION NOTICE | ||||
|  | ||||
| Copyright © 1991-2017 Unicode, Inc. All rights reserved. | ||||
| Distributed under the Terms of Use in http://www.unicode.org/copyright.html. | ||||
|  | ||||
| Permission is hereby granted, free of charge, to any person obtaining | ||||
| a copy of the Unicode data files and any associated documentation | ||||
| (the "Data Files") or Unicode software and any associated documentation | ||||
| (the "Software") to deal in the Data Files or Software | ||||
| without restriction, including without limitation the rights to use, | ||||
| copy, modify, merge, publish, distribute, and/or sell copies of | ||||
| the Data Files or Software, and to permit persons to whom the Data Files | ||||
| or Software are furnished to do so, provided that either | ||||
| (a) this copyright and permission notice appear with all copies | ||||
| of the Data Files or Software, or | ||||
| (b) this copyright and permission notice appear in associated | ||||
| Documentation. | ||||
|  | ||||
| THE DATA FILES AND SOFTWARE ARE PROVIDED "AS IS", WITHOUT WARRANTY OF | ||||
| ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE | ||||
| WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND | ||||
| NONINFRINGEMENT OF THIRD PARTY RIGHTS. | ||||
| IN NO EVENT SHALL THE COPYRIGHT HOLDER OR HOLDERS INCLUDED IN THIS | ||||
| NOTICE BE LIABLE FOR ANY CLAIM, OR ANY SPECIAL INDIRECT OR CONSEQUENTIAL | ||||
| DAMAGES, OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, | ||||
| DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER | ||||
| TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR | ||||
| PERFORMANCE OF THE DATA FILES OR SOFTWARE. | ||||
|  | ||||
| Except as contained in this notice, the name of a copyright holder | ||||
| shall not be used in advertising or otherwise to promote the sale, | ||||
| use or other dealings in these Data Files or Software without prior | ||||
| written authorization of the copyright holder. | ||||
							
								
								
									
										30
									
								
								vendor/github.com/apparentlymart/go-textseg/textseg/all_tokens.go
									
									
									
										generated
									
									
										vendored
									
									
								
							
							
						
						
									
										30
									
								
								vendor/github.com/apparentlymart/go-textseg/textseg/all_tokens.go
									
									
									
										generated
									
									
										vendored
									
									
								
							| @@ -1,30 +0,0 @@ | ||||
| package textseg | ||||
|  | ||||
| import ( | ||||
| 	"bufio" | ||||
| 	"bytes" | ||||
| ) | ||||
|  | ||||
| // AllTokens is a utility that uses a bufio.SplitFunc to produce a slice of | ||||
| // all of the recognized tokens in the given buffer. | ||||
| func AllTokens(buf []byte, splitFunc bufio.SplitFunc) ([][]byte, error) { | ||||
| 	scanner := bufio.NewScanner(bytes.NewReader(buf)) | ||||
| 	scanner.Split(splitFunc) | ||||
| 	var ret [][]byte | ||||
| 	for scanner.Scan() { | ||||
| 		ret = append(ret, scanner.Bytes()) | ||||
| 	} | ||||
| 	return ret, scanner.Err() | ||||
| } | ||||
|  | ||||
| // TokenCount is a utility that uses a bufio.SplitFunc to count the number of | ||||
| // recognized tokens in the given buffer. | ||||
| func TokenCount(buf []byte, splitFunc bufio.SplitFunc) (int, error) { | ||||
| 	scanner := bufio.NewScanner(bytes.NewReader(buf)) | ||||
| 	scanner.Split(splitFunc) | ||||
| 	var ret int | ||||
| 	for scanner.Scan() { | ||||
| 		ret++ | ||||
| 	} | ||||
| 	return ret, scanner.Err() | ||||
| } | ||||
							
								
								
									
										7
									
								
								vendor/github.com/apparentlymart/go-textseg/textseg/generate.go
									
									
									
										generated
									
									
										vendored
									
									
								
							
							
						
						
									
										7
									
								
								vendor/github.com/apparentlymart/go-textseg/textseg/generate.go
									
									
									
										generated
									
									
										vendored
									
									
								
							| @@ -1,7 +0,0 @@ | ||||
| package textseg | ||||
|  | ||||
| //go:generate go run make_tables.go -output tables.go | ||||
| //go:generate go run make_test_tables.go -output tables_test.go | ||||
| //go:generate ruby unicode2ragel.rb --url=http://www.unicode.org/Public/9.0.0/ucd/auxiliary/GraphemeBreakProperty.txt -m GraphemeCluster -p "Prepend,CR,LF,Control,Extend,Regional_Indicator,SpacingMark,L,V,T,LV,LVT,E_Base,E_Modifier,ZWJ,Glue_After_Zwj,E_Base_GAZ" -o grapheme_clusters_table.rl | ||||
| //go:generate ragel -Z grapheme_clusters.rl | ||||
| //go:generate gofmt -w grapheme_clusters.go | ||||
							
								
								
									
										5276
									
								
								vendor/github.com/apparentlymart/go-textseg/textseg/grapheme_clusters.go
									
									
									
										generated
									
									
										vendored
									
									
								
							
							
						
						
									
										5276
									
								
								vendor/github.com/apparentlymart/go-textseg/textseg/grapheme_clusters.go
									
									
									
										generated
									
									
										vendored
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										132
									
								
								vendor/github.com/apparentlymart/go-textseg/textseg/grapheme_clusters.rl
									
									
									
										generated
									
									
										vendored
									
									
								
							
							
						
						
									
										132
									
								
								vendor/github.com/apparentlymart/go-textseg/textseg/grapheme_clusters.rl
									
									
									
										generated
									
									
										vendored
									
									
								
							| @@ -1,132 +0,0 @@ | ||||
| package textseg | ||||
|  | ||||
| import ( | ||||
|     "errors" | ||||
|     "unicode/utf8" | ||||
| ) | ||||
|  | ||||
| // Generated from grapheme_clusters.rl. DO NOT EDIT | ||||
| %%{ | ||||
|   # (except you are actually in grapheme_clusters.rl here, so edit away!) | ||||
|  | ||||
|   machine graphclust; | ||||
|   write data; | ||||
| }%% | ||||
|  | ||||
| var Error = errors.New("invalid UTF8 text") | ||||
|  | ||||
| // ScanGraphemeClusters is a split function for bufio.Scanner that splits | ||||
| // on grapheme cluster boundaries. | ||||
| func ScanGraphemeClusters(data []byte, atEOF bool) (int, []byte, error) { | ||||
|     if len(data) == 0 { | ||||
|         return 0, nil, nil | ||||
|     } | ||||
|  | ||||
|     // Ragel state | ||||
| 	cs := 0 // Current State | ||||
| 	p := 0  // "Pointer" into data | ||||
| 	pe := len(data) // End-of-data "pointer" | ||||
|     ts := 0 | ||||
|     te := 0 | ||||
|     act := 0 | ||||
|     eof := pe | ||||
|  | ||||
|     // Make Go compiler happy | ||||
|     _ = ts | ||||
|     _ = te | ||||
|     _ = act | ||||
|     _ = eof | ||||
|  | ||||
|     startPos := 0 | ||||
|     endPos := 0 | ||||
|  | ||||
|     %%{ | ||||
|         include GraphemeCluster "grapheme_clusters_table.rl"; | ||||
|  | ||||
|         action start { | ||||
|             startPos = p | ||||
|         } | ||||
|  | ||||
|         action end { | ||||
|             endPos = p | ||||
|         } | ||||
|  | ||||
|         action emit { | ||||
|             return endPos+1, data[startPos:endPos+1], nil | ||||
|         } | ||||
|  | ||||
|         ZWJGlue = ZWJ (Glue_After_Zwj | E_Base_GAZ Extend* E_Modifier?)?; | ||||
|         AnyExtender = Extend | ZWJGlue | SpacingMark; | ||||
|         Extension = AnyExtender*; | ||||
|         ReplacementChar = (0xEF 0xBF 0xBD); | ||||
|  | ||||
|         CRLFSeq = CR LF; | ||||
|         ControlSeq = Control | ReplacementChar; | ||||
|         HangulSeq = ( | ||||
|             L+ (((LV? V+ | LVT) T*)?|LV?) | | ||||
|             LV V* T* | | ||||
|             V+ T* | | ||||
|             LVT T* | | ||||
|             T+ | ||||
|         ) Extension; | ||||
|         EmojiSeq = (E_Base | E_Base_GAZ) Extend* E_Modifier? Extension; | ||||
|         ZWJSeq = ZWJGlue Extension; | ||||
|         EmojiFlagSeq = Regional_Indicator Regional_Indicator? Extension; | ||||
|  | ||||
|         UTF8Cont = 0x80 .. 0xBF; | ||||
|         AnyUTF8 = ( | ||||
|             0x00..0x7F | | ||||
|             0xC0..0xDF . UTF8Cont | | ||||
|             0xE0..0xEF . UTF8Cont . UTF8Cont | | ||||
|             0xF0..0xF7 . UTF8Cont . UTF8Cont . UTF8Cont | ||||
|         ); | ||||
|  | ||||
|         # OtherSeq is any character that isn't at the start of one of the extended sequences above, followed by extension | ||||
|         OtherSeq = (AnyUTF8 - (CR|LF|Control|ReplacementChar|L|LV|V|LVT|T|E_Base|E_Base_GAZ|ZWJ|Regional_Indicator|Prepend)) Extension; | ||||
|  | ||||
|         # PrependSeq is prepend followed by any of the other patterns above, except control characters which explicitly break | ||||
|         PrependSeq = Prepend+ (HangulSeq|EmojiSeq|ZWJSeq|EmojiFlagSeq|OtherSeq)?; | ||||
|  | ||||
|         CRLFTok = CRLFSeq >start @end; | ||||
|         ControlTok = ControlSeq >start @end; | ||||
|         HangulTok = HangulSeq >start @end; | ||||
|         EmojiTok = EmojiSeq >start @end; | ||||
|         ZWJTok = ZWJSeq >start @end; | ||||
|         EmojiFlagTok = EmojiFlagSeq >start @end; | ||||
|         OtherTok = OtherSeq >start @end; | ||||
|         PrependTok = PrependSeq >start @end; | ||||
|  | ||||
|         main := |* | ||||
|             CRLFTok => emit; | ||||
|             ControlTok => emit; | ||||
|             HangulTok => emit; | ||||
|             EmojiTok => emit; | ||||
|             ZWJTok => emit; | ||||
|             EmojiFlagTok => emit; | ||||
|             PrependTok => emit; | ||||
|             OtherTok => emit; | ||||
|  | ||||
|             # any single valid UTF-8 character would also be valid per spec, | ||||
|             # but we'll handle that separately after the loop so we can deal | ||||
|             # with requesting more bytes if we're not at EOF. | ||||
|         *|; | ||||
|  | ||||
|         write init; | ||||
|         write exec; | ||||
|     }%% | ||||
|  | ||||
|     // If we fall out here then we were unable to complete a sequence. | ||||
|     // If we weren't able to complete a sequence then either we've | ||||
|     // reached the end of a partial buffer (so there's more data to come) | ||||
|     // or we have an isolated symbol that would normally be part of a | ||||
|     // grapheme cluster but has appeared in isolation here. | ||||
|  | ||||
|     if !atEOF { | ||||
|         // Request more | ||||
|         return 0, nil, nil | ||||
|     } | ||||
|  | ||||
|     // Just take the first UTF-8 sequence and return that. | ||||
|     _, seqLen := utf8.DecodeRune(data) | ||||
|     return seqLen, data[:seqLen], nil | ||||
| } | ||||
							
								
								
									
										1583
									
								
								vendor/github.com/apparentlymart/go-textseg/textseg/grapheme_clusters_table.rl
									
									
									
										generated
									
									
										vendored
									
									
								
							
							
						
						
									
										1583
									
								
								vendor/github.com/apparentlymart/go-textseg/textseg/grapheme_clusters_table.rl
									
									
									
										generated
									
									
										vendored
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										5700
									
								
								vendor/github.com/apparentlymart/go-textseg/textseg/tables.go
									
									
									
										generated
									
									
										vendored
									
									
								
							
							
						
						
									
										5700
									
								
								vendor/github.com/apparentlymart/go-textseg/textseg/tables.go
									
									
									
										generated
									
									
										vendored
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										335
									
								
								vendor/github.com/apparentlymart/go-textseg/textseg/unicode2ragel.rb
									
									
									
										generated
									
									
										vendored
									
									
								
							
							
						
						
									
										335
									
								
								vendor/github.com/apparentlymart/go-textseg/textseg/unicode2ragel.rb
									
									
									
										generated
									
									
										vendored
									
									
								
							| @@ -1,335 +0,0 @@ | ||||
| #!/usr/bin/env ruby | ||||
| # | ||||
| # This scripted has been updated to accept more command-line arguments: | ||||
| # | ||||
| #    -u, --url                        URL to process | ||||
| #    -m, --machine                    Machine name | ||||
| #    -p, --properties                 Properties to add to the machine | ||||
| #    -o, --output                     Write output to file | ||||
| # | ||||
| # Updated by: Marty Schoch <marty.schoch@gmail.com> | ||||
| #  | ||||
| # This script uses the unicode spec to generate a Ragel state machine | ||||
| # that recognizes unicode alphanumeric characters.  It generates 5 | ||||
| # character classes: uupper, ulower, ualpha, udigit, and ualnum. | ||||
| # Currently supported encodings are UTF-8 [default] and UCS-4. | ||||
| # | ||||
| # Usage: unicode2ragel.rb [options] | ||||
| #    -e, --encoding [ucs4 | utf8]     Data encoding | ||||
| #    -h, --help                       Show this message | ||||
| # | ||||
| # This script was originally written as part of the Ferret search | ||||
| # engine library. | ||||
| # | ||||
| # Author: Rakan El-Khalil <rakan@well.com> | ||||
|  | ||||
| require 'optparse' | ||||
| require 'open-uri' | ||||
|  | ||||
| ENCODINGS = [ :utf8, :ucs4 ] | ||||
| ALPHTYPES = { :utf8 => "byte", :ucs4 => "rune" } | ||||
| DEFAULT_CHART_URL = "http://www.unicode.org/Public/5.1.0/ucd/DerivedCoreProperties.txt" | ||||
| DEFAULT_MACHINE_NAME= "WChar" | ||||
|  | ||||
| ### | ||||
| # Display vars & default option | ||||
|  | ||||
| TOTAL_WIDTH = 80 | ||||
| RANGE_WIDTH = 23 | ||||
| @encoding = :utf8 | ||||
| @chart_url = DEFAULT_CHART_URL | ||||
| machine_name = DEFAULT_MACHINE_NAME | ||||
| properties = [] | ||||
| @output = $stdout | ||||
|  | ||||
| ### | ||||
| # Option parsing | ||||
|  | ||||
| cli_opts = OptionParser.new do |opts| | ||||
|   opts.on("-e", "--encoding [ucs4 | utf8]", "Data encoding") do |o| | ||||
|     @encoding = o.downcase.to_sym | ||||
|   end | ||||
|   opts.on("-h", "--help", "Show this message") do | ||||
|     puts opts | ||||
|     exit | ||||
|   end | ||||
|   opts.on("-u", "--url URL", "URL to process") do |o| | ||||
|     @chart_url = o  | ||||
|   end | ||||
|   opts.on("-m", "--machine MACHINE_NAME", "Machine name") do |o| | ||||
|     machine_name = o | ||||
|   end | ||||
|   opts.on("-p", "--properties x,y,z", Array, "Properties to add to machine") do |o| | ||||
|     properties = o | ||||
|   end | ||||
|   opts.on("-o", "--output FILE", "output file") do |o| | ||||
|     @output = File.new(o, "w+") | ||||
|   end | ||||
| end | ||||
|  | ||||
| cli_opts.parse(ARGV) | ||||
| unless ENCODINGS.member? @encoding | ||||
|   puts "Invalid encoding: #{@encoding}" | ||||
|   puts cli_opts | ||||
|   exit | ||||
| end | ||||
|  | ||||
| ## | ||||
| # Downloads the document at url and yields every alpha line's hex | ||||
| # range and description. | ||||
|  | ||||
| def each_alpha( url, property )  | ||||
|   open( url ) do |file| | ||||
|     file.each_line do |line| | ||||
|       next if line =~ /^#/; | ||||
|       next if line !~ /; #{property} #/; | ||||
|  | ||||
|       range, description = line.split(/;/) | ||||
|       range.strip! | ||||
|       description.gsub!(/.*#/, '').strip! | ||||
|  | ||||
|       if range =~ /\.\./ | ||||
|            start, stop = range.split '..' | ||||
|       else start = stop = range | ||||
|       end | ||||
|  | ||||
|       yield start.hex .. stop.hex, description | ||||
|     end | ||||
|   end | ||||
| end | ||||
|  | ||||
| ### | ||||
| # Formats to hex at minimum width | ||||
|  | ||||
| def to_hex( n ) | ||||
|   r = "%0X" % n | ||||
|   r = "0#{r}" unless (r.length % 2).zero? | ||||
|   r | ||||
| end | ||||
|  | ||||
| ### | ||||
| # UCS4 is just a straight hex conversion of the unicode codepoint. | ||||
|  | ||||
| def to_ucs4( range ) | ||||
|   rangestr  =   "0x" + to_hex(range.begin) | ||||
|   rangestr << "..0x" + to_hex(range.end) if range.begin != range.end | ||||
|   [ rangestr ] | ||||
| end | ||||
|  | ||||
| ## | ||||
| # 0x00     - 0x7f     -> 0zzzzzzz[7] | ||||
| # 0x80     - 0x7ff    -> 110yyyyy[5] 10zzzzzz[6] | ||||
| # 0x800    - 0xffff   -> 1110xxxx[4] 10yyyyyy[6] 10zzzzzz[6] | ||||
| # 0x010000 - 0x10ffff -> 11110www[3] 10xxxxxx[6] 10yyyyyy[6] 10zzzzzz[6]  | ||||
|  | ||||
| UTF8_BOUNDARIES = [0x7f, 0x7ff, 0xffff, 0x10ffff] | ||||
|  | ||||
| def to_utf8_enc( n ) | ||||
|   r = 0 | ||||
|   if n <= 0x7f | ||||
|     r = n | ||||
|   elsif n <= 0x7ff | ||||
|     y = 0xc0 | (n >> 6) | ||||
|     z = 0x80 | (n & 0x3f) | ||||
|     r = y << 8 | z | ||||
|   elsif n <= 0xffff | ||||
|     x = 0xe0 | (n >> 12) | ||||
|     y = 0x80 | (n >>  6) & 0x3f | ||||
|     z = 0x80 |  n        & 0x3f | ||||
|     r = x << 16 | y << 8 | z | ||||
|   elsif n <= 0x10ffff | ||||
|     w = 0xf0 | (n >> 18) | ||||
|     x = 0x80 | (n >> 12) & 0x3f | ||||
|     y = 0x80 | (n >>  6) & 0x3f | ||||
|     z = 0x80 |  n        & 0x3f | ||||
|     r = w << 24 | x << 16 | y << 8 | z | ||||
|   end | ||||
|  | ||||
|   to_hex(r) | ||||
| end | ||||
|  | ||||
| def from_utf8_enc( n ) | ||||
|   n = n.hex | ||||
|   r = 0 | ||||
|   if n <= 0x7f | ||||
|     r = n | ||||
|   elsif n <= 0xdfff | ||||
|     y = (n >> 8) & 0x1f | ||||
|     z =  n       & 0x3f | ||||
|     r = y << 6 | z | ||||
|   elsif n <= 0xefffff | ||||
|     x = (n >> 16) & 0x0f | ||||
|     y = (n >>  8) & 0x3f | ||||
|     z =  n        & 0x3f | ||||
|     r = x << 10 | y << 6 | z | ||||
|   elsif n <= 0xf7ffffff | ||||
|     w = (n >> 24) & 0x07 | ||||
|     x = (n >> 16) & 0x3f | ||||
|     y = (n >>  8) & 0x3f | ||||
|     z =  n        & 0x3f | ||||
|     r = w << 18 | x << 12 | y << 6 | z | ||||
|   end | ||||
|   r | ||||
| end | ||||
|  | ||||
| ### | ||||
| # Given a range, splits it up into ranges that can be continuously | ||||
| # encoded into utf8.  Eg: 0x00 .. 0xff => [0x00..0x7f, 0x80..0xff] | ||||
| # This is not strictly needed since the current [5.1] unicode standard | ||||
| # doesn't have ranges that straddle utf8 boundaries.  This is included | ||||
| # for completeness as there is no telling if that will ever change. | ||||
|  | ||||
| def utf8_ranges( range ) | ||||
|   ranges = [] | ||||
|   UTF8_BOUNDARIES.each do |max| | ||||
|     if range.begin <= max | ||||
|       if range.end <= max | ||||
|         ranges << range | ||||
|         return ranges | ||||
|       end | ||||
|  | ||||
|       ranges << (range.begin .. max) | ||||
|       range = (max + 1) .. range.end | ||||
|     end | ||||
|   end | ||||
|   ranges | ||||
| end | ||||
|  | ||||
| def build_range( start, stop ) | ||||
|   size = start.size/2 | ||||
|   left = size - 1 | ||||
|   return [""] if size < 1 | ||||
|  | ||||
|   a = start[0..1] | ||||
|   b = stop[0..1] | ||||
|  | ||||
|   ### | ||||
|   # Shared prefix | ||||
|  | ||||
|   if a == b | ||||
|     return build_range(start[2..-1], stop[2..-1]).map do |elt| | ||||
|       "0x#{a} " + elt | ||||
|     end | ||||
|   end | ||||
|  | ||||
|   ### | ||||
|   # Unshared prefix, end of run | ||||
|  | ||||
|   return ["0x#{a}..0x#{b} "] if left.zero? | ||||
|    | ||||
|   ### | ||||
|   # Unshared prefix, not end of run | ||||
|   # Range can be 0x123456..0x56789A | ||||
|   # Which is equivalent to: | ||||
|   #     0x123456 .. 0x12FFFF | ||||
|   #     0x130000 .. 0x55FFFF | ||||
|   #     0x560000 .. 0x56789A | ||||
|  | ||||
|   ret = [] | ||||
|   ret << build_range(start, a + "FF" * left) | ||||
|  | ||||
|   ### | ||||
|   # Only generate middle range if need be. | ||||
|  | ||||
|   if a.hex+1 != b.hex | ||||
|     max = to_hex(b.hex - 1) | ||||
|     max = "FF" if b == "FF" | ||||
|     ret << "0x#{to_hex(a.hex+1)}..0x#{max} " + "0x00..0xFF " * left | ||||
|   end | ||||
|  | ||||
|   ### | ||||
|   # Don't generate last range if it is covered by first range | ||||
|    | ||||
|   ret << build_range(b + "00" * left, stop) unless b == "FF" | ||||
|   ret.flatten! | ||||
| end | ||||
|  | ||||
| def to_utf8( range ) | ||||
|   utf8_ranges( range ).map do |r|    | ||||
|     begin_enc = to_utf8_enc(r.begin) | ||||
|     end_enc = to_utf8_enc(r.end) | ||||
|     build_range begin_enc, end_enc | ||||
|   end.flatten! | ||||
| end | ||||
|  | ||||
| ## | ||||
| # Perform a 3-way comparison of the number of codepoints advertised by | ||||
| # the unicode spec for the given range, the originally parsed range, | ||||
| # and the resulting utf8 encoded range. | ||||
|  | ||||
| def count_codepoints( code ) | ||||
|   code.split(' ').inject(1) do |acc, elt| | ||||
|     if elt =~ /0x(.+)\.\.0x(.+)/ | ||||
|       if @encoding == :utf8 | ||||
|         acc * (from_utf8_enc($2) - from_utf8_enc($1) + 1) | ||||
|       else | ||||
|         acc * ($2.hex - $1.hex + 1) | ||||
|       end | ||||
|     else | ||||
|       acc | ||||
|     end | ||||
|   end | ||||
| end | ||||
|  | ||||
| def is_valid?( range, desc, codes ) | ||||
|   spec_count  = 1 | ||||
|   spec_count  = $1.to_i if desc =~ /\[(\d+)\]/ | ||||
|   range_count = range.end - range.begin + 1 | ||||
|  | ||||
|   sum = codes.inject(0) { |acc, elt| acc + count_codepoints(elt) } | ||||
|   sum == spec_count and sum == range_count | ||||
| end | ||||
|  | ||||
| ## | ||||
| # Generate the state maching to stdout | ||||
|  | ||||
| def generate_machine( name, property ) | ||||
|   pipe = " " | ||||
|   @output.puts "    #{name} = " | ||||
|   each_alpha( @chart_url, property ) do |range, desc| | ||||
|  | ||||
|     codes = (@encoding == :ucs4) ? to_ucs4(range) : to_utf8(range) | ||||
|  | ||||
|     #raise "Invalid encoding of range #{range}: #{codes.inspect}" unless  | ||||
|     #  is_valid? range, desc, codes | ||||
|  | ||||
|     range_width = codes.map { |a| a.size }.max | ||||
|     range_width = RANGE_WIDTH if range_width < RANGE_WIDTH | ||||
|  | ||||
|     desc_width  = TOTAL_WIDTH - RANGE_WIDTH - 11 | ||||
|     desc_width -= (range_width - RANGE_WIDTH) if range_width > RANGE_WIDTH | ||||
|  | ||||
|     if desc.size > desc_width | ||||
|       desc = desc[0..desc_width - 4] + "..." | ||||
|     end | ||||
|  | ||||
|     codes.each_with_index do |r, idx| | ||||
|       desc = "" unless idx.zero? | ||||
|       code = "%-#{range_width}s" % r | ||||
|       @output.puts "      #{pipe} #{code} ##{desc}" | ||||
|       pipe = "|" | ||||
|     end | ||||
|   end | ||||
|   @output.puts "      ;" | ||||
|   @output.puts "" | ||||
| end | ||||
|  | ||||
| @output.puts <<EOF | ||||
| # The following Ragel file was autogenerated with #{$0}  | ||||
| # from: #{@chart_url} | ||||
| # | ||||
| # It defines #{properties}. | ||||
| # | ||||
| # To use this, make sure that your alphtype is set to #{ALPHTYPES[@encoding]}, | ||||
| # and that your input is in #{@encoding}. | ||||
|  | ||||
| %%{ | ||||
|     machine #{machine_name}; | ||||
|      | ||||
| EOF | ||||
|  | ||||
| properties.each { |x| generate_machine( x, x ) } | ||||
|  | ||||
| @output.puts <<EOF | ||||
| }%% | ||||
| EOF | ||||
							
								
								
									
										19
									
								
								vendor/github.com/apparentlymart/go-textseg/textseg/utf8_seqs.go
									
									
									
										generated
									
									
										vendored
									
									
								
							
							
						
						
									
										19
									
								
								vendor/github.com/apparentlymart/go-textseg/textseg/utf8_seqs.go
									
									
									
										generated
									
									
										vendored
									
									
								
							| @@ -1,19 +0,0 @@ | ||||
| package textseg | ||||
|  | ||||
| import "unicode/utf8" | ||||
|  | ||||
| // ScanGraphemeClusters is a split function for bufio.Scanner that splits | ||||
| // on UTF8 sequence boundaries. | ||||
| // | ||||
| // This is included largely for completeness, since this behavior is already | ||||
| // built in to Go when ranging over a string. | ||||
| func ScanUTF8Sequences(data []byte, atEOF bool) (int, []byte, error) { | ||||
| 	if len(data) == 0 { | ||||
| 		return 0, nil, nil | ||||
| 	} | ||||
| 	r, seqLen := utf8.DecodeRune(data) | ||||
| 	if r == utf8.RuneError && !atEOF { | ||||
| 		return 0, nil, nil | ||||
| 	} | ||||
| 	return seqLen, data[:seqLen], nil | ||||
| } | ||||
							
								
								
									
										202
									
								
								vendor/github.com/aws/aws-sdk-go/LICENSE.txt
									
									
									
										generated
									
									
										vendored
									
									
								
							
							
						
						
									
										202
									
								
								vendor/github.com/aws/aws-sdk-go/LICENSE.txt
									
									
									
										generated
									
									
										vendored
									
									
								
							| @@ -1,202 +0,0 @@ | ||||
|  | ||||
|                                  Apache License | ||||
|                            Version 2.0, January 2004 | ||||
|                         http://www.apache.org/licenses/ | ||||
|  | ||||
|    TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION | ||||
|  | ||||
|    1. Definitions. | ||||
|  | ||||
|       "License" shall mean the terms and conditions for use, reproduction, | ||||
|       and distribution as defined by Sections 1 through 9 of this document. | ||||
|  | ||||
|       "Licensor" shall mean the copyright owner or entity authorized by | ||||
|       the copyright owner that is granting the License. | ||||
|  | ||||
|       "Legal Entity" shall mean the union of the acting entity and all | ||||
|       other entities that control, are controlled by, or are under common | ||||
|       control with that entity. For the purposes of this definition, | ||||
|       "control" means (i) the power, direct or indirect, to cause the | ||||
|       direction or management of such entity, whether by contract or | ||||
|       otherwise, or (ii) ownership of fifty percent (50%) or more of the | ||||
|       outstanding shares, or (iii) beneficial ownership of such entity. | ||||
|  | ||||
|       "You" (or "Your") shall mean an individual or Legal Entity | ||||
|       exercising permissions granted by this License. | ||||
|  | ||||
|       "Source" form shall mean the preferred form for making modifications, | ||||
|       including but not limited to software source code, documentation | ||||
|       source, and configuration files. | ||||
|  | ||||
|       "Object" form shall mean any form resulting from mechanical | ||||
|       transformation or translation of a Source form, including but | ||||
|       not limited to compiled object code, generated documentation, | ||||
|       and conversions to other media types. | ||||
|  | ||||
|       "Work" shall mean the work of authorship, whether in Source or | ||||
|       Object form, made available under the License, as indicated by a | ||||
|       copyright notice that is included in or attached to the work | ||||
|       (an example is provided in the Appendix below). | ||||
|  | ||||
|       "Derivative Works" shall mean any work, whether in Source or Object | ||||
|       form, that is based on (or derived from) the Work and for which the | ||||
|       editorial revisions, annotations, elaborations, or other modifications | ||||
|       represent, as a whole, an original work of authorship. For the purposes | ||||
|       of this License, Derivative Works shall not include works that remain | ||||
|       separable from, or merely link (or bind by name) to the interfaces of, | ||||
|       the Work and Derivative Works thereof. | ||||
|  | ||||
|       "Contribution" shall mean any work of authorship, including | ||||
|       the original version of the Work and any modifications or additions | ||||
|       to that Work or Derivative Works thereof, that is intentionally | ||||
|       submitted to Licensor for inclusion in the Work by the copyright owner | ||||
|       or by an individual or Legal Entity authorized to submit on behalf of | ||||
|       the copyright owner. For the purposes of this definition, "submitted" | ||||
|       means any form of electronic, verbal, or written communication sent | ||||
|       to the Licensor or its representatives, including but not limited to | ||||
|       communication on electronic mailing lists, source code control systems, | ||||
|       and issue tracking systems that are managed by, or on behalf of, the | ||||
|       Licensor for the purpose of discussing and improving the Work, but | ||||
|       excluding communication that is conspicuously marked or otherwise | ||||
|       designated in writing by the copyright owner as "Not a Contribution." | ||||
|  | ||||
|       "Contributor" shall mean Licensor and any individual or Legal Entity | ||||
|       on behalf of whom a Contribution has been received by Licensor and | ||||
|       subsequently incorporated within the Work. | ||||
|  | ||||
|    2. Grant of Copyright License. Subject to the terms and conditions of | ||||
|       this License, each Contributor hereby grants to You a perpetual, | ||||
|       worldwide, non-exclusive, no-charge, royalty-free, irrevocable | ||||
|       copyright license to reproduce, prepare Derivative Works of, | ||||
|       publicly display, publicly perform, sublicense, and distribute the | ||||
|       Work and such Derivative Works in Source or Object form. | ||||
|  | ||||
|    3. Grant of Patent License. Subject to the terms and conditions of | ||||
|       this License, each Contributor hereby grants to You a perpetual, | ||||
|       worldwide, non-exclusive, no-charge, royalty-free, irrevocable | ||||
|       (except as stated in this section) patent license to make, have made, | ||||
|       use, offer to sell, sell, import, and otherwise transfer the Work, | ||||
|       where such license applies only to those patent claims licensable | ||||
|       by such Contributor that are necessarily infringed by their | ||||
|       Contribution(s) alone or by combination of their Contribution(s) | ||||
|       with the Work to which such Contribution(s) was submitted. If You | ||||
|       institute patent litigation against any entity (including a | ||||
|       cross-claim or counterclaim in a lawsuit) alleging that the Work | ||||
|       or a Contribution incorporated within the Work constitutes direct | ||||
|       or contributory patent infringement, then any patent licenses | ||||
|       granted to You under this License for that Work shall terminate | ||||
|       as of the date such litigation is filed. | ||||
|  | ||||
|    4. Redistribution. You may reproduce and distribute copies of the | ||||
|       Work or Derivative Works thereof in any medium, with or without | ||||
|       modifications, and in Source or Object form, provided that You | ||||
|       meet the following conditions: | ||||
|  | ||||
|       (a) You must give any other recipients of the Work or | ||||
|           Derivative Works a copy of this License; and | ||||
|  | ||||
|       (b) You must cause any modified files to carry prominent notices | ||||
|           stating that You changed the files; and | ||||
|  | ||||
|       (c) You must retain, in the Source form of any Derivative Works | ||||
|           that You distribute, all copyright, patent, trademark, and | ||||
|           attribution notices from the Source form of the Work, | ||||
|           excluding those notices that do not pertain to any part of | ||||
|           the Derivative Works; and | ||||
|  | ||||
|       (d) If the Work includes a "NOTICE" text file as part of its | ||||
|           distribution, then any Derivative Works that You distribute must | ||||
|           include a readable copy of the attribution notices contained | ||||
|           within such NOTICE file, excluding those notices that do not | ||||
|           pertain to any part of the Derivative Works, in at least one | ||||
|           of the following places: within a NOTICE text file distributed | ||||
|           as part of the Derivative Works; within the Source form or | ||||
|           documentation, if provided along with the Derivative Works; or, | ||||
|           within a display generated by the Derivative Works, if and | ||||
|           wherever such third-party notices normally appear. The contents | ||||
|           of the NOTICE file are for informational purposes only and | ||||
|           do not modify the License. You may add Your own attribution | ||||
|           notices within Derivative Works that You distribute, alongside | ||||
|           or as an addendum to the NOTICE text from the Work, provided | ||||
|           that such additional attribution notices cannot be construed | ||||
|           as modifying the License. | ||||
|  | ||||
|       You may add Your own copyright statement to Your modifications and | ||||
|       may provide additional or different license terms and conditions | ||||
|       for use, reproduction, or distribution of Your modifications, or | ||||
|       for any such Derivative Works as a whole, provided Your use, | ||||
|       reproduction, and distribution of the Work otherwise complies with | ||||
|       the conditions stated in this License. | ||||
|  | ||||
|    5. Submission of Contributions. Unless You explicitly state otherwise, | ||||
|       any Contribution intentionally submitted for inclusion in the Work | ||||
|       by You to the Licensor shall be under the terms and conditions of | ||||
|       this License, without any additional terms or conditions. | ||||
|       Notwithstanding the above, nothing herein shall supersede or modify | ||||
|       the terms of any separate license agreement you may have executed | ||||
|       with Licensor regarding such Contributions. | ||||
|  | ||||
|    6. Trademarks. This License does not grant permission to use the trade | ||||
|       names, trademarks, service marks, or product names of the Licensor, | ||||
|       except as required for reasonable and customary use in describing the | ||||
|       origin of the Work and reproducing the content of the NOTICE file. | ||||
|  | ||||
|    7. Disclaimer of Warranty. Unless required by applicable law or | ||||
|       agreed to in writing, Licensor provides the Work (and each | ||||
|       Contributor provides its Contributions) on an "AS IS" BASIS, | ||||
|       WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or | ||||
|       implied, including, without limitation, any warranties or conditions | ||||
|       of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A | ||||
|       PARTICULAR PURPOSE. You are solely responsible for determining the | ||||
|       appropriateness of using or redistributing the Work and assume any | ||||
|       risks associated with Your exercise of permissions under this License. | ||||
|  | ||||
|    8. Limitation of Liability. In no event and under no legal theory, | ||||
|       whether in tort (including negligence), contract, or otherwise, | ||||
|       unless required by applicable law (such as deliberate and grossly | ||||
|       negligent acts) or agreed to in writing, shall any Contributor be | ||||
|       liable to You for damages, including any direct, indirect, special, | ||||
|       incidental, or consequential damages of any character arising as a | ||||
|       result of this License or out of the use or inability to use the | ||||
|       Work (including but not limited to damages for loss of goodwill, | ||||
|       work stoppage, computer failure or malfunction, or any and all | ||||
|       other commercial damages or losses), even if such Contributor | ||||
|       has been advised of the possibility of such damages. | ||||
|  | ||||
|    9. Accepting Warranty or Additional Liability. While redistributing | ||||
|       the Work or Derivative Works thereof, You may choose to offer, | ||||
|       and charge a fee for, acceptance of support, warranty, indemnity, | ||||
|       or other liability obligations and/or rights consistent with this | ||||
|       License. However, in accepting such obligations, You may act only | ||||
|       on Your own behalf and on Your sole responsibility, not on behalf | ||||
|       of any other Contributor, and only if You agree to indemnify, | ||||
|       defend, and hold each Contributor harmless for any liability | ||||
|       incurred by, or claims asserted against, such Contributor by reason | ||||
|       of your accepting any such warranty or additional liability. | ||||
|  | ||||
|    END OF TERMS AND CONDITIONS | ||||
|  | ||||
|    APPENDIX: How to apply the Apache License to your work. | ||||
|  | ||||
|       To apply the Apache License to your work, attach the following | ||||
|       boilerplate notice, with the fields enclosed by brackets "[]" | ||||
|       replaced with your own identifying information. (Don't include | ||||
|       the brackets!)  The text should be enclosed in the appropriate | ||||
|       comment syntax for the file format. We also recommend that a | ||||
|       file or class name and description of purpose be included on the | ||||
|       same "printed page" as the copyright notice for easier | ||||
|       identification within third-party archives. | ||||
|  | ||||
|    Copyright [yyyy] [name of copyright owner] | ||||
|  | ||||
|    Licensed under the Apache License, Version 2.0 (the "License"); | ||||
|    you may not use this file except in compliance with the License. | ||||
|    You may obtain a copy of the License at | ||||
|  | ||||
|        http://www.apache.org/licenses/LICENSE-2.0 | ||||
|  | ||||
|    Unless required by applicable law or agreed to in writing, software | ||||
|    distributed under the License is distributed on an "AS IS" BASIS, | ||||
|    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||||
|    See the License for the specific language governing permissions and | ||||
|    limitations under the License. | ||||
							
								
								
									
										3
									
								
								vendor/github.com/aws/aws-sdk-go/NOTICE.txt
									
									
									
										generated
									
									
										vendored
									
									
								
							
							
						
						
									
										3
									
								
								vendor/github.com/aws/aws-sdk-go/NOTICE.txt
									
									
									
										generated
									
									
										vendored
									
									
								
							| @@ -1,3 +0,0 @@ | ||||
| AWS SDK for Go | ||||
| Copyright 2015 Amazon.com, Inc. or its affiliates. All Rights Reserved. | ||||
| Copyright 2014-2015 Stripe, Inc. | ||||
							
								
								
									
										164
									
								
								vendor/github.com/aws/aws-sdk-go/aws/awserr/error.go
									
									
									
										generated
									
									
										vendored
									
									
								
							
							
						
						
									
										164
									
								
								vendor/github.com/aws/aws-sdk-go/aws/awserr/error.go
									
									
									
										generated
									
									
										vendored
									
									
								
							| @@ -1,164 +0,0 @@ | ||||
| // Package awserr represents API error interface accessors for the SDK. | ||||
| package awserr | ||||
|  | ||||
| // An Error wraps lower level errors with code, message and an original error. | ||||
| // The underlying concrete error type may also satisfy other interfaces which | ||||
| // can be to used to obtain more specific information about the error. | ||||
| // | ||||
| // Calling Error() or String() will always include the full information about | ||||
| // an error based on its underlying type. | ||||
| // | ||||
| // Example: | ||||
| // | ||||
| //     output, err := s3manage.Upload(svc, input, opts) | ||||
| //     if err != nil { | ||||
| //         if awsErr, ok := err.(awserr.Error); ok { | ||||
| //             // Get error details | ||||
| //             log.Println("Error:", awsErr.Code(), awsErr.Message()) | ||||
| // | ||||
| //             // Prints out full error message, including original error if there was one. | ||||
| //             log.Println("Error:", awsErr.Error()) | ||||
| // | ||||
| //             // Get original error | ||||
| //             if origErr := awsErr.OrigErr(); origErr != nil { | ||||
| //                 // operate on original error. | ||||
| //             } | ||||
| //         } else { | ||||
| //             fmt.Println(err.Error()) | ||||
| //         } | ||||
| //     } | ||||
| // | ||||
| type Error interface { | ||||
| 	// Satisfy the generic error interface. | ||||
| 	error | ||||
|  | ||||
| 	// Returns the short phrase depicting the classification of the error. | ||||
| 	Code() string | ||||
|  | ||||
| 	// Returns the error details message. | ||||
| 	Message() string | ||||
|  | ||||
| 	// Returns the original error if one was set.  Nil is returned if not set. | ||||
| 	OrigErr() error | ||||
| } | ||||
|  | ||||
| // BatchError is a batch of errors which also wraps lower level errors with | ||||
| // code, message, and original errors. Calling Error() will include all errors | ||||
| // that occurred in the batch. | ||||
| // | ||||
| // Deprecated: Replaced with BatchedErrors. Only defined for backwards | ||||
| // compatibility. | ||||
| type BatchError interface { | ||||
| 	// Satisfy the generic error interface. | ||||
| 	error | ||||
|  | ||||
| 	// Returns the short phrase depicting the classification of the error. | ||||
| 	Code() string | ||||
|  | ||||
| 	// Returns the error details message. | ||||
| 	Message() string | ||||
|  | ||||
| 	// Returns the original error if one was set.  Nil is returned if not set. | ||||
| 	OrigErrs() []error | ||||
| } | ||||
|  | ||||
| // BatchedErrors is a batch of errors which also wraps lower level errors with | ||||
| // code, message, and original errors. Calling Error() will include all errors | ||||
| // that occurred in the batch. | ||||
| // | ||||
| // Replaces BatchError | ||||
| type BatchedErrors interface { | ||||
| 	// Satisfy the base Error interface. | ||||
| 	Error | ||||
|  | ||||
| 	// Returns the original error if one was set.  Nil is returned if not set. | ||||
| 	OrigErrs() []error | ||||
| } | ||||
|  | ||||
| // New returns an Error object described by the code, message, and origErr. | ||||
| // | ||||
| // If origErr satisfies the Error interface it will not be wrapped within a new | ||||
| // Error object and will instead be returned. | ||||
| func New(code, message string, origErr error) Error { | ||||
| 	var errs []error | ||||
| 	if origErr != nil { | ||||
| 		errs = append(errs, origErr) | ||||
| 	} | ||||
| 	return newBaseError(code, message, errs) | ||||
| } | ||||
|  | ||||
| // NewBatchError returns an BatchedErrors with a collection of errors as an | ||||
| // array of errors. | ||||
| func NewBatchError(code, message string, errs []error) BatchedErrors { | ||||
| 	return newBaseError(code, message, errs) | ||||
| } | ||||
|  | ||||
| // A RequestFailure is an interface to extract request failure information from | ||||
| // an Error such as the request ID of the failed request returned by a service. | ||||
| // RequestFailures may not always have a requestID value if the request failed | ||||
| // prior to reaching the service such as a connection error. | ||||
| // | ||||
| // Example: | ||||
| // | ||||
| //     output, err := s3manage.Upload(svc, input, opts) | ||||
| //     if err != nil { | ||||
| //         if reqerr, ok := err.(RequestFailure); ok { | ||||
| //             log.Println("Request failed", reqerr.Code(), reqerr.Message(), reqerr.RequestID()) | ||||
| //         } else { | ||||
| //             log.Println("Error:", err.Error()) | ||||
| //         } | ||||
| //     } | ||||
| // | ||||
| // Combined with awserr.Error: | ||||
| // | ||||
| //    output, err := s3manage.Upload(svc, input, opts) | ||||
| //    if err != nil { | ||||
| //        if awsErr, ok := err.(awserr.Error); ok { | ||||
| //            // Generic AWS Error with Code, Message, and original error (if any) | ||||
| //            fmt.Println(awsErr.Code(), awsErr.Message(), awsErr.OrigErr()) | ||||
| // | ||||
| //            if reqErr, ok := err.(awserr.RequestFailure); ok { | ||||
| //                // A service error occurred | ||||
| //                fmt.Println(reqErr.StatusCode(), reqErr.RequestID()) | ||||
| //            } | ||||
| //        } else { | ||||
| //            fmt.Println(err.Error()) | ||||
| //        } | ||||
| //    } | ||||
| // | ||||
| type RequestFailure interface { | ||||
| 	Error | ||||
|  | ||||
| 	// The status code of the HTTP response. | ||||
| 	StatusCode() int | ||||
|  | ||||
| 	// The request ID returned by the service for a request failure. This will | ||||
| 	// be empty if no request ID is available such as the request failed due | ||||
| 	// to a connection error. | ||||
| 	RequestID() string | ||||
| } | ||||
|  | ||||
| // NewRequestFailure returns a wrapped error with additional information for | ||||
| // request status code, and service requestID. | ||||
| // | ||||
| // Should be used to wrap all request which involve service requests. Even if | ||||
| // the request failed without a service response, but had an HTTP status code | ||||
| // that may be meaningful. | ||||
| func NewRequestFailure(err Error, statusCode int, reqID string) RequestFailure { | ||||
| 	return newRequestError(err, statusCode, reqID) | ||||
| } | ||||
|  | ||||
| // UnmarshalError provides the interface for the SDK failing to unmarshal data. | ||||
| type UnmarshalError interface { | ||||
| 	awsError | ||||
| 	Bytes() []byte | ||||
| } | ||||
|  | ||||
| // NewUnmarshalError returns an initialized UnmarshalError error wrapper adding | ||||
| // the bytes that fail to unmarshal to the error. | ||||
| func NewUnmarshalError(err error, msg string, bytes []byte) UnmarshalError { | ||||
| 	return &unmarshalError{ | ||||
| 		awsError: New("UnmarshalError", msg, err), | ||||
| 		bytes:    bytes, | ||||
| 	} | ||||
| } | ||||
							
								
								
									
										221
									
								
								vendor/github.com/aws/aws-sdk-go/aws/awserr/types.go
									
									
									
										generated
									
									
										vendored
									
									
								
							
							
						
						
									
										221
									
								
								vendor/github.com/aws/aws-sdk-go/aws/awserr/types.go
									
									
									
										generated
									
									
										vendored
									
									
								
							| @@ -1,221 +0,0 @@ | ||||
| package awserr | ||||
|  | ||||
| import ( | ||||
| 	"encoding/hex" | ||||
| 	"fmt" | ||||
| ) | ||||
|  | ||||
| // SprintError returns a string of the formatted error code. | ||||
| // | ||||
| // Both extra and origErr are optional.  If they are included their lines | ||||
| // will be added, but if they are not included their lines will be ignored. | ||||
| func SprintError(code, message, extra string, origErr error) string { | ||||
| 	msg := fmt.Sprintf("%s: %s", code, message) | ||||
| 	if extra != "" { | ||||
| 		msg = fmt.Sprintf("%s\n\t%s", msg, extra) | ||||
| 	} | ||||
| 	if origErr != nil { | ||||
| 		msg = fmt.Sprintf("%s\ncaused by: %s", msg, origErr.Error()) | ||||
| 	} | ||||
| 	return msg | ||||
| } | ||||
|  | ||||
| // A baseError wraps the code and message which defines an error. It also | ||||
| // can be used to wrap an original error object. | ||||
| // | ||||
| // Should be used as the root for errors satisfying the awserr.Error. Also | ||||
| // for any error which does not fit into a specific error wrapper type. | ||||
| type baseError struct { | ||||
| 	// Classification of error | ||||
| 	code string | ||||
|  | ||||
| 	// Detailed information about error | ||||
| 	message string | ||||
|  | ||||
| 	// Optional original error this error is based off of. Allows building | ||||
| 	// chained errors. | ||||
| 	errs []error | ||||
| } | ||||
|  | ||||
| // newBaseError returns an error object for the code, message, and errors. | ||||
| // | ||||
| // code is a short no whitespace phrase depicting the classification of | ||||
| // the error that is being created. | ||||
| // | ||||
| // message is the free flow string containing detailed information about the | ||||
| // error. | ||||
| // | ||||
| // origErrs is the error objects which will be nested under the new errors to | ||||
| // be returned. | ||||
| func newBaseError(code, message string, origErrs []error) *baseError { | ||||
| 	b := &baseError{ | ||||
| 		code:    code, | ||||
| 		message: message, | ||||
| 		errs:    origErrs, | ||||
| 	} | ||||
|  | ||||
| 	return b | ||||
| } | ||||
|  | ||||
| // Error returns the string representation of the error. | ||||
| // | ||||
| // See ErrorWithExtra for formatting. | ||||
| // | ||||
| // Satisfies the error interface. | ||||
| func (b baseError) Error() string { | ||||
| 	size := len(b.errs) | ||||
| 	if size > 0 { | ||||
| 		return SprintError(b.code, b.message, "", errorList(b.errs)) | ||||
| 	} | ||||
|  | ||||
| 	return SprintError(b.code, b.message, "", nil) | ||||
| } | ||||
|  | ||||
| // String returns the string representation of the error. | ||||
| // Alias for Error to satisfy the stringer interface. | ||||
| func (b baseError) String() string { | ||||
| 	return b.Error() | ||||
| } | ||||
|  | ||||
| // Code returns the short phrase depicting the classification of the error. | ||||
| func (b baseError) Code() string { | ||||
| 	return b.code | ||||
| } | ||||
|  | ||||
| // Message returns the error details message. | ||||
| func (b baseError) Message() string { | ||||
| 	return b.message | ||||
| } | ||||
|  | ||||
| // OrigErr returns the original error if one was set. Nil is returned if no | ||||
| // error was set. This only returns the first element in the list. If the full | ||||
| // list is needed, use BatchedErrors. | ||||
| func (b baseError) OrigErr() error { | ||||
| 	switch len(b.errs) { | ||||
| 	case 0: | ||||
| 		return nil | ||||
| 	case 1: | ||||
| 		return b.errs[0] | ||||
| 	default: | ||||
| 		if err, ok := b.errs[0].(Error); ok { | ||||
| 			return NewBatchError(err.Code(), err.Message(), b.errs[1:]) | ||||
| 		} | ||||
| 		return NewBatchError("BatchedErrors", | ||||
| 			"multiple errors occurred", b.errs) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // OrigErrs returns the original errors if one was set. An empty slice is | ||||
| // returned if no error was set. | ||||
| func (b baseError) OrigErrs() []error { | ||||
| 	return b.errs | ||||
| } | ||||
|  | ||||
| // So that the Error interface type can be included as an anonymous field | ||||
| // in the requestError struct and not conflict with the error.Error() method. | ||||
| type awsError Error | ||||
|  | ||||
| // A requestError wraps a request or service error. | ||||
| // | ||||
| // Composed of baseError for code, message, and original error. | ||||
| type requestError struct { | ||||
| 	awsError | ||||
| 	statusCode int | ||||
| 	requestID  string | ||||
| 	bytes      []byte | ||||
| } | ||||
|  | ||||
| // newRequestError returns a wrapped error with additional information for | ||||
| // request status code, and service requestID. | ||||
| // | ||||
| // Should be used to wrap all request which involve service requests. Even if | ||||
| // the request failed without a service response, but had an HTTP status code | ||||
| // that may be meaningful. | ||||
| // | ||||
| // Also wraps original errors via the baseError. | ||||
| func newRequestError(err Error, statusCode int, requestID string) *requestError { | ||||
| 	return &requestError{ | ||||
| 		awsError:   err, | ||||
| 		statusCode: statusCode, | ||||
| 		requestID:  requestID, | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // Error returns the string representation of the error. | ||||
| // Satisfies the error interface. | ||||
| func (r requestError) Error() string { | ||||
| 	extra := fmt.Sprintf("status code: %d, request id: %s", | ||||
| 		r.statusCode, r.requestID) | ||||
| 	return SprintError(r.Code(), r.Message(), extra, r.OrigErr()) | ||||
| } | ||||
|  | ||||
| // String returns the string representation of the error. | ||||
| // Alias for Error to satisfy the stringer interface. | ||||
| func (r requestError) String() string { | ||||
| 	return r.Error() | ||||
| } | ||||
|  | ||||
| // StatusCode returns the wrapped status code for the error | ||||
| func (r requestError) StatusCode() int { | ||||
| 	return r.statusCode | ||||
| } | ||||
|  | ||||
| // RequestID returns the wrapped requestID | ||||
| func (r requestError) RequestID() string { | ||||
| 	return r.requestID | ||||
| } | ||||
|  | ||||
| // OrigErrs returns the original errors if one was set. An empty slice is | ||||
| // returned if no error was set. | ||||
| func (r requestError) OrigErrs() []error { | ||||
| 	if b, ok := r.awsError.(BatchedErrors); ok { | ||||
| 		return b.OrigErrs() | ||||
| 	} | ||||
| 	return []error{r.OrigErr()} | ||||
| } | ||||
|  | ||||
| type unmarshalError struct { | ||||
| 	awsError | ||||
| 	bytes []byte | ||||
| } | ||||
|  | ||||
| // Error returns the string representation of the error. | ||||
| // Satisfies the error interface. | ||||
| func (e unmarshalError) Error() string { | ||||
| 	extra := hex.Dump(e.bytes) | ||||
| 	return SprintError(e.Code(), e.Message(), extra, e.OrigErr()) | ||||
| } | ||||
|  | ||||
| // String returns the string representation of the error. | ||||
| // Alias for Error to satisfy the stringer interface. | ||||
| func (e unmarshalError) String() string { | ||||
| 	return e.Error() | ||||
| } | ||||
|  | ||||
| // Bytes returns the bytes that failed to unmarshal. | ||||
| func (e unmarshalError) Bytes() []byte { | ||||
| 	return e.bytes | ||||
| } | ||||
|  | ||||
| // An error list that satisfies the golang interface | ||||
| type errorList []error | ||||
|  | ||||
| // Error returns the string representation of the error. | ||||
| // | ||||
| // Satisfies the error interface. | ||||
| func (e errorList) Error() string { | ||||
| 	msg := "" | ||||
| 	// How do we want to handle the array size being zero | ||||
| 	if size := len(e); size > 0 { | ||||
| 		for i := 0; i < size; i++ { | ||||
| 			msg += e[i].Error() | ||||
| 			// We check the next index to see if it is within the slice. | ||||
| 			// If it is, then we append a newline. We do this, because unit tests | ||||
| 			// could be broken with the additional '\n' | ||||
| 			if i+1 < size { | ||||
| 				msg += "\n" | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| 	return msg | ||||
| } | ||||
							
								
								
									
										108
									
								
								vendor/github.com/aws/aws-sdk-go/aws/awsutil/copy.go
									
									
									
										generated
									
									
										vendored
									
									
								
							
							
						
						
									
										108
									
								
								vendor/github.com/aws/aws-sdk-go/aws/awsutil/copy.go
									
									
									
										generated
									
									
										vendored
									
									
								
							| @@ -1,108 +0,0 @@ | ||||
| package awsutil | ||||
|  | ||||
| import ( | ||||
| 	"io" | ||||
| 	"reflect" | ||||
| 	"time" | ||||
| ) | ||||
|  | ||||
| // Copy deeply copies a src structure to dst. Useful for copying request and | ||||
| // response structures. | ||||
| // | ||||
| // Can copy between structs of different type, but will only copy fields which | ||||
| // are assignable, and exist in both structs. Fields which are not assignable, | ||||
| // or do not exist in both structs are ignored. | ||||
| func Copy(dst, src interface{}) { | ||||
| 	dstval := reflect.ValueOf(dst) | ||||
| 	if !dstval.IsValid() { | ||||
| 		panic("Copy dst cannot be nil") | ||||
| 	} | ||||
|  | ||||
| 	rcopy(dstval, reflect.ValueOf(src), true) | ||||
| } | ||||
|  | ||||
| // CopyOf returns a copy of src while also allocating the memory for dst. | ||||
| // src must be a pointer type or this operation will fail. | ||||
| func CopyOf(src interface{}) (dst interface{}) { | ||||
| 	dsti := reflect.New(reflect.TypeOf(src).Elem()) | ||||
| 	dst = dsti.Interface() | ||||
| 	rcopy(dsti, reflect.ValueOf(src), true) | ||||
| 	return | ||||
| } | ||||
|  | ||||
| // rcopy performs a recursive copy of values from the source to destination. | ||||
| // | ||||
| // root is used to skip certain aspects of the copy which are not valid | ||||
| // for the root node of a object. | ||||
| func rcopy(dst, src reflect.Value, root bool) { | ||||
| 	if !src.IsValid() { | ||||
| 		return | ||||
| 	} | ||||
|  | ||||
| 	switch src.Kind() { | ||||
| 	case reflect.Ptr: | ||||
| 		if _, ok := src.Interface().(io.Reader); ok { | ||||
| 			if dst.Kind() == reflect.Ptr && dst.Elem().CanSet() { | ||||
| 				dst.Elem().Set(src) | ||||
| 			} else if dst.CanSet() { | ||||
| 				dst.Set(src) | ||||
| 			} | ||||
| 		} else { | ||||
| 			e := src.Type().Elem() | ||||
| 			if dst.CanSet() && !src.IsNil() { | ||||
| 				if _, ok := src.Interface().(*time.Time); !ok { | ||||
| 					dst.Set(reflect.New(e)) | ||||
| 				} else { | ||||
| 					tempValue := reflect.New(e) | ||||
| 					tempValue.Elem().Set(src.Elem()) | ||||
| 					// Sets time.Time's unexported values | ||||
| 					dst.Set(tempValue) | ||||
| 				} | ||||
| 			} | ||||
| 			if src.Elem().IsValid() { | ||||
| 				// Keep the current root state since the depth hasn't changed | ||||
| 				rcopy(dst.Elem(), src.Elem(), root) | ||||
| 			} | ||||
| 		} | ||||
| 	case reflect.Struct: | ||||
| 		t := dst.Type() | ||||
| 		for i := 0; i < t.NumField(); i++ { | ||||
| 			name := t.Field(i).Name | ||||
| 			srcVal := src.FieldByName(name) | ||||
| 			dstVal := dst.FieldByName(name) | ||||
| 			if srcVal.IsValid() && dstVal.CanSet() { | ||||
| 				rcopy(dstVal, srcVal, false) | ||||
| 			} | ||||
| 		} | ||||
| 	case reflect.Slice: | ||||
| 		if src.IsNil() { | ||||
| 			break | ||||
| 		} | ||||
|  | ||||
| 		s := reflect.MakeSlice(src.Type(), src.Len(), src.Cap()) | ||||
| 		dst.Set(s) | ||||
| 		for i := 0; i < src.Len(); i++ { | ||||
| 			rcopy(dst.Index(i), src.Index(i), false) | ||||
| 		} | ||||
| 	case reflect.Map: | ||||
| 		if src.IsNil() { | ||||
| 			break | ||||
| 		} | ||||
|  | ||||
| 		s := reflect.MakeMap(src.Type()) | ||||
| 		dst.Set(s) | ||||
| 		for _, k := range src.MapKeys() { | ||||
| 			v := src.MapIndex(k) | ||||
| 			v2 := reflect.New(v.Type()).Elem() | ||||
| 			rcopy(v2, v, false) | ||||
| 			dst.SetMapIndex(k, v2) | ||||
| 		} | ||||
| 	default: | ||||
| 		// Assign the value if possible. If its not assignable, the value would | ||||
| 		// need to be converted and the impact of that may be unexpected, or is | ||||
| 		// not compatible with the dst type. | ||||
| 		if src.Type().AssignableTo(dst.Type()) { | ||||
| 			dst.Set(src) | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
							
								
								
									
										27
									
								
								vendor/github.com/aws/aws-sdk-go/aws/awsutil/equal.go
									
									
									
										generated
									
									
										vendored
									
									
								
							
							
						
						
									
										27
									
								
								vendor/github.com/aws/aws-sdk-go/aws/awsutil/equal.go
									
									
									
										generated
									
									
										vendored
									
									
								
							| @@ -1,27 +0,0 @@ | ||||
| package awsutil | ||||
|  | ||||
| import ( | ||||
| 	"reflect" | ||||
| ) | ||||
|  | ||||
| // DeepEqual returns if the two values are deeply equal like reflect.DeepEqual. | ||||
| // In addition to this, this method will also dereference the input values if | ||||
| // possible so the DeepEqual performed will not fail if one parameter is a | ||||
| // pointer and the other is not. | ||||
| // | ||||
| // DeepEqual will not perform indirection of nested values of the input parameters. | ||||
| func DeepEqual(a, b interface{}) bool { | ||||
| 	ra := reflect.Indirect(reflect.ValueOf(a)) | ||||
| 	rb := reflect.Indirect(reflect.ValueOf(b)) | ||||
|  | ||||
| 	if raValid, rbValid := ra.IsValid(), rb.IsValid(); !raValid && !rbValid { | ||||
| 		// If the elements are both nil, and of the same type they are equal | ||||
| 		// If they are of different types they are not equal | ||||
| 		return reflect.TypeOf(a) == reflect.TypeOf(b) | ||||
| 	} else if raValid != rbValid { | ||||
| 		// Both values must be valid to be equal | ||||
| 		return false | ||||
| 	} | ||||
|  | ||||
| 	return reflect.DeepEqual(ra.Interface(), rb.Interface()) | ||||
| } | ||||
							
								
								
									
										221
									
								
								vendor/github.com/aws/aws-sdk-go/aws/awsutil/path_value.go
									
									
									
										generated
									
									
										vendored
									
									
								
							
							
						
						
									
										221
									
								
								vendor/github.com/aws/aws-sdk-go/aws/awsutil/path_value.go
									
									
									
										generated
									
									
										vendored
									
									
								
							| @@ -1,221 +0,0 @@ | ||||
| package awsutil | ||||
|  | ||||
| import ( | ||||
| 	"reflect" | ||||
| 	"regexp" | ||||
| 	"strconv" | ||||
| 	"strings" | ||||
|  | ||||
| 	"github.com/jmespath/go-jmespath" | ||||
| ) | ||||
|  | ||||
| var indexRe = regexp.MustCompile(`(.+)\[(-?\d+)?\]$`) | ||||
|  | ||||
| // rValuesAtPath returns a slice of values found in value v. The values | ||||
| // in v are explored recursively so all nested values are collected. | ||||
| func rValuesAtPath(v interface{}, path string, createPath, caseSensitive, nilTerm bool) []reflect.Value { | ||||
| 	pathparts := strings.Split(path, "||") | ||||
| 	if len(pathparts) > 1 { | ||||
| 		for _, pathpart := range pathparts { | ||||
| 			vals := rValuesAtPath(v, pathpart, createPath, caseSensitive, nilTerm) | ||||
| 			if len(vals) > 0 { | ||||
| 				return vals | ||||
| 			} | ||||
| 		} | ||||
| 		return nil | ||||
| 	} | ||||
|  | ||||
| 	values := []reflect.Value{reflect.Indirect(reflect.ValueOf(v))} | ||||
| 	components := strings.Split(path, ".") | ||||
| 	for len(values) > 0 && len(components) > 0 { | ||||
| 		var index *int64 | ||||
| 		var indexStar bool | ||||
| 		c := strings.TrimSpace(components[0]) | ||||
| 		if c == "" { // no actual component, illegal syntax | ||||
| 			return nil | ||||
| 		} else if caseSensitive && c != "*" && strings.ToLower(c[0:1]) == c[0:1] { | ||||
| 			// TODO normalize case for user | ||||
| 			return nil // don't support unexported fields | ||||
| 		} | ||||
|  | ||||
| 		// parse this component | ||||
| 		if m := indexRe.FindStringSubmatch(c); m != nil { | ||||
| 			c = m[1] | ||||
| 			if m[2] == "" { | ||||
| 				index = nil | ||||
| 				indexStar = true | ||||
| 			} else { | ||||
| 				i, _ := strconv.ParseInt(m[2], 10, 32) | ||||
| 				index = &i | ||||
| 				indexStar = false | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		nextvals := []reflect.Value{} | ||||
| 		for _, value := range values { | ||||
| 			// pull component name out of struct member | ||||
| 			if value.Kind() != reflect.Struct { | ||||
| 				continue | ||||
| 			} | ||||
|  | ||||
| 			if c == "*" { // pull all members | ||||
| 				for i := 0; i < value.NumField(); i++ { | ||||
| 					if f := reflect.Indirect(value.Field(i)); f.IsValid() { | ||||
| 						nextvals = append(nextvals, f) | ||||
| 					} | ||||
| 				} | ||||
| 				continue | ||||
| 			} | ||||
|  | ||||
| 			value = value.FieldByNameFunc(func(name string) bool { | ||||
| 				if c == name { | ||||
| 					return true | ||||
| 				} else if !caseSensitive && strings.ToLower(name) == strings.ToLower(c) { | ||||
| 					return true | ||||
| 				} | ||||
| 				return false | ||||
| 			}) | ||||
|  | ||||
| 			if nilTerm && value.Kind() == reflect.Ptr && len(components[1:]) == 0 { | ||||
| 				if !value.IsNil() { | ||||
| 					value.Set(reflect.Zero(value.Type())) | ||||
| 				} | ||||
| 				return []reflect.Value{value} | ||||
| 			} | ||||
|  | ||||
| 			if createPath && value.Kind() == reflect.Ptr && value.IsNil() { | ||||
| 				// TODO if the value is the terminus it should not be created | ||||
| 				// if the value to be set to its position is nil. | ||||
| 				value.Set(reflect.New(value.Type().Elem())) | ||||
| 				value = value.Elem() | ||||
| 			} else { | ||||
| 				value = reflect.Indirect(value) | ||||
| 			} | ||||
|  | ||||
| 			if value.Kind() == reflect.Slice || value.Kind() == reflect.Map { | ||||
| 				if !createPath && value.IsNil() { | ||||
| 					value = reflect.ValueOf(nil) | ||||
| 				} | ||||
| 			} | ||||
|  | ||||
| 			if value.IsValid() { | ||||
| 				nextvals = append(nextvals, value) | ||||
| 			} | ||||
| 		} | ||||
| 		values = nextvals | ||||
|  | ||||
| 		if indexStar || index != nil { | ||||
| 			nextvals = []reflect.Value{} | ||||
| 			for _, valItem := range values { | ||||
| 				value := reflect.Indirect(valItem) | ||||
| 				if value.Kind() != reflect.Slice { | ||||
| 					continue | ||||
| 				} | ||||
|  | ||||
| 				if indexStar { // grab all indices | ||||
| 					for i := 0; i < value.Len(); i++ { | ||||
| 						idx := reflect.Indirect(value.Index(i)) | ||||
| 						if idx.IsValid() { | ||||
| 							nextvals = append(nextvals, idx) | ||||
| 						} | ||||
| 					} | ||||
| 					continue | ||||
| 				} | ||||
|  | ||||
| 				// pull out index | ||||
| 				i := int(*index) | ||||
| 				if i >= value.Len() { // check out of bounds | ||||
| 					if createPath { | ||||
| 						// TODO resize slice | ||||
| 					} else { | ||||
| 						continue | ||||
| 					} | ||||
| 				} else if i < 0 { // support negative indexing | ||||
| 					i = value.Len() + i | ||||
| 				} | ||||
| 				value = reflect.Indirect(value.Index(i)) | ||||
|  | ||||
| 				if value.Kind() == reflect.Slice || value.Kind() == reflect.Map { | ||||
| 					if !createPath && value.IsNil() { | ||||
| 						value = reflect.ValueOf(nil) | ||||
| 					} | ||||
| 				} | ||||
|  | ||||
| 				if value.IsValid() { | ||||
| 					nextvals = append(nextvals, value) | ||||
| 				} | ||||
| 			} | ||||
| 			values = nextvals | ||||
| 		} | ||||
|  | ||||
| 		components = components[1:] | ||||
| 	} | ||||
| 	return values | ||||
| } | ||||
|  | ||||
| // ValuesAtPath returns a list of values at the case insensitive lexical | ||||
| // path inside of a structure. | ||||
| func ValuesAtPath(i interface{}, path string) ([]interface{}, error) { | ||||
| 	result, err := jmespath.Search(path, i) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
|  | ||||
| 	v := reflect.ValueOf(result) | ||||
| 	if !v.IsValid() || (v.Kind() == reflect.Ptr && v.IsNil()) { | ||||
| 		return nil, nil | ||||
| 	} | ||||
| 	if s, ok := result.([]interface{}); ok { | ||||
| 		return s, err | ||||
| 	} | ||||
| 	if v.Kind() == reflect.Map && v.Len() == 0 { | ||||
| 		return nil, nil | ||||
| 	} | ||||
| 	if v.Kind() == reflect.Slice { | ||||
| 		out := make([]interface{}, v.Len()) | ||||
| 		for i := 0; i < v.Len(); i++ { | ||||
| 			out[i] = v.Index(i).Interface() | ||||
| 		} | ||||
| 		return out, nil | ||||
| 	} | ||||
|  | ||||
| 	return []interface{}{result}, nil | ||||
| } | ||||
|  | ||||
| // SetValueAtPath sets a value at the case insensitive lexical path inside | ||||
| // of a structure. | ||||
| func SetValueAtPath(i interface{}, path string, v interface{}) { | ||||
| 	rvals := rValuesAtPath(i, path, true, false, v == nil) | ||||
| 	for _, rval := range rvals { | ||||
| 		if rval.Kind() == reflect.Ptr && rval.IsNil() { | ||||
| 			continue | ||||
| 		} | ||||
| 		setValue(rval, v) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func setValue(dstVal reflect.Value, src interface{}) { | ||||
| 	if dstVal.Kind() == reflect.Ptr { | ||||
| 		dstVal = reflect.Indirect(dstVal) | ||||
| 	} | ||||
| 	srcVal := reflect.ValueOf(src) | ||||
|  | ||||
| 	if !srcVal.IsValid() { // src is literal nil | ||||
| 		if dstVal.CanAddr() { | ||||
| 			// Convert to pointer so that pointer's value can be nil'ed | ||||
| 			//                     dstVal = dstVal.Addr() | ||||
| 		} | ||||
| 		dstVal.Set(reflect.Zero(dstVal.Type())) | ||||
|  | ||||
| 	} else if srcVal.Kind() == reflect.Ptr { | ||||
| 		if srcVal.IsNil() { | ||||
| 			srcVal = reflect.Zero(dstVal.Type()) | ||||
| 		} else { | ||||
| 			srcVal = reflect.ValueOf(src).Elem() | ||||
| 		} | ||||
| 		dstVal.Set(srcVal) | ||||
| 	} else { | ||||
| 		dstVal.Set(srcVal) | ||||
| 	} | ||||
|  | ||||
| } | ||||
							
								
								
									
										113
									
								
								vendor/github.com/aws/aws-sdk-go/aws/awsutil/prettify.go
									
									
									
										generated
									
									
										vendored
									
									
								
							
							
						
						
									
										113
									
								
								vendor/github.com/aws/aws-sdk-go/aws/awsutil/prettify.go
									
									
									
										generated
									
									
										vendored
									
									
								
							| @@ -1,113 +0,0 @@ | ||||
| package awsutil | ||||
|  | ||||
| import ( | ||||
| 	"bytes" | ||||
| 	"fmt" | ||||
| 	"io" | ||||
| 	"reflect" | ||||
| 	"strings" | ||||
| ) | ||||
|  | ||||
| // Prettify returns the string representation of a value. | ||||
| func Prettify(i interface{}) string { | ||||
| 	var buf bytes.Buffer | ||||
| 	prettify(reflect.ValueOf(i), 0, &buf) | ||||
| 	return buf.String() | ||||
| } | ||||
|  | ||||
| // prettify will recursively walk value v to build a textual | ||||
| // representation of the value. | ||||
| func prettify(v reflect.Value, indent int, buf *bytes.Buffer) { | ||||
| 	for v.Kind() == reflect.Ptr { | ||||
| 		v = v.Elem() | ||||
| 	} | ||||
|  | ||||
| 	switch v.Kind() { | ||||
| 	case reflect.Struct: | ||||
| 		strtype := v.Type().String() | ||||
| 		if strtype == "time.Time" { | ||||
| 			fmt.Fprintf(buf, "%s", v.Interface()) | ||||
| 			break | ||||
| 		} else if strings.HasPrefix(strtype, "io.") { | ||||
| 			buf.WriteString("<buffer>") | ||||
| 			break | ||||
| 		} | ||||
|  | ||||
| 		buf.WriteString("{\n") | ||||
|  | ||||
| 		names := []string{} | ||||
| 		for i := 0; i < v.Type().NumField(); i++ { | ||||
| 			name := v.Type().Field(i).Name | ||||
| 			f := v.Field(i) | ||||
| 			if name[0:1] == strings.ToLower(name[0:1]) { | ||||
| 				continue // ignore unexported fields | ||||
| 			} | ||||
| 			if (f.Kind() == reflect.Ptr || f.Kind() == reflect.Slice || f.Kind() == reflect.Map) && f.IsNil() { | ||||
| 				continue // ignore unset fields | ||||
| 			} | ||||
| 			names = append(names, name) | ||||
| 		} | ||||
|  | ||||
| 		for i, n := range names { | ||||
| 			val := v.FieldByName(n) | ||||
| 			buf.WriteString(strings.Repeat(" ", indent+2)) | ||||
| 			buf.WriteString(n + ": ") | ||||
| 			prettify(val, indent+2, buf) | ||||
|  | ||||
| 			if i < len(names)-1 { | ||||
| 				buf.WriteString(",\n") | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		buf.WriteString("\n" + strings.Repeat(" ", indent) + "}") | ||||
| 	case reflect.Slice: | ||||
| 		strtype := v.Type().String() | ||||
| 		if strtype == "[]uint8" { | ||||
| 			fmt.Fprintf(buf, "<binary> len %d", v.Len()) | ||||
| 			break | ||||
| 		} | ||||
|  | ||||
| 		nl, id, id2 := "", "", "" | ||||
| 		if v.Len() > 3 { | ||||
| 			nl, id, id2 = "\n", strings.Repeat(" ", indent), strings.Repeat(" ", indent+2) | ||||
| 		} | ||||
| 		buf.WriteString("[" + nl) | ||||
| 		for i := 0; i < v.Len(); i++ { | ||||
| 			buf.WriteString(id2) | ||||
| 			prettify(v.Index(i), indent+2, buf) | ||||
|  | ||||
| 			if i < v.Len()-1 { | ||||
| 				buf.WriteString("," + nl) | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		buf.WriteString(nl + id + "]") | ||||
| 	case reflect.Map: | ||||
| 		buf.WriteString("{\n") | ||||
|  | ||||
| 		for i, k := range v.MapKeys() { | ||||
| 			buf.WriteString(strings.Repeat(" ", indent+2)) | ||||
| 			buf.WriteString(k.String() + ": ") | ||||
| 			prettify(v.MapIndex(k), indent+2, buf) | ||||
|  | ||||
| 			if i < v.Len()-1 { | ||||
| 				buf.WriteString(",\n") | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		buf.WriteString("\n" + strings.Repeat(" ", indent) + "}") | ||||
| 	default: | ||||
| 		if !v.IsValid() { | ||||
| 			fmt.Fprint(buf, "<invalid value>") | ||||
| 			return | ||||
| 		} | ||||
| 		format := "%v" | ||||
| 		switch v.Interface().(type) { | ||||
| 		case string: | ||||
| 			format = "%q" | ||||
| 		case io.ReadSeeker, io.Reader: | ||||
| 			format = "buffer(%p)" | ||||
| 		} | ||||
| 		fmt.Fprintf(buf, format, v.Interface()) | ||||
| 	} | ||||
| } | ||||
							
								
								
									
										88
									
								
								vendor/github.com/aws/aws-sdk-go/aws/awsutil/string_value.go
									
									
									
										generated
									
									
										vendored
									
									
								
							
							
						
						
									
										88
									
								
								vendor/github.com/aws/aws-sdk-go/aws/awsutil/string_value.go
									
									
									
										generated
									
									
										vendored
									
									
								
							| @@ -1,88 +0,0 @@ | ||||
| package awsutil | ||||
|  | ||||
| import ( | ||||
| 	"bytes" | ||||
| 	"fmt" | ||||
| 	"reflect" | ||||
| 	"strings" | ||||
| ) | ||||
|  | ||||
| // StringValue returns the string representation of a value. | ||||
| func StringValue(i interface{}) string { | ||||
| 	var buf bytes.Buffer | ||||
| 	stringValue(reflect.ValueOf(i), 0, &buf) | ||||
| 	return buf.String() | ||||
| } | ||||
|  | ||||
| func stringValue(v reflect.Value, indent int, buf *bytes.Buffer) { | ||||
| 	for v.Kind() == reflect.Ptr { | ||||
| 		v = v.Elem() | ||||
| 	} | ||||
|  | ||||
| 	switch v.Kind() { | ||||
| 	case reflect.Struct: | ||||
| 		buf.WriteString("{\n") | ||||
|  | ||||
| 		for i := 0; i < v.Type().NumField(); i++ { | ||||
| 			ft := v.Type().Field(i) | ||||
| 			fv := v.Field(i) | ||||
|  | ||||
| 			if ft.Name[0:1] == strings.ToLower(ft.Name[0:1]) { | ||||
| 				continue // ignore unexported fields | ||||
| 			} | ||||
| 			if (fv.Kind() == reflect.Ptr || fv.Kind() == reflect.Slice) && fv.IsNil() { | ||||
| 				continue // ignore unset fields | ||||
| 			} | ||||
|  | ||||
| 			buf.WriteString(strings.Repeat(" ", indent+2)) | ||||
| 			buf.WriteString(ft.Name + ": ") | ||||
|  | ||||
| 			if tag := ft.Tag.Get("sensitive"); tag == "true" { | ||||
| 				buf.WriteString("<sensitive>") | ||||
| 			} else { | ||||
| 				stringValue(fv, indent+2, buf) | ||||
| 			} | ||||
|  | ||||
| 			buf.WriteString(",\n") | ||||
| 		} | ||||
|  | ||||
| 		buf.WriteString("\n" + strings.Repeat(" ", indent) + "}") | ||||
| 	case reflect.Slice: | ||||
| 		nl, id, id2 := "", "", "" | ||||
| 		if v.Len() > 3 { | ||||
| 			nl, id, id2 = "\n", strings.Repeat(" ", indent), strings.Repeat(" ", indent+2) | ||||
| 		} | ||||
| 		buf.WriteString("[" + nl) | ||||
| 		for i := 0; i < v.Len(); i++ { | ||||
| 			buf.WriteString(id2) | ||||
| 			stringValue(v.Index(i), indent+2, buf) | ||||
|  | ||||
| 			if i < v.Len()-1 { | ||||
| 				buf.WriteString("," + nl) | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		buf.WriteString(nl + id + "]") | ||||
| 	case reflect.Map: | ||||
| 		buf.WriteString("{\n") | ||||
|  | ||||
| 		for i, k := range v.MapKeys() { | ||||
| 			buf.WriteString(strings.Repeat(" ", indent+2)) | ||||
| 			buf.WriteString(k.String() + ": ") | ||||
| 			stringValue(v.MapIndex(k), indent+2, buf) | ||||
|  | ||||
| 			if i < v.Len()-1 { | ||||
| 				buf.WriteString(",\n") | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		buf.WriteString("\n" + strings.Repeat(" ", indent) + "}") | ||||
| 	default: | ||||
| 		format := "%v" | ||||
| 		switch v.Interface().(type) { | ||||
| 		case string: | ||||
| 			format = "%q" | ||||
| 		} | ||||
| 		fmt.Fprintf(buf, format, v.Interface()) | ||||
| 	} | ||||
| } | ||||
							
								
								
									
										96
									
								
								vendor/github.com/aws/aws-sdk-go/aws/client/client.go
									
									
									
										generated
									
									
										vendored
									
									
								
							
							
						
						
									
										96
									
								
								vendor/github.com/aws/aws-sdk-go/aws/client/client.go
									
									
									
										generated
									
									
										vendored
									
									
								
							| @@ -1,96 +0,0 @@ | ||||
| package client | ||||
|  | ||||
| import ( | ||||
| 	"fmt" | ||||
|  | ||||
| 	"github.com/aws/aws-sdk-go/aws" | ||||
| 	"github.com/aws/aws-sdk-go/aws/client/metadata" | ||||
| 	"github.com/aws/aws-sdk-go/aws/request" | ||||
| ) | ||||
|  | ||||
| // A Config provides configuration to a service client instance. | ||||
| type Config struct { | ||||
| 	Config        *aws.Config | ||||
| 	Handlers      request.Handlers | ||||
| 	Endpoint      string | ||||
| 	SigningRegion string | ||||
| 	SigningName   string | ||||
|  | ||||
| 	// States that the signing name did not come from a modeled source but | ||||
| 	// was derived based on other data. Used by service client constructors | ||||
| 	// to determine if the signin name can be overridden based on metadata the | ||||
| 	// service has. | ||||
| 	SigningNameDerived bool | ||||
| } | ||||
|  | ||||
| // ConfigProvider provides a generic way for a service client to receive | ||||
| // the ClientConfig without circular dependencies. | ||||
| type ConfigProvider interface { | ||||
| 	ClientConfig(serviceName string, cfgs ...*aws.Config) Config | ||||
| } | ||||
|  | ||||
| // ConfigNoResolveEndpointProvider same as ConfigProvider except it will not | ||||
| // resolve the endpoint automatically. The service client's endpoint must be | ||||
| // provided via the aws.Config.Endpoint field. | ||||
| type ConfigNoResolveEndpointProvider interface { | ||||
| 	ClientConfigNoResolveEndpoint(cfgs ...*aws.Config) Config | ||||
| } | ||||
|  | ||||
| // A Client implements the base client request and response handling | ||||
| // used by all service clients. | ||||
| type Client struct { | ||||
| 	request.Retryer | ||||
| 	metadata.ClientInfo | ||||
|  | ||||
| 	Config   aws.Config | ||||
| 	Handlers request.Handlers | ||||
| } | ||||
|  | ||||
| // New will return a pointer to a new initialized service client. | ||||
| func New(cfg aws.Config, info metadata.ClientInfo, handlers request.Handlers, options ...func(*Client)) *Client { | ||||
| 	svc := &Client{ | ||||
| 		Config:     cfg, | ||||
| 		ClientInfo: info, | ||||
| 		Handlers:   handlers.Copy(), | ||||
| 	} | ||||
|  | ||||
| 	switch retryer, ok := cfg.Retryer.(request.Retryer); { | ||||
| 	case ok: | ||||
| 		svc.Retryer = retryer | ||||
| 	case cfg.Retryer != nil && cfg.Logger != nil: | ||||
| 		s := fmt.Sprintf("WARNING: %T does not implement request.Retryer; using DefaultRetryer instead", cfg.Retryer) | ||||
| 		cfg.Logger.Log(s) | ||||
| 		fallthrough | ||||
| 	default: | ||||
| 		maxRetries := aws.IntValue(cfg.MaxRetries) | ||||
| 		if cfg.MaxRetries == nil || maxRetries == aws.UseServiceDefaultRetries { | ||||
| 			maxRetries = DefaultRetryerMaxNumRetries | ||||
| 		} | ||||
| 		svc.Retryer = DefaultRetryer{NumMaxRetries: maxRetries} | ||||
| 	} | ||||
|  | ||||
| 	svc.AddDebugHandlers() | ||||
|  | ||||
| 	for _, option := range options { | ||||
| 		option(svc) | ||||
| 	} | ||||
|  | ||||
| 	return svc | ||||
| } | ||||
|  | ||||
| // NewRequest returns a new Request pointer for the service API | ||||
| // operation and parameters. | ||||
| func (c *Client) NewRequest(operation *request.Operation, params interface{}, data interface{}) *request.Request { | ||||
| 	return request.New(c.Config, c.ClientInfo, c.Handlers, c.Retryer, operation, params, data) | ||||
| } | ||||
|  | ||||
| // AddDebugHandlers injects debug logging handlers into the service to log request | ||||
| // debug information. | ||||
| func (c *Client) AddDebugHandlers() { | ||||
| 	if !c.Config.LogLevel.AtLeast(aws.LogDebug) { | ||||
| 		return | ||||
| 	} | ||||
|  | ||||
| 	c.Handlers.Send.PushFrontNamed(LogHTTPRequestHandler) | ||||
| 	c.Handlers.Send.PushBackNamed(LogHTTPResponseHandler) | ||||
| } | ||||
							
								
								
									
										177
									
								
								vendor/github.com/aws/aws-sdk-go/aws/client/default_retryer.go
									
									
									
										generated
									
									
										vendored
									
									
								
							
							
						
						
									
										177
									
								
								vendor/github.com/aws/aws-sdk-go/aws/client/default_retryer.go
									
									
									
										generated
									
									
										vendored
									
									
								
							| @@ -1,177 +0,0 @@ | ||||
| package client | ||||
|  | ||||
| import ( | ||||
| 	"math" | ||||
| 	"strconv" | ||||
| 	"time" | ||||
|  | ||||
| 	"github.com/aws/aws-sdk-go/aws/request" | ||||
| 	"github.com/aws/aws-sdk-go/internal/sdkrand" | ||||
| ) | ||||
|  | ||||
| // DefaultRetryer implements basic retry logic using exponential backoff for | ||||
| // most services. If you want to implement custom retry logic, you can implement the | ||||
| // request.Retryer interface. | ||||
| // | ||||
| type DefaultRetryer struct { | ||||
| 	// Num max Retries is the number of max retries that will be performed. | ||||
| 	// By default, this is zero. | ||||
| 	NumMaxRetries    int | ||||
|  | ||||
| 	// MinRetryDelay is the minimum retry delay after which retry will be performed. | ||||
| 	// If not set, the value is 0ns. | ||||
| 	MinRetryDelay    time.Duration | ||||
|  | ||||
| 	// MinThrottleRetryDelay is the minimum retry delay when throttled. | ||||
| 	// If not set, the value is 0ns. | ||||
| 	MinThrottleDelay time.Duration | ||||
|  | ||||
| 	// MaxRetryDelay is the maximum retry delay before which retry must be performed. | ||||
| 	// If not set, the value is 0ns. | ||||
| 	MaxRetryDelay    time.Duration | ||||
|  | ||||
| 	// MaxThrottleDelay is the maximum retry delay when throttled. | ||||
| 	// If not set, the value is 0ns. | ||||
| 	MaxThrottleDelay time.Duration | ||||
| } | ||||
|  | ||||
| const ( | ||||
| 	// DefaultRetryerMaxNumRetries sets maximum number of retries | ||||
| 	DefaultRetryerMaxNumRetries = 3 | ||||
|  | ||||
| 	// DefaultRetryerMinRetryDelay sets minimum retry delay | ||||
| 	DefaultRetryerMinRetryDelay = 30 * time.Millisecond | ||||
|  | ||||
| 	// DefaultRetryerMinThrottleDelay sets minimum delay when throttled | ||||
| 	DefaultRetryerMinThrottleDelay = 500 * time.Millisecond | ||||
|  | ||||
| 	// DefaultRetryerMaxRetryDelay sets maximum retry delay | ||||
| 	DefaultRetryerMaxRetryDelay = 300 * time.Second | ||||
|  | ||||
| 	// DefaultRetryerMaxThrottleDelay sets maximum delay when throttled | ||||
| 	DefaultRetryerMaxThrottleDelay = 300 * time.Second | ||||
| ) | ||||
|  | ||||
| // MaxRetries returns the number of maximum returns the service will use to make | ||||
| // an individual API request. | ||||
| func (d DefaultRetryer) MaxRetries() int { | ||||
| 	return d.NumMaxRetries | ||||
| } | ||||
|  | ||||
| // setRetryerDefaults sets the default values of the retryer if not set | ||||
| func (d *DefaultRetryer) setRetryerDefaults() { | ||||
| 	if d.MinRetryDelay == 0 { | ||||
| 		d.MinRetryDelay = DefaultRetryerMinRetryDelay | ||||
| 	} | ||||
| 	if d.MaxRetryDelay == 0 { | ||||
| 		d.MaxRetryDelay = DefaultRetryerMaxRetryDelay | ||||
| 	} | ||||
| 	if d.MinThrottleDelay == 0 { | ||||
| 		d.MinThrottleDelay = DefaultRetryerMinThrottleDelay | ||||
| 	} | ||||
| 	if d.MaxThrottleDelay == 0 { | ||||
| 		d.MaxThrottleDelay = DefaultRetryerMaxThrottleDelay | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // RetryRules returns the delay duration before retrying this request again | ||||
| func (d DefaultRetryer) RetryRules(r *request.Request) time.Duration { | ||||
|  | ||||
| 	// if number of max retries is zero, no retries will be performed. | ||||
| 	if d.NumMaxRetries == 0 { | ||||
| 		return 0 | ||||
| 	} | ||||
|  | ||||
| 	// Sets default value for retryer members | ||||
| 	d.setRetryerDefaults() | ||||
|  | ||||
| 	// minDelay is the minimum retryer delay | ||||
| 	minDelay := d.MinRetryDelay | ||||
|  | ||||
| 	var initialDelay time.Duration | ||||
|  | ||||
| 	isThrottle := r.IsErrorThrottle() | ||||
| 	if isThrottle { | ||||
| 		if delay, ok := getRetryAfterDelay(r); ok { | ||||
| 			initialDelay = delay | ||||
| 		} | ||||
| 		minDelay = d.MinThrottleDelay | ||||
| 	} | ||||
|  | ||||
| 	retryCount := r.RetryCount | ||||
|  | ||||
| 	// maxDelay the maximum retryer delay | ||||
| 	maxDelay := d.MaxRetryDelay | ||||
|  | ||||
| 	if isThrottle { | ||||
| 		maxDelay = d.MaxThrottleDelay | ||||
| 	} | ||||
|  | ||||
| 	var delay time.Duration | ||||
|  | ||||
| 	// Logic to cap the retry count based on the minDelay provided | ||||
| 	actualRetryCount := int(math.Log2(float64(minDelay))) + 1 | ||||
| 	if actualRetryCount < 63-retryCount { | ||||
| 		delay = time.Duration(1<<uint64(retryCount)) * getJitterDelay(minDelay) | ||||
| 		if delay > maxDelay { | ||||
| 			delay = getJitterDelay(maxDelay / 2) | ||||
| 		} | ||||
| 	} else { | ||||
| 		delay = getJitterDelay(maxDelay / 2) | ||||
| 	} | ||||
| 	return delay + initialDelay | ||||
| } | ||||
|  | ||||
| // getJitterDelay returns a jittered delay for retry | ||||
| func getJitterDelay(duration time.Duration) time.Duration { | ||||
| 	return time.Duration(sdkrand.SeededRand.Int63n(int64(duration)) + int64(duration)) | ||||
| } | ||||
|  | ||||
| // ShouldRetry returns true if the request should be retried. | ||||
| func (d DefaultRetryer) ShouldRetry(r *request.Request) bool { | ||||
|  | ||||
| 	// ShouldRetry returns false if number of max retries is 0. | ||||
| 	if d.NumMaxRetries == 0 { | ||||
| 		return false | ||||
| 	} | ||||
|  | ||||
| 	// If one of the other handlers already set the retry state | ||||
| 	// we don't want to override it based on the service's state | ||||
| 	if r.Retryable != nil { | ||||
| 		return *r.Retryable | ||||
| 	} | ||||
| 	return r.IsErrorRetryable() || r.IsErrorThrottle() | ||||
| } | ||||
|  | ||||
| // This will look in the Retry-After header, RFC 7231, for how long | ||||
| // it will wait before attempting another request | ||||
| func getRetryAfterDelay(r *request.Request) (time.Duration, bool) { | ||||
| 	if !canUseRetryAfterHeader(r) { | ||||
| 		return 0, false | ||||
| 	} | ||||
|  | ||||
| 	delayStr := r.HTTPResponse.Header.Get("Retry-After") | ||||
| 	if len(delayStr) == 0 { | ||||
| 		return 0, false | ||||
| 	} | ||||
|  | ||||
| 	delay, err := strconv.Atoi(delayStr) | ||||
| 	if err != nil { | ||||
| 		return 0, false | ||||
| 	} | ||||
|  | ||||
| 	return time.Duration(delay) * time.Second, true | ||||
| } | ||||
|  | ||||
| // Will look at the status code to see if the retry header pertains to | ||||
| // the status code. | ||||
| func canUseRetryAfterHeader(r *request.Request) bool { | ||||
| 	switch r.HTTPResponse.StatusCode { | ||||
| 	case 429: | ||||
| 	case 503: | ||||
| 	default: | ||||
| 		return false | ||||
| 	} | ||||
|  | ||||
| 	return true | ||||
| } | ||||
							
								
								
									
										194
									
								
								vendor/github.com/aws/aws-sdk-go/aws/client/logger.go
									
									
									
										generated
									
									
										vendored
									
									
								
							
							
						
						
									
										194
									
								
								vendor/github.com/aws/aws-sdk-go/aws/client/logger.go
									
									
									
										generated
									
									
										vendored
									
									
								
							| @@ -1,194 +0,0 @@ | ||||
| package client | ||||
|  | ||||
| import ( | ||||
| 	"bytes" | ||||
| 	"fmt" | ||||
| 	"io" | ||||
| 	"io/ioutil" | ||||
| 	"net/http/httputil" | ||||
|  | ||||
| 	"github.com/aws/aws-sdk-go/aws" | ||||
| 	"github.com/aws/aws-sdk-go/aws/request" | ||||
| ) | ||||
|  | ||||
| const logReqMsg = `DEBUG: Request %s/%s Details: | ||||
| ---[ REQUEST POST-SIGN ]----------------------------- | ||||
| %s | ||||
| -----------------------------------------------------` | ||||
|  | ||||
| const logReqErrMsg = `DEBUG ERROR: Request %s/%s: | ||||
| ---[ REQUEST DUMP ERROR ]----------------------------- | ||||
| %s | ||||
| ------------------------------------------------------` | ||||
|  | ||||
| type logWriter struct { | ||||
| 	// Logger is what we will use to log the payload of a response. | ||||
| 	Logger aws.Logger | ||||
| 	// buf stores the contents of what has been read | ||||
| 	buf *bytes.Buffer | ||||
| } | ||||
|  | ||||
| func (logger *logWriter) Write(b []byte) (int, error) { | ||||
| 	return logger.buf.Write(b) | ||||
| } | ||||
|  | ||||
| type teeReaderCloser struct { | ||||
| 	// io.Reader will be a tee reader that is used during logging. | ||||
| 	// This structure will read from a body and write the contents to a logger. | ||||
| 	io.Reader | ||||
| 	// Source is used just to close when we are done reading. | ||||
| 	Source io.ReadCloser | ||||
| } | ||||
|  | ||||
| func (reader *teeReaderCloser) Close() error { | ||||
| 	return reader.Source.Close() | ||||
| } | ||||
|  | ||||
| // LogHTTPRequestHandler is a SDK request handler to log the HTTP request sent | ||||
| // to a service. Will include the HTTP request body if the LogLevel of the | ||||
| // request matches LogDebugWithHTTPBody. | ||||
| var LogHTTPRequestHandler = request.NamedHandler{ | ||||
| 	Name: "awssdk.client.LogRequest", | ||||
| 	Fn:   logRequest, | ||||
| } | ||||
|  | ||||
| func logRequest(r *request.Request) { | ||||
| 	logBody := r.Config.LogLevel.Matches(aws.LogDebugWithHTTPBody) | ||||
| 	bodySeekable := aws.IsReaderSeekable(r.Body) | ||||
|  | ||||
| 	b, err := httputil.DumpRequestOut(r.HTTPRequest, logBody) | ||||
| 	if err != nil { | ||||
| 		r.Config.Logger.Log(fmt.Sprintf(logReqErrMsg, | ||||
| 			r.ClientInfo.ServiceName, r.Operation.Name, err)) | ||||
| 		return | ||||
| 	} | ||||
|  | ||||
| 	if logBody { | ||||
| 		if !bodySeekable { | ||||
| 			r.SetReaderBody(aws.ReadSeekCloser(r.HTTPRequest.Body)) | ||||
| 		} | ||||
| 		// Reset the request body because dumpRequest will re-wrap the | ||||
| 		// r.HTTPRequest's Body as a NoOpCloser and will not be reset after | ||||
| 		// read by the HTTP client reader. | ||||
| 		if err := r.Error; err != nil { | ||||
| 			r.Config.Logger.Log(fmt.Sprintf(logReqErrMsg, | ||||
| 				r.ClientInfo.ServiceName, r.Operation.Name, err)) | ||||
| 			return | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	r.Config.Logger.Log(fmt.Sprintf(logReqMsg, | ||||
| 		r.ClientInfo.ServiceName, r.Operation.Name, string(b))) | ||||
| } | ||||
|  | ||||
| // LogHTTPRequestHeaderHandler is a SDK request handler to log the HTTP request sent | ||||
| // to a service. Will only log the HTTP request's headers. The request payload | ||||
| // will not be read. | ||||
| var LogHTTPRequestHeaderHandler = request.NamedHandler{ | ||||
| 	Name: "awssdk.client.LogRequestHeader", | ||||
| 	Fn:   logRequestHeader, | ||||
| } | ||||
|  | ||||
| func logRequestHeader(r *request.Request) { | ||||
| 	b, err := httputil.DumpRequestOut(r.HTTPRequest, false) | ||||
| 	if err != nil { | ||||
| 		r.Config.Logger.Log(fmt.Sprintf(logReqErrMsg, | ||||
| 			r.ClientInfo.ServiceName, r.Operation.Name, err)) | ||||
| 		return | ||||
| 	} | ||||
|  | ||||
| 	r.Config.Logger.Log(fmt.Sprintf(logReqMsg, | ||||
| 		r.ClientInfo.ServiceName, r.Operation.Name, string(b))) | ||||
| } | ||||
|  | ||||
| const logRespMsg = `DEBUG: Response %s/%s Details: | ||||
| ---[ RESPONSE ]-------------------------------------- | ||||
| %s | ||||
| -----------------------------------------------------` | ||||
|  | ||||
| const logRespErrMsg = `DEBUG ERROR: Response %s/%s: | ||||
| ---[ RESPONSE DUMP ERROR ]----------------------------- | ||||
| %s | ||||
| -----------------------------------------------------` | ||||
|  | ||||
| // LogHTTPResponseHandler is a SDK request handler to log the HTTP response | ||||
| // received from a service. Will include the HTTP response body if the LogLevel | ||||
| // of the request matches LogDebugWithHTTPBody. | ||||
| var LogHTTPResponseHandler = request.NamedHandler{ | ||||
| 	Name: "awssdk.client.LogResponse", | ||||
| 	Fn:   logResponse, | ||||
| } | ||||
|  | ||||
| func logResponse(r *request.Request) { | ||||
| 	lw := &logWriter{r.Config.Logger, bytes.NewBuffer(nil)} | ||||
|  | ||||
| 	if r.HTTPResponse == nil { | ||||
| 		lw.Logger.Log(fmt.Sprintf(logRespErrMsg, | ||||
| 			r.ClientInfo.ServiceName, r.Operation.Name, "request's HTTPResponse is nil")) | ||||
| 		return | ||||
| 	} | ||||
|  | ||||
| 	logBody := r.Config.LogLevel.Matches(aws.LogDebugWithHTTPBody) | ||||
| 	if logBody { | ||||
| 		r.HTTPResponse.Body = &teeReaderCloser{ | ||||
| 			Reader: io.TeeReader(r.HTTPResponse.Body, lw), | ||||
| 			Source: r.HTTPResponse.Body, | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	handlerFn := func(req *request.Request) { | ||||
| 		b, err := httputil.DumpResponse(req.HTTPResponse, false) | ||||
| 		if err != nil { | ||||
| 			lw.Logger.Log(fmt.Sprintf(logRespErrMsg, | ||||
| 				req.ClientInfo.ServiceName, req.Operation.Name, err)) | ||||
| 			return | ||||
| 		} | ||||
|  | ||||
| 		lw.Logger.Log(fmt.Sprintf(logRespMsg, | ||||
| 			req.ClientInfo.ServiceName, req.Operation.Name, string(b))) | ||||
|  | ||||
| 		if logBody { | ||||
| 			b, err := ioutil.ReadAll(lw.buf) | ||||
| 			if err != nil { | ||||
| 				lw.Logger.Log(fmt.Sprintf(logRespErrMsg, | ||||
| 					req.ClientInfo.ServiceName, req.Operation.Name, err)) | ||||
| 				return | ||||
| 			} | ||||
|  | ||||
| 			lw.Logger.Log(string(b)) | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	const handlerName = "awsdk.client.LogResponse.ResponseBody" | ||||
|  | ||||
| 	r.Handlers.Unmarshal.SetBackNamed(request.NamedHandler{ | ||||
| 		Name: handlerName, Fn: handlerFn, | ||||
| 	}) | ||||
| 	r.Handlers.UnmarshalError.SetBackNamed(request.NamedHandler{ | ||||
| 		Name: handlerName, Fn: handlerFn, | ||||
| 	}) | ||||
| } | ||||
|  | ||||
| // LogHTTPResponseHeaderHandler is a SDK request handler to log the HTTP | ||||
| // response received from a service. Will only log the HTTP response's headers. | ||||
| // The response payload will not be read. | ||||
| var LogHTTPResponseHeaderHandler = request.NamedHandler{ | ||||
| 	Name: "awssdk.client.LogResponseHeader", | ||||
| 	Fn:   logResponseHeader, | ||||
| } | ||||
|  | ||||
| func logResponseHeader(r *request.Request) { | ||||
| 	if r.Config.Logger == nil { | ||||
| 		return | ||||
| 	} | ||||
|  | ||||
| 	b, err := httputil.DumpResponse(r.HTTPResponse, false) | ||||
| 	if err != nil { | ||||
| 		r.Config.Logger.Log(fmt.Sprintf(logRespErrMsg, | ||||
| 			r.ClientInfo.ServiceName, r.Operation.Name, err)) | ||||
| 		return | ||||
| 	} | ||||
|  | ||||
| 	r.Config.Logger.Log(fmt.Sprintf(logRespMsg, | ||||
| 		r.ClientInfo.ServiceName, r.Operation.Name, string(b))) | ||||
| } | ||||
							
								
								
									
										13
									
								
								vendor/github.com/aws/aws-sdk-go/aws/client/metadata/client_info.go
									
									
									
										generated
									
									
										vendored
									
									
								
							
							
						
						
									
										13
									
								
								vendor/github.com/aws/aws-sdk-go/aws/client/metadata/client_info.go
									
									
									
										generated
									
									
										vendored
									
									
								
							| @@ -1,13 +0,0 @@ | ||||
| package metadata | ||||
|  | ||||
| // ClientInfo wraps immutable data from the client.Client structure. | ||||
| type ClientInfo struct { | ||||
| 	ServiceName   string | ||||
| 	ServiceID     string | ||||
| 	APIVersion    string | ||||
| 	Endpoint      string | ||||
| 	SigningName   string | ||||
| 	SigningRegion string | ||||
| 	JSONVersion   string | ||||
| 	TargetPrefix  string | ||||
| } | ||||
							
								
								
									
										28
									
								
								vendor/github.com/aws/aws-sdk-go/aws/client/no_op_retryer.go
									
									
									
										generated
									
									
										vendored
									
									
								
							
							
						
						
									
										28
									
								
								vendor/github.com/aws/aws-sdk-go/aws/client/no_op_retryer.go
									
									
									
										generated
									
									
										vendored
									
									
								
							| @@ -1,28 +0,0 @@ | ||||
| package client | ||||
|  | ||||
| import ( | ||||
| 	"time" | ||||
|  | ||||
| 	"github.com/aws/aws-sdk-go/aws/request" | ||||
| ) | ||||
|  | ||||
| // NoOpRetryer provides a retryer that performs no retries. | ||||
| // It should be used when we do not want retries to be performed. | ||||
| type NoOpRetryer struct{} | ||||
|  | ||||
| // MaxRetries returns the number of maximum returns the service will use to make | ||||
| // an individual API; For NoOpRetryer the MaxRetries will always be zero. | ||||
| func (d NoOpRetryer) MaxRetries() int { | ||||
| 	return 0 | ||||
| } | ||||
|  | ||||
| // ShouldRetry will always return false for NoOpRetryer, as it should never retry. | ||||
| func (d NoOpRetryer) ShouldRetry(_ *request.Request) bool { | ||||
| 	return false | ||||
| } | ||||
|  | ||||
| // RetryRules returns the delay duration before retrying this request again; | ||||
| // since NoOpRetryer does not retry, RetryRules always returns 0. | ||||
| func (d NoOpRetryer) RetryRules(_ *request.Request) time.Duration { | ||||
| 	return 0 | ||||
| } | ||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user