diff --git a/.github/dependabot.yml b/.github/dependabot.yml new file mode 100644 index 0000000..bcab5e5 --- /dev/null +++ b/.github/dependabot.yml @@ -0,0 +1,32 @@ +# To get started with Dependabot version updates, you'll need to specify which +# package ecosystems to update and where the package manifests are located. +# Please see the documentation for all configuration options: +# https://docs.github.com/en/code-security/dependabot/dependabot-version-updates/configuration-options-for-the-dependabot.yml-file + +version: 2 +updates: + # Maintain dependencies for GitHub Actions + - + package-ecosystem: "github-actions" + directory: "/" + schedule: + interval: "weekly" + target-branch: "dev" + commit-message: + prefix: "chore" + include: "scope" + labels: + - "Type: Maintenance" + + # Maintain dependencies for go modules + - + package-ecosystem: "gomod" + directory: "/" + schedule: + interval: "weekly" + target-branch: "dev" + commit-message: + prefix: "chore" + include: "scope" + labels: + - "Type: Maintenance" \ No newline at end of file diff --git a/.github/workflows/codeql.yml b/.github/workflows/codeql.yml new file mode 100644 index 0000000..5c29437 --- /dev/null +++ b/.github/workflows/codeql.yml @@ -0,0 +1,46 @@ +name: 🚨 Analyze Code (CodeQL) + +on: + push: + branches: + - "main" + paths: + - '**.go' + - '**.mod' + pull_request: + branches: + - "main" + paths: + - '**.go' + - '**.mod' + workflow_dispatch: + +jobs: + analyze: + name: Analyze Code (CodeQL) + strategy: + fail-fast: false + matrix: + language: [ 'go' ] + runs-on: ubuntu-latest + permissions: + actions: read + contents: read + security-events: write + steps: + - + name: Checkout the repository + uses: actions/checkout@v4 + with: + fetch-depth: 0 + - + name: Initialize CodeQL + uses: github/codeql-action/init@v3 + with: + languages: ${{ matrix.language }} + - + name: Autobuild + uses: github/codeql-action/autobuild@v3 + - + name: Perform CodeQL Analysis + uses: github/codeql-action/analyze@v3 \ No newline at end of file diff --git a/.github/workflows/linting.yml b/.github/workflows/linting.yml new file mode 100644 index 0000000..7904b01 --- /dev/null +++ b/.github/workflows/linting.yml @@ -0,0 +1,43 @@ +name: 💅🏻 Lint + +on: + push: + branches: + - "main" + paths: + - '**.go' + - '**.mod' + pull_request: + branches: + - "main" + paths: + - '**.go' + - '**.mod' + workflow_dispatch: + +permissions: + contents: read + +jobs: + lint: + name: Lint + runs-on: ubuntu-latest + steps: + - + name: Set up Go + uses: actions/setup-go@v5 + with: + go-version: '>=1.23' + cache: false + - + name: Checkout the repository + uses: actions/checkout@v4 + with: + fetch-depth: 0 + - + name: Run golangci-lint + uses: golangci/golangci-lint-action@v6 + with: + version: v1.61.0 + args: --timeout 5m + working-directory: . \ No newline at end of file diff --git a/.github/workflows/testing.yml b/.github/workflows/testing.yml new file mode 100644 index 0000000..34235c0 --- /dev/null +++ b/.github/workflows/testing.yml @@ -0,0 +1,45 @@ +name: 🧪 Test + +on: + push: + branches: + - "main" + paths: + - '**.go' + - '**.mod' + pull_request: + branches: + - "main" + paths: + - '**.go' + - '**.mod' + workflow_dispatch: + +jobs: + test: + name: Test + strategy: + matrix: + os: [ubuntu-latest, windows-latest, macOS-12] + runs-on: ${{ matrix.os }} + steps: + - + name: Set up Go + uses: actions/setup-go@v5 + with: + go-version: '>=1.23' + - + name: Checkout the repository + uses: actions/checkout@v4 + with: + fetch-depth: 0 + - + name: Go modules hygine + run: | + go clean -modcache + go mod tidy + working-directory: . + - + name: Go test + run: go test -v ./... + working-directory: . \ No newline at end of file diff --git a/.gitignore b/.gitignore new file mode 100644 index 0000000..e69de29 diff --git a/.golangci.yaml b/.golangci.yaml new file mode 100644 index 0000000..14d7876 --- /dev/null +++ b/.golangci.yaml @@ -0,0 +1,284 @@ +# Options for analysis running. +run: + # Number of operating system threads (`GOMAXPROCS`) that can execute golangci-lint simultaneously. + # If it is explicitly set to 0 (i.e. not the default) then golangci-lint will automatically set the value to match Linux container CPU quota. + # Default: the number of logical CPUs in the machine + # concurrency: 4 + # Timeout for analysis, e.g. 30s, 5m. + # Default: 1m + timeout: 5m + # Exit code when at least one issue was found. + # Default: 1 + issues-exit-code: 1 + # Include test files or not. + # Default: true + tests: true + # List of build tags, all linters use it. + # Default: [] + build-tags: [] + # If set, we pass it to "go list -mod={option}". From "go help modules": + # If invoked with -mod=readonly, the go command is disallowed from the implicit + # automatic updating of go.mod described above. Instead, it fails when any changes + # to go.mod are needed. This setting is most useful to check that go.mod does + # not need updates, such as in a continuous integration and testing system. + # If invoked with -mod=vendor, the go command assumes that the vendor + # directory holds the correct copies of dependencies and ignores + # the dependency descriptions in go.mod. + # + # Allowed values: readonly|vendor|mod + # Default: "" + modules-download-mode: readonly + # Allow multiple parallel golangci-lint instances running. + # If false, golangci-lint acquires file lock on start. + # Default: false + allow-parallel-runners: true + # Allow multiple golangci-lint instances running, but serialize them around a lock. + # If false, golangci-lint exits with an error if it fails to acquire file lock on start. + # Default: false + allow-serial-runners: true + # Define the Go version limit. + # Mainly related to generics support since go1.18. + # Default: use Go version from the go.mod file, fallback on the env var `GOVERSION`, fallback on 1.17 + go: '1.23' + +# output configuration options +output: + # The formats used to render issues. + # Formats: + # - `colored-line-number` + # - `line-number` + # - `json` + # - `colored-tab` + # - `tab` + # - `html` + # - `checkstyle` + # - `code-climate` + # - `junit-xml` + # - `junit-xml-extended` + # - `github-actions` + # - `teamcity` + # - `sarif` + # Output path can be either `stdout`, `stderr` or path to the file to write to. + # + # For the CLI flag (`--out-format`), multiple formats can be specified by separating them by comma. + # The output can be specified for each of them by separating format name and path by colon symbol. + # Example: "--out-format=checkstyle:report.xml,json:stdout,colored-line-number" + # The CLI flag (`--out-format`) override the configuration file. + # + # Default: + # formats: + # - format: colored-line-number + # path: stdout + formats: + # - + # format: json + # path: stderr + # - + # format: checkstyle + # path: report.xml + - + format: colored-line-number + path: stderr + # Print lines of code with issue. + # Default: true + print-issued-lines: true + # Print linter name in the end of issue text. + # Default: true + print-linter-name: true + # Make issues output unique by line. + # Default: true + uniq-by-line: false + # Add a prefix to the output file references. + # Default: "" + path-prefix: "" + # Sort results by the order defined in `sort-order`. + # Default: false + sort-results: true + # Order to use when sorting results. + # Require `sort-results` to `true`. + # Possible values: `file`, `linter`, and `severity`. + # + # If the severity values are inside the following list, they are ordered in this order: + # 1. error + # 2. warning + # 3. high + # 4. medium + # 5. low + # Either they are sorted alphabetically. + # + # Default: ["file"] + sort-order: + - linter + - severity + - file # filepath, line, and column. + # Show statistics per linter. + # Default: false + show-stats: false + +linters: + # Disable all linters. + # Default: false + disable-all: true + # Enable specific linter + # https://golangci-lint.run/usage/linters/#enabled-by-default + enable: + - asasalint + - asciicheck + - bidichk + - bodyclose + - canonicalheader + - containedctx + - contextcheck + - copyloopvar + # - cyclop + - decorder + # - depguard + - dogsled + - dupl + - dupword + - durationcheck + - err113 + - errcheck + - errchkjson + - errname + - errorlint + - exhaustive + # - exhaustruct + - fatcontext + - forbidigo + - forcetypeassert + # - funlen + - gci + - ginkgolinter + - gocheckcompilerdirectives + # - gochecknoglobals + # - gochecknoinits + - gochecksumtype + - gocognit + - goconst + - gocritic + - gocyclo + - godot + - godox + - gofmt + - gofumpt + - goheader + - goimports + - gomoddirectives + - gomodguard + - goprintffuncname + - gosec + - gosimple + - gosmopolitan + - govet + - grouper + - importas + - inamedparam + - ineffassign + - interfacebloat + - intrange + - ireturn + # - lll + - loggercheck + - maintidx + - makezero + - mirror + - misspell + - mnd + - musttag + # - nakedret + - nestif + - nilerr + - nilnil + - nlreturn + - noctx + - nolintlint + # - nonamedreturns + - nosprintfhostport + - paralleltest + - perfsprint + - prealloc + - predeclared + - promlinter + - protogetter + - reassign + - revive + - rowserrcheck + - sloglint + - spancheck + - sqlclosecheck + - staticcheck + - stylecheck + - tagalign + - tagliatelle + - tenv + - testableexamples + - testifylint + - testpackage + - thelper + - tparallel + - unconvert + - unparam + - unused + - usestdlibvars + # - varnamelen + - wastedassign + - whitespace + - wrapcheck + - wsl + - zerologlint + +linters-settings: + goconst: + min-len: 2 + min-occurrences: 3 + gocritic: + enabled-tags: + - performance + - experimental + - style + - opinionated + disabled-checks: + - captLocal + - whyNoLint + gocyclo: + # Minimal code complexity to report. + # Default: 30 (but we recommend 10-20) + min-complexity: 10 + # varnamelen: + # # The minimum length of a variable's name that is considered "long". + # # Variable names that are at least this long will be ignored. + # # Default: 3 + # min-name-length: 2 + # # Check method receivers. + # # Default: false + # check-receiver: true + # # Check named return values. + # # Default: false + # check-return: true + # # Check type parameters. + # # Default: false + # check-type-param: true + whitespace: + # Enforces newlines (or comments) after every multi-line if statement. + # Default: false + multi-if: true + # Enforces newlines (or comments) after every multi-line function signature. + # Default: false + multi-func: true + +issues: + # Which dirs to exclude: issues from them won't be reported. + # Can use regexp here: `generated.*`, regexp is applied on full path, + # including the path prefix if one is set. + # Default dirs are skipped independently of this option's value (see exclude-dirs-use-default). + # "/" will be replaced by current OS file path separator to properly work on Windows. + # Default: [] + exclude-dirs: + - gen + # Show issues in any part of update files (requires new-from-rev or new-from-patch). + # Default: false + whole-files: false + # Fix found issues (if it's supported by the linter). + # Default: false + fix: true \ No newline at end of file diff --git a/.vscode/extenstions.json b/.vscode/extenstions.json new file mode 100644 index 0000000..7203cb3 --- /dev/null +++ b/.vscode/extenstions.json @@ -0,0 +1,5 @@ +{ + "recommendations": [ + "golang.go" + ] +} \ No newline at end of file diff --git a/.vscode/settings.json b/.vscode/settings.json new file mode 100644 index 0000000..1a653cd --- /dev/null +++ b/.vscode/settings.json @@ -0,0 +1,6 @@ +{ + "go.lintTool": "golangci-lint", + "go.lintFlags": [ + "--fast" + ] +} \ No newline at end of file diff --git a/CONTRIBUTING.md b/CONTRIBUTING.md new file mode 100644 index 0000000..d21b461 --- /dev/null +++ b/CONTRIBUTING.md @@ -0,0 +1,54 @@ +# Contributing + +Contributions are welcomed! + +Please review the following guidelines before contributing. Also, feel free to propose changes to these guidelines by updating this file and submitting a pull request. + +* [I have a question...](#have-a-question) +* [I found a bug...](#found-a-bug) +* [I have a feature request...](#have-a-feature-request) +* [I have a contribution to share...](#ready-to-contribute) + +## Have a Question? + +Please don't open a GitHub issue for questions about how to use `hq-go-url`, as the goal is to use issues for managing bugs and feature requests. Issues that are related to general support will be closed. + +## Found a Bug? + +If you've identified a bug in `hq-go-url`, please [submit an issue](#create-an-issue) to our GitHub repo: [hueristiq/hq-go-url](https://github.com/hueristiq/hq-go-url/issues/new). Please also feel free to submit a [Pull Request](#pull-requests) with a fix for the bug! + +## Have a Feature Request? + +All feature requests should start with [submitting an issue](#create-an-issue) documenting the user story and acceptance criteria. Again, feel free to submit a [Pull Request](#pull-requests) with a proposed implementation of the feature. + +## Ready to Contribute + +### Create an issue + +Before submitting a new issue, please search the issues to make sure there isn't a similar issue doesn't already exist. + +Assuming no existing issues exist, please ensure you include required information when submitting the issue to ensure we can quickly reproduce your issue. + +We may have additional questions and will communicate through the GitHub issue, so please respond back to our questions to help reproduce and resolve the issue as quickly as possible. + +New issues can be created with in our [GitHub repo](https://github.com/hueristiq/hq-go-url/issues/new). + +### Pull Requests + +Pull requests should target the `dev` branch. Please also reference the issue from the description of the pull request using [special keyword syntax](https://help.github.com/articles/closing-issues-via-commit-messages/) to auto close the issue when the PR is merged. For example, include the phrase `fixes #14` in the PR description to have issue #14 auto close. + +### Styleguide + +When submitting code, please make every effort to follow existing conventions and style in order to keep the code as readable as possible. Here are a few points to keep in mind: + +* All dependencies must be defined in the `go.mod` file. + * Advanced IDEs and code editors (like VSCode) will take care of that, but to be sure, run `go mod tidy` to validate dependencies. +* Please run `go fmt ./...` before committing to ensure code aligns with go standards. +* We use [`golangci-lint`](https://golangci-lint.run/) for linting Go code, run `golangci-lint run --fix` before submitting PR. Editors such as Visual Studio Code or JetBrains IntelliJ; with Go support plugin will offer `golangci-lint` automatically. +* For details on the approved style, check out [Effective Go](https://golang.org/doc/effective_go.html). + +### License + +By contributing your code, you agree to license your contribution under the terms of the [MIT License](https://github.com/hueristiq/hq-go-url/blob/master/LICENSE). + +All files are released with the MIT license. diff --git a/LICENSE b/LICENSE new file mode 100644 index 0000000..f0d452f --- /dev/null +++ b/LICENSE @@ -0,0 +1,22 @@ +MIT License + +Copyright (c) 2023 Hueristiq + +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. + diff --git a/Makefile b/Makefile new file mode 100644 index 0000000..0ddf6d0 --- /dev/null +++ b/Makefile @@ -0,0 +1,108 @@ +# Set the default shell to `/bin/sh` for executing commands in the Makefile. +SHELL = /bin/sh + +# Define the project name for easy reference. +PROJECT = "hq-go-url" + +# --- Go(Golang) ------------------------------------------------------------------------------------ + +# Define common Go commands with variables for reusability and easier updates. +GOCMD=go # The main Go command. +GOMOD=$(GOCMD) mod # Go mod command for managing modules. +GOGET=$(GOCMD) get # Go get command for retrieving packages. +GOFMT=$(GOCMD) fmt # Go fmt command for formatting Go code. +GOTEST=$(GOCMD) test # Go test command for running tests. + +# Define Go build flags for verbosity and linking. +GOFLAGS := -v # Verbose flag for Go commands to print detailed output. +LDFLAGS := -s -w # Linker flags to strip debug information (-s) and reduce binary size (-w). + +# Set static linking flags for systems that are not macOS (darwin). +# Static linking allows the binary to include all required libraries in the executable. +ifneq ($(shell go env GOOS),darwin) + LDFLAGS := -extldflags "-static" +endif + +# Define Golangci-lint command for linting Go code. +GOLANGCILINTCMD=golangci-lint +GOLANGCILINTRUN=$(GOLANGCILINTCMD) run + +# --- Go Module Management + +# Tidy Go modules +# This target cleans up `go.mod` and `go.sum` files by removing any unused dependencies. +# Use this command to ensure that the module files are in a clean state. +.PHONY: go-mod-tidy +go-mod-tidy: + $(GOMOD) tidy + +# Update Go modules +# This target updates the Go module dependencies to their latest versions. +# It fetches and updates all modules, and any indirect dependencies. +.PHONY: go-mod-update +go-mod-update: + $(GOGET) -f -t -u ./... # Update test dependencies. + $(GOGET) -f -u ./... # Update other dependencies. + +# --- Go Code Quality and Testing + +# Format Go code +# This target formats all Go source files according to Go's standard formatting rules using `go fmt`. +.PHONY: go-fmt +go-fmt: + $(GOFMT) ./... + +# Lint Go code +# This target lints the Go source code to ensure it adheres to best practices. +# It uses `golangci-lint` to run various static analysis checks on the code. +# It first runs the `go-fmt` target to ensure the code is properly formatted. +.PHONY: go-lint +go-lint: go-fmt + $(GOLANGCILINTRUN) $(GOLANGCILINT) ./... + +# Run Go tests +# This target runs all Go tests in the current module. +# The `GOFLAGS` flag ensures that tests are run with verbose output, providing more detailed information. +.PHONY: go-test +go-test: + $(GOTEST) $(GOFLAGS) ./... + +# --- Go Code Generation --------------------------------------------------------------------------- + +# Run Go generate +# This target runs go generate ./..., which will process any //go:generate directives found in your Go source files. +# The ./... pattern ensures that the command is run on all packages within the module. +.PHONY: go-generate +go-generate: + $(GOCMD) generate ./... + +# --- Help ----------------------------------------------------------------------------------------- + +# Display help information +# This target prints out a detailed list of all available Makefile commands for ease of use. +# It's a helpful reference for developers using the Makefile. +.PHONY: help +help: + @echo "" + @echo "*****************************************************************************" + @echo "" + @echo "PROJECT : $(PROJECT)" + @echo "" + @echo "*****************************************************************************" + @echo "" + @echo "Available commands:" + @echo "" + @echo " Go Commands:" + @echo " go-mod-tidy .............. Tidy Go modules." + @echo " go-mod-update ............ Update Go modules." + @echo " go-fmt ................... Format Go code." + @echo " go-lint .................. Lint Go code." + @echo " go-test .................. Run Go tests." + @echo " go-generate .............. Run Go generate." + @echo "" + @echo " Help Commands:" + @echo " help ..................... Display this help information" + @echo "" + +# Default goal when `make` is run without arguments +.DEFAULT_GOAL = help \ No newline at end of file diff --git a/README.md b/README.md new file mode 100644 index 0000000..78fc345 --- /dev/null +++ b/README.md @@ -0,0 +1,171 @@ +# hq-go-url + +[![go report card](https://goreportcard.com/badge/github.com/hueristiq/hq-go-url)](https://goreportcard.com/report/github.com/hueristiq/hq-go-url) [![open issues](https://img.shields.io/github/issues-raw/hueristiq/hq-go-url.svg?style=flat&color=1E90FF)](https://github.com/hueristiq/hq-go-url/issues?q=is:issue+is:open) [![closed issues](https://img.shields.io/github/issues-closed-raw/hueristiq/hq-go-url.svg?style=flat&color=1E90FF)](https://github.com/hueristiq/hq-go-url/issues?q=is:issue+is:closed) [![license](https://img.shields.io/badge/license-MIT-gray.svg?color=1E90FF)](https://github.com/hueristiq/hq-go-url/blob/master/LICENSE) ![maintenance](https://img.shields.io/badge/maintained%3F-yes-1E90FF.svg) [![contribution](https://img.shields.io/badge/contributions-welcome-1E90FF.svg)](https://github.com/hueristiq/hq-go-url/blob/master/CONTRIBUTING.md) + +`hq-go-url` is a [Go (Golang)](http://golang.org/) package designed for extracting, parsing, and manipulating URLs with ease. This library is useful for developers who need to work with URLs in a structured way. + +## Resources + +* [Features](#features) +* [Usage](#usage) + * [URL Extraction](#url-extraction) + * [Customizing URL Extraction](#customizing-url-extraction) + * [Domain Parsing](#domain-parsingn) + * [URL Parsing](#url-parsing) +* [Contributing](#contributing) +* [Licensing](#licensing) +* [Credits](#credits) + * [Contributors](#contributors) + * [Similar Projects](#similar-projects) + +## Features + +* **Flexible URL Extraction:** Extract URLs from text using regular expressions. +* **Domain Parsing:** Parse domains into subdomains, root domains, and top-level domains (TLDs). +* **Extended URL Parsing:** Extend the standard `net/url` package in Go with additional fields and capabilities. + +## Installation + +To install the package, run the following command in your terminal: + +```bash +go get -v -u github.com/hueristiq/hq-go-url +``` + +This command will download and install the `hq-go-url` package into your Go workspace, making it available for use in your projects. + +## Usage + +Below are examples demonstrating how to use the different features of the `hq-go-url` package. + +### URL Extraction + +You can extract URLs from a given text string using the Extractor. Here's a simple example: + +```go +package main + +import ( + "fmt" + hqgourl "github.com/hueristiq/hq-go-url" + "regexp" +) + +func main() { + extr := hqgourl.NewExtractor() + text := "Check out this website: https://example.com and send an email to info@example.com." + + regex := extr.CompileRegex() + matches := regex.FindAllString(text, -1) + + fmt.Println("Found URLs:", matches) +} +``` + +#### Customizing URL Extraction + +You can customize how URLs are extracted by specifying URL schemes, hosts, or providing custom regular expression patterns. + +* Extract URLs with Specific Schemes (e.g., HTTP, HTTPS, FTP): + + ```go + extr := hqgourl.NewExtractor( + hqgourl.ExtractorWithSchemePattern(`(?:https?|ftp)://`), + ) + ``` + + This configuration will extract only URLs starting with http, https, or ftp schemes. + +* Extract URLs with Custom Host Patterns (e.g., example.com): + + ```go + extr := hqgourl.NewExtractor( + hqgourl.ExtractorWithHostPattern(`(?:www\.)?example\.com`), + ) + + ``` + + This setup will extract URLs that have hosts matching www.example.com or example.com. + +> [!NOTE] +> Since API is centered around [regexp.Regexp](https://golang.org/pkg/regexp/#Regexp), many other methods are available + +### Domain Parsing + +The `DomainParser` can parse domains into their components, such as subdomains, root domains, and TLDs: + +```go +package main + +import ( + "fmt" + hqgourl "github.com/hueristiq/hq-go-url" +) + +func main() { + dp := hqgourl.NewDomainParser() + + parsedDomain := dp.Parse("subdomain.example.com") + + fmt.Printf("Subdomain: %s, Root Domain: %s, TLD: %s\n", parsedDomain.Sub, parsedDomain.Root, parsedDomain.TopLevel) +} +``` + +### URL Parsing + +The `Parser` provides an extended way to parse URLs, including additional fields like port and file extension: + +```go +package main + +import ( + "fmt" + hqgourl "github.com/hueristiq/hq-go-url" +) + +func main() { + up := hqgourl.NewParser() + + parsedURL, err := up.Parse("https://subdomain.example.com:8080/path/file.txt") + if err != nil { + fmt.Println("Error parsing URL:", err) + + return + } + + fmt.Printf("Subdomain: %s\n", parsedURL.Domain.Sub) + fmt.Printf("Root Domain: %s\n", parsedURL.Domain.Root) + fmt.Printf("TLD: %s\n", parsedURL.Domain.TopLevel) + fmt.Printf("Port: %d\n", parsedURL.Port) + fmt.Printf("File Extension: %s\n", parsedURL.Extension) +} +``` + +Set a default scheme: + +```go +up := hqgourl.NewParser(hqgourl.ParserWithDefaultScheme("https")) +``` + +## Contributing + +We welcome contributions! Feel free to submit [Pull Requests](https://github.com/hueristiq/hq-go-url/pulls) or report [Issues](https://github.com/hueristiq/hq-go-url/issues). For more details, check out the [contribution guidelines](https://github.com/hueristiq/hq-go-url/blob/master/CONTRIBUTING.md). + + +## Licensing + +This package is licensed under the [MIT license](https://opensource.org/license/mit). You are free to use, modify, and distribute it, as long as you follow the terms of the license. You can find the full license text in the repository - [Full MIT license text](https://github.com/hueristiq/hq-go-url/blob/master/LICENSE). + +## Credits + +### Contributors + +A huge thanks to all the contributors who have helped make `hq-go-url` what it is today! + +[![contributors](https://contrib.rocks/image?repo=hueristiq/hq-go-url&max=500)](https://github.com/hueristiq/hq-go-url/graphs/contributors) + +### Similar Projects + +If you're interested in more packages like this, check out: + +[DomainParser](https://github.com/Cgboal/DomainParser) ◇ [urlx](https://github.com/goware/urlx) ◇ [xurls](https://github.com/mvdan/xurls) ◇ [goware's tldomains](https://github.com/goware/tldomains) ◇ [jakewarren's tldomains](https://github.com/jakewarren/tldomains) \ No newline at end of file diff --git a/domain.go b/domain.go new file mode 100644 index 0000000..87bb4c4 --- /dev/null +++ b/domain.go @@ -0,0 +1,36 @@ +package url + +import "strings" + +// Domain struct represents the structure of a parsed domain name, including its subdomain, root domain, and top-level domain (TLD). +type Domain struct { + Sub string + Root string + TopLevel string +} + +// String assembles the domain components back into a full domain string. +func (d *Domain) String() (domain string) { + var parts []string + + if d.Sub != "" { + parts = append(parts, d.Sub) + } + + if d.Root != "" { + parts = append(parts, d.Root) + } + + if d.TopLevel != "" { + parts = append(parts, d.TopLevel) + } + + domain = strings.Join(parts, ".") + + return +} + +// DomainInterface defines a standard interface for any domain representation. +type DomainInterface interface { + String() (domain string) +} diff --git a/domain_parser.go b/domain_parser.go new file mode 100644 index 0000000..6e72704 --- /dev/null +++ b/domain_parser.go @@ -0,0 +1,121 @@ +package url + +import ( + "index/suffixarray" + "strings" + + "github.com/hueristiq/hq-go-url/tlds" +) + +// DomainParser encapsulates the logic for parsing full domain strings into their constituent parts: +// subdomains, root domains, and top-level domains (TLDs). It leverages a suffix array for efficient +// search and extraction of these components from a full domain string. +type DomainParser struct { + sa *suffixarray.Index +} + +// Parse takes a full domain string and splits it into its constituent parts: subdomain, +// root domain, and TLD. This method efficiently identifies the TLD using a suffix array +// and separates the remaining parts of the domain accordingly. +func (dp *DomainParser) Parse(domain string) (parsedDomain *Domain) { + parsedDomain = &Domain{} + + // Split the domain into parts based on '.' + parts := strings.Split(domain, ".") + + if len(parts) <= 1 { + parsedDomain.Root = domain + + return + } + + // Identify the index where the TLD begins using the findTLDOffset method. + TLDOffset := dp.findTLDOffset(parts) + + if TLDOffset < 0 { + parsedDomain.Root = domain + + return + } + + // Based on the TLD offset, separate the domain string into subdomain, root domain, and TLD. + parsedDomain.Sub = strings.Join(parts[:TLDOffset], ".") + parsedDomain.Root = parts[TLDOffset] + parsedDomain.TopLevel = strings.Join(parts[TLDOffset+1:], ".") + + return +} + +// findTLDOffset determines the starting index of the TLD within a domain split into parts. +// It reverses through the parts of the domain to accurately handle cases where subdomains may +// mimic TLDs. The method uses the suffix array to find known TLDs efficiently. +func (dp *DomainParser) findTLDOffset(parts []string) (offset int) { + offset = -1 + + partsLength := len(parts) + partsLastIndex := partsLength - 1 + + for i := partsLastIndex; i >= 0; i-- { + // Construct a potential TLD from the current part to the end. + TLD := strings.Join(parts[i:], ".") + + // Search for the TLD in the suffix array. + indices := dp.sa.Lookup([]byte(TLD), -1) + + // If a match is found, update the offset, else break. + if len(indices) > 0 { + offset = i - 1 + } else { + break + } + } + + return +} + +// DomainParserInterface defines a standard interface for any DomainParser representation. +type DomainParserInterface interface { + Parse(domain string) (parsedDomain *Domain) + findTLDOffset(parts []string) (offset int) +} + +// DomainParserOptionsFunc is a function type designed for configuring a DomainParser instance. +// It allows for the application of customization options, such as specifying custom TLDs. +type DomainParserOptionsFunc func(*DomainParser) + +// Ensure type compatibility with interfaces. +var ( + _ DomainInterface = &Domain{} + _ DomainParserInterface = &DomainParser{} +) + +// NewDomainParser creates and initializes a DomainParser with a comprehensive list of TLDs, +// including both standard and pseudo-TLDs. This setup ensures accurate parsing across a wide +// range of domain names. Additional options can be applied to customize the parser further. +func NewDomainParser(opts ...DomainParserOptionsFunc) (dp *DomainParser) { + dp = &DomainParser{} + + // Combine standard and pseudo-TLDs for comprehensive coverage. + TLDs := []string{} + + TLDs = append(TLDs, tlds.Official...) + TLDs = append(TLDs, tlds.Pseudo...) + + // Initialize the suffix array with TLD data. + dp.sa = suffixarray.New([]byte("\x00" + strings.Join(TLDs, "\x00") + "\x00")) + + // Apply any additional options + for _, opt := range opts { + opt(dp) + } + + return +} + +// DomainParserWithTLDs allows for the initialization of the DomainParser with a custom set of TLDs. +// This is particularly useful for applications requiring parsing of non-standard or niche TLDs. +func DomainParserWithTLDs(TLDs ...string) DomainParserOptionsFunc { + return func(dp *DomainParser) { + dp.sa = suffixarray.New([]byte("\x00" + strings.Join(TLDs, "\x00") + "\x00")) + } +} diff --git a/domain_parser_test.go b/domain_parser_test.go new file mode 100644 index 0000000..f6db5f1 --- /dev/null +++ b/domain_parser_test.go @@ -0,0 +1,215 @@ +package url_test + +import ( + "fmt" + "reflect" + "testing" + + hqgourl "github.com/hueristiq/hq-go-url" + "github.com/hueristiq/hq-go-url/tlds" +) + +func TestDomain_String(t *testing.T) { + t.Parallel() + + cases := []struct { + domainStruct *hqgourl.Domain + expectedDomainString string + }{ + { + &hqgourl.Domain{ + "", + "example", + "", + }, + "example", + }, + { + &hqgourl.Domain{ + "", + "example", + "com", + }, + "example.com", + }, + { + &hqgourl.Domain{ + "www", + "example", + "com", + }, + "www.example.com", + }, + { + &hqgourl.Domain{ + "blog.www", + "example", + "com", + }, + "blog.www.example.com", + }, + } + + for _, c := range cases { + t.Run(fmt.Sprintf("Domain.String(%q)", c.domainStruct), func(t *testing.T) { + t.Parallel() + + domainString := c.domainStruct.String() + + if domainString != c.expectedDomainString { + t.Errorf("Domain.String(%q) = %v, want %v", c.domainStruct, domainString, c.expectedDomainString) + } + }) + } +} + +func TestNewDomainParser(t *testing.T) { + t.Parallel() + + dp := hqgourl.NewDomainParser() + if dp == nil { + t.Error("NewDomainParser() = nil; want non-nil") + } +} + +func TestDomainParsing(t *testing.T) { + t.Parallel() + + cases := []struct { + rawDomain string + expectedParsedDomain *hqgourl.Domain + }{ + { + "localhost", + &hqgourl.Domain{ + Sub: "", + Root: "localhost", + TopLevel: "", + }, + }, + { + "co.uk", + &hqgourl.Domain{ + Sub: "", + Root: "co.uk", + TopLevel: "", + }, + }, + { + "example.com", + &hqgourl.Domain{ + Sub: "", + Root: "example", + TopLevel: "com", + }, + }, + { + "www.example.com", + &hqgourl.Domain{ + Sub: "www", + Root: "example", + TopLevel: "com", + }, + }, + { + "example.co.uk", + &hqgourl.Domain{ + Sub: "", + Root: "example", + TopLevel: "co.uk", + }, + }, + { + "www.example.co.uk", + &hqgourl.Domain{ + Sub: "www", + Root: "example", + TopLevel: "co.uk", + }, + }, + { + "www.example.custom", + &hqgourl.Domain{ + Sub: "", + Root: "www.example.custom", + TopLevel: "", + }, + }, + } + + dp := hqgourl.NewDomainParser() + + for _, c := range cases { + t.Run(fmt.Sprintf("Parse(%q)", c.rawDomain), func(t *testing.T) { + t.Parallel() + + parsedDomain := dp.Parse(c.rawDomain) + + if !reflect.DeepEqual(parsedDomain, c.expectedParsedDomain) { + t.Errorf("Parse(%q) = %+v, want %+v", c.rawDomain, parsedDomain, c.expectedParsedDomain) + } + }) + } +} + +func TestDomainParsingWithCustomTLDs(t *testing.T) { + t.Parallel() + + cases := []struct { + rawDomain string + expectedParsedDomain *hqgourl.Domain + }{ + { + "example.custom", + &hqgourl.Domain{ + Sub: "", + Root: "example", + TopLevel: "custom", + }, + }, + { + "subdomain.example.custom", + &hqgourl.Domain{ + Sub: "subdomain", + Root: "example", + TopLevel: "custom", + }, + }, + } + + dp := hqgourl.NewDomainParser( + hqgourl.DomainParserWithTLDs("custom"), + ) + + for _, c := range cases { + t.Run(fmt.Sprintf("Parse(%q)", c.rawDomain), func(t *testing.T) { + t.Parallel() + + parsedDomain := dp.Parse(c.rawDomain) + + if !reflect.DeepEqual(parsedDomain, c.expectedParsedDomain) { + t.Errorf("Parse(%q) = %+v, want %+v", c.rawDomain, parsedDomain, c.expectedParsedDomain) + } + }) + } +} + +func TestDomainParserWithStandardAndTLDsPseudo(t *testing.T) { + t.Parallel() + + dp := hqgourl.NewDomainParser() + + TLDs := []string{} + + TLDs = append(TLDs, tlds.Official...) + TLDs = append(TLDs, tlds.Pseudo...) + + for _, TLD := range TLDs { + domain := "example." + TLD + + parsedDomain := dp.Parse(domain) + if parsedDomain.TopLevel != TLD { + t.Errorf("Parse(%q) = %q, %q, %q; want %q, %q, %q", domain, "", "example", TLD, "", "example", parsedDomain.TopLevel) + } + } +} diff --git a/extractor.go b/extractor.go new file mode 100644 index 0000000..5c6302c --- /dev/null +++ b/extractor.go @@ -0,0 +1,255 @@ +package url + +import ( + "regexp" + "strings" + "unicode/utf8" + + "github.com/hueristiq/hq-go-url/schemes" + "github.com/hueristiq/hq-go-url/tlds" + "github.com/hueristiq/hq-go-url/unicodes" +) + +// Extractor is a struct that configures the URL extraction process. +// It allows specifying whether to include URL schemes and hosts in the extraction and supports +// custom regex patterns for these components. +type Extractor struct { + withScheme bool // Indicates if the scheme part is mandatory in the URLs to be extracted. + withSchemePattern string // Custom regex pattern for matching URL schemes, if provided. + withHost bool // Indicates if the host part is mandatory in the URLs to be extracted. + withHostPattern string // Custom regex pattern for matching URL hosts, if provided. +} + +// CompileRegex compiles a regex pattern based on the Extractor configuration. +// It dynamically constructs a regex pattern to accurately capture URLs from text, +// supporting various URL formats and components. The method ensures the regex captures +// the longest possible match for a URL, enhancing the accuracy of the extraction process. +func (e *Extractor) CompileRegex() (regex *regexp.Regexp) { + schemePattern := ExtractorSchemePattern + + if e.withScheme && e.withSchemePattern != "" { + schemePattern = e.withSchemePattern + } + + var asciiTLDs, unicodeTLDs []string + + for i, tld := range tlds.Official { + if tld[0] >= utf8.RuneSelf { + asciiTLDs = tlds.Official[:i:i] + unicodeTLDs = tlds.Official[i:] + + break + } + } + + punycode := `xn--[a-z0-9-]+` + knownTLDPattern := `(?:(?i)` + punycode + `|` + anyOf(append(asciiTLDs, tlds.Pseudo...)...) + `\b|` + anyOf(unicodeTLDs...) + `)` + domainPattern := `(?:` + _subdomainPattern + knownTLDPattern + `|localhost)` + + hostWithoutPortPattern := `(?:` + domainPattern + `|\[` + ExtractorIPv6Pattern + `\]|\b` + ExtractorIPv4Pattern + `\b)` + hostWithPortOptionalPattern := `(?:` + hostWithoutPortPattern + ExtractorPortOptionalPattern + `)` + + if e.withHost && e.withHostPattern != "" { + hostWithPortOptionalPattern = e.withHostPattern + } + + _IAuthorityPattern := `(?:` + _IUserInfoOptionalPattern + hostWithPortOptionalPattern + `)` + _IAuthorityOptionalPattern := _IAuthorityPattern + `?` + + webURL := _IAuthorityPattern + `(?:/` + pathCont + `|/)?` + + // Emails pattern. + email := `(?P[a-zA-Z0-9._%\-+]+@` + hostWithPortOptionalPattern + `)` + + URLsWithSchemePattern := schemePattern + _IAuthorityOptionalPattern + pathCont + + if e.withHostPattern != "" { + URLsWithSchemePattern = schemePattern + _IAuthorityPattern + `(?:/` + pathCont + `|/)?` + } + + URLsWithHostPattern := webURL + `|` + email + + RelativeURLsPattern := `(\/[\w\/?=&#.-]*)|([\w\/?=&#.-]+?(?:\/[\w\/?=&#.-]+)+)` + + var pattern string + + switch { + case e.withScheme: + pattern = URLsWithSchemePattern + case e.withHost: + pattern = URLsWithSchemePattern + `|` + URLsWithHostPattern + default: + pattern = URLsWithSchemePattern + `|` + URLsWithHostPattern + `|` + RelativeURLsPattern + } + + // Compiling the final regex pattern. + regex = regexp.MustCompile(pattern) + // Ensures the longest possible match is found. + regex.Longest() + + return +} + +// ExtractorOptionsFunc defines a function type for configuring Extractor instances. +// This approach allows for flexible and fluent configuration of the extractor. +type ExtractorOptionsFunc func(*Extractor) + +// ExtractorInterface defines the interface for Extractor, ensuring it implements certain methods. +type ExtractorInterface interface { + CompileRegex() (regex *regexp.Regexp) +} + +const ( + _alphaCharacterSet = `a-zA-Z` + _digitCHaracterSet = `0-9` + _IUnreservedCharacterSet = _alphaCharacterSet + _digitCHaracterSet + `\-\._~` + unicodes.AllowedUcsChar + _IEndUnreservedCharacterSet = _alphaCharacterSet + _digitCHaracterSet + `\-_~` + unicodes.AllowedUcsCharMinusPunc + _subDelimsCharacterSet = `!\$&'\(\)\*\+,;=` + _endSubDelimsCharacterSet = `\$&\+=` + _pctEncodingPattern = `%[0-9a-fA-F]{2}` + + _IUserInfoPattern = `(?:(?:[` + _IUnreservedCharacterSet + _subDelimsCharacterSet + `:]|` + _pctEncodingPattern + `)+@)` + _IUserInfoOptionalPattern = _IUserInfoPattern + `?` + + ExtractorIPv4Pattern = `(?:25[0-5]|2[0-4][0-9]|1[0-9]{2}|[1-9][0-9]|[0-9])\.(?:25[0-5]|2[0-4][0-9]|1[0-9]{2}|[1-9][0-9]|[0-9])\.(?:25[0-5]|2[0-4][0-9]|1[0-9]{2}|[1-9][0-9]|[0-9])\.(?:25[0-5]|2[0-4][0-9]|1[0-9]{2}|[1-9][0-9]|[0-9])` + ExtractorNonEmptyIPv6Pattern = `(?:` + + // 7 colon-terminated chomps, followed by a final chomp or the rest of an elision. + `(?:[0-9a-fA-F]{1,4}:){7}(?:[0-9a-fA-F]{1,4}|:)|` + + // 6 chomps, followed by an IPv4 address or elision with final chomp or final elision. + `(?:[0-9a-fA-F]{1,4}:){6}(?:` + ExtractorIPv4Pattern + `|:[0-9a-fA-F]{1,4}|:)|` + + // 5 chomps, followed by an elision with optional IPv4 or up to 2 final chomps. + `(?:[0-9a-fA-F]{1,4}:){5}(?::` + ExtractorIPv4Pattern + `|(?::[0-9a-fA-F]{1,4}){1,2}|:)|` + + // 4 chomps, followed by an elision with optional IPv4 (optionally preceded by a chomp) or + // up to 3 final chomps. + `(?:[0-9a-fA-F]{1,4}:){4}(?:(?::[0-9a-fA-F]{1,4}){0,1}:` + ExtractorIPv4Pattern + `|(?::[0-9a-fA-F]{1,4}){1,3}|:)|` + + // 3 chomps, followed by an elision with optional IPv4 (preceded by up to 2 chomps) or + // up to 4 final chomps. + `(?:[0-9a-fA-F]{1,4}:){3}(?:(?::[0-9a-fA-F]{1,4}){0,2}:` + ExtractorIPv4Pattern + `|(?::[0-9a-fA-F]{1,4}){1,4}|:)|` + + // 2 chomps, followed by an elision with optional IPv4 (preceded by up to 3 chomps) or + // up to 5 final chomps. + `(?:[0-9a-fA-F]{1,4}:){2}(?:(?::[0-9a-fA-F]{1,4}){0,3}:` + ExtractorIPv4Pattern + `|(?::[0-9a-fA-F]{1,4}){1,5}|:)|` + + // 1 chomp, followed by an elision with optional IPv4 (preceded by up to 4 chomps) or + // up to 6 final chomps. + `(?:[0-9a-fA-F]{1,4}:){1}(?:(?::[0-9a-fA-F]{1,4}){0,4}:` + ExtractorIPv4Pattern + `|(?::[0-9a-fA-F]{1,4}){1,6}|:)|` + + // elision, followed by optional IPv4 (preceded by up to 5 chomps) or up to 7 final chomps. + // `:` is an intentionally omitted alternative, to avoid matching `::`. + `:(?:(?::[0-9a-fA-F]{1,4}){0,5}:` + ExtractorIPv4Pattern + `|(?::[0-9a-fA-F]{1,4}){1,7})` + + `)` + ExtractorIPv6Pattern = `(?:` + ExtractorNonEmptyIPv6Pattern + `|::)` + + ExtractorPortPattern = `(?::[0-9]{1,4}|[1-5][0-9]{4}|6[0-5][0-9]{3}\b)` + ExtractorPortOptionalPattern = ExtractorPortPattern + `?` + + midIPathSegmentChar = _IUnreservedCharacterSet + `%` + _subDelimsCharacterSet + `:@` + endIPathSegmentChar = _IEndUnreservedCharacterSet + `%` + _endSubDelimsCharacterSet + + _IPrivateCharacters = `\x{E000}-\x{F8FF}\x{F0000}-\x{FFFFD}\x{100000}-\x{10FFFD}` + + midIChar = `/?#\\` + midIPathSegmentChar + _IPrivateCharacters + endIChar = `/#` + endIPathSegmentChar + _IPrivateCharacters + wellParen = `\((?:[` + midIChar + `]|\([` + midIChar + `]*\))*\)` + wellBrack = `\[(?:[` + midIChar + `]|\[[` + midIChar + `]*\])*\]` + wellBrace = `\{(?:[` + midIChar + `]|\{[` + midIChar + `]*\})*\}` + wellAll = wellParen + `|` + wellBrack + `|` + wellBrace + pathCont = `(?:[` + midIChar + `]*(?:` + wellAll + `|[` + endIChar + `]))+` + + _letter = `\p{L}` + _mark = `\p{M}` + _number = `\p{N}` + _IRICharctersPattern = `[` + _letter + _mark + _number + `](?:[` + _letter + _mark + _number + `\-]*[` + _letter + _mark + _number + `])?` + + _subdomainPattern = `(?:` + _IRICharctersPattern + `\.)+` +) + +var ( + // ExtractorSchemePattern defines a general pattern for matching URL schemes. + // It matches any scheme that starts with alphabetical characters followed by any combination + // of alphabets, dots, hyphens, or pluses, and ends with "://". It also matches any scheme + // from a predefined list that does not require authority (host), ending with ":". + ExtractorSchemePattern = `(?:[a-zA-Z][a-zA-Z.\-+]*://|` + anyOf(schemes.NoAuthority...) + `:)` + // ExtractorKnownOfficialSchemePattern defines a pattern for matching officially recognized + // URL schemes. This includes schemes like "http", "https", "ftp", etc., and is strictly based + // on the schemes defined in the schemes.Schemes slice, ensuring a match ends with "://". + ExtractorKnownOfficialSchemePattern = `(?:` + anyOf(schemes.Official...) + `://)` + // ExtractorKnownUnofficialSchemePattern defines a pattern for matching unofficial or + // less commonly used URL schemes. Similar to the official pattern but based on the + // schemes.SchemesUnofficial slice, it supports schemes that might not be universally recognized + // but are valid in specific contexts, ending with "://". + ExtractorKnownUnofficialSchemePattern = `(?:` + anyOf(schemes.Unofficial...) + `://)` + // ExtractorKnownNoAuthoritySchemePattern defines a pattern for matching schemes that + // do not require an authority (host) component. This is useful for schemes like "mailto:", + // "tel:", and others where a host is not applicable, ending with ":". + ExtractorKnownNoAuthoritySchemePattern = `(?:` + anyOf(schemes.NoAuthority...) + `:)` + // ExtractorKnownSchemePattern combines the patterns for officially recognized, + // unofficial, and no-authority-required schemes into one comprehensive pattern. It is + // case-insensitive (noted by "(?i)") and designed to match a wide range of schemes, accommodating + // the broadest possible set of URLs. + ExtractorKnownSchemePattern = `(?:(?i)(?:` + anyOf(schemes.Official...) + `|` + anyOf(schemes.Unofficial...) + `)://|` + anyOf(schemes.NoAuthority...) + `:)` + + _ ExtractorInterface = &Extractor{} +) + +// NewExtractor creates a new Extractor instance with optional configuration. +// It applies the provided options to the extractor, allowing for customized behavior. +func NewExtractor(opts ...ExtractorOptionsFunc) (extractor *Extractor) { + extractor = &Extractor{} + + for _, opt := range opts { + opt(extractor) + } + + return +} + +// ExtractorWithScheme returns an option function to include URL schemes in the extraction process. +func ExtractorWithScheme() ExtractorOptionsFunc { + return func(e *Extractor) { + e.withScheme = true + } +} + +// ExtractorWithSchemePattern returns an option function to specify a custom regex pattern +// for matching URL schemes. This allows for fine-tuned control over which schemes are considered valid. +func ExtractorWithSchemePattern(pattern string) ExtractorOptionsFunc { + return func(e *Extractor) { + e.withScheme = true + e.withSchemePattern = pattern + } +} + +// ExtractorWithHost returns an option function to include hosts in the URLs to be extracted. +// This can be used to ensure that only URLs with specified host components are captured. +func ExtractorWithHost() ExtractorOptionsFunc { + return func(e *Extractor) { + e.withHost = true + } +} + +// ExtractorWithHostPattern returns an option function to specify a custom regex pattern +// for matching URL hosts. This is useful for targeting specific domain names or IP address formats. +func ExtractorWithHostPattern(pattern string) ExtractorOptionsFunc { + return func(e *Extractor) { + e.withHost = true + e.withHostPattern = pattern + } +} + +// anyOf is a helper function that constructs a regex pattern for a set of strings. +// It simplifies the creation of regex patterns by automatically escaping and joining the provided strings. +func anyOf(strs ...string) string { + var b strings.Builder + + b.WriteString("(?:") + + for i, s := range strs { + if i != 0 { + b.WriteByte('|') + } + + b.WriteString(regexp.QuoteMeta(s)) + } + + b.WriteByte(')') + + return b.String() +} diff --git a/extractor_test.go b/extractor_test.go new file mode 100644 index 0000000..c5140e2 --- /dev/null +++ b/extractor_test.go @@ -0,0 +1,653 @@ +package url_test + +import ( + "testing" + + hqgourl "github.com/hueristiq/hq-go-url" +) + +func TestNewExtractor(t *testing.T) { + t.Parallel() + + extr := hqgourl.NewExtractor() + + if extr == nil { + t.Error("NewExtractor() = nil; want non-nil") + } +} + +func TestCompileRegex(t *testing.T) { + t.Parallel() + + extr := hqgourl.NewExtractor() + + regex := extr.CompileRegex() + + if regex == nil { + t.Errorf("CompileRegex() = nil; want non-nil") + } +} + +// func TestURLExtraction(t *testing.T) { +// t.Parallel() + +// hqgourl := hqgourl.NewExtractor() + +// regex := hqgourl.CompileRegex() + +// testCases := []struct { +// text string +// want []string +// }{ +// { +// ` +// Localhost URL: http://localhost +// Localhost URL with Port: http://localhost:8000/home +// Standard URL: https://www.example.com +// URL with Port: http://www.example.com:8080/page +// URL with HTTPS and Query: https://www.example.com/search?q=openai +// URL with Path: https://www.example.com/resources/docs/guide +// URL with Fragment: https://www.example.com/about#team +// URL with User Info: https://user:password@example.com +// Complex URL: https://www.example.com:8080/search?q=openai#results +// International URL: https://www.例子.公司.cn +// URL with IPv4 Address: http://192.168.1.1/setup +// URL with IPv4 and Port: http://192.168.1.1:8080/setup +// URL with IPv6 Address: http://[2001:0db8:85a3:0000:0000:8a2e:0370:7334] +// URL with IPv6 and Port: http://[2001:0db8:85a3:0000:0000:8a2e:0370:7334]:8080 +// URL with Special Characters: https://example.com/products?id=50&category=books +// URL with Encoded Characters: https://example.com/search?name=John%20Doe&age=30 +// URL with Multiple Subdomains: https://subdomain.example.co.uk +// File Protocol URL: file:///C:/path/to/file.txt +// Mailto Protocol URL: mailto:user@example.com +// FTP URL with Port: ftp://user:password@ftp.example.com:21 +// Data URL: data:text/plain;base64,aGVsbG8= +// URL with JavaScript Protocol: javascript:alert('Hello World'); +// URL with Spaces: https://www.example.com/test space +// URL in Text: Check out this link: https://www.example.com, it's cool! +// Relative URL: /path/to/resource +// URL without Scheme: www.example.com +// URL without Scheme and with Path: www.example.com/resources +// URL without Scheme and Query: www.example.com/search?q=openai +// URL without Scheme, IPv4: 192.168.1.1 +// URL without Scheme, IPv4, and Port: 192.168.1.1:8080 +// URL without Scheme, IPv6: [2001:0db8:85a3:0000:0000:8a2e:0370:7334] +// URL without Scheme, IPv6, and Port: [2001:0db8:85a3:0000:0000:8a2e:0370:7334]:8080 +// `, +// []string{ +// "http://localhost", +// "http://localhost:8000/home", +// "https://www.example.com", +// "http://www.example.com:8080/page", +// "https://www.example.com/search?q=openai", +// "https://www.example.com/resources/docs/guide", +// "https://www.example.com/about#team", +// "https://user:password@example.com", +// "https://www.example.com:8080/search?q=openai#results", +// "https://www.例子.公司.cn", +// "http://192.168.1.1/setup", +// "http://192.168.1.1:8080/setup", +// "http://[2001:0db8:85a3:0000:0000:8a2e:0370:7334]", +// "http://[2001:0db8:85a3:0000:0000:8a2e:0370:7334]:8080", +// "https://example.com/products?id=50&category=books", +// "https://example.com/search?name=John%20Doe&age=30", +// "https://subdomain.example.co.uk", +// "file:///C:/path/to/file.txt", +// "mailto:user@example.com", +// "ftp://user:password@ftp.example.com:21", +// "text/plain", +// "https://www.example.com/test", +// "https://www.example.com", +// "/path/to/resource", +// "www.example.com", +// "www.example.com/resources", +// "www.example.com/search?q=openai", +// "192.168.1.1", +// "192.168.1.1:8080", +// "[2001:0db8:85a3:0000:0000:8a2e:0370:7334]", +// "[2001:0db8:85a3:0000:0000:8a2e:0370:7334]:8080", +// }, +// }, +// } + +// for _, tc := range testCases { +// got := regex.FindAllString(tc.text, -1) + +// if !equalSlices(got, tc.want) { +// t.Errorf("Extracted URLs = %v, want %v", got, tc.want) +// } +// } +// } + +func TestURLExtractionWithScheme(t *testing.T) { + t.Parallel() + + extr := hqgourl.NewExtractor( + hqgourl.ExtractorWithScheme(), + ) + + regex := extr.CompileRegex() + + testCases := []struct { + text string + want []string + }{ + { + ` + http://localhost + http://localhost:8000/home + + https://www.example.com + http://www.example.com:8080/page + https://www.example.com/search?q=openai + https://www.example.com/resources/docs/guide + https://www.example.com/about#team + https://user:password@example.com + https://www.example.com:8080/search?q=openai#results + + https://www.例子.公司.cn + http://中国.中国/中国 + http://中国.中国/foo中国 + + http://192.168.1.1/setup + http://192.168.1.1:8080/setup + http://[2001:0db8:85a3:0000:0000:8a2e:0370:7334] + http://[2001:0db8:85a3:0000:0000:8a2e:0370:7334]:8080 + + https://example.com/products?id=50&category=books + https://example.com/search?name=John%20Doe&age=30 + + http://foo.com/🐼 + https://shmibbles.me/tmp/自殺でも?.png + https://subdomain.example.co.uk + + foo://example.com + file:///C:/path/to/file.txt + mailto:user@example.com + ftp://user:password@ftp.example.com:21 + data:text/plain;base64,aGVsbG8= + javascript:alert('Hello World'); + sms:123 + xmpp:foo@bar + bitcoin:Addr23?amount=1&message=foo + cid:foo-32x32.v2_fe0f1423.png + mid:960830.1639@XIson.com + postgres://user:pass@host.com:5432/path?k=v#f + zoommtg://zoom.us/join?confno=1234&pwd=xxx + + https://www.example.com/test space + https://www.example.com, it's cool! + + /path/to/resource + www.example.com + www.example.com/resources + www.example.com/search?q=openai + 192.168.1.1 + 192.168.1.1:8080 + ::1 + ::ffff:0:0 + 64:ff9b:: + 64:ff9b:1:: + 100:: + 2001:: + 2001:1::1 + 2001:1::2 + 2001:2:: + 2001:3:: + 2001:4:112:: + 2001:10:: + 2001:20:: + 2002:: + 2620:4f:8000:: + fc00:: + fe80:: + [2001:0db8:85a3:0000:0000:8a2e:0370:7334] + [2001:0db8:85a3:0000:0000:8a2e:0370:7334]:8080 + `, + []string{ + "http://localhost", + "http://localhost:8000/home", + "https://www.example.com", + "http://www.example.com:8080/page", + "https://www.example.com/search?q=openai", + "https://www.example.com/resources/docs/guide", + "https://www.example.com/about#team", + "https://user:password@example.com", + "https://www.example.com:8080/search?q=openai#results", + "https://www.例子.公司.cn", + "http://中国.中国/中国", + "http://中国.中国/foo中国", + // "http://उदाहरण.परीकषा", + "http://192.168.1.1/setup", + "http://192.168.1.1:8080/setup", + "http://[2001:0db8:85a3:0000:0000:8a2e:0370:7334]", + "http://[2001:0db8:85a3:0000:0000:8a2e:0370:7334]:8080", + "https://example.com/products?id=50&category=books", + "https://example.com/search?name=John%20Doe&age=30", + // "http://✪foo.bar/pa✪th©more", + "http://foo.com/🐼", + "https://shmibbles.me/tmp/自殺でも?.png", + "https://subdomain.example.co.uk", + "foo://example.com", + "file:///C:/path/to/file.txt", + "mailto:user@example.com", + "ftp://user:password@ftp.example.com:21", + "sms:123", + "xmpp:foo@bar", + "bitcoin:Addr23?amount=1&message=foo", + "cid:foo-32x32.v2_fe0f1423.png", + "mid:960830.1639@XIson.com", + "postgres://user:pass@host.com:5432/path?k=v#f", + "zoommtg://zoom.us/join?confno=1234&pwd=xxx", + "https://www.example.com/test", + "https://www.example.com", + }, + }, + } + + for _, tc := range testCases { + got := regex.FindAllString(tc.text, -1) + + if !equalSlices(got, tc.want) { + t.Errorf("Extracted URLs = %v, want %v", got, tc.want) + } + } +} + +func TestURLExtractionWithSchemePattern(t *testing.T) { + t.Parallel() + + extr := hqgourl.NewExtractor( + hqgourl.ExtractorWithSchemePattern(`(?:https?://)`), + ) + + regex := extr.CompileRegex() + + testCases := []struct { + text string + want []string + }{ + { + ` + http://localhost + http://localhost:8000/home + + https://www.example.com + http://www.example.com:8080/page + https://www.example.com/search?q=openai + https://www.example.com/resources/docs/guide + https://www.example.com/about#team + https://user:password@example.com + https://www.example.com:8080/search?q=openai#results + + https://www.例子.公司.cn + http://中国.中国/中国 + http://中国.中国/foo中国 + + http://192.168.1.1/setup + http://192.168.1.1:8080/setup + http://[2001:0db8:85a3:0000:0000:8a2e:0370:7334] + http://[2001:0db8:85a3:0000:0000:8a2e:0370:7334]:8080 + + https://example.com/products?id=50&category=books + https://example.com/search?name=John%20Doe&age=30 + http://foo.com/🐼 + https://shmibbles.me/tmp/自殺でも?.png + https://subdomain.example.co.uk + + foo://example.com + file:///C:/path/to/file.txt + mailto:user@example.com + ftp://user:password@ftp.example.com:21 + data:text/plain;base64,aGVsbG8= + javascript:alert('Hello World'); + sms:123 + xmpp:foo@bar + bitcoin:Addr23?amount=1&message=foo + cid:foo-32x32.v2_fe0f1423.png + mid:960830.1639@XIson.com + postgres://user:pass@host.com:5432/path?k=v#f + zoommtg://zoom.us/join?confno=1234&pwd=xxx + + https://www.example.com/test space + https://www.example.com, it's cool! + + /path/to/resource + www.example.com + www.example.com/resources + www.example.com/search?q=openai + 192.168.1.1 + 192.168.1.1:8080 + ::1 + ::ffff:0:0 + 64:ff9b:: + 64:ff9b:1:: + 100:: + 2001:: + 2001:1::1 + 2001:1::2 + 2001:2:: + 2001:3:: + 2001:4:112:: + 2001:10:: + 2001:20:: + 2002:: + 2620:4f:8000:: + fc00:: + fe80:: + [2001:0db8:85a3:0000:0000:8a2e:0370:7334] + [2001:0db8:85a3:0000:0000:8a2e:0370:7334]:8080 + `, + []string{ + "http://localhost", + "http://localhost:8000/home", + "https://www.example.com", + "http://www.example.com:8080/page", + "https://www.example.com/search?q=openai", + "https://www.example.com/resources/docs/guide", + "https://www.example.com/about#team", + "https://user:password@example.com", + "https://www.example.com:8080/search?q=openai#results", + "https://www.例子.公司.cn", + "http://中国.中国/中国", + "http://中国.中国/foo中国", + // "http://उदाहरण.परीकषा", + "http://192.168.1.1/setup", + "http://192.168.1.1:8080/setup", + "http://[2001:0db8:85a3:0000:0000:8a2e:0370:7334]", + "http://[2001:0db8:85a3:0000:0000:8a2e:0370:7334]:8080", + "https://example.com/products?id=50&category=books", + "https://example.com/search?name=John%20Doe&age=30", + // "http://✪foo.bar/pa✪th©more", + "http://foo.com/🐼", + "https://shmibbles.me/tmp/自殺でも?.png", + "https://subdomain.example.co.uk", + "https://www.example.com/test", + "https://www.example.com", + }, + }, + } + + for _, tc := range testCases { + got := regex.FindAllString(tc.text, -1) + + if !equalSlices(got, tc.want) { + t.Errorf("Extracted URLs = %v, want %v", got, tc.want) + } + } +} + +func TestURLExtractionWithHost(t *testing.T) { + t.Parallel() + + extr := hqgourl.NewExtractor( + hqgourl.ExtractorWithHost(), + ) + + regex := extr.CompileRegex() + + testCases := []struct { + text string + want []string + }{ + { + ` + http://localhost + http://localhost:8000/home + + https://www.example.com + http://www.example.com:8080/page + https://www.example.com/search?q=openai + https://www.example.com/resources/docs/guide + https://www.example.com/about#team + https://user:password@example.com + https://www.example.com:8080/search?q=openai#results + + https://www.例子.公司.cn + http://中国.中国/中国 + http://中国.中国/foo中国 + http://उदाहरण.परीकषा + + http://192.168.1.1/setup + http://192.168.1.1:8080/setup + http://[2001:0db8:85a3:0000:0000:8a2e:0370:7334] + http://[2001:0db8:85a3:0000:0000:8a2e:0370:7334]:8080 + + https://example.com/products?id=50&category=books + https://example.com/search?name=John%20Doe&age=30 + http://✪foo.bar/pa✪th©more + http://foo.com/🐼 + https://shmibbles.me/tmp/自殺でも?.png + https://subdomain.example.co.uk + + foo://example.com + file:///C:/path/to/file.txt + mailto:user@example.com + ftp://user:password@ftp.example.com:21 + data:text/plain;base64,aGVsbG8= + javascript:alert('Hello World'); + sms:123 + xmpp:foo@bar + bitcoin:Addr23?amount=1&message=foo + cid:foo-32x32.v2_fe0f1423.png + mid:960830.1639@XIson.com + postgres://user:pass@host.com:5432/path?k=v#f + zoommtg://zoom.us/join?confno=1234&pwd=xxx + + https://www.example.com/test space + https://www.example.com, it's cool! + + /path/to/resource + www.example.com + www.example.com/resources + www.example.com/search?q=openai + 192.168.1.1 + 192.168.1.1:8080 + ::1 + ::ffff:0:0 + 64:ff9b:: + 64:ff9b:1:: + 100:: + 2001:: + 2001:1::1 + 2001:1::2 + 2001:2:: + 2001:3:: + 2001:4:112:: + 2001:10:: + 2001:20:: + 2002:: + 2620:4f:8000:: + fc00:: + fe80:: + [2001:0db8:85a3:0000:0000:8a2e:0370:7334] + [2001:0db8:85a3:0000:0000:8a2e:0370:7334]:8080 + `, + []string{ + "http://localhost", + "http://localhost:8000/home", + "https://www.example.com", + "http://www.example.com:8080/page", + "https://www.example.com/search?q=openai", + "https://www.example.com/resources/docs/guide", + "https://www.example.com/about#team", + "https://user:password@example.com", + "https://www.example.com:8080/search?q=openai#results", + "https://www.例子.公司.cn", + "http://中国.中国/中国", + "http://中国.中国/foo中国", + "http://उदाहरण.परीकषा", + "http://192.168.1.1/setup", + "http://192.168.1.1:8080/setup", + "http://[2001:0db8:85a3:0000:0000:8a2e:0370:7334]", + "http://[2001:0db8:85a3:0000:0000:8a2e:0370:7334]:8080", + "https://example.com/products?id=50&category=books", + "https://example.com/search?name=John%20Doe&age=30", + "http://✪foo.bar/pa✪th©more", + "http://foo.com/🐼", + "https://shmibbles.me/tmp/自殺でも?.png", + "https://subdomain.example.co.uk", + "foo://example.com", + "file:///C:/path/to/file.txt", + "mailto:user@example.com", + "ftp://user:password@ftp.example.com:21", + "sms:123", + "xmpp:foo@bar", + "bitcoin:Addr23?amount=1&message=foo", + "cid:foo-32x32.v2_fe0f1423.png", + "mid:960830.1639@XIson.com", + "postgres://user:pass@host.com:5432/path?k=v#f", + "zoommtg://zoom.us/join?confno=1234&pwd=xxx", + "https://www.example.com/test", + "https://www.example.com", + "www.example.com", + "www.example.com/resources", + "www.example.com/search?q=openai", + "192.168.1.1", + "192.168.1.1:8080", + "[2001:0db8:85a3:0000:0000:8a2e:0370:7334]", + "[2001:0db8:85a3:0000:0000:8a2e:0370:7334]:8080", + }, + }, + } + + for _, tc := range testCases { + got := regex.FindAllString(tc.text, -1) + + if !equalSlices(got, tc.want) { + t.Errorf("Extracted URLs = %v, want %v", got, tc.want) + } + } +} + +func TestURLExtractionWithHostPattern(t *testing.T) { + t.Parallel() + + extr := hqgourl.NewExtractor( + hqgourl.ExtractorWithHostPattern(`(?:(?:\w+[.])*example\.com` + hqgourl.ExtractorPortOptionalPattern + `)`), + ) + + regex := extr.CompileRegex() + + testCases := []struct { + text string + want []string + }{ + { + ` + http://localhost + http://localhost:8000/home + + https://www.example.com + http://www.example.com:8080/page + https://www.example.com/search?q=openai + https://www.example.com/resources/docs/guide + https://www.example.com/about#team + https://user:password@example.com + https://www.example.com:8080/search?q=openai#results + + https://www.例子.公司.cn + http://中国.中国/中国 + http://中国.中国/foo中国 + http://उदाहरण.परीकषा + + http://192.168.1.1/setup + http://192.168.1.1:8080/setup + http://[2001:0db8:85a3:0000:0000:8a2e:0370:7334] + http://[2001:0db8:85a3:0000:0000:8a2e:0370:7334]:8080 + + https://example.com/products?id=50&category=books + https://example.com/search?name=John%20Doe&age=30 + http://✪foo.bar/pa✪th©more + http://foo.com/🐼 + https://shmibbles.me/tmp/自殺でも?.png + https://subdomain.example.co.uk + + foo://example.com + file:///C:/path/to/file.txt + mailto:user@example.com + ftp://user:password@ftp.example.com:21 + data:text/plain;base64,aGVsbG8= + javascript:alert('Hello World'); + sms:123 + xmpp:foo@bar + bitcoin:Addr23?amount=1&message=foo + cid:foo-32x32.v2_fe0f1423.png + mid:960830.1639@XIson.com + postgres://user:pass@host.com:5432/path?k=v#f + zoommtg://zoom.us/join?confno=1234&pwd=xxx + + https://www.example.com/test space + https://www.example.com, it's cool! + + /path/to/resource + www.example.com + www.example.com/resources + www.example.com/search?q=openai + 192.168.1.1 + 192.168.1.1:8080 + ::1 + ::ffff:0:0 + 64:ff9b:: + 64:ff9b:1:: + 100:: + 2001:: + 2001:1::1 + 2001:1::2 + 2001:2:: + 2001:3:: + 2001:4:112:: + 2001:10:: + 2001:20:: + 2002:: + 2620:4f:8000:: + fc00:: + fe80:: + [2001:0db8:85a3:0000:0000:8a2e:0370:7334] + [2001:0db8:85a3:0000:0000:8a2e:0370:7334]:8080 + `, + []string{ + "https://www.example.com", + "http://www.example.com:8080/page", + "https://www.example.com/search?q=openai", + "https://www.example.com/resources/docs/guide", + "https://www.example.com/about#team", + "https://user:password@example.com", + "https://www.example.com:8080/search?q=openai#results", + "https://example.com/products?id=50&category=books", + "https://example.com/search?name=John%20Doe&age=30", + "foo://example.com", + "mailto:user@example.com", + "ftp://user:password@ftp.example.com:21", + "https://www.example.com/test", + "https://www.example.com", + "www.example.com", + "www.example.com/resources", + "www.example.com/search?q=openai", + }, + }, + } + + for _, tc := range testCases { + got := regex.FindAllString(tc.text, -1) + + if !equalSlices(got, tc.want) { + t.Errorf("Extracted URLs = %v, want %v", got, tc.want) + } + } +} + +// equalSlices checks if two slices of strings are equal. +func equalSlices(a, b []string) bool { + if len(a) != len(b) { + return false + } + + for i := range a { + if a[i] != b[i] { + return false + } + } + + return true +} diff --git a/gen/TLDs/main.go b/gen/TLDs/main.go new file mode 100644 index 0000000..dd03487 --- /dev/null +++ b/gen/TLDs/main.go @@ -0,0 +1,235 @@ +package main + +import ( + "bufio" + "flag" + "fmt" + "log" + "net/http" + "os" + "regexp" + "sort" + "strings" + "text/template" +) + +var ( + // Output file path for the generated Go source file + output string + + // Template for the autogenerated Go file containing the list of TLDs + tmpl = template.Must(template.New("schemes").Parse(`// This file is autogenerated by the TLDs generator. Please do not edit manually. +package tlds + +// Official is a sorted list of public TLDs and eTLDs. +// The list is fetched from: +// - https://data.iana.org/TLD/tlds-alpha-by-domain.txt +// - https://publicsuffix.org/list/public_suffix_list.dat +var Official = []string{ +{{- range $_, $TLD := .TLDs}} + "{{$TLD}}", +{{- end}} +} +`)) +) + +func init() { + // Define the command-line flag for output file path + flag.StringVar(&output, "output", "", "Specify the output file path for the generated Go source file.") + + // Custom usage message for the command-line flag + flag.Usage = func() { + h := "USAGE:\n" + h += " schemes [OPTIONS]\n" + + h += "\nOPTIONS:\n" + h += " -output string Specify the output file path for the generated Go source file.\n" + + fmt.Fprintln(os.Stderr, h) + } + + // Parse command-line flags + flag.Parse() +} + +func main() { + // Ensure that an output file path is specified + if output == "" { + log.Fatalln("Output file path is required. Use -output to specify the output file path.") + } + + log.Printf("Generating %s...\n", output) + + // Fetch TLDs from IANA + TLDs, err := getTLDsFromIANA() + if err != nil { + log.Fatalf("Failed to get TLDs from IANA: %v\n", err) + } + + // Fetch effective TLDs from the Public Suffix list + eTLDs, err := getEffectiveTLDsFromPublicSuffix() + if err != nil { + log.Fatalf("Failed to get effective TLDs from Public Suffix: %v\n", err) + } + + // Combine both TLDs and eTLDs + TLDs = append(TLDs, eTLDs...) + + // Sort the combined list of TLDs + sort.Strings(TLDs) + + // Remove duplicate entries + TLDs = removeDuplicates(TLDs) + + // Write the TLDs to the output file + if err := writeTLDsToFile(TLDs, output); err != nil { + log.Fatalf("Failed to write schemes to file: %v\n", err) + } + + log.Println("TLDs file generated successfully.") +} + +// getTLDsFromIANA fetches the list of TLDs from the IANA TLD list and returns them. +func getTLDsFromIANA() (TLDs []string, err error) { + // Perform HTTP GET request to fetch the IANA TLD list + var res *http.Response + + res, err = http.Get("https://data.iana.org/TLD/tlds-alpha-by-domain.txt") + if err != nil { + err = fmt.Errorf("failed to fetch IANA TLDs: %w", err) + + return + } + + defer res.Body.Close() + + // Regular expression to match valid TLD entries (ignore comments) + re := regexp.MustCompile(`^[^#]+$`) + + // Scan through the response body line by line + scanner := bufio.NewScanner(res.Body) + + for scanner.Scan() { + line := scanner.Text() + + line = strings.TrimSpace(line) + line = strings.ToLower(line) + + // Extract valid TLDs (skip comments and entries starting with "xn--") + TLD := re.FindString(line) + + if TLD == "" || strings.HasPrefix(TLD, "xn--") { + continue + } + + TLDs = append(TLDs, TLD) + } + + // Check for errors during scanning + if err = scanner.Err(); err != nil { + err = fmt.Errorf("scanner error: %w", err) + + return + } + + return +} + +func getEffectiveTLDsFromPublicSuffix() (eTLDs []string, err error) { + // Perform HTTP GET request to fetch the Public Suffix list + var res *http.Response + + res, err = http.Get("https://publicsuffix.org/list/effective_tld_names.dat") + if err != nil { + err = fmt.Errorf("failed to fetch Public Suffix TLDs: %w", err) + + return + } + + defer res.Body.Close() + + // Scan through the response body line by line + scanner := bufio.NewScanner(res.Body) + + for scanner.Scan() { + line := scanner.Text() + + line = strings.TrimSpace(line) + + // Stop reading when encountering private domain section + if strings.HasPrefix(line, "// ===BEGIN PRIVATE DOMAINS") { + break + } + + // Skip comments + if strings.HasPrefix(line, "//") { + continue + } + + TLD := line + + // Remove special characters + TLD = strings.ReplaceAll(TLD, "*.", "") + TLD = strings.ReplaceAll(TLD, "!", "") + + if TLD == "" { + continue + } + + eTLDs = append(eTLDs, TLD) + } + + // Check for errors during scanning + if err = scanner.Err(); err != nil { + err = fmt.Errorf("scanner error: %w", err) + + return + } + + return +} + +// removeDuplicates +// removes duplicate elements from a slice of any type that satisfies the comparable constraint. +func removeDuplicates[T comparable](slice []T) []T { + keys := make(map[T]bool) + + var list []T + + for _, entry := range slice { + if _, exists := keys[entry]; !exists { + keys[entry] = true + + list = append(list, entry) + } + } + + return list +} + +// writeTLDsToFile writes the generated list of URI schemes to the specified file +// using a Go source file template. +func writeTLDsToFile(TLDs []string, output string) (err error) { + // Create the output file + file, err := os.Create(output) + if err != nil { + err = fmt.Errorf("failed to create output file: %w", err) + + return + } + + defer file.Close() + + // Execute the template and write to the output file + data := struct { + TLDs []string + }{ + TLDs: TLDs, + } + + if err := tmpl.Execute(file, data); err != nil { + return fmt.Errorf("failed to execute template: %w", err) + } + + return +} diff --git a/gen/schemes/main.go b/gen/schemes/main.go new file mode 100644 index 0000000..56df3e9 --- /dev/null +++ b/gen/schemes/main.go @@ -0,0 +1,158 @@ +package main + +import ( + "encoding/csv" + "errors" + "flag" + "fmt" + "io" + "log" + "net/http" + "os" + "strings" + "text/template" +) + +var ( + // Output file path for the generated Go source file + output string + + // Template for the autogenerated Go file containing the list of schemes + schemesTmpl = template.Must(template.New("schemes").Parse(`// This file is autogenerated by the schemes generator. Please do not edit manually. +package schemes + +// Official is a sorted list of all IANA assigned schemes. +// This list is fetched from: +// - https://www.iana.org/assignments/uri-schemes/uri-schemes-1.csv +var Official = []string{ +{{- range $scheme := .Schemes}} + "{{$scheme}}", +{{- end}} +} +`)) +) + +func init() { + // Define the command-line flag for output file path + flag.StringVar(&output, "output", "", "Specify the output file path for the generated Go source file.") + + // Custom usage message for the command-line flag + flag.Usage = func() { + h := "USAGE:\n" + h += " schemes [OPTIONS]\n" + + h += "\nOPTIONS:\n" + h += " -output string Specify the output file path for the generated Go source file.\n" + + fmt.Fprintln(os.Stderr, h) + } + + // Parse command-line flags + flag.Parse() +} + +func main() { + // Ensure that an output file path is specified + if output == "" { + log.Fatalln("Output file path is required. Use -output to specify the output file path.") + } + + log.Printf("Generating %s...\n", output) + + // Fetch and generate the list of URI schemes + schemes, err := fetchSchemesList() + if err != nil { + log.Fatalf("Failed to fetch schemes: %v\n", err) + } + + // Write the schemes to the output file + if err := writeSchemesToFile(schemes, output); err != nil { + log.Fatalf("Failed to write schemes to file: %v\n", err) + } + + log.Println("Schemes file generated successfully.") +} + +// fetchSchemesList fetches the list of URI schemes from the IANA CSV file +// and returns a slice of valid scheme names. +func fetchSchemesList() (schemes []string, err error) { + // Perform HTTP GET request to fetch the CSV file + schemesSourcesURL := "https://www.iana.org/assignments/uri-schemes/uri-schemes-1.csv" + + var res *http.Response + + res, err = http.Get(schemesSourcesURL) + if err != nil { + err = fmt.Errorf("failed to fetch the schemes CSV: %w", err) + + return + } + + defer res.Body.Close() + + // Create a new CSV reader for parsing the response body + reader := csv.NewReader(res.Body) + + // Skip the CSV header row + if _, err = reader.Read(); err != nil { + err = fmt.Errorf("failed to read CSV header: %w", err) + + return + } + + for { + var record []string + + record, err = reader.Read() + + // End of file reached + if errors.Is(err, io.EOF) { + err = nil + + break + } + + if err != nil { + err = fmt.Errorf("error reading CSV row: %w", err) + + return + } + + // Skip rows marked as OBSOLETE + if strings.Contains(record[0], "OBSOLETE") { + continue // skip obsolete schemes; note the scheme column is abused + } + + // Append valid scheme to the list + schemes = append(schemes, record[0]) + } + + return +} + +// writeSchemesToFile writes the generated list of URI schemes to the specified file +// using a Go source file template. +func writeSchemesToFile(schemes []string, output string) (err error) { + // Create the output file + file, err := os.Create(output) + if err != nil { + err = fmt.Errorf("failed to create output file: %w", err) + + return + } + + defer file.Close() + + // Execute the template and write to the output file + data := struct { + Schemes []string + }{ + Schemes: schemes, + } + + if err := schemesTmpl.Execute(file, data); err != nil { + return fmt.Errorf("failed to execute template: %w", err) + } + + return +} diff --git a/gen/unicodes/main.go b/gen/unicodes/main.go new file mode 100644 index 0000000..033b89f --- /dev/null +++ b/gen/unicodes/main.go @@ -0,0 +1,195 @@ +package main + +import ( + "flag" + "fmt" + "log" + "os" + "strconv" + "strings" + "text/template" + "unicode" +) + +var ( + // Output file path for the generated Go source file + output string + + // Template for the autogenerated Go file containing the list of schemes + tmpl = template.Must(template.New("schemes").Parse(`// This file is autogenerated by the unicodes generator. Please do not edit manually. +package unicodes + +const AllowedUcsChar = {{.withPunc}} + +const AllowedUcsCharMinusPunc = {{.withoutPunc}} +`)) +) + +func init() { + // Define the command-line flag for output file path + flag.StringVar(&output, "output", "", "Specify the output file path for the generated Go source file.") + + // Custom usage message for the command-line flag + flag.Usage = func() { + h := "USAGE:\n" + h += " schemes [OPTIONS]\n" + + h += "\nOPTIONS:\n" + h += " -output string Specify the output file path for the generated Go source file.\n" + + fmt.Fprintln(os.Stderr, h) + } + + // Parse command-line flags + flag.Parse() +} + +func main() { + // Ensure that an output file path is specified + if output == "" { + log.Fatalln("Output file path is required. Use -output to specify the output file path.") + } + + log.Printf("Generating %s...\n", output) + + if err := writeUnicode(); err != nil { + log.Fatal(err) + } + + log.Println("Unicodes file generated successfully.") +} + +func visit(rt *unicode.RangeTable, fn func(rune)) { + for _, r16 := range rt.R16 { + for r := rune(r16.Lo); r <= rune(r16.Hi); r += rune(r16.Stride) { + fn(r) + } + } + + for _, r32 := range rt.R32 { + for r := rune(r32.Lo); r <= rune(r32.Hi); r += rune(r32.Stride) { + fn(r) + } + } +} + +func writeUnicode() error { + // rfc3987Ranges contains the ranges of valid code points specified by RFC 3987. + rfc3987Ranges := [][2]rune{ + {0xA0, 0xD7FF}, + {0xF900, 0xFDCF}, + {0xFDF0, 0xFFEF}, + {0x10000, 0x1FFFD}, + {0x20000, 0x2FFFD}, + {0x30000, 0x3FFFD}, + {0x40000, 0x4FFFD}, + {0x50000, 0x5FFFD}, + {0x60000, 0x6FFFD}, + {0x70000, 0x7FFFD}, + {0x80000, 0x8FFFD}, + {0x90000, 0x9FFFD}, + {0xA0000, 0xAFFFD}, + {0xB0000, 0xBFFFD}, + {0xC0000, 0xCFFFD}, + {0xD0000, 0xDFFFD}, + {0xE1000, 0xEFFFD}, + } + + // removeRune accepts a slice of inclusive code point ranges (in ascending order) + // and returns a new slice that is equivalent except for excluding a specified rune + // by removing/replacing/splitting any range containing it. + // Its linear searches over the ranges (including those added by previous invocations) + // are inefficient, but acceptable because this code runs only at build time. + removeRune := func(ranges [][2]rune, cp rune) [][2]rune { + for i, r := range ranges { + // Ranges are in ascending order. Skip any that precede `cp`, + // and bail out upon reaching one that follows `cp`. + if r[1] < cp { + continue + } else if cp < r[0] { + break + } + + // `cp` is in this range and must be removed from it. + if cp == r[0] && cp == r[1] { + // Remove this single-element range. + return append(ranges[0:i], ranges[i+1:]...) + } else if cp == r[0] { + // Remove the first element of this range. + newRange := [2]rune{r[0] + 1, r[1]} + newTail := append([][2]rune{newRange}, ranges[i+1:]...) + + return append(ranges[0:i], newTail...) + } else if cp == r[1] { + // Remove the last element of this range. + newRange := [2]rune{r[0], r[1] - 1} + newTail := append([][2]rune{newRange}, ranges[i+1:]...) + + return append(ranges[0:i], newTail...) + } else { + // Split this range. + newTail := append( + [][2]rune{ + {r[0], cp - 1}, + {cp + 1, r[1]}, + }, + ranges[i+1:]...) + + return append(ranges[0:i], newTail...) + } + } + + return ranges + } + + // sepFreeRanges excludes separators from rfc3987Ranges. + sepFreeRanges := append([][2]rune{}, rfc3987Ranges...) + + visit(unicode.Z, func(cp rune) { + sepFreeRanges = removeRune(sepFreeRanges, cp) + }) + + // puncFreeRanges excludes punctuation from sepFreeRanges. + puncFreeRanges := append([][2]rune{}, sepFreeRanges...) + + visit(unicode.Po, func(cp rune) { + puncFreeRanges = removeRune(puncFreeRanges, cp) + }) + + // Build the corresponding regular expression character class contents. + characterClassContents := func(ranges [][2]rune) strings.Builder { + var builder strings.Builder + + for _, r := range ranges { + // regexp.QuoteMeta is not necessary because all metacharacters are ASCII. + // cf. https://golang.org/s/re2syntax and + // https://cs.opensource.google/go/go/+/refs/tags/go1.17.6:src/regexp/regexp.go;l=721 + builder.WriteRune(r[0]) + + if r[0] == r[1] { + continue + } + + builder.WriteRune('-') + builder.WriteRune(r[1]) + } + + return builder + } + + allowedUcsChar := characterClassContents(sepFreeRanges) + allowedUcsCharMinusPunc := characterClassContents(puncFreeRanges) + + // Write to file. + f, err := os.Create(output) + if err != nil { + return err + } + + defer f.Close() + + return tmpl.Execute(f, map[string]string{ + "withPunc": strconv.Quote(allowedUcsChar.String()), + "withoutPunc": strconv.Quote(allowedUcsCharMinusPunc.String()), + }) +} diff --git a/go.mod b/go.mod new file mode 100644 index 0000000..9f7e329 --- /dev/null +++ b/go.mod @@ -0,0 +1,3 @@ +module github.com/hueristiq/hq-go-url + +go 1.23.1 diff --git a/go.sum b/go.sum new file mode 100644 index 0000000..e69de29 diff --git a/schemes/schemes_no_authority.go b/schemes/schemes_no_authority.go new file mode 100644 index 0000000..d00f6a7 --- /dev/null +++ b/schemes/schemes_no_authority.go @@ -0,0 +1,16 @@ +package schemes + +// NoAuthority is a sorted list of some well-known url schemes that are +// followed by ":" instead of "://". The list includes both officially +// registered and unofficial schemes. +var NoAuthority = []string{ + `bitcoin`, // Bitcoin + `cid`, // Content-ID + `file`, // Files + `magnet`, // Torrent magnets + `mailto`, // Mail + `mid`, // Message-ID + `sms`, // SMS + `tel`, // Telephone + `xmpp`, // XMPP +} diff --git a/schemes/schemes_official.go b/schemes/schemes_official.go new file mode 100644 index 0000000..740f5f4 --- /dev/null +++ b/schemes/schemes_official.go @@ -0,0 +1,391 @@ +// This file is autogenerated by the schemes generator. Please do not edit manually. +package schemes + +// Official is a sorted list of all IANA assigned schemes. +// This list is fetched from: +// - https://www.iana.org/assignments/uri-schemes/uri-schemes-1.csv +var Official = []string{ + "aaa", + "aaas", + "about", + "acap", + "acct", + "acd", + "acr", + "adiumxtra", + "adt", + "afp", + "afs", + "aim", + "amss", + "android", + "appdata", + "apt", + "ar", + "ark", + "at", + "attachment", + "aw", + "barion", + "bb", + "beshare", + "bitcoin", + "bitcoincash", + "blob", + "bluetooth", + "bolo", + "brid", + "browserext", + "cabal", + "calculator", + "callto", + "cap", + "cast", + "casts", + "chrome", + "chrome-extension", + "cid", + "coap", + "coap+tcp", + "coap+ws", + "coaps", + "coaps+tcp", + "coaps+ws", + "com-eventbrite-attendee", + "content", + "content-type", + "crid", + "cstr", + "cvs", + "dab", + "dat", + "data", + "dav", + "dhttp", + "diaspora", + "dict", + "did", + "dis", + "dlna-playcontainer", + "dlna-playsingle", + "dns", + "dntp", + "doi", + "dpp", + "drm", + "drop", + "dtmi", + "dtn", + "dvb", + "dvx", + "dweb", + "ed2k", + "eid", + "elsi", + "embedded", + "ens", + "ethereum", + "example", + "facetime", + "fax", + "feed", + "feedready", + "fido", + "file", + "filesystem", + "finger", + "first-run-pen-experience", + "fish", + "fm", + "ftp", + "fuchsia-pkg", + "geo", + "gg", + "git", + "gitoid", + "gizmoproject", + "go", + "gopher", + "graph", + "grd", + "gtalk", + "h323", + "ham", + "hcap", + "hcp", + "hs20", + "http", + "https", + "hxxp", + "hxxps", + "hydrazone", + "hyper", + "iax", + "icap", + "icon", + "im", + "imap", + "info", + "iotdisco", + "ipfs", + "ipn", + "ipns", + "ipp", + "ipps", + "irc", + "irc6", + "ircs", + "iris", + "iris.beep", + "iris.lwz", + "iris.xpc", + "iris.xpcs", + "isostore", + "itms", + "jabber", + "jar", + "jms", + "keyparc", + "lastfm", + "lbry", + "ldap", + "ldaps", + "leaptofrogans", + "lid", + "lorawan", + "lpa", + "lvlt", + "machineProvisioningProgressReporter", + "magnet", + "mailserver", + "mailto", + "maps", + "market", + "matrix", + "message", + "microsoft.windows.camera", + "microsoft.windows.camera.multipicker", + "microsoft.windows.camera.picker", + "mid", + "mms", + "modem", + "mongodb", + "moz", + "ms-access", + "ms-appinstaller", + "ms-browser-extension", + "ms-calculator", + "ms-drive-to", + "ms-enrollment", + "ms-excel", + "ms-eyecontrolspeech", + "ms-gamebarservices", + "ms-gamingoverlay", + "ms-getoffice", + "ms-help", + "ms-infopath", + "ms-inputapp", + "ms-launchremotedesktop", + "ms-lockscreencomponent-config", + "ms-media-stream-id", + "ms-meetnow", + "ms-mixedrealitycapture", + "ms-mobileplans", + "ms-newsandinterests", + "ms-officeapp", + "ms-people", + "ms-personacard", + "ms-project", + "ms-powerpoint", + "ms-publisher", + "ms-recall", + "ms-remotedesktop", + "ms-remotedesktop-launch", + "ms-restoretabcompanion", + "ms-screenclip", + "ms-screensketch", + "ms-search", + "ms-search-repair", + "ms-secondary-screen-controller", + "ms-secondary-screen-setup", + "ms-settings", + "ms-settings-airplanemode", + "ms-settings-bluetooth", + "ms-settings-camera", + "ms-settings-cellular", + "ms-settings-cloudstorage", + "ms-settings-connectabledevices", + "ms-settings-displays-topology", + "ms-settings-emailandaccounts", + "ms-settings-language", + "ms-settings-location", + "ms-settings-lock", + "ms-settings-nfctransactions", + "ms-settings-notifications", + "ms-settings-power", + "ms-settings-privacy", + "ms-settings-proximity", + "ms-settings-screenrotation", + "ms-settings-wifi", + "ms-settings-workplace", + "ms-spd", + "ms-stickers", + "ms-sttoverlay", + "ms-transit-to", + "ms-useractivityset", + "ms-virtualtouchpad", + "ms-visio", + "ms-walk-to", + "ms-whiteboard", + "ms-whiteboard-cmd", + "ms-word", + "msnim", + "msrp", + "msrps", + "mss", + "mt", + "mtqp", + "mumble", + "mupdate", + "mvn", + "mvrp", + "mvrps", + "news", + "nfs", + "ni", + "nih", + "nntp", + "notes", + "num", + "ocf", + "oid", + "onenote", + "onenote-cmd", + "opaquelocktoken", + "openid", + "openpgp4fpr", + "otpauth", + "p1", + "pack", + "palm", + "paparazzi", + "payment", + "payto", + "pkcs11", + "platform", + "pop", + "pres", + "prospero", + "proxy", + "pwid", + "psyc", + "pttp", + "qb", + "query", + "quic-transport", + "redis", + "rediss", + "reload", + "res", + "resource", + "rmi", + "rsync", + "rtmfp", + "rtmp", + "rtsp", + "rtsps", + "rtspu", + "sarif", + "secondlife", + "secret-token", + "service", + "session", + "sftp", + "sgn", + "shc", + "sieve", + "simpleledger", + "simplex", + "sip", + "sips", + "skype", + "smb", + "smp", + "sms", + "smtp", + "snews", + "snmp", + "soap.beep", + "soap.beeps", + "soldat", + "spiffe", + "spotify", + "ssb", + "ssh", + "starknet", + "steam", + "stun", + "stuns", + "submit", + "svn", + "swh", + "swid", + "swidpath", + "tag", + "taler", + "teamspeak", + "tel", + "teliaeid", + "telnet", + "tftp", + "things", + "thismessage", + "thzp", + "tip", + "tn3270", + "tool", + "turn", + "turns", + "tv", + "udp", + "unreal", + "upt", + "urn", + "ut2004", + "uuid-in-package", + "v-event", + "vemmi", + "ventrilo", + "ves", + "videotex", + "vnc", + "view-source", + "vscode", + "vscode-insiders", + "vsls", + "w3", + "wais", + "web3", + "wcr", + "webcal", + "web+ap", + "wifi", + "wpid", + "ws", + "wss", + "wtai", + "wyciwyg", + "xcon", + "xcon-userid", + "xfire", + "xmlrpc.beep", + "xmlrpc.beeps", + "xmpp", + "xftp", + "xrcp", + "xri", + "ymsgr", + "z39.50", + "z39.50r", + "z39.50s", +} diff --git a/schemes/schemes_unoficial.go b/schemes/schemes_unoficial.go new file mode 100644 index 0000000..40f8e9d --- /dev/null +++ b/schemes/schemes_unoficial.go @@ -0,0 +1,16 @@ +package schemes + +// Unofficial is a sorted list of some well-known url schemes which +// aren't officially registered just yet. They tend to correspond to software. +// +// Mostly collected from https://en.wikipedia.org/wiki/List_of_URI_schemes#Unofficial_but_common_URI_schemes. +var Unofficial = []string{ + `gemini`, // gemini + `jdbc`, // Java database Connectivity + `moz-extension`, // Firefox extension + `postgres`, // PostgreSQL (short form) + `postgresql`, // PostgreSQL + `slack`, // Slack + `zoommtg`, // Zoom (desktop) + `zoomus`, // Zoom (mobile) +} diff --git a/tlds/tlds_official.go b/tlds/tlds_official.go new file mode 100644 index 0000000..3ee6806 --- /dev/null +++ b/tlds/tlds_official.go @@ -0,0 +1,6884 @@ +// This file is autogenerated by the TLDs generator. Please do not edit manually. +package tlds + +// Official is a sorted list of public TLDs and eTLDs. +// The list is fetched from: +// - https://data.iana.org/TLD/tlds-alpha-by-domain.txt +// - https://publicsuffix.org/list/public_suffix_list.dat +var Official = []string{ + "0.bg", + "1.bg", + "2.bg", + "2000.hu", + "3.bg", + "4.bg", + "5.bg", + "5g.in", + "6.bg", + "6g.in", + "7.bg", + "8.bg", + "9.bg", + "9guacu.br", + "a.bg", + "a.se", + "aa.no", + "aaa", + "aaa.pro", + "aarborte.no", + "aarp", + "ab.ca", + "abashiri.hokkaido.jp", + "abb", + "abbott", + "abbvie", + "abc", + "abc.br", + "abeno.osaka.jp", + "abiko.chiba.jp", + "abira.hokkaido.jp", + "able", + "abo.pa", + "abogado", + "abr.it", + "abruzzo.it", + "abu.yamaguchi.jp", + "abudhabi", + "ac", + "ac.ae", + "ac.at", + "ac.be", + "ac.ci", + "ac.cn", + "ac.cr", + "ac.cy", + "ac.fj", + "ac.gn", + "ac.gov.br", + "ac.id", + "ac.il", + "ac.im", + "ac.in", + "ac.ir", + "ac.jp", + "ac.ke", + "ac.kr", + "ac.lk", + "ac.ls", + "ac.ma", + "ac.me", + "ac.mu", + "ac.mw", + "ac.mz", + "ac.ni", + "ac.nz", + "ac.pa", + "ac.pk", + "ac.pr", + "ac.rs", + "ac.rw", + "ac.se", + "ac.sz", + "ac.th", + "ac.tj", + "ac.tz", + "ac.ug", + "ac.uk", + "ac.vn", + "ac.za", + "ac.zm", + "ac.zw", + "aca.pro", + "academia.bo", + "academy", + "accenture", + "accident-investigation.aero", + "accident-prevention.aero", + "accountant", + "accountants", + "acct.pro", + "achi.nagano.jp", + "aco", + "act.au", + "act.edu.au", + "actor", + "ad", + "ad.jp", + "adachi.tokyo.jp", + "adm.br", + "ads", + "adult", + "adult.ht", + "adv.br", + "adv.mz", + "ae", + "aeg", + "aejrie.no", + "aero", + "aero.mv", + "aero.tt", + "aerobatic.aero", + "aeroclub.aero", + "aerodrome.aero", + "aetna", + "af", + "afjord.no", + "afl", + "africa", + "africa.bj", + "ag", + "ag.it", + "aga.niigata.jp", + "agakhan", + "agano.niigata.jp", + "agdenes.no", + "agematsu.nagano.jp", + "agency", + "agents.aero", + "agr.br", + "agrar.hu", + "agric.za", + "agrigento.it", + "agro.bj", + "agro.bo", + "agro.pl", + "aguni.okinawa.jp", + "ah.cn", + "ah.no", + "ai", + "ai.in", + "ai.vn", + "aibetsu.hokkaido.jp", + "aichi.jp", + "aid.pl", + "aig", + "aikawa.kanagawa.jp", + "ainan.ehime.jp", + "aioi.hyogo.jp", + "aip.ee", + "air-surveillance.aero", + "air-traffic-control.aero", + "airbus", + "aircraft.aero", + "airforce", + "airline.aero", + "airport.aero", + "airtel", + "airtraffic.aero", + "aisai.aichi.jp", + "aisho.shiga.jp", + "aizubange.fukushima.jp", + "aizumi.tokushima.jp", + "aizumisato.fukushima.jp", + "aizuwakamatsu.fukushima.jp", + "aju.br", + "ak.us", + "akabira.hokkaido.jp", + "akagi.shimane.jp", + "akaiwa.okayama.jp", + "akashi.hyogo.jp", + "akdn", + "aki.kochi.jp", + "akiruno.tokyo.jp", + "akishima.tokyo.jp", + "akita.akita.jp", + "akita.jp", + "akkeshi.hokkaido.jp", + "aknoluokta.no", + "ako.hyogo.jp", + "akrehamn.no", + "akune.kagoshima.jp", + "al", + "al.gov.br", + "al.it", + "al.no", + "al.us", + "alaheadju.no", + "aland.fi", + "alessandria.it", + "alesund.no", + "algard.no", + "alibaba", + "alipay", + "allfinanz", + "allstate", + "ally", + "alsace", + "alstahaug.no", + "alstom", + "alt.za", + "alta.no", + "alto-adige.it", + "altoadige.it", + "alvdal.no", + "am", + "am.br", + "am.gov.br", + "am.in", + "ama.aichi.jp", + "ama.shimane.jp", + "amagasaki.hyogo.jp", + "amakusa.kumamoto.jp", + "amami.kagoshima.jp", + "amazon", + "ambulance.aero", + "americanexpress", + "americanfamily", + "amex", + "amfam", + "ami.ibaraki.jp", + "amica", + "amli.no", + "amot.no", + "amsterdam", + "an.it", + "analytics", + "anamizu.ishikawa.jp", + "anan.nagano.jp", + "anan.tokushima.jp", + "anani.br", + "ancona.it", + "andasuolo.no", + "andebu.no", + "ando.nara.jp", + "andoy.no", + "andria-barletta-trani.it", + "andria-trani-barletta.it", + "andriabarlettatrani.it", + "andriatranibarletta.it", + "android", + "andøy.no", + "angiang.vn", + "anjo.aichi.jp", + "ann-arbor.mi.us", + "annaka.gunma.jp", + "anpachi.gifu.jp", + "anquan", + "anz", + "ao", + "ao.it", + "aogaki.hyogo.jp", + "aogashima.tokyo.jp", + "aoki.nagano.jp", + "aol", + "aomori.aomori.jp", + "aomori.jp", + "aosta-valley.it", + "aosta.it", + "aostavalley.it", + "aoste.it", + "ap.gov.br", + "ap.gov.pl", + "ap.it", + "aparecida.br", + "apartments", + "app", + "app.br", + "apple", + "aq", + "aq.it", + "aquarelle", + "aquila.it", + "ar", + "ar.it", + "ar.us", + "arab", + "arai.shizuoka.jp", + "arakawa.saitama.jp", + "arakawa.tokyo.jp", + "aramco", + "arao.kumamoto.jp", + "archi", + "architectes.bj", + "ardal.no", + "aremark.no", + "arendal.no", + "arezzo.it", + "ariake.saga.jp", + "arida.wakayama.jp", + "aridagawa.wakayama.jp", + "arita.saga.jp", + "army", + "arna.no", + "arpa", + "arq.br", + "art", + "art.br", + "art.do", + "art.dz", + "art.ht", + "art.sn", + "arte", + "arte.bo", + "arts.co", + "arts.nf", + "arts.ro", + "arts.ve", + "as", + "as.us", + "asago.hyogo.jp", + "asahi.chiba.jp", + "asahi.ibaraki.jp", + "asahi.mie.jp", + "asahi.nagano.jp", + "asahi.toyama.jp", + "asahi.yamagata.jp", + "asahikawa.hokkaido.jp", + "asaka.saitama.jp", + "asakawa.fukushima.jp", + "asakuchi.okayama.jp", + "asaminami.hiroshima.jp", + "ascoli-piceno.it", + "ascolipiceno.it", + "asda", + "aseral.no", + "ashibetsu.hokkaido.jp", + "ashikaga.tochigi.jp", + "ashiya.fukuoka.jp", + "ashiya.hyogo.jp", + "ashoro.hokkaido.jp", + "asia", + "asker.no", + "askim.no", + "askoy.no", + "askvoll.no", + "askøy.no", + "asn.au", + "asn.lv", + "asnes.no", + "aso.kumamoto.jp", + "ass.km", + "assabu.hokkaido.jp", + "assn.lk", + "asso.ci", + "asso.dz", + "asso.fr", + "asso.gp", + "asso.ht", + "asso.km", + "asso.mc", + "asso.nc", + "asso.re", + "associates", + "association.aero", + "assur.bj", + "asti.it", + "asuke.aichi.jp", + "at", + "at.it", + "atami.shizuoka.jp", + "athleta", + "atm.pl", + "ato.br", + "atsugi.kanagawa.jp", + "atsuma.hokkaido.jp", + "attorney", + "au", + "auction", + "audi", + "audible", + "audio", + "audnedaln.no", + "augustow.pl", + "aukra.no", + "aure.no", + "aurland.no", + "aurskog-holand.no", + "aurskog-høland.no", + "auspost", + "austevoll.no", + "austrheim.no", + "author", + "author.aero", + "auto", + "auto.pl", + "autos", + "av.it", + "av.tr", + "avellino.it", + "averoy.no", + "averøy.no", + "avocat.pro", + "avocats.bj", + "avoues.fr", + "aw", + "awaji.hyogo.jp", + "aws", + "ax", + "axa", + "aya.miyazaki.jp", + "ayabe.kyoto.jp", + "ayagawa.kagawa.jp", + "ayase.kanagawa.jp", + "az", + "az.us", + "azumino.nagano.jp", + "azure", + "aéroport.ci", + "b.bg", + "b.br", + "b.se", + "ba", + "ba.gov.br", + "ba.it", + "babia-gora.pl", + "baby", + "bacgiang.vn", + "backan.vn", + "baclieu.vn", + "bacninh.vn", + "badaddja.no", + "bahcavuotna.no", + "bahccavuotna.no", + "baidar.no", + "baidu", + "bajddar.no", + "balat.no", + "balestrand.no", + "ballangen.no", + "ballooning.aero", + "balsan-sudtirol.it", + "balsan-suedtirol.it", + "balsan-südtirol.it", + "balsan.it", + "balsfjord.no", + "bamble.no", + "banamex", + "band", + "bandai.fukushima.jp", + "bando.ibaraki.jp", + "bank", + "bar", + "bar.pro", + "barcelona", + "barclaycard", + "barclays", + "bardu.no", + "barefoot", + "bargains", + "bari.it", + "baria-vungtau.vn", + "barletta-trani-andria.it", + "barlettatraniandria.it", + "barueri.br", + "barum.no", + "bas.it", + "baseball", + "basilicata.it", + "basketball", + "bato.tochigi.jp", + "batsfjord.no", + "bauhaus", + "bayern", + "bb", + "bbc", + "bbs.tr", + "bbt", + "bbva", + "bc.ca", + "bcg", + "bcn", + "bd", + "bd.se", + "be", + "bearalvahki.no", + "bearalváhki.no", + "beardu.no", + "beats", + "beauty", + "bedzin.pl", + "beer", + "beiarn.no", + "bel.tr", + "belau.pw", + "belem.br", + "belluno.it", + "benevento.it", + "bentley", + "bentre.vn", + "beppu.oita.jp", + "berg.no", + "bergamo.it", + "bergen.no", + "berlevag.no", + "berlevåg.no", + "berlin", + "beskidy.pl", + "best", + "bestbuy", + "bet", + "bet.ar", + "bet.br", + "bf", + "bg", + "bg.it", + "bh", + "bharti", + "bhz.br", + "bi", + "bi.it", + "bialowieza.pl", + "bialystok.pl", + "bib.br", + "bib.ve", + "bibai.hokkaido.jp", + "bible", + "bid", + "biei.hokkaido.jp", + "bielawa.pl", + "biella.it", + "bieszczady.pl", + "bievat.no", + "bievát.no", + "bifuka.hokkaido.jp", + "bihar.in", + "bihoro.hokkaido.jp", + "bike", + "bindal.no", + "bing", + "bingo", + "binhdinh.vn", + "binhduong.vn", + "binhphuoc.vn", + "binhthuan.vn", + "bio", + "bio.br", + "biratori.hokkaido.jp", + "birkenes.no", + "biz", + "biz.az", + "biz.bb", + "biz.cy", + "biz.et", + "biz.fj", + "biz.id", + "biz.in", + "biz.ki", + "biz.ls", + "biz.mv", + "biz.mw", + "biz.my", + "biz.ni", + "biz.nr", + "biz.pk", + "biz.pl", + "biz.pr", + "biz.ss", + "biz.tj", + "biz.tr", + "biz.tt", + "biz.vn", + "biz.zm", + "bizen.okayama.jp", + "bj", + "bj.cn", + "bjarkoy.no", + "bjarkøy.no", + "bjerkreim.no", + "bjugn.no", + "bl.it", + "black", + "blackfriday", + "blockbuster", + "blog", + "blog.bo", + "blog.br", + "bloomberg", + "blue", + "bm", + "bmd.br", + "bms", + "bmw", + "bn", + "bn.it", + "bnpparibas", + "bo", + "bo.it", + "bo.nordland.no", + "bo.telemark.no", + "boats", + "boavista.br", + "bodo.no", + "bodø.no", + "boehringer", + "bofa", + "bokn.no", + "boleslawiec.pl", + "bolivia.bo", + "bologna.it", + "bolt.hu", + "bolzano-altoadige.it", + "bolzano.it", + "bom", + "bomlo.no", + "bond", + "boo", + "book", + "booking", + "bosch", + "bostik", + "boston", + "bot", + "boutique", + "box", + "bozen-sudtirol.it", + "bozen-suedtirol.it", + "bozen-südtirol.it", + "bozen.it", + "br", + "br.it", + "bradesco", + "brand.se", + "bremanger.no", + "brescia.it", + "bridgestone", + "brindisi.it", + "broadway", + "broker", + "broker.aero", + "bronnoy.no", + "bronnoysund.no", + "brother", + "brumunddal.no", + "brussels", + "bryne.no", + "brønnøy.no", + "brønnøysund.no", + "bs", + "bs.it", + "bsb.br", + "bt", + "bt.it", + "bu.no", + "budejju.no", + "build", + "builders", + "bulsan-sudtirol.it", + "bulsan-suedtirol.it", + "bulsan-südtirol.it", + "bulsan.it", + "bungoono.oita.jp", + "bungotakada.oita.jp", + "bunkyo.tokyo.jp", + "busan.kr", + "business", + "business.in", + "buy", + "buzen.fukuoka.jp", + "buzz", + "bv", + "bw", + "by", + "bydgoszcz.pl", + "bygland.no", + "bykle.no", + "bytom.pl", + "bz", + "bz.it", + "bzh", + "báhcavuotna.no", + "báhccavuotna.no", + "báidár.no", + "bájddar.no", + "bálát.no", + "bådåddjå.no", + "båtsfjord.no", + "bærum.no", + "bø.nordland.no", + "bø.telemark.no", + "bømlo.no", + "c.bg", + "c.se", + "ca", + "ca.in", + "ca.it", + "ca.na", + "ca.us", + "caa.aero", + "cab", + "cafe", + "cagliari.it", + "cahcesuolo.no", + "cal", + "cal.it", + "calabria.it", + "call", + "caltanissetta.it", + "calvinklein", + "cam", + "cam.it", + "camau.vn", + "camera", + "camp", + "campania.it", + "campidano-medio.it", + "campidanomedio.it", + "campinagrande.br", + "campinas.br", + "campobasso.it", + "canon", + "cantho.vn", + "caobang.vn", + "capetown", + "capital", + "capitalone", + "car", + "caravan", + "carbonia-iglesias.it", + "carboniaiglesias.it", + "cards", + "care", + "career", + "careers", + "cargo.aero", + "carrara-massa.it", + "carraramassa.it", + "cars", + "casa", + "case", + "caserta.it", + "cash", + "casino", + "casino.hu", + "cat", + "catania.it", + "catanzaro.it", + "catering", + "catering.aero", + "catholic", + "catholic.edu.au", + "caxias.br", + "cb.it", + "cba", + "cbn", + "cbre", + "cc", + "cc.ak.us", + "cc.al.us", + "cc.ar.us", + "cc.as.us", + "cc.az.us", + "cc.ca.us", + "cc.co.us", + "cc.ct.us", + "cc.dc.us", + "cc.de.us", + "cc.fl.us", + "cc.ga.us", + "cc.gu.us", + "cc.hi.us", + "cc.ia.us", + "cc.id.us", + "cc.il.us", + "cc.in.us", + "cc.ks.us", + "cc.ky.us", + "cc.la.us", + "cc.ma.us", + "cc.md.us", + "cc.me.us", + "cc.mi.us", + "cc.mn.us", + "cc.mo.us", + "cc.ms.us", + "cc.mt.us", + "cc.na", + "cc.nc.us", + "cc.nd.us", + "cc.ne.us", + "cc.nh.us", + "cc.nj.us", + "cc.nm.us", + "cc.nv.us", + "cc.ny.us", + "cc.oh.us", + "cc.ok.us", + "cc.or.us", + "cc.pa.us", + "cc.pr.us", + "cc.ri.us", + "cc.sc.us", + "cc.sd.us", + "cc.tn.us", + "cc.tx.us", + "cc.ut.us", + "cc.va.us", + "cc.vi.us", + "cc.vt.us", + "cc.wa.us", + "cc.wi.us", + "cc.wv.us", + "cc.wy.us", + "cci.fr", + "cd", + "ce.gov.br", + "ce.it", + "center", + "ceo", + "cern", + "certification.aero", + "cesena-forli.it", + "cesena-forlì.it", + "cesenaforli.it", + "cesenaforlì.it", + "cf", + "cfa", + "cfd", + "cg", + "ch", + "ch.it", + "championship.aero", + "chanel", + "channel", + "charity", + "charter.aero", + "chase", + "chat", + "cheap", + "cherkassy.ua", + "cherkasy.ua", + "chernigov.ua", + "chernihiv.ua", + "chernivtsi.ua", + "chernovtsy.ua", + "chiba.jp", + "chichibu.saitama.jp", + "chieti.it", + "chigasaki.kanagawa.jp", + "chihayaakasaka.osaka.jp", + "chijiwa.nagasaki.jp", + "chikugo.fukuoka.jp", + "chikuho.fukuoka.jp", + "chikuhoku.nagano.jp", + "chikujo.fukuoka.jp", + "chikuma.nagano.jp", + "chikusei.ibaraki.jp", + "chikushino.fukuoka.jp", + "chikuzen.fukuoka.jp", + "chino.nagano.jp", + "chintai", + "chippubetsu.hokkaido.jp", + "chiryu.aichi.jp", + "chita.aichi.jp", + "chitose.hokkaido.jp", + "chiyoda.gunma.jp", + "chiyoda.tokyo.jp", + "chizu.tottori.jp", + "chofu.tokyo.jp", + "chonan.chiba.jp", + "chosei.chiba.jp", + "choshi.chiba.jp", + "choyo.kumamoto.jp", + "christmas", + "chrome", + "chtr.k12.ma.us", + "chungbuk.kr", + "chungnam.kr", + "chuo.chiba.jp", + "chuo.fukuoka.jp", + "chuo.osaka.jp", + "chuo.tokyo.jp", + "chuo.yamanashi.jp", + "church", + "ci", + "ci.it", + "ciencia.bo", + "cieszyn.pl", + "cim.br", + "cipriani", + "circle", + "cisco", + "citadel", + "citi", + "citic", + "city", + "city.hu", + "city.kawasaki.jp", + "city.kitakyushu.jp", + "city.kobe.jp", + "city.nagoya.jp", + "city.sapporo.jp", + "city.sendai.jp", + "city.yokohama.jp", + "civilaviation.aero", + "ck", + "ck.ua", + "cl", + "cl.it", + "claims", + "cleaning", + "click", + "clinic", + "clinique", + "clothing", + "cloud", + "club", + "club.aero", + "club.tw", + "clubmed", + "cm", + "cn", + "cn.in", + "cn.it", + "cn.ua", + "cng.br", + "cnt.br", + "co", + "co.ae", + "co.ag", + "co.am", + "co.ao", + "co.at", + "co.bb", + "co.bi", + "co.bj", + "co.bw", + "co.ci", + "co.cl", + "co.cm", + "co.cr", + "co.gg", + "co.gl", + "co.gy", + "co.hu", + "co.id", + "co.il", + "co.im", + "co.in", + "co.ir", + "co.it", + "co.je", + "co.jp", + "co.ke", + "co.kr", + "co.lc", + "co.ls", + "co.ma", + "co.me", + "co.mg", + "co.mu", + "co.mw", + "co.mz", + "co.na", + "co.ni", + "co.nz", + "co.om", + "co.pn", + "co.pw", + "co.rs", + "co.rw", + "co.st", + "co.sz", + "co.th", + "co.tj", + "co.tm", + "co.tt", + "co.tz", + "co.ug", + "co.uk", + "co.us", + "co.uz", + "co.ve", + "co.vi", + "co.za", + "co.zm", + "co.zw", + "coach", + "codes", + "coffee", + "cog.mi.us", + "college", + "cologne", + "com", + "com.ac", + "com.af", + "com.ag", + "com.ai", + "com.al", + "com.am", + "com.ar", + "com.au", + "com.aw", + "com.az", + "com.ba", + "com.bb", + "com.bh", + "com.bi", + "com.bj", + "com.bm", + "com.bn", + "com.bo", + "com.br", + "com.bs", + "com.bt", + "com.by", + "com.bz", + "com.ci", + "com.cm", + "com.cn", + "com.co", + "com.cu", + "com.cv", + "com.cw", + "com.cy", + "com.dm", + "com.do", + "com.dz", + "com.ec", + "com.ee", + "com.eg", + "com.es", + "com.et", + "com.fj", + "com.fm", + "com.fr", + "com.ge", + "com.gh", + "com.gi", + "com.gl", + "com.gn", + "com.gp", + "com.gr", + "com.gt", + "com.gu", + "com.gy", + "com.hk", + "com.hn", + "com.hr", + "com.ht", + "com.im", + "com.in", + "com.io", + "com.iq", + "com.is", + "com.jo", + "com.kg", + "com.ki", + "com.km", + "com.kp", + "com.kw", + "com.ky", + "com.kz", + "com.la", + "com.lb", + "com.lc", + "com.lk", + "com.lr", + "com.lv", + "com.ly", + "com.mg", + "com.mk", + "com.ml", + "com.mo", + "com.ms", + "com.mt", + "com.mu", + "com.mv", + "com.mw", + "com.mx", + "com.my", + "com.na", + "com.nf", + "com.ng", + "com.ni", + "com.nr", + "com.om", + "com.pa", + "com.pe", + "com.pf", + "com.ph", + "com.pk", + "com.pl", + "com.pr", + "com.ps", + "com.pt", + "com.py", + "com.qa", + "com.re", + "com.ro", + "com.sa", + "com.sb", + "com.sc", + "com.sd", + "com.sg", + "com.sh", + "com.sl", + "com.sn", + "com.so", + "com.ss", + "com.st", + "com.sv", + "com.sy", + "com.tj", + "com.tm", + "com.tn", + "com.to", + "com.tr", + "com.tt", + "com.tw", + "com.ua", + "com.ug", + "com.uy", + "com.uz", + "com.vc", + "com.ve", + "com.vi", + "com.vn", + "com.vu", + "com.ws", + "com.ye", + "com.zm", + "commbank", + "commune.am", + "community", + "como.it", + "company", + "compare", + "computer", + "comsec", + "condos", + "conf.au", + "conf.lv", + "conference.aero", + "construction", + "consulado.st", + "consultant.aero", + "consulting", + "consulting.aero", + "contact", + "contagem.br", + "contractors", + "control.aero", + "cooking", + "cool", + "coop", + "coop.ar", + "coop.br", + "coop.ht", + "coop.in", + "coop.km", + "coop.mv", + "coop.mw", + "coop.py", + "coop.rw", + "coop.tt", + "cooperativa.bo", + "corsica", + "cosenza.it", + "council.aero", + "country", + "coupon", + "coupons", + "courses", + "coz.br", + "cpa", + "cpa.pro", + "cq.cn", + "cr", + "cr.it", + "cr.ua", + "credit", + "creditcard", + "creditunion", + "cremona.it", + "crew.aero", + "cri.br", + "cri.nz", + "cricket", + "crimea.ua", + "crotone.it", + "crown", + "crs", + "cruise", + "cruises", + "cs.in", + "cs.it", + "ct.it", + "ct.us", + "cu", + "cuiaba.br", + "cuisinella", + "cuneo.it", + "curitiba.br", + "cv", + "cv.ua", + "cw", + "cx", + "cy", + "cymru", + "cyou", + "cz", + "cz.it", + "czeladz.pl", + "czest.pl", + "d.bg", + "d.se", + "dad", + "daegu.kr", + "daejeon.kr", + "daigo.ibaraki.jp", + "daisen.akita.jp", + "daito.osaka.jp", + "daiwa.hiroshima.jp", + "daklak.vn", + "daknong.vn", + "danang.vn", + "dance", + "data", + "date", + "date.fukushima.jp", + "date.hokkaido.jp", + "dating", + "datsun", + "davvenjarga.no", + "davvenjárga.no", + "davvesiida.no", + "day", + "dazaifu.fukuoka.jp", + "dc.us", + "dclk", + "dds", + "de", + "de.us", + "deal", + "dealer", + "deals", + "deatnu.no", + "def.br", + "degree", + "delhi.in", + "delivery", + "dell", + "dell-ogliastra.it", + "dellogliastra.it", + "deloitte", + "delta", + "democracia.bo", + "democrat", + "dental", + "dentist", + "dep.no", + "deporte.bo", + "des.br", + "desa.id", + "desi", + "design", + "design.aero", + "det.br", + "dev", + "dev.br", + "df.gov.br", + "dgca.aero", + "dhl", + "diamonds", + "dielddanuorri.no", + "dienbien.vn", + "diet", + "digital", + "direct", + "directory", + "discount", + "discover", + "dish", + "divtasvuodna.no", + "divttasvuotna.no", + "diy", + "dj", + "dk", + "dlugoleka.pl", + "dm", + "dn.ua", + "dnepropetrovsk.ua", + "dni.us", + "dnipropetrovsk.ua", + "dnp", + "do", + "docs", + "doctor", + "dog", + "domains", + "donetsk.ua", + "dongnai.vn", + "dongthap.vn", + "donna.no", + "doshi.yamanashi.jp", + "dot", + "dovre.no", + "download", + "dp.ua", + "dr.in", + "dr.na", + "dr.tr", + "drammen.no", + "drangedal.no", + "drive", + "drobak.no", + "drøbak.no", + "dst.mi.us", + "dtv", + "dubai", + "dunlop", + "dupont", + "durban", + "dvag", + "dvr", + "dyroy.no", + "dyrøy.no", + "dz", + "dønna.no", + "e.bg", + "e.se", + "e12.ve", + "e164.arpa", + "earth", + "eat", + "eaton.mi.us", + "ebetsu.hokkaido.jp", + "ebina.kanagawa.jp", + "ebino.miyazaki.jp", + "ebiz.tw", + "ec", + "echizen.fukui.jp", + "ecn.br", + "eco", + "eco.bj", + "eco.br", + "ecologia.bo", + "econo.bj", + "economia.bo", + "ed.ao", + "ed.ci", + "ed.cr", + "ed.jp", + "ed.pw", + "edeka", + "edogawa.tokyo.jp", + "edu", + "edu.ac", + "edu.af", + "edu.al", + "edu.ar", + "edu.au", + "edu.az", + "edu.ba", + "edu.bb", + "edu.bh", + "edu.bi", + "edu.bj", + "edu.bm", + "edu.bn", + "edu.bo", + "edu.br", + "edu.bs", + "edu.bt", + "edu.bz", + "edu.ci", + "edu.cn", + "edu.co", + "edu.cu", + "edu.cv", + "edu.cw", + "edu.dm", + "edu.do", + "edu.dz", + "edu.ec", + "edu.ee", + "edu.eg", + "edu.es", + "edu.et", + "edu.fm", + "edu.gd", + "edu.ge", + "edu.gh", + "edu.gi", + "edu.gl", + "edu.gn", + "edu.gp", + "edu.gr", + "edu.gt", + "edu.gu", + "edu.gy", + "edu.hk", + "edu.hn", + "edu.ht", + "edu.in", + "edu.iq", + "edu.is", + "edu.it", + "edu.jo", + "edu.kg", + "edu.ki", + "edu.km", + "edu.kn", + "edu.kp", + "edu.kw", + "edu.ky", + "edu.kz", + "edu.la", + "edu.lb", + "edu.lc", + "edu.lk", + "edu.lr", + "edu.ls", + "edu.lv", + "edu.ly", + "edu.me", + "edu.mg", + "edu.mk", + "edu.ml", + "edu.mn", + "edu.mo", + "edu.ms", + "edu.mt", + "edu.mv", + "edu.mw", + "edu.mx", + "edu.my", + "edu.mz", + "edu.ng", + "edu.ni", + "edu.nr", + "edu.om", + "edu.pa", + "edu.pe", + "edu.pf", + "edu.ph", + "edu.pk", + "edu.pl", + "edu.pn", + "edu.pr", + "edu.ps", + "edu.pt", + "edu.py", + "edu.qa", + "edu.rs", + "edu.sa", + "edu.sb", + "edu.sc", + "edu.sd", + "edu.sg", + "edu.sl", + "edu.sn", + "edu.so", + "edu.ss", + "edu.st", + "edu.sv", + "edu.sy", + "edu.tj", + "edu.tm", + "edu.to", + "edu.tr", + "edu.tt", + "edu.tw", + "edu.ua", + "edu.uy", + "edu.vc", + "edu.ve", + "edu.vn", + "edu.vu", + "edu.ws", + "edu.ye", + "edu.za", + "edu.zm", + "education", + "educator.aero", + "ee", + "eg", + "egersund.no", + "ehime.jp", + "eid.no", + "eidfjord.no", + "eidsberg.no", + "eidskog.no", + "eidsvoll.no", + "eigersund.no", + "eiheiji.fukui.jp", + "ekloges.cy", + "elblag.pl", + "elk.pl", + "elverum.no", + "email", + "emb.kw", + "embaixada.st", + "embetsu.hokkaido.jp", + "emerck", + "emergency.aero", + "emilia-romagna.it", + "emiliaromagna.it", + "emp.br", + "empresa.bo", + "emr.it", + "en.it", + "ena.gifu.jp", + "enebakk.no", + "energy", + "enf.br", + "eng.br", + "eng.pro", + "engerdal.no", + "engine.aero", + "engineer", + "engineer.aero", + "engineering", + "eniwa.hokkaido.jp", + "enna.it", + "ens.tn", + "enterprises", + "entertainment.aero", + "epson", + "equipment", + "equipment.aero", + "er", + "er.in", + "ericsson", + "erimo.hokkaido.jp", + "erni", + "erotica.hu", + "erotika.hu", + "es", + "es.gov.br", + "es.kr", + "esan.hokkaido.jp", + "esashi.hokkaido.jp", + "esp.br", + "esq", + "est.pr", + "estate", + "et", + "etajima.hiroshima.jp", + "etc.br", + "eti.br", + "etne.no", + "etnedal.no", + "eu", + "eu.int", + "eun.eg", + "eurovision", + "eus", + "evenassi.no", + "evenes.no", + "events", + "evenášši.no", + "evje-og-hornnes.no", + "exchange", + "exchange.aero", + "expert", + "exposed", + "express", + "express.aero", + "extraspace", + "f.bg", + "f.se", + "fage", + "fail", + "fairwinds", + "faith", + "fam.pk", + "family", + "fan", + "fans", + "far.br", + "farm", + "farmers", + "farsund.no", + "fashion", + "fast", + "fauske.no", + "fc.it", + "fe.it", + "fed.us", + "federation.aero", + "fedex", + "fedje.no", + "feedback", + "feira.br", + "fermo.it", + "ferrara.it", + "ferrari", + "ferrero", + "fet.no", + "fetsund.no", + "fg.it", + "fh.se", + "fhs.no", + "fhsk.se", + "fhv.se", + "fi", + "fi.cr", + "fi.it", + "fidelity", + "fido", + "fie.ee", + "film", + "film.hu", + "fin.ec", + "fin.tn", + "final", + "finance", + "financial", + "finnoy.no", + "finnøy.no", + "fire", + "firenze.it", + "firestone", + "firm.co", + "firm.ht", + "firm.in", + "firm.nf", + "firm.ro", + "firm.ve", + "firmdale", + "fish", + "fishing", + "fit", + "fitjar.no", + "fitness", + "fj", + "fj.cn", + "fjaler.no", + "fjell.no", + "fk", + "fl.us", + "fla.no", + "flakstad.no", + "flatanger.no", + "flekkefjord.no", + "flesberg.no", + "flickr", + "flight.aero", + "flights", + "flir", + "flog.br", + "flora.no", + "florence.it", + "floripa.br", + "florist", + "floro.no", + "florø.no", + "flowers", + "fly", + "flå.no", + "fm", + "fm.br", + "fm.it", + "fm.no", + "fnd.br", + "fo", + "foggia.it", + "folkebibl.no", + "folldal.no", + "foo", + "food", + "football", + "ford", + "forde.no", + "forex", + "forli-cesena.it", + "forlicesena.it", + "forlì-cesena.it", + "forlìcesena.it", + "forsale", + "forsand.no", + "fortal.br", + "forum", + "forum.hu", + "fosnes.no", + "fot.br", + "foundation", + "fox", + "foz.br", + "fr", + "fr.it", + "frana.no", + "fredrikstad.no", + "free", + "frei.no", + "freight.aero", + "fresenius", + "friuli-v-giulia.it", + "friuli-ve-giulia.it", + "friuli-vegiulia.it", + "friuli-venezia-giulia.it", + "friuli-veneziagiulia.it", + "friuli-vgiulia.it", + "friuliv-giulia.it", + "friulive-giulia.it", + "friulivegiulia.it", + "friulivenezia-giulia.it", + "friuliveneziagiulia.it", + "friulivgiulia.it", + "frl", + "frogans", + "frogn.no", + "froland.no", + "from.hr", + "frontier", + "frosinone.it", + "frosta.no", + "froya.no", + "fræna.no", + "frøya.no", + "fst.br", + "ftr", + "fuchu.hiroshima.jp", + "fuchu.tokyo.jp", + "fuchu.toyama.jp", + "fudai.iwate.jp", + "fuefuki.yamanashi.jp", + "fuel.aero", + "fuji.shizuoka.jp", + "fujieda.shizuoka.jp", + "fujiidera.osaka.jp", + "fujikawa.shizuoka.jp", + "fujikawa.yamanashi.jp", + "fujikawaguchiko.yamanashi.jp", + "fujimi.nagano.jp", + "fujimi.saitama.jp", + "fujimino.saitama.jp", + "fujinomiya.shizuoka.jp", + "fujioka.gunma.jp", + "fujisato.akita.jp", + "fujisawa.iwate.jp", + "fujisawa.kanagawa.jp", + "fujishiro.ibaraki.jp", + "fujitsu", + "fujiyoshida.yamanashi.jp", + "fukagawa.hokkaido.jp", + "fukaya.saitama.jp", + "fukuchi.fukuoka.jp", + "fukuchiyama.kyoto.jp", + "fukudomi.saga.jp", + "fukui.fukui.jp", + "fukui.jp", + "fukumitsu.toyama.jp", + "fukuoka.jp", + "fukuroi.shizuoka.jp", + "fukusaki.hyogo.jp", + "fukushima.fukushima.jp", + "fukushima.hokkaido.jp", + "fukushima.jp", + "fukuyama.hiroshima.jp", + "fun", + "funabashi.chiba.jp", + "funagata.yamagata.jp", + "funahashi.toyama.jp", + "fund", + "fuoisku.no", + "fuossko.no", + "furano.hokkaido.jp", + "furniture", + "furubira.hokkaido.jp", + "furudono.fukushima.jp", + "furukawa.miyagi.jp", + "fusa.no", + "fuso.aichi.jp", + "fussa.tokyo.jp", + "futaba.fukushima.jp", + "futbol", + "futsu.nagasaki.jp", + "futtsu.chiba.jp", + "fvg.it", + "fyi", + "fylkesbibl.no", + "fyresdal.no", + "førde.no", + "g.bg", + "g.se", + "g12.br", + "ga", + "ga.us", + "gaivuotna.no", + "gal", + "gallery", + "gallo", + "gallup", + "galsa.no", + "gamagori.aichi.jp", + "game", + "game.tw", + "games", + "games.hu", + "gamo.shiga.jp", + "gamvik.no", + "gangaviika.no", + "gangwon.kr", + "gap", + "garden", + "gaular.no", + "gausdal.no", + "gay", + "gb", + "gbiz", + "gc.ca", + "gd", + "gd.cn", + "gdn", + "ge", + "ge.it", + "gea", + "geek.nz", + "geisei.kochi.jp", + "gen.in", + "gen.mi.us", + "gen.nz", + "gen.tr", + "genkai.saga.jp", + "genoa.it", + "genova.it", + "gent", + "genting", + "geo.br", + "george", + "gf", + "gg", + "ggee", + "ggf.br", + "gh", + "gi", + "gialai.vn", + "giehtavuoatna.no", + "gift", + "gifts", + "gifu.gifu.jp", + "gifu.jp", + "gildeskal.no", + "gildeskål.no", + "ginan.gifu.jp", + "ginowan.okinawa.jp", + "ginoza.okinawa.jp", + "giske.no", + "gives", + "giving", + "gjemnes.no", + "gjerdrum.no", + "gjerstad.no", + "gjesdal.no", + "gjovik.no", + "gjøvik.no", + "gkp.pk", + "gl", + "glass", + "gle", + "gliding.aero", + "global", + "globo", + "glogow.pl", + "gloppen.no", + "gm", + "gmail", + "gmbh", + "gmina.pl", + "gmo", + "gmx", + "gn", + "gniezno.pl", + "go.ci", + "go.cr", + "go.gov.br", + "go.id", + "go.it", + "go.jp", + "go.ke", + "go.kr", + "go.pw", + "go.th", + "go.tj", + "go.tz", + "go.ug", + "gob.ar", + "gob.bo", + "gob.cl", + "gob.cu", + "gob.do", + "gob.ec", + "gob.es", + "gob.gt", + "gob.hn", + "gob.mx", + "gob.ni", + "gob.pa", + "gob.pe", + "gob.pk", + "gob.sv", + "gob.ve", + "gobo.wakayama.jp", + "godaddy", + "godo.gifu.jp", + "gog.pk", + "goiania.br", + "gojome.akita.jp", + "gok.pk", + "gokase.miyazaki.jp", + "gol.no", + "gold", + "goldpoint", + "golf", + "gon.pk", + "gonohe.aomori.jp", + "goo", + "goodyear", + "goog", + "google", + "gop", + "gop.pk", + "gorizia.it", + "gorlice.pl", + "gos.pk", + "gose.nara.jp", + "gosen.niigata.jp", + "goshiki.hyogo.jp", + "got", + "gotemba.shizuoka.jp", + "goto.nagasaki.jp", + "gotsu.shimane.jp", + "gouv.ci", + "gouv.fr", + "gouv.ht", + "gouv.km", + "gouv.ml", + "gouv.sn", + "gov", + "gov.ac", + "gov.ae", + "gov.af", + "gov.al", + "gov.ar", + "gov.as", + "gov.au", + "gov.az", + "gov.ba", + "gov.bb", + "gov.bf", + "gov.bh", + "gov.bm", + "gov.bn", + "gov.br", + "gov.bs", + "gov.bt", + "gov.by", + "gov.bz", + "gov.cd", + "gov.cl", + "gov.cm", + "gov.cn", + "gov.co", + "gov.cu", + "gov.cx", + "gov.cy", + "gov.dm", + "gov.do", + "gov.dz", + "gov.ec", + "gov.ee", + "gov.eg", + "gov.et", + "gov.fj", + "gov.gd", + "gov.ge", + "gov.gh", + "gov.gi", + "gov.gn", + "gov.gr", + "gov.gu", + "gov.gy", + "gov.hk", + "gov.ie", + "gov.il", + "gov.in", + "gov.iq", + "gov.ir", + "gov.is", + "gov.it", + "gov.jo", + "gov.kg", + "gov.ki", + "gov.km", + "gov.kn", + "gov.kp", + "gov.kw", + "gov.kz", + "gov.la", + "gov.lb", + "gov.lc", + "gov.lk", + "gov.lr", + "gov.ls", + "gov.lt", + "gov.lv", + "gov.ly", + "gov.ma", + "gov.me", + "gov.mg", + "gov.mk", + "gov.ml", + "gov.mn", + "gov.mo", + "gov.mr", + "gov.ms", + "gov.mu", + "gov.mv", + "gov.mw", + "gov.my", + "gov.mz", + "gov.nc.tr", + "gov.ng", + "gov.nr", + "gov.om", + "gov.ph", + "gov.pk", + "gov.pl", + "gov.pn", + "gov.pr", + "gov.ps", + "gov.pt", + "gov.py", + "gov.qa", + "gov.rs", + "gov.rw", + "gov.sa", + "gov.sb", + "gov.sc", + "gov.sd", + "gov.sg", + "gov.sh", + "gov.sl", + "gov.so", + "gov.ss", + "gov.sx", + "gov.sy", + "gov.tj", + "gov.tl", + "gov.tm", + "gov.tn", + "gov.to", + "gov.tr", + "gov.tt", + "gov.tw", + "gov.ua", + "gov.uk", + "gov.vc", + "gov.ve", + "gov.vn", + "gov.ws", + "gov.ye", + "gov.za", + "gov.zm", + "gov.zw", + "government.aero", + "govt.nz", + "gp", + "gq", + "gr", + "gr.it", + "gr.jp", + "grainger", + "grajewo.pl", + "gran.no", + "grane.no", + "granvin.no", + "graphics", + "gratangen.no", + "gratis", + "green", + "greta.fr", + "grimstad.no", + "gripe", + "griw.gov.pl", + "grocery", + "grondar.za", + "grong.no", + "grosseto.it", + "groundhandling.aero", + "group", + "group.aero", + "grp.lk", + "gru.br", + "grue.no", + "gs", + "gs.aa.no", + "gs.ah.no", + "gs.bu.no", + "gs.cn", + "gs.fm.no", + "gs.hl.no", + "gs.hm.no", + "gs.jan-mayen.no", + "gs.mr.no", + "gs.nl.no", + "gs.nt.no", + "gs.of.no", + "gs.ol.no", + "gs.oslo.no", + "gs.rl.no", + "gs.sf.no", + "gs.st.no", + "gs.svalbard.no", + "gs.tm.no", + "gs.tr.no", + "gs.va.no", + "gs.vf.no", + "gsm.pl", + "gt", + "gu", + "gu.us", + "guam.gu", + "gub.uy", + "gucci", + "guge", + "guide", + "guitars", + "gujarat.in", + "gujo.gifu.jp", + "gulen.no", + "gunma.jp", + "guovdageaidnu.no", + "guru", + "gushikami.okinawa.jp", + "gv.ao", + "gv.at", + "gw", + "gwangju.kr", + "gx.cn", + "gy", + "gyeongbuk.kr", + "gyeonggi.kr", + "gyeongnam.kr", + "gyokuto.kumamoto.jp", + "gz.cn", + "gáivuotna.no", + "gálsá.no", + "gáŋgaviika.no", + "h.bg", + "h.se", + "ha.cn", + "ha.no", + "habikino.osaka.jp", + "habmer.no", + "haboro.hokkaido.jp", + "hachijo.tokyo.jp", + "hachinohe.aomori.jp", + "hachioji.tokyo.jp", + "hachirogata.akita.jp", + "hadano.kanagawa.jp", + "hadsel.no", + "haebaru.okinawa.jp", + "haga.tochigi.jp", + "hagebostad.no", + "hagi.yamaguchi.jp", + "hagiang.vn", + "haibara.shizuoka.jp", + "haiduong.vn", + "haiphong.vn", + "hair", + "hakata.fukuoka.jp", + "hakodate.hokkaido.jp", + "hakone.kanagawa.jp", + "hakuba.nagano.jp", + "hakui.ishikawa.jp", + "hakusan.ishikawa.jp", + "halden.no", + "halsa.no", + "hamada.shimane.jp", + "hamamatsu.shizuoka.jp", + "hamar.no", + "hamaroy.no", + "hamatama.saga.jp", + "hamatonbetsu.hokkaido.jp", + "hamburg", + "hammarfeasta.no", + "hammerfest.no", + "hamura.tokyo.jp", + "hanam.vn", + "hanamaki.iwate.jp", + "hanamigawa.chiba.jp", + "hanawa.fukushima.jp", + "handa.aichi.jp", + "hanggliding.aero", + "hangout", + "hannan.osaka.jp", + "hanno.saitama.jp", + "hanoi.vn", + "hanyu.saitama.jp", + "hapmir.no", + "happou.akita.jp", + "hara.nagano.jp", + "haram.no", + "hareid.no", + "harima.hyogo.jp", + "harstad.no", + "hasama.oita.jp", + "hasami.nagasaki.jp", + "hashikami.aomori.jp", + "hashima.gifu.jp", + "hashimoto.wakayama.jp", + "hasuda.saitama.jp", + "hasvik.no", + "hatinh.vn", + "hatogaya.saitama.jp", + "hatoyama.saitama.jp", + "hatsukaichi.hiroshima.jp", + "hattfjelldal.no", + "haugesund.no", + "haugiang.vn", + "haus", + "hayakawa.yamanashi.jp", + "hayashima.okayama.jp", + "hazu.aichi.jp", + "hb.cn", + "hbo", + "hdfc", + "hdfcbank", + "he.cn", + "health", + "health.nz", + "health.vn", + "healthcare", + "heguri.nara.jp", + "hekinan.aichi.jp", + "help", + "helsinki", + "hemne.no", + "hemnes.no", + "hemsedal.no", + "herad.no", + "here", + "hermes", + "heroy.more-og-romsdal.no", + "heroy.nordland.no", + "herøy.møre-og-romsdal.no", + "herøy.nordland.no", + "hi.cn", + "hi.us", + "hichiso.gifu.jp", + "hida.gifu.jp", + "hidaka.hokkaido.jp", + "hidaka.kochi.jp", + "hidaka.saitama.jp", + "hidaka.wakayama.jp", + "higashi.fukuoka.jp", + "higashi.fukushima.jp", + "higashi.okinawa.jp", + "higashiagatsuma.gunma.jp", + "higashichichibu.saitama.jp", + "higashihiroshima.hiroshima.jp", + "higashiizu.shizuoka.jp", + "higashiizumo.shimane.jp", + "higashikagawa.kagawa.jp", + "higashikagura.hokkaido.jp", + "higashikawa.hokkaido.jp", + "higashikurume.tokyo.jp", + "higashimatsushima.miyagi.jp", + "higashimatsuyama.saitama.jp", + "higashimurayama.tokyo.jp", + "higashinaruse.akita.jp", + "higashine.yamagata.jp", + "higashiomi.shiga.jp", + "higashiosaka.osaka.jp", + "higashishirakawa.gifu.jp", + "higashisumiyoshi.osaka.jp", + "higashitsuno.kochi.jp", + "higashiura.aichi.jp", + "higashiyama.kyoto.jp", + "higashiyamato.tokyo.jp", + "higashiyodogawa.osaka.jp", + "higashiyoshino.nara.jp", + "hiji.oita.jp", + "hikari.yamaguchi.jp", + "hikawa.shimane.jp", + "hikimi.shimane.jp", + "hikone.shiga.jp", + "himeji.hyogo.jp", + "himeshima.oita.jp", + "himi.toyama.jp", + "hino.tokyo.jp", + "hino.tottori.jp", + "hinode.tokyo.jp", + "hinohara.tokyo.jp", + "hioki.kagoshima.jp", + "hiphop", + "hirado.nagasaki.jp", + "hiraizumi.iwate.jp", + "hirakata.osaka.jp", + "hiranai.aomori.jp", + "hirara.okinawa.jp", + "hirata.fukushima.jp", + "hiratsuka.kanagawa.jp", + "hiraya.nagano.jp", + "hirogawa.wakayama.jp", + "hirokawa.fukuoka.jp", + "hirono.fukushima.jp", + "hirono.iwate.jp", + "hiroo.hokkaido.jp", + "hirosaki.aomori.jp", + "hiroshima.jp", + "hisamitsu", + "hisayama.fukuoka.jp", + "hita.oita.jp", + "hitachi", + "hitachi.ibaraki.jp", + "hitachinaka.ibaraki.jp", + "hitachiomiya.ibaraki.jp", + "hitachiota.ibaraki.jp", + "hitra.no", + "hiv", + "hizen.saga.jp", + "hjartdal.no", + "hjelmeland.no", + "hk", + "hk.cn", + "hkt", + "hl.cn", + "hl.no", + "hm", + "hm.no", + "hn", + "hn.cn", + "hoabinh.vn", + "hobol.no", + "hobøl.no", + "hockey", + "hof.no", + "hofu.yamaguchi.jp", + "hokkaido.jp", + "hokksund.no", + "hokuryu.hokkaido.jp", + "hokuto.hokkaido.jp", + "hokuto.yamanashi.jp", + "hol.no", + "holdings", + "hole.no", + "holiday", + "holmestrand.no", + "holtalen.no", + "holtålen.no", + "homebuilt.aero", + "homedepot", + "homegoods", + "homes", + "homesense", + "honai.ehime.jp", + "honbetsu.hokkaido.jp", + "honda", + "honefoss.no", + "hongo.hiroshima.jp", + "honjo.akita.jp", + "honjo.saitama.jp", + "honjyo.akita.jp", + "hornindal.no", + "horokanai.hokkaido.jp", + "horonobe.hokkaido.jp", + "horse", + "horten.no", + "hospital", + "host", + "hosting", + "hot", + "hotel.hu", + "hotel.lk", + "hotel.tz", + "hotels", + "hotmail", + "house", + "how", + "hoyanger.no", + "hoylandet.no", + "hr", + "hs.kr", + "hsbc", + "ht", + "hu", + "hughes", + "huissier-justice.fr", + "hungyen.vn", + "hurdal.no", + "hurum.no", + "hvaler.no", + "hyatt", + "hyllestad.no", + "hyogo.jp", + "hyuga.miyazaki.jp", + "hyundai", + "hábmer.no", + "hámmárfeasta.no", + "hápmir.no", + "hå.no", + "hægebostad.no", + "hønefoss.no", + "høyanger.no", + "høylandet.no", + "i.bg", + "i.ng", + "i.ph", + "i.se", + "ia.us", + "ibara.okayama.jp", + "ibaraki.ibaraki.jp", + "ibaraki.jp", + "ibaraki.osaka.jp", + "ibestad.no", + "ibigawa.gifu.jp", + "ibm", + "ic.gov.pl", + "icbc", + "ice", + "ichiba.tokushima.jp", + "ichihara.chiba.jp", + "ichikai.tochigi.jp", + "ichikawa.chiba.jp", + "ichikawa.hyogo.jp", + "ichikawamisato.yamanashi.jp", + "ichinohe.iwate.jp", + "ichinomiya.aichi.jp", + "ichinomiya.chiba.jp", + "ichinoseki.iwate.jp", + "icu", + "id", + "id.au", + "id.ir", + "id.lv", + "id.ly", + "id.us", + "id.vn", + "ide.kyoto.jp", + "idf.il", + "idrett.no", + "idv.hk", + "idv.tw", + "ie", + "ieee", + "if.ua", + "ifm", + "iglesias-carbonia.it", + "iglesiascarbonia.it", + "iheya.okinawa.jp", + "iida.nagano.jp", + "iide.yamagata.jp", + "iijima.nagano.jp", + "iitate.fukushima.jp", + "iiyama.nagano.jp", + "iizuka.fukuoka.jp", + "iizuna.nagano.jp", + "ikano", + "ikaruga.nara.jp", + "ikata.ehime.jp", + "ikawa.akita.jp", + "ikeda.fukui.jp", + "ikeda.gifu.jp", + "ikeda.hokkaido.jp", + "ikeda.nagano.jp", + "ikeda.osaka.jp", + "iki.nagasaki.jp", + "ikoma.nara.jp", + "ikusaka.nagano.jp", + "il", + "il.us", + "ilawa.pl", + "im", + "im.it", + "imabari.ehime.jp", + "imakane.hokkaido.jp", + "imamat", + "imari.saga.jp", + "imb.br", + "imdb", + "imizu.toyama.jp", + "immo", + "immobilien", + "imperia.it", + "in", + "in-addr.arpa", + "in.na", + "in.ni", + "in.rs", + "in.th", + "in.ua", + "in.us", + "ina.ibaraki.jp", + "ina.nagano.jp", + "ina.saitama.jp", + "inabe.mie.jp", + "inagawa.hyogo.jp", + "inagi.tokyo.jp", + "inami.toyama.jp", + "inami.wakayama.jp", + "inashiki.ibaraki.jp", + "inatsuki.fukuoka.jp", + "inawashiro.fukushima.jp", + "inazawa.aichi.jp", + "inc", + "incheon.kr", + "ind.br", + "ind.gt", + "ind.in", + "ind.kw", + "ind.tn", + "inderoy.no", + "inderøy.no", + "indigena.bo", + "industria.bo", + "industries", + "ine.kyoto.jp", + "inf.br", + "inf.cu", + "inf.mk", + "infiniti", + "info", + "info.au", + "info.az", + "info.bb", + "info.bj", + "info.bo", + "info.co", + "info.ec", + "info.et", + "info.fj", + "info.gu", + "info.ht", + "info.hu", + "info.in", + "info.ke", + "info.ki", + "info.la", + "info.ls", + "info.mv", + "info.na", + "info.nf", + "info.ni", + "info.nr", + "info.pl", + "info.pr", + "info.ro", + "info.sd", + "info.tn", + "info.tr", + "info.tt", + "info.tz", + "info.ve", + "info.vn", + "info.zm", + "ing", + "ing.pa", + "ingatlan.hu", + "ink", + "ino.kochi.jp", + "institute", + "insurance", + "insurance.aero", + "insure", + "int", + "int.ar", + "int.az", + "int.bo", + "int.ci", + "int.co", + "int.cv", + "int.in", + "int.is", + "int.la", + "int.lk", + "int.mv", + "int.mw", + "int.ni", + "int.pt", + "int.tj", + "int.tt", + "int.ve", + "int.vn", + "international", + "internet.in", + "intl.tn", + "intuit", + "inuyama.aichi.jp", + "investments", + "inzai.chiba.jp", + "io", + "io.in", + "io.vn", + "ip6.arpa", + "ipiranga", + "iq", + "ir", + "iris.arpa", + "irish", + "iruma.saitama.jp", + "is", + "is.gov.pl", + "is.it", + "isa.kagoshima.jp", + "isa.us", + "isahaya.nagasaki.jp", + "ise.mie.jp", + "isehara.kanagawa.jp", + "isen.kagoshima.jp", + "isernia.it", + "isesaki.gunma.jp", + "ishigaki.okinawa.jp", + "ishikari.hokkaido.jp", + "ishikawa.fukushima.jp", + "ishikawa.jp", + "ishikawa.okinawa.jp", + "ishinomaki.miyagi.jp", + "isla.pr", + "ismaili", + "isshiki.aichi.jp", + "ist", + "istanbul", + "isumi.chiba.jp", + "it", + "it.ao", + "itabashi.tokyo.jp", + "itako.ibaraki.jp", + "itakura.gunma.jp", + "itami.hyogo.jp", + "itano.tokushima.jp", + "itau", + "itayanagi.aomori.jp", + "ito.shizuoka.jp", + "itoigawa.niigata.jp", + "itoman.okinawa.jp", + "its.me", + "itv", + "ivano-frankivsk.ua", + "iveland.no", + "ivgu.no", + "iwade.wakayama.jp", + "iwafune.tochigi.jp", + "iwaizumi.iwate.jp", + "iwaki.fukushima.jp", + "iwakuni.yamaguchi.jp", + "iwakura.aichi.jp", + "iwama.ibaraki.jp", + "iwamizawa.hokkaido.jp", + "iwanai.hokkaido.jp", + "iwanuma.miyagi.jp", + "iwata.shizuoka.jp", + "iwate.iwate.jp", + "iwate.jp", + "iwatsuki.saitama.jp", + "iwi.nz", + "iyo.ehime.jp", + "iz.hr", + "izena.okinawa.jp", + "izu.shizuoka.jp", + "izumi.kagoshima.jp", + "izumi.osaka.jp", + "izumiotsu.osaka.jp", + "izumisano.osaka.jp", + "izumizaki.fukushima.jp", + "izumo.shimane.jp", + "izumozaki.niigata.jp", + "izunokuni.shizuoka.jp", + "j.bg", + "jab.br", + "jaguar", + "jampa.br", + "jan-mayen.no", + "java", + "jaworzno.pl", + "jcb", + "jdf.br", + "je", + "jeep", + "jeju.kr", + "jelenia-gora.pl", + "jeonbuk.kr", + "jeonnam.kr", + "jessheim.no", + "jetzt", + "jevnaker.no", + "jewelry", + "jgora.pl", + "jinsekikogen.hiroshima.jp", + "jio", + "jl.cn", + "jll", + "jm", + "jmp", + "jnj", + "jo", + "joboji.iwate.jp", + "jobs", + "jobs.tt", + "joburg", + "joetsu.niigata.jp", + "jogasz.hu", + "johana.toyama.jp", + "joinville.br", + "jolster.no", + "jondal.no", + "jor.br", + "jorpeland.no", + "joso.ibaraki.jp", + "jot", + "journal.aero", + "journalist.aero", + "joy", + "joyo.kyoto.jp", + "jp", + "jpmorgan", + "jprs", + "js.cn", + "juegos", + "juniper", + "jur.pro", + "jus.br", + "jx.cn", + "jølster.no", + "jørpeland.no", + "k.bg", + "k.se", + "k12.ak.us", + "k12.al.us", + "k12.ar.us", + "k12.as.us", + "k12.az.us", + "k12.ca.us", + "k12.co.us", + "k12.ct.us", + "k12.dc.us", + "k12.ec", + "k12.fl.us", + "k12.ga.us", + "k12.gu.us", + "k12.ia.us", + "k12.id.us", + "k12.il", + "k12.il.us", + "k12.in.us", + "k12.ks.us", + "k12.ky.us", + "k12.la.us", + "k12.ma.us", + "k12.md.us", + "k12.me.us", + "k12.mi.us", + "k12.mn.us", + "k12.mo.us", + "k12.ms.us", + "k12.mt.us", + "k12.nc.us", + "k12.ne.us", + "k12.nh.us", + "k12.nj.us", + "k12.nm.us", + "k12.nv.us", + "k12.ny.us", + "k12.oh.us", + "k12.ok.us", + "k12.or.us", + "k12.pa.us", + "k12.pr.us", + "k12.sc.us", + "k12.tn.us", + "k12.tr", + "k12.tx.us", + "k12.ut.us", + "k12.va.us", + "k12.vi", + "k12.vi.us", + "k12.vt.us", + "k12.wa.us", + "k12.wi.us", + "k12.wy.us", + "kadena.okinawa.jp", + "kadogawa.miyazaki.jp", + "kadoma.osaka.jp", + "kafjord.no", + "kaga.ishikawa.jp", + "kagami.kochi.jp", + "kagamiishi.fukushima.jp", + "kagamino.okayama.jp", + "kagawa.jp", + "kagoshima.jp", + "kagoshima.kagoshima.jp", + "kaho.fukuoka.jp", + "kahoku.ishikawa.jp", + "kahoku.yamagata.jp", + "kai.yamanashi.jp", + "kainan.tokushima.jp", + "kainan.wakayama.jp", + "kaisei.kanagawa.jp", + "kaita.hiroshima.jp", + "kaizuka.osaka.jp", + "kakamigahara.gifu.jp", + "kakegawa.shizuoka.jp", + "kakinoki.shimane.jp", + "kakogawa.hyogo.jp", + "kakuda.miyagi.jp", + "kalisz.pl", + "kamagaya.chiba.jp", + "kamaishi.iwate.jp", + "kamakura.kanagawa.jp", + "kameoka.kyoto.jp", + "kameyama.mie.jp", + "kami.kochi.jp", + "kami.miyagi.jp", + "kamiamakusa.kumamoto.jp", + "kamifurano.hokkaido.jp", + "kamigori.hyogo.jp", + "kamiichi.toyama.jp", + "kamiizumi.saitama.jp", + "kamijima.ehime.jp", + "kamikawa.hokkaido.jp", + "kamikawa.hyogo.jp", + "kamikawa.saitama.jp", + "kamikitayama.nara.jp", + "kamikoani.akita.jp", + "kamimine.saga.jp", + "kaminokawa.tochigi.jp", + "kaminoyama.yamagata.jp", + "kamioka.akita.jp", + "kamisato.saitama.jp", + "kamishihoro.hokkaido.jp", + "kamisu.ibaraki.jp", + "kamisunagawa.hokkaido.jp", + "kamitonda.wakayama.jp", + "kamitsue.oita.jp", + "kamo.kyoto.jp", + "kamo.niigata.jp", + "kamoenai.hokkaido.jp", + "kamogawa.chiba.jp", + "kanagawa.jp", + "kanan.osaka.jp", + "kanazawa.ishikawa.jp", + "kanegasaki.iwate.jp", + "kaneyama.fukushima.jp", + "kaneyama.yamagata.jp", + "kani.gifu.jp", + "kanie.aichi.jp", + "kanmaki.nara.jp", + "kanna.gunma.jp", + "kannami.shizuoka.jp", + "kanonji.kagawa.jp", + "kanoya.kagoshima.jp", + "kanra.gunma.jp", + "kanuma.tochigi.jp", + "kanzaki.saga.jp", + "karasjohka.no", + "karasjok.no", + "karasuyama.tochigi.jp", + "karatsu.saga.jp", + "kariwa.niigata.jp", + "kariya.aichi.jp", + "karlsoy.no", + "karmoy.no", + "karmøy.no", + "karpacz.pl", + "kartuzy.pl", + "karuizawa.nagano.jp", + "karumai.iwate.jp", + "kasahara.gifu.jp", + "kasai.hyogo.jp", + "kasama.ibaraki.jp", + "kasamatsu.gifu.jp", + "kasaoka.okayama.jp", + "kashiba.nara.jp", + "kashihara.nara.jp", + "kashima.ibaraki.jp", + "kashima.saga.jp", + "kashiwa.chiba.jp", + "kashiwara.osaka.jp", + "kashiwazaki.niigata.jp", + "kasuga.fukuoka.jp", + "kasuga.hyogo.jp", + "kasugai.aichi.jp", + "kasukabe.saitama.jp", + "kasumigaura.ibaraki.jp", + "kasuya.fukuoka.jp", + "kaszuby.pl", + "katagami.akita.jp", + "katano.osaka.jp", + "katashina.gunma.jp", + "katori.chiba.jp", + "katowice.pl", + "katsuragi.nara.jp", + "katsuragi.wakayama.jp", + "katsushika.tokyo.jp", + "katsuura.chiba.jp", + "katsuyama.fukui.jp", + "kaufen", + "kautokeino.no", + "kawaba.gunma.jp", + "kawachinagano.osaka.jp", + "kawagoe.mie.jp", + "kawagoe.saitama.jp", + "kawaguchi.saitama.jp", + "kawahara.tottori.jp", + "kawai.iwate.jp", + "kawai.nara.jp", + "kawajima.saitama.jp", + "kawakami.nagano.jp", + "kawakami.nara.jp", + "kawakita.ishikawa.jp", + "kawamata.fukushima.jp", + "kawaminami.miyazaki.jp", + "kawanabe.kagoshima.jp", + "kawanehon.shizuoka.jp", + "kawanishi.hyogo.jp", + "kawanishi.nara.jp", + "kawanishi.yamagata.jp", + "kawara.fukuoka.jp", + "kawasaki.jp", + "kawasaki.miyagi.jp", + "kawatana.nagasaki.jp", + "kawaue.gifu.jp", + "kawazu.shizuoka.jp", + "kayabe.hokkaido.jp", + "kazimierz-dolny.pl", + "kazo.saitama.jp", + "kazuno.akita.jp", + "kddi", + "ke", + "keisen.fukuoka.jp", + "kembuchi.hokkaido.jp", + "kep.tr", + "kepno.pl", + "kerryhotels", + "kerrylogistics", + "kerryproperties", + "ketrzyn.pl", + "kfh", + "kg", + "kg.kr", + "kh", + "kh.ua", + "khanhhoa.vn", + "kharkiv.ua", + "kharkov.ua", + "kherson.ua", + "khmelnitskiy.ua", + "khmelnytskyi.ua", + "ki", + "kia", + "kibichuo.okayama.jp", + "kids", + "kids.us", + "kiengiang.vn", + "kiev.ua", + "kiho.mie.jp", + "kihoku.ehime.jp", + "kijo.miyazaki.jp", + "kikonai.hokkaido.jp", + "kikuchi.kumamoto.jp", + "kikugawa.shizuoka.jp", + "kim", + "kimino.wakayama.jp", + "kimitsu.chiba.jp", + "kimobetsu.hokkaido.jp", + "kin.okinawa.jp", + "kindle", + "kinko.kagoshima.jp", + "kinokawa.wakayama.jp", + "kira.aichi.jp", + "kirkenes.no", + "kirovograd.ua", + "kiryu.gunma.jp", + "kisarazu.chiba.jp", + "kishiwada.osaka.jp", + "kiso.nagano.jp", + "kisofukushima.nagano.jp", + "kisosaki.mie.jp", + "kita.kyoto.jp", + "kita.osaka.jp", + "kita.tokyo.jp", + "kitaaiki.nagano.jp", + "kitaakita.akita.jp", + "kitadaito.okinawa.jp", + "kitagata.gifu.jp", + "kitagata.saga.jp", + "kitagawa.kochi.jp", + "kitagawa.miyazaki.jp", + "kitahata.saga.jp", + "kitahiroshima.hokkaido.jp", + "kitakami.iwate.jp", + "kitakata.fukushima.jp", + "kitakata.miyazaki.jp", + "kitakyushu.jp", + "kitami.hokkaido.jp", + "kitamoto.saitama.jp", + "kitanakagusuku.okinawa.jp", + "kitashiobara.fukushima.jp", + "kitaura.miyazaki.jp", + "kitayama.wakayama.jp", + "kitchen", + "kiwa.mie.jp", + "kiwi", + "kiwi.nz", + "kiyama.saga.jp", + "kiyokawa.kanagawa.jp", + "kiyosato.hokkaido.jp", + "kiyose.tokyo.jp", + "kiyosu.aichi.jp", + "kizu.kyoto.jp", + "klabu.no", + "klepp.no", + "klodzko.pl", + "klæbu.no", + "km", + "km.ua", + "kmpsp.gov.pl", + "kn", + "kobayashi.miyazaki.jp", + "kobe.jp", + "kobierzyce.pl", + "kochi.jp", + "kochi.kochi.jp", + "kodaira.tokyo.jp", + "koeln", + "kofu.yamanashi.jp", + "koga.fukuoka.jp", + "koga.ibaraki.jp", + "koganei.tokyo.jp", + "koge.tottori.jp", + "koka.shiga.jp", + "kokonoe.oita.jp", + "kokubunji.tokyo.jp", + "kolobrzeg.pl", + "komae.tokyo.jp", + "komagane.nagano.jp", + "komaki.aichi.jp", + "komatsu", + "komatsu.ishikawa.jp", + "komatsushima.tokushima.jp", + "komforb.se", + "kommunalforbund.se", + "kommune.no", + "komono.mie.jp", + "komoro.nagano.jp", + "komvux.se", + "konan.aichi.jp", + "konan.shiga.jp", + "kongsberg.no", + "kongsvinger.no", + "konin.pl", + "konskowola.pl", + "konsulat.gov.pl", + "kontum.vn", + "konyvelo.hu", + "koori.fukushima.jp", + "kopervik.no", + "koriyama.fukushima.jp", + "koryo.nara.jp", + "kosai.shizuoka.jp", + "kosaka.akita.jp", + "kosei.shiga.jp", + "kosher", + "koshigaya.saitama.jp", + "koshimizu.hokkaido.jp", + "koshu.yamanashi.jp", + "kosuge.yamanashi.jp", + "kota.aichi.jp", + "koto.shiga.jp", + "koto.tokyo.jp", + "kotohira.kagawa.jp", + "kotoura.tottori.jp", + "kouhoku.saga.jp", + "kounosu.saitama.jp", + "kouyama.kagoshima.jp", + "kouzushima.tokyo.jp", + "koya.wakayama.jp", + "koza.wakayama.jp", + "kozagawa.wakayama.jp", + "kozaki.chiba.jp", + "kp", + "kpmg", + "kpn", + "kppsp.gov.pl", + "kr", + "kr.it", + "kr.ua", + "kraanghke.no", + "kragero.no", + "kragerø.no", + "krd", + "kred", + "kristiansand.no", + "kristiansund.no", + "krodsherad.no", + "krokstadelva.no", + "kropyvnytskyi.ua", + "krym.ua", + "kråanghke.no", + "krødsherad.no", + "ks.ua", + "ks.us", + "kuchinotsu.nagasaki.jp", + "kudamatsu.yamaguchi.jp", + "kudoyama.wakayama.jp", + "kui.hiroshima.jp", + "kuji.iwate.jp", + "kuju.oita.jp", + "kujukuri.chiba.jp", + "kuki.saitama.jp", + "kumagaya.saitama.jp", + "kumakogen.ehime.jp", + "kumamoto.jp", + "kumamoto.kumamoto.jp", + "kumano.hiroshima.jp", + "kumano.mie.jp", + "kumatori.osaka.jp", + "kumejima.okinawa.jp", + "kumenan.okayama.jp", + "kumiyama.kyoto.jp", + "kunigami.okinawa.jp", + "kunimi.fukushima.jp", + "kunisaki.oita.jp", + "kunitachi.tokyo.jp", + "kunitomi.miyazaki.jp", + "kunneppu.hokkaido.jp", + "kunohe.iwate.jp", + "kuokgroup", + "kurashiki.okayama.jp", + "kurate.fukuoka.jp", + "kure.hiroshima.jp", + "kuriyama.hokkaido.jp", + "kurobe.toyama.jp", + "kurogi.fukuoka.jp", + "kuroishi.aomori.jp", + "kuroiso.tochigi.jp", + "kuromatsunai.hokkaido.jp", + "kurotaki.nara.jp", + "kurume.fukuoka.jp", + "kusatsu.gunma.jp", + "kusatsu.shiga.jp", + "kushima.miyazaki.jp", + "kushimoto.wakayama.jp", + "kushiro.hokkaido.jp", + "kusu.oita.jp", + "kutchan.hokkaido.jp", + "kutno.pl", + "kuwana.mie.jp", + "kuzumaki.iwate.jp", + "kv.ua", + "kvafjord.no", + "kvalsund.no", + "kvam.no", + "kvanangen.no", + "kvinesdal.no", + "kvinnherad.no", + "kviteseid.no", + "kvitsoy.no", + "kvitsøy.no", + "kvæfjord.no", + "kvænangen.no", + "kw", + "kwp.gov.pl", + "kwpsp.gov.pl", + "ky", + "ky.us", + "kyiv.ua", + "kyonan.chiba.jp", + "kyotamba.kyoto.jp", + "kyotanabe.kyoto.jp", + "kyotango.kyoto.jp", + "kyoto", + "kyoto.jp", + "kyowa.akita.jp", + "kyowa.hokkaido.jp", + "kyuragi.saga.jp", + "kz", + "kárášjohka.no", + "kåfjord.no", + "l.bg", + "l.se", + "la", + "la-spezia.it", + "la.us", + "laakesvuemie.no", + "lacaixa", + "lahppi.no", + "laichau.vn", + "lakas.hu", + "lamborghini", + "lamdong.vn", + "lamer", + "lanbib.se", + "lancaster", + "land", + "landrover", + "langevag.no", + "langevåg.no", + "langson.vn", + "lanxess", + "laocai.vn", + "lapy.pl", + "laquila.it", + "lardal.no", + "larvik.no", + "lasalle", + "laspezia.it", + "lat", + "latina.it", + "latino", + "latrobe", + "lavagis.no", + "lavangen.no", + "law", + "law.pro", + "law.za", + "lawyer", + "laz.it", + "lazio.it", + "lb", + "lc", + "lc.it", + "lds", + "le.it", + "leangaviika.no", + "lease", + "leasing.aero", + "leaŋgaviika.no", + "lebesby.no", + "lebork.pl", + "lecce.it", + "lecco.it", + "leclerc", + "lefrak", + "leg.br", + "legal", + "legnica.pl", + "lego", + "leikanger.no", + "leilao.br", + "leirfjord.no", + "leirvik.no", + "leka.no", + "leksvik.no", + "lel.br", + "lenvik.no", + "lerdal.no", + "lesja.no", + "levanger.no", + "lexus", + "lezajsk.pl", + "lg.jp", + "lg.ua", + "lgbt", + "li", + "li.it", + "lib.ak.us", + "lib.al.us", + "lib.ar.us", + "lib.as.us", + "lib.az.us", + "lib.ca.us", + "lib.co.us", + "lib.ct.us", + "lib.dc.us", + "lib.ee", + "lib.fl.us", + "lib.ga.us", + "lib.gu.us", + "lib.hi.us", + "lib.ia.us", + "lib.id.us", + "lib.il.us", + "lib.in.us", + "lib.ks.us", + "lib.ky.us", + "lib.la.us", + "lib.ma.us", + "lib.md.us", + "lib.me.us", + "lib.mi.us", + "lib.mn.us", + "lib.mo.us", + "lib.ms.us", + "lib.mt.us", + "lib.nc.us", + "lib.nd.us", + "lib.ne.us", + "lib.nh.us", + "lib.nj.us", + "lib.nm.us", + "lib.nv.us", + "lib.ny.us", + "lib.oh.us", + "lib.ok.us", + "lib.or.us", + "lib.pa.us", + "lib.pr.us", + "lib.ri.us", + "lib.sc.us", + "lib.sd.us", + "lib.tn.us", + "lib.tx.us", + "lib.ut.us", + "lib.va.us", + "lib.vi.us", + "lib.vt.us", + "lib.wa.us", + "lib.wi.us", + "lib.wy.us", + "lidl", + "lier.no", + "lierne.no", + "life", + "lifeinsurance", + "lifestyle", + "lig.it", + "lighting", + "liguria.it", + "like", + "lillehammer.no", + "lillesand.no", + "lilly", + "limanowa.pl", + "limited", + "limo", + "lincoln", + "lindas.no", + "lindesnes.no", + "lindås.no", + "link", + "lipsy", + "live", + "living", + "livorno.it", + "lk", + "llc", + "llp", + "ln.cn", + "lo.it", + "loabat.no", + "loabát.no", + "loan", + "loans", + "locker", + "locus", + "lodi.it", + "lodingen.no", + "log.br", + "logistics.aero", + "loisirs.bj", + "lol", + "lom.it", + "lom.no", + "lombardia.it", + "lombardy.it", + "lomza.pl", + "london", + "londrina.br", + "longan.vn", + "loppa.no", + "lorenskog.no", + "loten.no", + "lotte", + "lotto", + "love", + "lowicz.pl", + "lpl", + "lplfinancial", + "lr", + "ls", + "lt", + "lt.it", + "lt.ua", + "ltd", + "ltd.co.im", + "ltd.cy", + "ltd.gi", + "ltd.lk", + "ltd.uk", + "ltda", + "lu", + "lu.it", + "lubin.pl", + "lucania.it", + "lucca.it", + "lugansk.ua", + "luhansk.ua", + "lukow.pl", + "lund.no", + "lundbeck", + "lunner.no", + "luroy.no", + "lurøy.no", + "luster.no", + "lutsk.ua", + "luxe", + "luxury", + "lv", + "lv.ua", + "lviv.ua", + "ly", + "lyngdal.no", + "lyngen.no", + "láhppi.no", + "lærdal.no", + "lødingen.no", + "lørenskog.no", + "løten.no", + "m.bg", + "m.se", + "ma", + "ma.gov.br", + "ma.us", + "macapa.br", + "maceio.br", + "macerata.it", + "machida.tokyo.jp", + "madrid", + "maebashi.gunma.jp", + "magazine.aero", + "maibara.shiga.jp", + "maif", + "mail.pl", + "maintenance.aero", + "maison", + "maizuru.kyoto.jp", + "makeup", + "makinohara.shizuoka.jp", + "makurazaki.kagoshima.jp", + "malatvuopmi.no", + "malbork.pl", + "malopolska.pl", + "malselv.no", + "malvik.no", + "mamurogawa.yamagata.jp", + "man", + "management", + "manaus.br", + "mandal.no", + "mango", + "maniwa.okayama.jp", + "manno.kagawa.jp", + "mantova.it", + "maori.nz", + "map", + "mar.it", + "marche.it", + "maringa.br", + "marker.no", + "market", + "marketing", + "marketplace.aero", + "markets", + "marnardal.no", + "marriott", + "marshalls", + "marugame.kagawa.jp", + "marumori.miyagi.jp", + "masaki.ehime.jp", + "masfjorden.no", + "mashike.hokkaido.jp", + "mashiki.kumamoto.jp", + "mashiko.tochigi.jp", + "masoy.no", + "massa-carrara.it", + "massacarrara.it", + "masuda.shimane.jp", + "mat.br", + "matera.it", + "matsubara.osaka.jp", + "matsubushi.saitama.jp", + "matsuda.kanagawa.jp", + "matsudo.chiba.jp", + "matsue.shimane.jp", + "matsukawa.nagano.jp", + "matsumae.hokkaido.jp", + "matsumoto.kagoshima.jp", + "matsumoto.nagano.jp", + "matsuno.ehime.jp", + "matsusaka.mie.jp", + "matsushige.tokushima.jp", + "matsushima.miyagi.jp", + "matsuura.nagasaki.jp", + "matsuyama.ehime.jp", + "matsuzaki.shizuoka.jp", + "matta-varjjat.no", + "mattel", + "mazowsze.pl", + "mazury.pl", + "mb.ca", + "mb.it", + "mba", + "mc", + "mc.it", + "mckinsey", + "md", + "md.ci", + "md.us", + "me", + "me.in", + "me.it", + "me.ke", + "me.so", + "me.ss", + "me.tz", + "me.uk", + "me.us", + "med", + "med.br", + "med.ec", + "med.ee", + "med.ht", + "med.ly", + "med.om", + "med.pa", + "med.pro", + "med.sa", + "med.sd", + "medecin.km", + "media", + "media.aero", + "media.hu", + "media.pl", + "medicina.bo", + "medio-campidano.it", + "mediocampidano.it", + "meet", + "meguro.tokyo.jp", + "meiwa.gunma.jp", + "meiwa.mie.jp", + "meland.no", + "melbourne", + "meldal.no", + "melhus.no", + "meloy.no", + "meløy.no", + "meme", + "memorial", + "men", + "menu", + "meraker.no", + "merck", + "merckmsd", + "meråker.no", + "messina.it", + "mg", + "mg.gov.br", + "mh", + "mi.it", + "mi.th", + "mi.us", + "miami", + "miasa.nagano.jp", + "miasta.pl", + "mibu.tochigi.jp", + "microlight.aero", + "microsoft", + "midori.chiba.jp", + "midori.gunma.jp", + "midsund.no", + "midtre-gauldal.no", + "mie.jp", + "mielec.pl", + "mielno.pl", + "mifune.kumamoto.jp", + "mihama.aichi.jp", + "mihama.chiba.jp", + "mihama.fukui.jp", + "mihama.mie.jp", + "mihama.wakayama.jp", + "mihara.hiroshima.jp", + "mihara.kochi.jp", + "miharu.fukushima.jp", + "miho.ibaraki.jp", + "mikasa.hokkaido.jp", + "mikawa.yamagata.jp", + "miki.hyogo.jp", + "mil", + "mil.ac", + "mil.ae", + "mil.al", + "mil.ar", + "mil.az", + "mil.ba", + "mil.bo", + "mil.br", + "mil.by", + "mil.cl", + "mil.cn", + "mil.co", + "mil.cy", + "mil.do", + "mil.ec", + "mil.eg", + "mil.fj", + "mil.ge", + "mil.gh", + "mil.gt", + "mil.hn", + "mil.id", + "mil.in", + "mil.iq", + "mil.jo", + "mil.kg", + "mil.km", + "mil.kr", + "mil.kz", + "mil.lv", + "mil.mg", + "mil.mv", + "mil.my", + "mil.mz", + "mil.ng", + "mil.ni", + "mil.no", + "mil.nz", + "mil.pe", + "mil.ph", + "mil.pl", + "mil.py", + "mil.qa", + "mil.rw", + "mil.sh", + "mil.st", + "mil.sy", + "mil.tj", + "mil.tm", + "mil.to", + "mil.tr", + "mil.tw", + "mil.tz", + "mil.uy", + "mil.vc", + "mil.ve", + "mil.ye", + "mil.za", + "mil.zm", + "mil.zw", + "milan.it", + "milano.it", + "mima.tokushima.jp", + "mimata.miyazaki.jp", + "minakami.gunma.jp", + "minamata.kumamoto.jp", + "minami-alps.yamanashi.jp", + "minami.fukuoka.jp", + "minami.kyoto.jp", + "minami.tokushima.jp", + "minamiaiki.nagano.jp", + "minamiashigara.kanagawa.jp", + "minamiawaji.hyogo.jp", + "minamiboso.chiba.jp", + "minamidaito.okinawa.jp", + "minamiechizen.fukui.jp", + "minamifurano.hokkaido.jp", + "minamiise.mie.jp", + "minamiizu.shizuoka.jp", + "minamimaki.nagano.jp", + "minamiminowa.nagano.jp", + "minamioguni.kumamoto.jp", + "minamisanriku.miyagi.jp", + "minamitane.kagoshima.jp", + "minamiuonuma.niigata.jp", + "minamiyamashiro.kyoto.jp", + "minano.saitama.jp", + "minato.osaka.jp", + "minato.tokyo.jp", + "mincom.tn", + "mini", + "mino.gifu.jp", + "minobu.yamanashi.jp", + "minoh.osaka.jp", + "minokamo.gifu.jp", + "minowa.nagano.jp", + "mint", + "misaki.okayama.jp", + "misaki.osaka.jp", + "misasa.tottori.jp", + "misato.akita.jp", + "misato.miyagi.jp", + "misato.saitama.jp", + "misato.shimane.jp", + "misato.wakayama.jp", + "misawa.aomori.jp", + "mishima.fukushima.jp", + "mishima.shizuoka.jp", + "misugi.mie.jp", + "mit", + "mitaka.tokyo.jp", + "mitake.gifu.jp", + "mitane.akita.jp", + "mito.ibaraki.jp", + "mitou.yamaguchi.jp", + "mitoyo.kagawa.jp", + "mitsubishi", + "mitsue.nara.jp", + "mitsuke.niigata.jp", + "miura.kanagawa.jp", + "miyada.nagano.jp", + "miyagi.jp", + "miyake.nara.jp", + "miyako.fukuoka.jp", + "miyako.iwate.jp", + "miyakonojo.miyazaki.jp", + "miyama.fukuoka.jp", + "miyama.mie.jp", + "miyashiro.saitama.jp", + "miyawaka.fukuoka.jp", + "miyazaki.jp", + "miyazaki.miyazaki.jp", + "miyazu.kyoto.jp", + "miyoshi.aichi.jp", + "miyoshi.hiroshima.jp", + "miyoshi.saitama.jp", + "miyoshi.tokushima.jp", + "miyota.nagano.jp", + "mizuho.tokyo.jp", + "mizumaki.fukuoka.jp", + "mizunami.gifu.jp", + "mizusawa.iwate.jp", + "mjondalen.no", + "mjøndalen.no", + "mk", + "mk.ua", + "ml", + "mlb", + "mls", + "mm", + "mma", + "mn", + "mn.it", + "mn.us", + "mo", + "mo-i-rana.no", + "mo.cn", + "mo.it", + "mo.us", + "moareke.no", + "mobara.chiba.jp", + "mobi", + "mobi.gp", + "mobi.ke", + "mobi.na", + "mobi.ng", + "mobi.tt", + "mobi.tz", + "mobile", + "mochizuki.nagano.jp", + "mod.gi", + "moda", + "modalen.no", + "modelling.aero", + "modena.it", + "modum.no", + "moe", + "moi", + "moka.tochigi.jp", + "mol.it", + "molde.no", + "molise.it", + "mom", + "mombetsu.hokkaido.jp", + "monash", + "money", + "money.bj", + "monster", + "monza-brianza.it", + "monza-e-della-brianza.it", + "monza.it", + "monzabrianza.it", + "monzaebrianza.it", + "monzaedellabrianza.it", + "morena.br", + "moriguchi.osaka.jp", + "morimachi.shizuoka.jp", + "morioka.iwate.jp", + "moriya.ibaraki.jp", + "moriyama.shiga.jp", + "moriyoshi.akita.jp", + "mormon", + "morotsuka.miyazaki.jp", + "moroyama.saitama.jp", + "mortgage", + "moscow", + "moseushi.hokkaido.jp", + "mosjoen.no", + "mosjøen.no", + "moskenes.no", + "moss.no", + "mosvik.no", + "motegi.tochigi.jp", + "moto", + "motobu.okinawa.jp", + "motorcycles", + "motosu.gifu.jp", + "motoyama.kochi.jp", + "mov", + "movie", + "movimiento.bo", + "moåreke.no", + "mp", + "mp.br", + "mq", + "mr", + "mr.no", + "mragowo.pl", + "ms", + "ms.gov.br", + "ms.it", + "ms.kr", + "ms.us", + "msd", + "mt", + "mt.gov.br", + "mt.it", + "mt.us", + "mtn", + "mtr", + "mu", + "mugi.tokushima.jp", + "muika.niigata.jp", + "mukawa.hokkaido.jp", + "muko.kyoto.jp", + "munakata.fukuoka.jp", + "muni.il", + "muosat.no", + "muosát.no", + "mup.gov.pl", + "murakami.niigata.jp", + "murata.miyagi.jp", + "murayama.yamagata.jp", + "muroran.hokkaido.jp", + "muroto.kochi.jp", + "mus.br", + "mus.mi.us", + "musashimurayama.tokyo.jp", + "musashino.tokyo.jp", + "museum", + "museum.mv", + "museum.mw", + "museum.no", + "museum.om", + "museum.tt", + "music", + "musica.ar", + "musica.bo", + "mutsu.aomori.jp", + "mutsuzawa.chiba.jp", + "mutual.ar", + "mv", + "mw", + "mw.gov.pl", + "mx", + "mx.na", + "my", + "my.id", + "mykolaiv.ua", + "myoko.niigata.jp", + "mz", + "málatvuopmi.no", + "mátta-várjjat.no", + "målselv.no", + "måsøy.no", + "māori.nz", + "n.bg", + "n.se", + "na", + "na.it", + "naamesjevuemie.no", + "nab", + "nabari.mie.jp", + "nachikatsuura.wakayama.jp", + "nagahama.shiga.jp", + "nagai.yamagata.jp", + "nagano.jp", + "nagano.nagano.jp", + "naganohara.gunma.jp", + "nagaoka.niigata.jp", + "nagaokakyo.kyoto.jp", + "nagara.chiba.jp", + "nagareyama.chiba.jp", + "nagasaki.jp", + "nagasaki.nagasaki.jp", + "nagasu.kumamoto.jp", + "nagato.yamaguchi.jp", + "nagatoro.saitama.jp", + "nagawa.nagano.jp", + "nagi.okayama.jp", + "nagiso.nagano.jp", + "nago.okinawa.jp", + "nagoya", + "nagoya.jp", + "naha.okinawa.jp", + "nahari.kochi.jp", + "naie.hokkaido.jp", + "naka.hiroshima.jp", + "naka.ibaraki.jp", + "nakadomari.aomori.jp", + "nakagawa.fukuoka.jp", + "nakagawa.hokkaido.jp", + "nakagawa.nagano.jp", + "nakagawa.tokushima.jp", + "nakagusuku.okinawa.jp", + "nakagyo.kyoto.jp", + "nakai.kanagawa.jp", + "nakama.fukuoka.jp", + "nakamichi.yamanashi.jp", + "nakamura.kochi.jp", + "nakaniikawa.toyama.jp", + "nakano.nagano.jp", + "nakano.tokyo.jp", + "nakanojo.gunma.jp", + "nakanoto.ishikawa.jp", + "nakasatsunai.hokkaido.jp", + "nakatane.kagoshima.jp", + "nakatombetsu.hokkaido.jp", + "nakatsugawa.gifu.jp", + "nakayama.yamagata.jp", + "nakijin.okinawa.jp", + "naklo.pl", + "namdalseid.no", + "namdinh.vn", + "name", + "name.az", + "name.eg", + "name.et", + "name.fj", + "name.hr", + "name.jo", + "name.mk", + "name.mv", + "name.my", + "name.na", + "name.ng", + "name.pr", + "name.qa", + "name.tj", + "name.tr", + "name.tt", + "name.vn", + "namegata.ibaraki.jp", + "namegawa.saitama.jp", + "namerikawa.toyama.jp", + "namie.fukushima.jp", + "namikata.ehime.jp", + "namsos.no", + "namsskogan.no", + "nanae.hokkaido.jp", + "nanao.ishikawa.jp", + "nanbu.tottori.jp", + "nanbu.yamanashi.jp", + "nango.fukushima.jp", + "nanjo.okinawa.jp", + "nankoku.kochi.jp", + "nanmoku.gunma.jp", + "nannestad.no", + "nanporo.hokkaido.jp", + "nantan.kyoto.jp", + "nanto.toyama.jp", + "nanyo.yamagata.jp", + "naoshima.kagawa.jp", + "naples.it", + "napoli.it", + "nara.jp", + "nara.nara.jp", + "narashino.chiba.jp", + "narita.chiba.jp", + "naroy.no", + "narusawa.yamanashi.jp", + "naruto.tokushima.jp", + "narviika.no", + "narvik.no", + "nasu.tochigi.jp", + "nasushiobara.tochigi.jp", + "nat.cu", + "nat.tn", + "natal.br", + "natori.miyagi.jp", + "natural.bo", + "naturbruksgymn.se", + "naustdal.no", + "navigation.aero", + "navuotna.no", + "navy", + "nayoro.hokkaido.jp", + "nb.ca", + "nba", + "nc", + "nc.tr", + "nc.us", + "nd.us", + "ne", + "ne.jp", + "ne.ke", + "ne.kr", + "ne.pw", + "ne.tz", + "ne.ug", + "ne.us", + "nec", + "nedre-eiker.no", + "nemuro.hokkaido.jp", + "nerima.tokyo.jp", + "nes.akershus.no", + "nes.buskerud.no", + "nesna.no", + "nesodden.no", + "nesoddtangen.no", + "nesseby.no", + "nesset.no", + "net", + "net.ac", + "net.ae", + "net.af", + "net.ag", + "net.ai", + "net.al", + "net.am", + "net.ar", + "net.au", + "net.az", + "net.ba", + "net.bb", + "net.bh", + "net.bj", + "net.bm", + "net.bn", + "net.bo", + "net.br", + "net.bs", + "net.bt", + "net.bz", + "net.ci", + "net.cm", + "net.cn", + "net.co", + "net.cu", + "net.cw", + "net.cy", + "net.dm", + "net.do", + "net.dz", + "net.ec", + "net.eg", + "net.et", + "net.fj", + "net.fm", + "net.ge", + "net.gg", + "net.gl", + "net.gn", + "net.gp", + "net.gr", + "net.gt", + "net.gu", + "net.gy", + "net.hk", + "net.hn", + "net.ht", + "net.id", + "net.il", + "net.im", + "net.in", + "net.iq", + "net.ir", + "net.is", + "net.je", + "net.jo", + "net.kg", + "net.ki", + "net.kn", + "net.kw", + "net.ky", + "net.kz", + "net.la", + "net.lb", + "net.lc", + "net.lk", + "net.lr", + "net.ls", + "net.lv", + "net.ly", + "net.ma", + "net.me", + "net.mk", + "net.ml", + "net.mo", + "net.ms", + "net.mt", + "net.mu", + "net.mv", + "net.mw", + "net.mx", + "net.my", + "net.mz", + "net.nf", + "net.ng", + "net.ni", + "net.nr", + "net.nz", + "net.om", + "net.pa", + "net.pe", + "net.ph", + "net.pk", + "net.pl", + "net.pn", + "net.pr", + "net.ps", + "net.pt", + "net.py", + "net.qa", + "net.rw", + "net.sa", + "net.sb", + "net.sc", + "net.sd", + "net.sg", + "net.sh", + "net.sl", + "net.so", + "net.ss", + "net.st", + "net.sy", + "net.th", + "net.tj", + "net.tm", + "net.tn", + "net.to", + "net.tr", + "net.tt", + "net.tw", + "net.ua", + "net.uk", + "net.uy", + "net.uz", + "net.vc", + "net.ve", + "net.vi", + "net.vn", + "net.vu", + "net.ws", + "net.ye", + "net.za", + "net.zm", + "netbank", + "netflix", + "network", + "neustar", + "new", + "news", + "news.hu", + "next", + "nextdirect", + "nexus", + "neyagawa.osaka.jp", + "nf", + "nf.ca", + "nfl", + "ng", + "nghean.vn", + "ngo", + "ngo.lk", + "ngo.ph", + "ngo.za", + "nh.us", + "nhk", + "nhs.uk", + "ni", + "nic.in", + "nic.tj", + "nic.za", + "nichinan.miyazaki.jp", + "nichinan.tottori.jp", + "nico", + "nieruchomosci.pl", + "niigata.jp", + "niigata.niigata.jp", + "niihama.ehime.jp", + "niikappu.hokkaido.jp", + "niimi.okayama.jp", + "niiza.saitama.jp", + "nikaho.akita.jp", + "nike", + "niki.hokkaido.jp", + "nikko.tochigi.jp", + "nikolaev.ua", + "nikon", + "ninhbinh.vn", + "ninhthuan.vn", + "ninja", + "ninohe.iwate.jp", + "ninomiya.kanagawa.jp", + "nirasaki.yamanashi.jp", + "nis.za", + "nishi.fukuoka.jp", + "nishi.osaka.jp", + "nishiaizu.fukushima.jp", + "nishiarita.saga.jp", + "nishiawakura.okayama.jp", + "nishiazai.shiga.jp", + "nishigo.fukushima.jp", + "nishihara.kumamoto.jp", + "nishihara.okinawa.jp", + "nishiizu.shizuoka.jp", + "nishikata.tochigi.jp", + "nishikatsura.yamanashi.jp", + "nishikawa.yamagata.jp", + "nishimera.miyazaki.jp", + "nishinomiya.hyogo.jp", + "nishinoomote.kagoshima.jp", + "nishinoshima.shimane.jp", + "nishio.aichi.jp", + "nishiokoppe.hokkaido.jp", + "nishitosa.kochi.jp", + "nishiwaki.hyogo.jp", + "nissan", + "nissay", + "nissedal.no", + "nisshin.aichi.jp", + "niteroi.br", + "nittedal.no", + "niyodogawa.kochi.jp", + "nj.us", + "nl", + "nl.ca", + "nl.no", + "nm.cn", + "nm.us", + "no", + "no.it", + "nobeoka.miyazaki.jp", + "noboribetsu.hokkaido.jp", + "noda.chiba.jp", + "noda.iwate.jp", + "nogata.fukuoka.jp", + "nogi.tochigi.jp", + "noheji.aomori.jp", + "nokia", + "nom.ad", + "nom.ag", + "nom.br", + "nom.co", + "nom.es", + "nom.fr", + "nom.km", + "nom.mg", + "nom.nc", + "nom.ni", + "nom.pa", + "nom.pe", + "nom.pl", + "nom.re", + "nom.ro", + "nom.tm", + "nom.ve", + "nom.za", + "nombre.bo", + "nome.cv", + "nome.pt", + "nomi.ishikawa.jp", + "nonoichi.ishikawa.jp", + "nord-aurdal.no", + "nord-fron.no", + "nord-odal.no", + "norddal.no", + "nordkapp.no", + "nordre-land.no", + "nordreisa.no", + "nore-og-uvdal.no", + "norton", + "nose.osaka.jp", + "nosegawa.nara.jp", + "noshiro.akita.jp", + "not.br", + "notaires.km", + "noticias.bo", + "noto.ishikawa.jp", + "notodden.no", + "notogawa.shiga.jp", + "notteroy.no", + "novara.it", + "now", + "nowaruda.pl", + "nowruz", + "nowtv", + "nozawaonsen.nagano.jp", + "np", + "nr", + "nra", + "nrw", + "ns.ca", + "nsn.us", + "nsw.au", + "nsw.edu.au", + "nt.au", + "nt.ca", + "nt.edu.au", + "nt.no", + "nt.ro", + "ntr.br", + "ntt", + "nu", + "nu.ca", + "nu.it", + "numata.gunma.jp", + "numata.hokkaido.jp", + "numazu.shizuoka.jp", + "nuoro.it", + "nv.us", + "nx.cn", + "ny.us", + "nyc", + "nysa.pl", + "nyuzen.toyama.jp", + "nz", + "návuotna.no", + "nååmesjevuemie.no", + "nærøy.no", + "nøtterøy.no", + "o.bg", + "o.se", + "oamishirasato.chiba.jp", + "oarai.ibaraki.jp", + "obama.fukui.jp", + "obama.nagasaki.jp", + "obanazawa.yamagata.jp", + "obi", + "obihiro.hokkaido.jp", + "obira.hokkaido.jp", + "observer", + "obu.aichi.jp", + "obuse.nagano.jp", + "ochi.kochi.jp", + "od.ua", + "odate.akita.jp", + "odawara.kanagawa.jp", + "odda.no", + "odesa.ua", + "odessa.ua", + "odo.br", + "oe.yamagata.jp", + "of.by", + "of.no", + "off.ai", + "office", + "ofunato.iwate.jp", + "og.ao", + "og.it", + "oga.akita.jp", + "ogaki.gifu.jp", + "ogano.saitama.jp", + "ogasawara.tokyo.jp", + "ogata.akita.jp", + "ogawa.ibaraki.jp", + "ogawa.nagano.jp", + "ogawa.saitama.jp", + "ogawara.miyagi.jp", + "ogi.saga.jp", + "ogimi.okinawa.jp", + "ogliastra.it", + "ogori.fukuoka.jp", + "ogose.saitama.jp", + "oguchi.aichi.jp", + "oguni.kumamoto.jp", + "oguni.yamagata.jp", + "oh.us", + "oharu.aichi.jp", + "ohda.shimane.jp", + "ohi.fukui.jp", + "ohira.miyagi.jp", + "ohira.tochigi.jp", + "ohkura.yamagata.jp", + "ohtawara.tochigi.jp", + "oi.kanagawa.jp", + "oia.gov.pl", + "oirase.aomori.jp", + "oirm.gov.pl", + "oishida.yamagata.jp", + "oiso.kanagawa.jp", + "oita.jp", + "oita.oita.jp", + "oizumi.gunma.jp", + "oji.nara.jp", + "ojiya.niigata.jp", + "ok.us", + "okagaki.fukuoka.jp", + "okawa.fukuoka.jp", + "okawa.kochi.jp", + "okaya.nagano.jp", + "okayama.jp", + "okayama.okayama.jp", + "okazaki.aichi.jp", + "oke.gov.pl", + "okegawa.saitama.jp", + "oketo.hokkaido.jp", + "oki.fukuoka.jp", + "okinawa", + "okinawa.jp", + "okinawa.okinawa.jp", + "okinoshima.shimane.jp", + "okoppe.hokkaido.jp", + "oksnes.no", + "okuizumo.shimane.jp", + "okuma.fukushima.jp", + "okutama.tokyo.jp", + "ol.no", + "olawa.pl", + "olayan", + "olayangroup", + "olbia-tempio.it", + "olbiatempio.it", + "olecko.pl", + "olkusz.pl", + "ollo", + "olsztyn.pl", + "om", + "omachi.nagano.jp", + "omachi.saga.jp", + "omaezaki.shizuoka.jp", + "omasvuotna.no", + "ome.tokyo.jp", + "omega", + "omi.nagano.jp", + "omi.niigata.jp", + "omigawa.chiba.jp", + "omihachiman.shiga.jp", + "omitama.ibaraki.jp", + "omiya.saitama.jp", + "omotego.fukushima.jp", + "omura.nagasaki.jp", + "omuta.fukuoka.jp", + "on.ca", + "onagawa.miyagi.jp", + "one", + "ong", + "ong.br", + "onga.fukuoka.jp", + "onion", + "onjuku.chiba.jp", + "onl", + "online", + "onna.okinawa.jp", + "ono.fukui.jp", + "ono.fukushima.jp", + "ono.hyogo.jp", + "onojo.fukuoka.jp", + "onomichi.hiroshima.jp", + "ookuwa.nagano.jp", + "ooo", + "ooshika.nagano.jp", + "oow.gov.pl", + "open", + "opoczno.pl", + "opole.pl", + "oppdal.no", + "oppegard.no", + "oppegård.no", + "or.at", + "or.bi", + "or.ci", + "or.cr", + "or.id", + "or.it", + "or.jp", + "or.ke", + "or.kr", + "or.mu", + "or.na", + "or.pw", + "or.th", + "or.tz", + "or.ug", + "or.us", + "ora.gunma.jp", + "oracle", + "orange", + "org", + "org.ac", + "org.ae", + "org.af", + "org.ag", + "org.ai", + "org.al", + "org.am", + "org.ar", + "org.au", + "org.az", + "org.ba", + "org.bb", + "org.bh", + "org.bi", + "org.bj", + "org.bm", + "org.bn", + "org.bo", + "org.br", + "org.bs", + "org.bt", + "org.bw", + "org.bz", + "org.ci", + "org.cn", + "org.co", + "org.cu", + "org.cv", + "org.cw", + "org.cy", + "org.dm", + "org.do", + "org.dz", + "org.ec", + "org.ee", + "org.eg", + "org.es", + "org.et", + "org.fj", + "org.fm", + "org.ge", + "org.gg", + "org.gh", + "org.gi", + "org.gl", + "org.gn", + "org.gp", + "org.gr", + "org.gt", + "org.gu", + "org.gy", + "org.hk", + "org.hn", + "org.ht", + "org.hu", + "org.il", + "org.im", + "org.in", + "org.iq", + "org.ir", + "org.is", + "org.je", + "org.jo", + "org.kg", + "org.ki", + "org.km", + "org.kn", + "org.kp", + "org.kw", + "org.ky", + "org.kz", + "org.la", + "org.lb", + "org.lc", + "org.lk", + "org.lr", + "org.ls", + "org.lv", + "org.ly", + "org.ma", + "org.me", + "org.mg", + "org.mk", + "org.ml", + "org.mn", + "org.mo", + "org.ms", + "org.mt", + "org.mu", + "org.mv", + "org.mw", + "org.mx", + "org.my", + "org.mz", + "org.na", + "org.ng", + "org.ni", + "org.nr", + "org.nz", + "org.om", + "org.pa", + "org.pe", + "org.pf", + "org.ph", + "org.pk", + "org.pl", + "org.pn", + "org.pr", + "org.ps", + "org.pt", + "org.py", + "org.qa", + "org.ro", + "org.rs", + "org.rw", + "org.sa", + "org.sb", + "org.sc", + "org.sd", + "org.se", + "org.sg", + "org.sh", + "org.sl", + "org.sn", + "org.so", + "org.ss", + "org.st", + "org.sv", + "org.sy", + "org.sz", + "org.tj", + "org.tm", + "org.tn", + "org.to", + "org.tr", + "org.tt", + "org.tw", + "org.ua", + "org.ug", + "org.uk", + "org.uy", + "org.uz", + "org.vc", + "org.ve", + "org.vi", + "org.vn", + "org.vu", + "org.ws", + "org.ye", + "org.za", + "org.zm", + "org.zw", + "organic", + "origins", + "oristano.it", + "orkanger.no", + "orkdal.no", + "orland.no", + "orskog.no", + "orsta.no", + "os.hedmark.no", + "os.hordaland.no", + "osaka", + "osaka.jp", + "osakasayama.osaka.jp", + "osaki.miyagi.jp", + "osakikamijima.hiroshima.jp", + "osasco.br", + "oschr.gov.pl", + "osen.no", + "oseto.nagasaki.jp", + "oshima.tokyo.jp", + "oshima.yamaguchi.jp", + "oshino.yamanashi.jp", + "oshu.iwate.jp", + "oslo.no", + "osoyro.no", + "osteroy.no", + "osterøy.no", + "ostre-toten.no", + "ostroda.pl", + "ostroleka.pl", + "ostrowiec.pl", + "ostrowwlkp.pl", + "osøyro.no", + "ot.it", + "ota.gunma.jp", + "ota.tokyo.jp", + "otake.hiroshima.jp", + "otaki.chiba.jp", + "otaki.nagano.jp", + "otaki.saitama.jp", + "otama.fukushima.jp", + "otari.nagano.jp", + "otaru.hokkaido.jp", + "ote.bj", + "other.nf", + "oto.fukuoka.jp", + "otobe.hokkaido.jp", + "otofuke.hokkaido.jp", + "otoineppu.hokkaido.jp", + "otoyo.kochi.jp", + "otsu.shiga.jp", + "otsuchi.iwate.jp", + "otsuka", + "otsuki.kochi.jp", + "otsuki.yamanashi.jp", + "ott", + "ouchi.saga.jp", + "ouda.nara.jp", + "oum.gov.pl", + "oumu.hokkaido.jp", + "overhalla.no", + "ovh", + "ovre-eiker.no", + "owani.aomori.jp", + "owariasahi.aichi.jp", + "oyabe.toyama.jp", + "oyama.tochigi.jp", + "oyamazaki.kyoto.jp", + "oyer.no", + "oygarden.no", + "oyodo.nara.jp", + "oystre-slidre.no", + "oz.au", + "ozora.hokkaido.jp", + "ozu.ehime.jp", + "ozu.kumamoto.jp", + "p.bg", + "p.se", + "pa", + "pa.gov.br", + "pa.gov.pl", + "pa.it", + "pa.us", + "padova.it", + "padua.it", + "page", + "palermo.it", + "palmas.br", + "panasonic", + "parachuting.aero", + "paragliding.aero", + "paris", + "parliament.nz", + "parma.it", + "paroch.k12.ma.us", + "pars", + "parti.se", + "partners", + "parts", + "party", + "passenger-association.aero", + "patria.bo", + "pavia.it", + "pay", + "pb.ao", + "pb.gov.br", + "pc.it", + "pc.pl", + "pccw", + "pd.it", + "pe", + "pe.ca", + "pe.gov.br", + "pe.it", + "pe.kr", + "per.la", + "per.nf", + "per.sg", + "perso.ht", + "perso.sn", + "perso.tn", + "perugia.it", + "pesaro-urbino.it", + "pesarourbino.it", + "pescara.it", + "pet", + "pf", + "pfizer", + "pg", + "pg.in", + "pg.it", + "ph", + "pharmaciens.km", + "pharmacy", + "phd", + "philips", + "phone", + "photo", + "photography", + "photos", + "phutho.vn", + "phuyen.vn", + "physio", + "pi.gov.br", + "pi.it", + "piacenza.it", + "pics", + "pictet", + "pictures", + "pid", + "piedmont.it", + "piemonte.it", + "pila.pl", + "pilot.aero", + "pin", + "pinb.gov.pl", + "ping", + "pink", + "pioneer", + "pippu.hokkaido.jp", + "pisa.it", + "pistoia.it", + "pisz.pl", + "piw.gov.pl", + "pizza", + "pk", + "pl", + "pl.ua", + "place", + "play", + "playstation", + "plc.co.im", + "plc.ly", + "plc.uk", + "plo.ps", + "plumbing", + "plurinacional.bo", + "plus", + "pm", + "pmn.it", + "pn", + "pn.it", + "pnc", + "po.gov.pl", + "po.it", + "poa.br", + "podhale.pl", + "podlasie.pl", + "pohl", + "poker", + "pol.dz", + "pol.ht", + "pol.tr", + "police.uk", + "politica.bo", + "politie", + "polkowice.pl", + "poltava.ua", + "pomorskie.pl", + "pomorze.pl", + "ponpes.id", + "pordenone.it", + "porn", + "porsanger.no", + "porsangu.no", + "porsgrunn.no", + "porsáŋgu.no", + "post", + "post.in", + "potenza.it", + "powiat.pl", + "pp.az", + "pp.se", + "ppg.br", + "pr", + "pr.gov.br", + "pr.gov.pl", + "pr.it", + "pr.us", + "pramerica", + "prato.it", + "praxi", + "prd.fr", + "prd.km", + "prd.mg", + "press", + "press.aero", + "press.cy", + "press.ma", + "press.se", + "presse.ci", + "presse.km", + "presse.ml", + "pri.ee", + "prime", + "principe.st", + "priv.hu", + "priv.me", + "priv.no", + "priv.pl", + "pro", + "pro.az", + "pro.br", + "pro.cy", + "pro.ec", + "pro.fj", + "pro.ht", + "pro.in", + "pro.mv", + "pro.na", + "pro.om", + "pro.pr", + "pro.tt", + "pro.vn", + "prochowice.pl", + "prod", + "production.aero", + "productions", + "prof", + "prof.pr", + "profesional.bo", + "progressive", + "promo", + "properties", + "property", + "protection", + "pru", + "prudential", + "pruszkow.pl", + "przeworsk.pl", + "ps", + "psc.br", + "psi.br", + "psp.gov.pl", + "psse.gov.pl", + "pt", + "pt.it", + "pu.it", + "pub", + "pub.sa", + "publ.pt", + "pueblo.bo", + "pug.it", + "puglia.it", + "pulawy.pl", + "pup.gov.pl", + "pv.it", + "pvh.br", + "pvt.ge", + "pvt.k12.ma.us", + "pw", + "pwc", + "py", + "pz.it", + "q.bg", + "qa", + "qc.ca", + "qh.cn", + "qld.au", + "qld.edu.au", + "qld.gov.au", + "qpon", + "qsl.br", + "quangbinh.vn", + "quangnam.vn", + "quangngai.vn", + "quangninh.vn", + "quangtri.vn", + "quebec", + "quest", + "r.bg", + "r.se", + "ra.it", + "racing", + "rade.no", + "radio", + "radio.br", + "radom.pl", + "radoy.no", + "radøy.no", + "ragusa.it", + "rahkkeravju.no", + "raholt.no", + "raisa.no", + "rakkestad.no", + "ralingen.no", + "rana.no", + "randaberg.no", + "rankoshi.hokkaido.jp", + "ranzan.saitama.jp", + "rar.ve", + "rauma.no", + "ravenna.it", + "rawa-maz.pl", + "rc.it", + "re", + "re.it", + "re.kr", + "read", + "realestate", + "realestate.pl", + "realtor", + "realty", + "rebun.hokkaido.jp", + "rec.br", + "rec.co", + "rec.nf", + "rec.ro", + "rec.ve", + "recht.pro", + "recife.br", + "recipes", + "recreation.aero", + "red", + "red.sv", + "redstone", + "redumbrella", + "reggio-calabria.it", + "reggio-emilia.it", + "reggiocalabria.it", + "reggioemilia.it", + "rehab", + "reise", + "reisen", + "reit", + "reklam.hu", + "rel.ht", + "rel.pl", + "reliance", + "ren", + "rendalen.no", + "rennebu.no", + "rennesoy.no", + "rennesøy.no", + "rent", + "rentals", + "rep.br", + "rep.kp", + "repair", + "repbody.aero", + "report", + "republican", + "res.aero", + "res.in", + "research.aero", + "rest", + "restaurant", + "restaurant.bj", + "resto.bj", + "review", + "reviews", + "revista.bo", + "rexroth", + "rg.it", + "ri.it", + "ri.us", + "ribeirao.br", + "rich", + "richardli", + "ricoh", + "rieti.it", + "rifu.miyagi.jp", + "riik.ee", + "rikubetsu.hokkaido.jp", + "rikuzentakata.iwate.jp", + "ril", + "rimini.it", + "rindal.no", + "ringebu.no", + "ringerike.no", + "ringsaker.no", + "rio", + "rio.br", + "riobranco.br", + "riopreto.br", + "rip", + "rishiri.hokkaido.jp", + "rishirifuji.hokkaido.jp", + "risor.no", + "rissa.no", + "risør.no", + "ritto.shiga.jp", + "rivne.ua", + "rj.gov.br", + "rl.no", + "rm.it", + "rn.gov.br", + "rn.it", + "ro", + "ro.gov.br", + "ro.it", + "roan.no", + "rocks", + "rodeo", + "rodoy.no", + "rogers", + "rokunohe.aomori.jp", + "rollag.no", + "roma.it", + "rome.it", + "romsa.no", + "romskog.no", + "room", + "roros.no", + "rost.no", + "rotorcraft.aero", + "rovigo.it", + "rovno.ua", + "royken.no", + "royrvik.no", + "rr.gov.br", + "rs", + "rs.gov.br", + "rsvp", + "ru", + "rugby", + "ruhr", + "run", + "ruovat.no", + "rv.ua", + "rw", + "rwe", + "rybnik.pl", + "rygge.no", + "ryokami.saitama.jp", + "ryugasaki.ibaraki.jp", + "ryukyu", + "ryuoh.shiga.jp", + "rzeszow.pl", + "rzgw.gov.pl", + "ráhkkerávju.no", + "ráisa.no", + "råde.no", + "råholt.no", + "rælingen.no", + "rødøy.no", + "rømskog.no", + "røros.no", + "røst.no", + "røyken.no", + "røyrvik.no", + "s.bg", + "s.se", + "sa", + "sa.au", + "sa.cr", + "sa.edu.au", + "sa.gov.au", + "sa.gov.pl", + "sa.it", + "saarland", + "sabae.fukui.jp", + "sado.niigata.jp", + "safe", + "safety", + "safety.aero", + "saga.jp", + "saga.saga.jp", + "sagae.yamagata.jp", + "sagamihara.kanagawa.jp", + "saigawa.fukuoka.jp", + "saijo.ehime.jp", + "saikai.nagasaki.jp", + "saiki.oita.jp", + "saitama.jp", + "saitama.saitama.jp", + "saito.miyazaki.jp", + "saka.hiroshima.jp", + "sakado.saitama.jp", + "sakae.chiba.jp", + "sakae.nagano.jp", + "sakahogi.gifu.jp", + "sakai.fukui.jp", + "sakai.ibaraki.jp", + "sakai.osaka.jp", + "sakaiminato.tottori.jp", + "sakaki.nagano.jp", + "sakata.yamagata.jp", + "sakawa.kochi.jp", + "sakegawa.yamagata.jp", + "saku.nagano.jp", + "sakuho.nagano.jp", + "sakura", + "sakura.chiba.jp", + "sakura.tochigi.jp", + "sakuragawa.ibaraki.jp", + "sakurai.nara.jp", + "sakyo.kyoto.jp", + "salangen.no", + "salat.no", + "sale", + "salerno.it", + "salon", + "saltdal.no", + "salud.bo", + "salvador.br", + "samegawa.fukushima.jp", + "samnanger.no", + "sampa.br", + "samsclub", + "samsung", + "samukawa.kanagawa.jp", + "sanagochi.tokushima.jp", + "sanda.hyogo.jp", + "sande.more-og-romsdal.no", + "sande.møre-og-romsdal.no", + "sande.vestfold.no", + "sandefjord.no", + "sandnes.no", + "sandnessjoen.no", + "sandnessjøen.no", + "sandoy.no", + "sandvik", + "sandvikcoromant", + "sandøy.no", + "sango.nara.jp", + "sanjo.niigata.jp", + "sannan.hyogo.jp", + "sannohe.aomori.jp", + "sano.tochigi.jp", + "sanofi", + "sanok.pl", + "santamaria.br", + "santoandre.br", + "sanuki.kagawa.jp", + "saobernardo.br", + "saogonca.br", + "saotome.st", + "sap", + "sapporo.jp", + "sar.it", + "sardegna.it", + "sardinia.it", + "sarl", + "saroma.hokkaido.jp", + "sarpsborg.no", + "sarufutsu.hokkaido.jp", + "sas", + "sasaguri.fukuoka.jp", + "sasayama.hyogo.jp", + "sasebo.nagasaki.jp", + "sassari.it", + "satosho.okayama.jp", + "satsumasendai.kagoshima.jp", + "satte.saitama.jp", + "sauda.no", + "sauherad.no", + "save", + "savona.it", + "saxo", + "sayama.osaka.jp", + "sayama.saitama.jp", + "sayo.hyogo.jp", + "sb", + "sb.ua", + "sbi", + "sbs", + "sc", + "sc.cn", + "sc.gov.br", + "sc.ke", + "sc.kr", + "sc.ls", + "sc.tz", + "sc.ug", + "sc.us", + "scb", + "sch.ae", + "sch.id", + "sch.ir", + "sch.jo", + "sch.lk", + "sch.ly", + "sch.ng", + "sch.qa", + "sch.sa", + "sch.ss", + "sch.uk", + "sch.zm", + "schaeffler", + "schmidt", + "scholarships", + "school", + "school.na", + "school.nz", + "school.za", + "schools.nsw.edu.au", + "schule", + "schwarz", + "sci.eg", + "science", + "scientist.aero", + "scot", + "sd", + "sd.cn", + "sd.us", + "sdn.gov.pl", + "se", + "se.gov.br", + "search", + "seat", + "sebastopol.ua", + "sec.ps", + "secure", + "security", + "seek", + "seg.br", + "seihi.nagasaki.jp", + "seika.kyoto.jp", + "seiro.niigata.jp", + "seirou.niigata.jp", + "seiyo.ehime.jp", + "sejny.pl", + "seki.gifu.jp", + "sekigahara.gifu.jp", + "sekikawa.niigata.jp", + "sel.no", + "selbu.no", + "select", + "selje.no", + "seljord.no", + "semboku.akita.jp", + "semine.miyagi.jp", + "senasa.ar", + "sendai.jp", + "sener", + "sennan.osaka.jp", + "seoul.kr", + "sera.hiroshima.jp", + "seranishi.hiroshima.jp", + "services", + "services.aero", + "setagaya.tokyo.jp", + "seto.aichi.jp", + "setouchi.okayama.jp", + "settsu.osaka.jp", + "sevastopol.ua", + "seven", + "sew", + "sex", + "sex.hu", + "sex.pl", + "sexy", + "sf.no", + "sfr", + "sg", + "sh", + "sh.cn", + "shakotan.hokkaido.jp", + "shangrila", + "shari.hokkaido.jp", + "sharp", + "shell", + "shia", + "shibata.miyagi.jp", + "shibata.niigata.jp", + "shibecha.hokkaido.jp", + "shibetsu.hokkaido.jp", + "shibukawa.gunma.jp", + "shibuya.tokyo.jp", + "shichikashuku.miyagi.jp", + "shichinohe.aomori.jp", + "shiga.jp", + "shiiba.miyazaki.jp", + "shijonawate.osaka.jp", + "shika.ishikawa.jp", + "shikabe.hokkaido.jp", + "shikama.miyagi.jp", + "shikaoi.hokkaido.jp", + "shikatsu.aichi.jp", + "shiki.saitama.jp", + "shikokuchuo.ehime.jp", + "shiksha", + "shima.mie.jp", + "shimabara.nagasaki.jp", + "shimada.shizuoka.jp", + "shimamaki.hokkaido.jp", + "shimamoto.osaka.jp", + "shimane.jp", + "shimane.shimane.jp", + "shimizu.hokkaido.jp", + "shimizu.shizuoka.jp", + "shimoda.shizuoka.jp", + "shimodate.ibaraki.jp", + "shimofusa.chiba.jp", + "shimogo.fukushima.jp", + "shimoichi.nara.jp", + "shimoji.okinawa.jp", + "shimokawa.hokkaido.jp", + "shimokitayama.nara.jp", + "shimonita.gunma.jp", + "shimonoseki.yamaguchi.jp", + "shimosuwa.nagano.jp", + "shimotsuke.tochigi.jp", + "shimotsuma.ibaraki.jp", + "shinagawa.tokyo.jp", + "shinanomachi.nagano.jp", + "shingo.aomori.jp", + "shingu.fukuoka.jp", + "shingu.hyogo.jp", + "shingu.wakayama.jp", + "shinichi.hiroshima.jp", + "shinjo.nara.jp", + "shinjo.okayama.jp", + "shinjo.yamagata.jp", + "shinjuku.tokyo.jp", + "shinkamigoto.nagasaki.jp", + "shinonsen.hyogo.jp", + "shinshinotsu.hokkaido.jp", + "shinshiro.aichi.jp", + "shinto.gunma.jp", + "shintoku.hokkaido.jp", + "shintomi.miyazaki.jp", + "shinyoshitomi.fukuoka.jp", + "shiogama.miyagi.jp", + "shiojiri.nagano.jp", + "shioya.tochigi.jp", + "shirahama.wakayama.jp", + "shirakawa.fukushima.jp", + "shirakawa.gifu.jp", + "shirako.chiba.jp", + "shiranuka.hokkaido.jp", + "shiraoi.hokkaido.jp", + "shiraoka.saitama.jp", + "shirataka.yamagata.jp", + "shiriuchi.hokkaido.jp", + "shiroi.chiba.jp", + "shiroishi.miyagi.jp", + "shiroishi.saga.jp", + "shirosato.ibaraki.jp", + "shishikui.tokushima.jp", + "shiso.hyogo.jp", + "shisui.chiba.jp", + "shitara.aichi.jp", + "shiwa.iwate.jp", + "shizukuishi.iwate.jp", + "shizuoka.jp", + "shizuoka.shizuoka.jp", + "shobara.hiroshima.jp", + "shoes", + "shonai.fukuoka.jp", + "shonai.yamagata.jp", + "shoo.okayama.jp", + "shop", + "shop.ht", + "shop.hu", + "shop.pl", + "shopping", + "shouji", + "show", + "show.aero", + "showa.fukushima.jp", + "showa.gunma.jp", + "showa.yamanashi.jp", + "shunan.yamaguchi.jp", + "si", + "si.it", + "sic.it", + "sicilia.it", + "sicily.it", + "siellak.no", + "siena.it", + "sigdal.no", + "siljan.no", + "silk", + "sina", + "singles", + "siracusa.it", + "sirdal.no", + "site", + "sj", + "sjc.br", + "sk", + "sk.ca", + "skanit.no", + "skanland.no", + "skaun.no", + "skedsmo.no", + "skedsmokorset.no", + "ski", + "ski.no", + "skien.no", + "skierva.no", + "skiervá.no", + "skin", + "skiptvet.no", + "skjak.no", + "skjervoy.no", + "skjervøy.no", + "skjåk.no", + "sklep.pl", + "sko.gov.pl", + "skoczow.pl", + "skodje.no", + "sky", + "skydiving.aero", + "skype", + "skánit.no", + "skånland.no", + "sl", + "slask.pl", + "slattum.no", + "sld.do", + "sld.pa", + "slg.br", + "sling", + "slupsk.pl", + "slz.br", + "sm", + "sm.ua", + "smart", + "smile", + "smola.no", + "smøla.no", + "sn", + "sn.cn", + "snaase.no", + "snasa.no", + "sncf", + "snillfjord.no", + "snoasa.no", + "snåase.no", + "snåsa.no", + "so", + "so.gov.pl", + "so.it", + "sobetsu.hokkaido.jp", + "soc.dz", + "soc.lk", + "soccer", + "social", + "soctrang.vn", + "sodegaura.chiba.jp", + "soeda.fukuoka.jp", + "softbank", + "software", + "software.aero", + "sogndal.no", + "sogne.no", + "sohu", + "soja.okayama.jp", + "soka.saitama.jp", + "sokndal.no", + "sola.no", + "solar", + "solund.no", + "solutions", + "soma.fukushima.jp", + "somna.no", + "sondre-land.no", + "sondrio.it", + "song", + "songdalen.no", + "soni.nara.jp", + "sonla.vn", + "sony", + "soo.kagoshima.jp", + "sor-aurdal.no", + "sor-fron.no", + "sor-odal.no", + "sor-varanger.no", + "sorfold.no", + "sorocaba.br", + "sorreisa.no", + "sortland.no", + "sorum.no", + "sos.pl", + "sosa.chiba.jp", + "sosnowiec.pl", + "sowa.ibaraki.jp", + "soy", + "sp.gov.br", + "sp.it", + "spa", + "space", + "spjelkavik.no", + "sport", + "sport.hu", + "spot", + "spydeberg.no", + "sr", + "sr.gov.pl", + "sr.it", + "srl", + "srv.br", + "ss", + "ss.it", + "st", + "st.no", + "stada", + "stalowa-wola.pl", + "stange.no", + "staples", + "star", + "starachowice.pl", + "stargard.pl", + "starostwo.gov.pl", + "stat.no", + "statebank", + "statefarm", + "stathelle.no", + "stavanger.no", + "stavern.no", + "stc", + "stcgroup", + "steigen.no", + "steinkjer.no", + "sth.ac.at", + "stjordal.no", + "stjordalshalsen.no", + "stjørdal.no", + "stjørdalshalsen.no", + "stockholm", + "stokke.no", + "stor-elvdal.no", + "storage", + "stord.no", + "stordal.no", + "store", + "store.bb", + "store.nf", + "store.ro", + "store.st", + "store.ve", + "storfjord.no", + "strand.no", + "stranda.no", + "stream", + "stryn.no", + "student.aero", + "studio", + "study", + "style", + "su", + "sucks", + "sue.fukuoka.jp", + "suedtirol.it", + "suginami.tokyo.jp", + "sugito.saitama.jp", + "suifu.ibaraki.jp", + "suita.osaka.jp", + "sukagawa.fukushima.jp", + "sukumo.kochi.jp", + "sula.no", + "suldal.no", + "suli.hu", + "sumida.tokyo.jp", + "sumita.iwate.jp", + "sumoto.hyogo.jp", + "sumoto.kumamoto.jp", + "sumy.ua", + "sunagawa.hokkaido.jp", + "sund.no", + "sunndal.no", + "supplies", + "supply", + "support", + "surf", + "surgery", + "surnadal.no", + "susaki.kochi.jp", + "susono.shizuoka.jp", + "suwa.nagano.jp", + "suwalki.pl", + "suzaka.nagano.jp", + "suzu.ishikawa.jp", + "suzuka.mie.jp", + "suzuki", + "sv", + "sv.it", + "svalbard.no", + "sveio.no", + "svelvik.no", + "swatch", + "swidnica.pl", + "swiebodzin.pl", + "swinoujscie.pl", + "swiss", + "sx", + "sx.cn", + "sy", + "sydney", + "sykkylven.no", + "systems", + "sz", + "szczecin.pl", + "szczytno.pl", + "szex.hu", + "szkola.pl", + "sálat.no", + "sálát.no", + "søgne.no", + "sømna.no", + "søndre-land.no", + "sør-aurdal.no", + "sør-fron.no", + "sør-odal.no", + "sør-varanger.no", + "sørfold.no", + "sørreisa.no", + "sørum.no", + "südtirol.it", + "t.bg", + "t.se", + "ta.it", + "taa.it", + "tab", + "tabayama.yamanashi.jp", + "tabuse.yamaguchi.jp", + "tachiarai.fukuoka.jp", + "tachikawa.tokyo.jp", + "tadaoka.osaka.jp", + "tado.mie.jp", + "tadotsu.kagawa.jp", + "tagajo.miyagi.jp", + "tagami.niigata.jp", + "tagawa.fukuoka.jp", + "tahara.aichi.jp", + "taiji.wakayama.jp", + "taiki.hokkaido.jp", + "taiki.mie.jp", + "tainai.niigata.jp", + "taipei", + "taira.toyama.jp", + "taishi.hyogo.jp", + "taishi.osaka.jp", + "taishin.fukushima.jp", + "taito.tokyo.jp", + "taiwa.miyagi.jp", + "tajimi.gifu.jp", + "tajiri.osaka.jp", + "taka.hyogo.jp", + "takagi.nagano.jp", + "takahagi.ibaraki.jp", + "takahama.aichi.jp", + "takahama.fukui.jp", + "takaharu.miyazaki.jp", + "takahashi.okayama.jp", + "takahata.yamagata.jp", + "takaishi.osaka.jp", + "takamatsu.kagawa.jp", + "takamori.kumamoto.jp", + "takamori.nagano.jp", + "takanabe.miyazaki.jp", + "takanezawa.tochigi.jp", + "takaoka.toyama.jp", + "takarazuka.hyogo.jp", + "takasago.hyogo.jp", + "takasaki.gunma.jp", + "takashima.shiga.jp", + "takasu.hokkaido.jp", + "takata.fukuoka.jp", + "takatori.nara.jp", + "takatsuki.osaka.jp", + "takatsuki.shiga.jp", + "takayama.gifu.jp", + "takayama.gunma.jp", + "takayama.nagano.jp", + "takazaki.miyazaki.jp", + "takehara.hiroshima.jp", + "taketa.oita.jp", + "taketomi.okinawa.jp", + "taki.mie.jp", + "takikawa.hokkaido.jp", + "takino.hyogo.jp", + "takinoue.hokkaido.jp", + "takko.aomori.jp", + "tako.chiba.jp", + "taku.saga.jp", + "talk", + "tama.tokyo.jp", + "tamakawa.fukushima.jp", + "tamaki.mie.jp", + "tamamura.gunma.jp", + "tamano.okayama.jp", + "tamatsukuri.ibaraki.jp", + "tamayu.shimane.jp", + "tamba.hyogo.jp", + "tana.no", + "tanabe.kyoto.jp", + "tanabe.wakayama.jp", + "tanagura.fukushima.jp", + "tananger.no", + "tanohata.iwate.jp", + "taobao", + "tara.saga.jp", + "tarama.okinawa.jp", + "taranto.it", + "target", + "targi.pl", + "tarnobrzeg.pl", + "tarui.gifu.jp", + "tarumizu.kagoshima.jp", + "tas.au", + "tas.edu.au", + "tas.gov.au", + "tatamotors", + "tatar", + "tatebayashi.gunma.jp", + "tateshina.nagano.jp", + "tateyama.chiba.jp", + "tateyama.toyama.jp", + "tatsuno.hyogo.jp", + "tatsuno.nagano.jp", + "tattoo", + "tawaramoto.nara.jp", + "tax", + "taxi", + "taxi.aero", + "taxi.br", + "tayninh.vn", + "tc", + "tc.br", + "tci", + "td", + "tdk", + "te.it", + "te.ua", + "team", + "tec.br", + "tec.mi.us", + "tec.ve", + "tech", + "technology", + "tecnologia.bo", + "tel", + "tel.tr", + "temasek", + "tempio-olbia.it", + "tempioolbia.it", + "tendo.yamagata.jp", + "tenei.fukushima.jp", + "tenkawa.nara.jp", + "tennis", + "tenri.nara.jp", + "teo.br", + "teramo.it", + "terni.it", + "ternopil.ua", + "teshikaga.hokkaido.jp", + "test.tj", + "teva", + "tf", + "tg", + "tgory.pl", + "th", + "thaibinh.vn", + "thainguyen.vn", + "thanhhoa.vn", + "thanhphohochiminh.vn", + "thd", + "the.br", + "theater", + "theatre", + "thuathienhue.vn", + "tiaa", + "tickets", + "tienda", + "tiengiang.vn", + "time.no", + "tingvoll.no", + "tinn.no", + "tips", + "tires", + "tirol", + "tj", + "tj.cn", + "tjeldsund.no", + "tjmaxx", + "tjome.no", + "tjx", + "tjøme.no", + "tk", + "tkmaxx", + "tksat.bo", + "tl", + "tm", + "tm.cy", + "tm.dz", + "tm.fr", + "tm.hu", + "tm.km", + "tm.mc", + "tm.mg", + "tm.no", + "tm.pl", + "tm.ro", + "tm.se", + "tm.za", + "tmall", + "tmp.br", + "tn", + "tn.it", + "tn.us", + "to", + "to.gov.br", + "to.it", + "toba.mie.jp", + "tobe.ehime.jp", + "tobetsu.hokkaido.jp", + "tobishima.aichi.jp", + "tochigi.jp", + "tochigi.tochigi.jp", + "tochio.niigata.jp", + "toda.saitama.jp", + "today", + "toei.aichi.jp", + "toga.toyama.jp", + "togakushi.nagano.jp", + "togane.chiba.jp", + "togitsu.nagasaki.jp", + "togo.aichi.jp", + "togura.nagano.jp", + "tohma.hokkaido.jp", + "tohnosho.chiba.jp", + "toho.fukuoka.jp", + "tokai.aichi.jp", + "tokai.ibaraki.jp", + "tokamachi.niigata.jp", + "tokashiki.okinawa.jp", + "toki.gifu.jp", + "tokigawa.saitama.jp", + "tokke.no", + "tokoname.aichi.jp", + "tokorozawa.saitama.jp", + "tokushima.jp", + "tokushima.tokushima.jp", + "tokuyama.yamaguchi.jp", + "tokyo", + "tokyo.jp", + "tolga.no", + "tomakomai.hokkaido.jp", + "tomari.hokkaido.jp", + "tome.miyagi.jp", + "tomi.nagano.jp", + "tomigusuku.okinawa.jp", + "tomika.gifu.jp", + "tomioka.gunma.jp", + "tomisato.chiba.jp", + "tomiya.miyagi.jp", + "tomobe.ibaraki.jp", + "tonaki.okinawa.jp", + "tonami.toyama.jp", + "tondabayashi.osaka.jp", + "tone.ibaraki.jp", + "tono.iwate.jp", + "tonosho.kagawa.jp", + "tonsberg.no", + "tools", + "toon.ehime.jp", + "top", + "torahime.shiga.jp", + "toray", + "toride.ibaraki.jp", + "torino.it", + "torsken.no", + "tos.it", + "tosa.kochi.jp", + "tosashimizu.kochi.jp", + "toscana.it", + "toshiba", + "toshima.tokyo.jp", + "tosu.saga.jp", + "total", + "tottori.jp", + "tottori.tottori.jp", + "tourism.bj", + "tourism.pl", + "tourism.tn", + "tours", + "towada.aomori.jp", + "town", + "toya.hokkaido.jp", + "toyako.hokkaido.jp", + "toyama.jp", + "toyama.toyama.jp", + "toyo.kochi.jp", + "toyoake.aichi.jp", + "toyohashi.aichi.jp", + "toyokawa.aichi.jp", + "toyonaka.osaka.jp", + "toyone.aichi.jp", + "toyono.osaka.jp", + "toyooka.hyogo.jp", + "toyosato.shiga.jp", + "toyota", + "toyota.aichi.jp", + "toyota.yamaguchi.jp", + "toyotomi.hokkaido.jp", + "toyotsu.fukuoka.jp", + "toyoura.hokkaido.jp", + "toys", + "tozawa.yamagata.jp", + "tozsde.hu", + "tp.it", + "tr", + "tr.it", + "tr.no", + "tra.kp", + "trade", + "trader.aero", + "trading", + "trading.aero", + "trainer.aero", + "training", + "trana.no", + "tranby.no", + "trani-andria-barletta.it", + "trani-barletta-andria.it", + "traniandriabarletta.it", + "tranibarlettaandria.it", + "tranoy.no", + "transporte.bo", + "tranøy.no", + "trapani.it", + "travel", + "travel.in", + "travel.pl", + "travel.tt", + "travelers", + "travelersinsurance", + "travinh.vn", + "trd.br", + "trentin-sud-tirol.it", + "trentin-sudtirol.it", + "trentin-sued-tirol.it", + "trentin-suedtirol.it", + "trentin-süd-tirol.it", + "trentin-südtirol.it", + "trentino-a-adige.it", + "trentino-aadige.it", + "trentino-alto-adige.it", + "trentino-altoadige.it", + "trentino-s-tirol.it", + "trentino-stirol.it", + "trentino-sud-tirol.it", + "trentino-sudtirol.it", + "trentino-sued-tirol.it", + "trentino-suedtirol.it", + "trentino-süd-tirol.it", + "trentino-südtirol.it", + "trentino.it", + "trentinoa-adige.it", + "trentinoaadige.it", + "trentinoalto-adige.it", + "trentinoaltoadige.it", + "trentinos-tirol.it", + "trentinostirol.it", + "trentinosud-tirol.it", + "trentinosudtirol.it", + "trentinosued-tirol.it", + "trentinosuedtirol.it", + "trentinosüd-tirol.it", + "trentinosüdtirol.it", + "trentinsud-tirol.it", + "trentinsudtirol.it", + "trentinsued-tirol.it", + "trentinsuedtirol.it", + "trentinsüd-tirol.it", + "trentinsüdtirol.it", + "trento.it", + "treviso.it", + "trieste.it", + "troandin.no", + "trogstad.no", + "tromsa.no", + "tromso.no", + "tromsø.no", + "trondheim.no", + "trust", + "trv", + "trysil.no", + "træna.no", + "trøgstad.no", + "ts.it", + "tsk.tr", + "tsu.mie.jp", + "tsubame.niigata.jp", + "tsubata.ishikawa.jp", + "tsubetsu.hokkaido.jp", + "tsuchiura.ibaraki.jp", + "tsuga.tochigi.jp", + "tsugaru.aomori.jp", + "tsuiki.fukuoka.jp", + "tsukigata.hokkaido.jp", + "tsukiyono.gunma.jp", + "tsukuba.ibaraki.jp", + "tsukui.kanagawa.jp", + "tsukumi.oita.jp", + "tsumagoi.gunma.jp", + "tsunan.niigata.jp", + "tsuno.kochi.jp", + "tsuno.miyazaki.jp", + "tsuru.yamanashi.jp", + "tsuruga.fukui.jp", + "tsurugashima.saitama.jp", + "tsurugi.ishikawa.jp", + "tsuruoka.yamagata.jp", + "tsuruta.aomori.jp", + "tsushima.aichi.jp", + "tsushima.nagasaki.jp", + "tsuwano.shimane.jp", + "tsuyama.okayama.jp", + "tt", + "tt.im", + "tube", + "tui", + "tunes", + "tur.ar", + "tur.br", + "turek.pl", + "turin.it", + "turystyka.pl", + "tuscany.it", + "tushu", + "tuyenquang.vn", + "tv", + "tv.bb", + "tv.bo", + "tv.br", + "tv.im", + "tv.in", + "tv.it", + "tv.na", + "tv.sd", + "tv.tr", + "tv.tz", + "tvedestrand.no", + "tvs", + "tw", + "tw.cn", + "tx.us", + "tychy.pl", + "tydal.no", + "tynset.no", + "tysfjord.no", + "tysnes.no", + "tysvar.no", + "tysvær.no", + "tz", + "tønsberg.no", + "u.bg", + "u.se", + "ua", + "ubank", + "ube.yamaguchi.jp", + "ubs", + "uchihara.ibaraki.jp", + "uchiko.ehime.jp", + "uchinada.ishikawa.jp", + "uchinomi.kagawa.jp", + "ud.it", + "uda.nara.jp", + "udi.br", + "udine.it", + "udono.mie.jp", + "ueda.nagano.jp", + "ueno.gunma.jp", + "uenohara.yamanashi.jp", + "ug", + "ug.gov.pl", + "ugim.gov.pl", + "uji.kyoto.jp", + "ujiie.tochigi.jp", + "ujitawara.kyoto.jp", + "uk", + "uk.in", + "uki.kumamoto.jp", + "ukiha.fukuoka.jp", + "ullensaker.no", + "ullensvang.no", + "ulsan.kr", + "ulvik.no", + "um.gov.pl", + "umaji.kochi.jp", + "umb.it", + "umbria.it", + "umi.fukuoka.jp", + "umig.gov.pl", + "unazuki.toyama.jp", + "unicom", + "union.aero", + "univ.bj", + "univ.sn", + "university", + "unjarga.no", + "unjárga.no", + "unnan.shimane.jp", + "uno", + "unzen.nagasaki.jp", + "uol", + "uonuma.niigata.jp", + "uozu.toyama.jp", + "up.in", + "upow.gov.pl", + "uppo.gov.pl", + "ups", + "urakawa.hokkaido.jp", + "urasoe.okinawa.jp", + "urausu.hokkaido.jp", + "urawa.saitama.jp", + "urayasu.chiba.jp", + "urbino-pesaro.it", + "urbinopesaro.it", + "ureshino.mie.jp", + "uri.arpa", + "urn.arpa", + "uruma.okinawa.jp", + "uryu.hokkaido.jp", + "us", + "us.gov.pl", + "us.in", + "us.na", + "usa.oita.jp", + "ushiku.ibaraki.jp", + "ustka.pl", + "usui.fukuoka.jp", + "usuki.oita.jp", + "ut.us", + "utashinai.hokkaido.jp", + "utazas.hu", + "utazu.kagawa.jp", + "uto.kumamoto.jp", + "utsira.no", + "utsunomiya.tochigi.jp", + "uw.gov.pl", + "uwajima.ehime.jp", + "uy", + "uz", + "uz.ua", + "uzhgorod.ua", + "uzhhorod.ua", + "uzs.gov.pl", + "v.bg", + "va", + "va.it", + "va.no", + "va.us", + "vaapste.no", + "vacations", + "vadso.no", + "vadsø.no", + "vaga.no", + "vagan.no", + "vagsoy.no", + "vaksdal.no", + "val-d-aosta.it", + "val-daosta.it", + "vald-aosta.it", + "valdaosta.it", + "valer.hedmark.no", + "valer.ostfold.no", + "valle-aosta.it", + "valle-d-aosta.it", + "valle-daosta.it", + "valle.no", + "valleaosta.it", + "valled-aosta.it", + "valledaosta.it", + "vallee-aoste.it", + "vallee-d-aoste.it", + "valleeaoste.it", + "valleedaoste.it", + "vallée-aoste.it", + "vallée-d-aoste.it", + "valléeaoste.it", + "valléedaoste.it", + "vana", + "vang.no", + "vanguard", + "vanylven.no", + "vao.it", + "vardo.no", + "vardø.no", + "varese.it", + "varggat.no", + "varoy.no", + "vb.it", + "vc", + "vc.it", + "vda.it", + "ve", + "ve.it", + "vefsn.no", + "vega.no", + "vegarshei.no", + "vegas", + "vegårshei.no", + "ven.it", + "veneto.it", + "venezia.it", + "venice.it", + "vennesla.no", + "ventures", + "verbania.it", + "vercelli.it", + "verdal.no", + "verisign", + "vermögensberater", + "vermögensberatung", + "verona.it", + "verran.no", + "versicherung", + "vestby.no", + "vestnes.no", + "vestre-slidre.no", + "vestre-toten.no", + "vestvagoy.no", + "vestvågøy.no", + "vet", + "vet.br", + "veterinaire.km", + "vevelstad.no", + "vf.no", + "vg", + "vgs.no", + "vi", + "vi.it", + "vi.us", + "viajes", + "vibo-valentia.it", + "vibovalentia.it", + "vic.au", + "vic.edu.au", + "vic.gov.au", + "vicenza.it", + "video", + "video.hu", + "vig", + "vik.no", + "viking", + "vikna.no", + "villas", + "vin", + "vindafjord.no", + "vinhlong.vn", + "vinhphuc.vn", + "vinnica.ua", + "vinnytsia.ua", + "vip", + "virgin", + "visa", + "vision", + "viterbo.it", + "viva", + "vivo", + "vix.br", + "vlaanderen", + "vlog.br", + "vn", + "vn.ua", + "voagat.no", + "vodka", + "volda.no", + "volvo", + "volyn.ua", + "voss.no", + "vossevangen.no", + "vote", + "voting", + "voto", + "voyage", + "vr.it", + "vs.it", + "vt.it", + "vt.us", + "vu", + "vv.it", + "várggát.no", + "vågan.no", + "vågsøy.no", + "vågå.no", + "våler.hedmark.no", + "våler.østfold.no", + "værøy.no", + "w.bg", + "w.se", + "wa.au", + "wa.edu.au", + "wa.gov.au", + "wa.us", + "wada.nagano.jp", + "wajiki.tokushima.jp", + "wajima.ishikawa.jp", + "wakasa.fukui.jp", + "wakasa.tottori.jp", + "wakayama.jp", + "wakayama.wakayama.jp", + "wake.okayama.jp", + "wakkanai.hokkaido.jp", + "wakuya.miyagi.jp", + "walbrzych.pl", + "wales", + "walmart", + "walter", + "wang", + "wanggou", + "wanouchi.gifu.jp", + "warabi.saitama.jp", + "warmia.pl", + "warszawa.pl", + "washtenaw.mi.us", + "wassamu.hokkaido.jp", + "watarai.mie.jp", + "watari.miyagi.jp", + "watch", + "watches", + "waw.pl", + "wazuka.kyoto.jp", + "weather", + "weatherchannel", + "web.bo", + "web.co", + "web.do", + "web.gu", + "web.id", + "web.lk", + "web.nf", + "web.ni", + "web.pk", + "web.tj", + "web.tr", + "web.ve", + "web.za", + "webcam", + "weber", + "website", + "wed", + "wedding", + "wegrow.pl", + "weibo", + "weir", + "wf", + "whoswho", + "wi.us", + "wielun.pl", + "wien", + "wif.gov.pl", + "wiih.gov.pl", + "wiki", + "wiki.bo", + "wiki.br", + "williamhill", + "win", + "winb.gov.pl", + "windows", + "wine", + "winners", + "wios.gov.pl", + "witd.gov.pl", + "wiw.gov.pl", + "wkz.gov.pl", + "wlocl.pl", + "wloclawek.pl", + "wme", + "wodzislaw.pl", + "wolomin.pl", + "wolterskluwer", + "woodside", + "work", + "workinggroup.aero", + "works", + "works.aero", + "world", + "wow", + "wroclaw.pl", + "ws", + "ws.na", + "wsa.gov.pl", + "wskr.gov.pl", + "wsse.gov.pl", + "wtc", + "wtf", + "wuoz.gov.pl", + "wv.us", + "www.ck", + "www.ro", + "wy.us", + "wzmiuw.gov.pl", + "x.bg", + "x.se", + "xbox", + "xerox", + "xihuan", + "xin", + "xj.cn", + "xxx", + "xyz", + "xz.cn", + "y.bg", + "y.se", + "yabu.hyogo.jp", + "yabuki.fukushima.jp", + "yachimata.chiba.jp", + "yachiyo.chiba.jp", + "yachiyo.ibaraki.jp", + "yachts", + "yaese.okinawa.jp", + "yahaba.iwate.jp", + "yahiko.niigata.jp", + "yahoo", + "yaita.tochigi.jp", + "yaizu.shizuoka.jp", + "yakage.okayama.jp", + "yakumo.hokkaido.jp", + "yakumo.shimane.jp", + "yalta.ua", + "yamada.fukuoka.jp", + "yamada.iwate.jp", + "yamada.toyama.jp", + "yamaga.kumamoto.jp", + "yamagata.gifu.jp", + "yamagata.ibaraki.jp", + "yamagata.jp", + "yamagata.nagano.jp", + "yamagata.yamagata.jp", + "yamaguchi.jp", + "yamakita.kanagawa.jp", + "yamamoto.miyagi.jp", + "yamanakako.yamanashi.jp", + "yamanashi.jp", + "yamanashi.yamanashi.jp", + "yamanobe.yamagata.jp", + "yamanouchi.nagano.jp", + "yamashina.kyoto.jp", + "yamato.fukushima.jp", + "yamato.kanagawa.jp", + "yamato.kumamoto.jp", + "yamatokoriyama.nara.jp", + "yamatotakada.nara.jp", + "yamatsuri.fukushima.jp", + "yamaxun", + "yamazoe.nara.jp", + "yame.fukuoka.jp", + "yanagawa.fukuoka.jp", + "yanaizu.fukushima.jp", + "yandex", + "yao.osaka.jp", + "yaotsu.gifu.jp", + "yasaka.nagano.jp", + "yashio.saitama.jp", + "yashiro.hyogo.jp", + "yasu.shiga.jp", + "yasuda.kochi.jp", + "yasugi.shimane.jp", + "yasuoka.nagano.jp", + "yatomi.aichi.jp", + "yatsuka.shimane.jp", + "yatsushiro.kumamoto.jp", + "yawara.ibaraki.jp", + "yawata.kyoto.jp", + "yawatahama.ehime.jp", + "yazu.tottori.jp", + "ye", + "yenbai.vn", + "yk.ca", + "yn.cn", + "yodobashi", + "yoga", + "yoichi.hokkaido.jp", + "yoita.niigata.jp", + "yoka.hyogo.jp", + "yokaichiba.chiba.jp", + "yokawa.hyogo.jp", + "yokkaichi.mie.jp", + "yokohama", + "yokohama.jp", + "yokoshibahikari.chiba.jp", + "yokosuka.kanagawa.jp", + "yokote.akita.jp", + "yokoze.saitama.jp", + "yomitan.okinawa.jp", + "yonabaru.okinawa.jp", + "yonago.tottori.jp", + "yonaguni.okinawa.jp", + "yonezawa.yamagata.jp", + "yono.saitama.jp", + "yorii.saitama.jp", + "yoro.gifu.jp", + "yoshida.saitama.jp", + "yoshida.shizuoka.jp", + "yoshikawa.saitama.jp", + "yoshimi.saitama.jp", + "yoshino.nara.jp", + "yoshinogari.saga.jp", + "yoshioka.gunma.jp", + "yotsukaido.chiba.jp", + "you", + "youtube", + "yt", + "yuasa.wakayama.jp", + "yufu.oita.jp", + "yugawa.fukushima.jp", + "yugawara.kanagawa.jp", + "yuki.ibaraki.jp", + "yukuhashi.fukuoka.jp", + "yun", + "yura.wakayama.jp", + "yurihonjo.akita.jp", + "yusuhara.kochi.jp", + "yusui.kagoshima.jp", + "yuu.yamaguchi.jp", + "yuza.yamagata.jp", + "yuzawa.niigata.jp", + "z.bg", + "z.se", + "za", + "zachpomor.pl", + "zagan.pl", + "zakarpattia.ua", + "zama.kanagawa.jp", + "zamami.okinawa.jp", + "zao.miyagi.jp", + "zaporizhzhe.ua", + "zaporizhzhia.ua", + "zappos", + "zara", + "zarow.pl", + "zentsuji.kagawa.jp", + "zero", + "zgora.pl", + "zgorzelec.pl", + "zhitomir.ua", + "zhytomyr.ua", + "zip", + "zj.cn", + "zlg.br", + "zm", + "zone", + "zp.gov.pl", + "zp.ua", + "zpisdn.gov.pl", + "zt.ua", + "zuerich", + "zushi.kanagawa.jp", + "zw", + "ákŋoluokta.no", + "álaheadju.no", + "áltá.no", + "åfjord.no", + "åkrehamn.no", + "ål.no", + "ålesund.no", + "ålgård.no", + "åmli.no", + "åmot.no", + "årdal.no", + "ås.no", + "åseral.no", + "åsnes.no", + "øksnes.no", + "ørland.no", + "ørskog.no", + "ørsta.no", + "østre-toten.no", + "øvre-eiker.no", + "øyer.no", + "øygarden.no", + "øystre-slidre.no", + "čáhcesuolo.no", + "ελ", + "ευ", + "ак.срб", + "бг", + "бел", + "дети", + "ею", + "католик", + "ком", + "мкд", + "мон", + "москва", + "обр.срб", + "од.срб", + "онлайн", + "орг", + "орг.срб", + "пр.срб", + "рус", + "рф", + "сайт", + "срб", + "укр", + "упр.срб", + "қаз", + "հայ", + "אקדמיה.ישראל", + "ישוב.ישראל", + "ישראל", + "ממשל.ישראל", + "צהל.ישראל", + "קום", + "ابوظبي", + "ارامكو", + "الاردن", + "البحرين", + "الجزائر", + "السعودية", + "السعوديه", + "السعودیة", + "السعودیۃ", + "العليان", + "المغرب", + "اليمن", + "امارات", + "ايران", + "ايران.ir", + "ایران", + "ایران.ir", + "بارت", + "بازار", + "بيتك", + "بھارت", + "تونس", + "سودان", + "سوريا", + "سورية", + "شبكة", + "عراق", + "عرب", + "عمان", + "فلسطين", + "قطر", + "كاثوليك", + "كوم", + "مصر", + "مليسيا", + "موريتانيا", + "موقع", + "همراه", + "پاكستان", + "پاکستان", + "ڀارت", + "कॉम", + "नेट", + "भारत", + "भारतम्", + "भारोत", + "संगठन", + "বাংলা", + "ভারত", + "ভাৰত", + "ਭਾਰਤ", + "ભારત", + "ଭାରତ", + "இந்தியா", + "இலங்கை", + "சிங்கப்பூர்", + "భారత్", + "ಭಾರತ", + "ഭാരതം", + "ලංකා", + "คอม", + "ทหาร.ไทย", + "ธุรกิจ.ไทย", + "รัฐบาล.ไทย", + "ศึกษา.ไทย", + "องค์กร.ไทย", + "เน็ต.ไทย", + "ไทย", + "ລາວ", + "გე", + "みんな", + "アマゾン", + "クラウド", + "グーグル", + "コム", + "ストア", + "セール", + "ファッション", + "ポイント", + "三重.jp", + "世界", + "个人.hk", + "中信", + "中国", + "中國", + "中文网", + "亚马逊", + "京都.jp", + "企业", + "佐賀.jp", + "佛山", + "信息", + "個人.hk", + "個人.香港", + "健康", + "八卦", + "公司", + "公司.cn", + "公司.hk", + "公司.香港", + "公益", + "兵庫.jp", + "北海道.jp", + "千葉.jp", + "台湾", + "台灣", + "和歌山.jp", + "商城", + "商店", + "商标", + "商業.tw", + "嘉里", + "嘉里大酒店", + "在线", + "埼玉.jp", + "大分.jp", + "大拿", + "大阪.jp", + "天主教", + "奈良.jp", + "娱乐", + "宮城.jp", + "宮崎.jp", + "家電", + "富山.jp", + "山口.jp", + "山形.jp", + "山梨.jp", + "岐阜.jp", + "岡山.jp", + "岩手.jp", + "島根.jp", + "广东", + "広島.jp", + "微博", + "徳島.jp", + "愛媛.jp", + "愛知.jp", + "慈善", + "我爱你", + "手机", + "招聘", + "政务", + "政府", + "政府.hk", + "政府.香港", + "敎育.hk", + "教育.hk", + "教育.香港", + "新加坡", + "新潟.jp", + "新闻", + "时尚", + "書籍", + "机构", + "東京.jp", + "栃木.jp", + "沖縄.jp", + "淡马锡", + "游戏", + "滋賀.jp", + "澳門", + "澳门", + "点看", + "熊本.jp", + "石川.jp", + "神奈川.jp", + "福井.jp", + "福岡.jp", + "福島.jp", + "秋田.jp", + "移动", + "箇人.hk", + "組織.hk", + "組織.tw", + "組織.香港", + "組织.hk", + "網絡.cn", + "網絡.hk", + "網絡.香港", + "網络.hk", + "網路.tw", + "组織.hk", + "组织.hk", + "组织机构", + "网址", + "网店", + "网站", + "网絡.hk", + "网络", + "网络.cn", + "网络.hk", + "群馬.jp", + "联通", + "臺灣", + "茨城.jp", + "谷歌", + "购物", + "通販", + "長崎.jp", + "長野.jp", + "集团", + "電訊盈科", + "青森.jp", + "静岡.jp", + "飞利浦", + "食品", + "餐厅", + "香川.jp", + "香格里拉", + "香港", + "高知.jp", + "鳥取.jp", + "鹿児島.jp", + "닷넷", + "닷컴", + "삼성", + "한국", +} diff --git a/tlds/tlds_pseudo.go b/tlds/tlds_pseudo.go new file mode 100644 index 0000000..8bb570c --- /dev/null +++ b/tlds/tlds_pseudo.go @@ -0,0 +1,20 @@ +package tlds + +// Pseudo is a sorted list of some widely used unofficial TLDs. +// The list is fetched from: +// - https://en.wikipedia.org/wiki/Pseudo-top-level_domain +// - https://en.wikipedia.org/wiki/Category:Pseudo-top-level_domains +// - https://tools.ietf.org/html/draft-grothoff-iesg-special-use-p2p-names-00 +// - https://www.iana.org/assignments/special-use-domain-names/special-use-domain-names.xhtml +var Pseudo = []string{ + `bit`, // Namecoin + `example`, // Example domain + `exit`, // Tor exit node + `gnu`, // GNS by public key + `i2p`, // I2P network + `invalid`, // Invalid domain + `local`, // Local network + `localhost`, // Local network + `test`, // Test domain + `zkey`, // GNS domain name +} diff --git a/unicodes/unicodes.go b/unicodes/unicodes.go new file mode 100644 index 0000000..f873194 --- /dev/null +++ b/unicodes/unicodes.go @@ -0,0 +1,6 @@ +// This file is autogenerated by the unicodes generator. Please do not edit manually. +package unicodes + +const AllowedUcsChar = "¡-ᙿᚁ-\u1fff\u200b-‧\u202a-\u202e‰-⁞\u2060-\u2fff、-\ud7ff豈-﷏ﷰ-\uffef𐀀-\U0001fffd𠀀-\U0002fffd𰀀-\U0003fffd\U00040000-\U0004fffd\U00050000-\U0005fffd\U00060000-\U0006fffd\U00070000-\U0007fffd\U00080000-\U0008fffd\U00090000-\U0009fffd\U000a0000-\U000afffd\U000b0000-\U000bfffd\U000c0000-\U000cfffd\U000d0000-\U000dfffd\U000e1000-\U000efffd" + +const AllowedUcsCharMinusPunc = "¢-¦¨-µ¸-¾À-ͽͿ-ΆΈ-ՙՠ-ֈ֊-ֿׁ-ׂׄ-ׇׅ-ײ\u05f5-؈؋؎-ؚ\u061cؠ-٩ٮ-ۓە-ۿ\u070e-߶ߺ-\u082f\u083f-\u085d\u085f-ॣ०-९ॱ-ৼ৾-ੵ\u0a77-૯૱-\u0c76౸-ಃಅ-ෳ\u0df5-๎๐-๙\u0e5c-༃༓༕-྄྆-࿏࿕-࿘\u0fdb-၉ၐ-ჺჼ-፟፩-᙭ᙯ-ᙿᚁ-ᛪᛮ-᜴\u1737-៓ៗ៛-\u17ff᠆᠋-\u1943᥆-\u1a1dᨠ-\u1a9fᪧ\u1aae-᭙᭡-᭼\u1b7f-\u1bfbᰀ-\u1c3a᱀-ᱽᲀ-Ჿ\u1cc8-᳔᳒-\u1fff\u200b-―‘-‟\u202a-\u202e‹-›‿-⁀⁄-⁆⁒⁔\u2060-\u2cf8⳽ⴀ-ⵯ\u2d71-ⷿ⸂-⸅⸉-⸊⸌-⸍⸗⸚⸜-⸝⸠-⸩ⸯ⸺-⸻⹀⹂⹐-⹑⹕-\u2fff〄-〼〾-ヺー-ꓽꔀ-ꘌꘐ-꙲ꙴ-꙽ꙿ-꛱\ua6f8-ꡳ\ua878-\ua8cd꣐-ꣷꣻꣽ-꤭ꤰ-\ua95eꥠ-꧀\ua9ce-\ua9ddꧠ-\uaa5bꩠ-ꫝꫠ-ꫯꫲ-ꯪ꯬-\ud7ff豈-﷏ﷰ-️︗-︘\ufe1a-︯︱-﹄﹇-﹈﹍-﹏\ufe53﹘-﹞﹢-\ufe67﹩\ufe6c-\uff00$(-)+-0-9<->A-[]-⦆「-」ヲ-\uffef𐀀-\U000100ff\U00010103-\U0001039e𐎠-𐏏𐏑-\U0001056e𐕰-\U00010856𐡘-\U0001091e𐤠-\U0001093e\U00010940-\U00010a4f\U00010a59-𐩾𐪀-𐫯\U00010af7-\U00010b38𐭀-\U00010b98\U00010b9d-𐽔\U00010f5a-𐾅\U00010f8a-𑁆\U0001104e-𑂺\U000110bd𑃂-𑄿𑅄-𑅳𑅶-𑇄𑇉-𑇌𑇎-𑇚𑇜\U000111e0-𑈷𑈾-𑊨\U000112aa-𑑊𑑐-𑑙\U0001145c𑑞-𑓅𑓇-𑗀𑗘-𑙀𑙄-\U0001165f\U0001166d-𑚸\U000116ba-𑜻𑜿-𑠺\U0001183c-𑥃\U00011947-𑧡𑧣-𑨾𑩇-𑪙𑪝\U00011aa3-\U00011aff\U00011b0a-𑱀\U00011c46-\U00011c6f𑱲-𑻶\U00011ef9-𑽂𑽐-\U00011ffe𒀀-\U0001246f\U00012475-𒿰\U00012ff3-\U00016a6d𖩰-𖫴\U00016af6-𖬶𖬼-𖭃𖭅-𖺖\U00016e9b-𖿡𖿣-𛲞\U0001bca0-𝪆\U0001da8c-\U0001e95d\U0001e960-\U0001fffd𠀀-\U0002fffd𰀀-\U0003fffd\U00040000-\U0004fffd\U00050000-\U0005fffd\U00060000-\U0006fffd\U00070000-\U0007fffd\U00080000-\U0008fffd\U00090000-\U0009fffd\U000a0000-\U000afffd\U000b0000-\U000bfffd\U000c0000-\U000cfffd\U000d0000-\U000dfffd\U000e1000-\U000efffd" diff --git a/url.go b/url.go new file mode 100644 index 0000000..c1d513a --- /dev/null +++ b/url.go @@ -0,0 +1,18 @@ +package url + +import "net/url" + +//go:generate go run gen/schemes/main.go -output ./schemes/schemes_official.go +//go:generate go run gen/TLDs/main.go -output ./tlds/tlds_official.go +//go:generate go run gen/unicodes/main.go -output ./unicodes/unicodes.go + +// URL extends the standard net/url URL struct with additional domain-related fields. +// It includes details like subdomain, root domain, and Top-Level Domain (TLD), along with +// standard URL components. This struct provides a comprehensive representation of a URL. +type URL struct { + *url.URL // Embedding the standard URL struct for base functionalities. + + Domain *Domain + Port int // Port number used in the URL. + Extension string // File extension derived from the URL path. +} diff --git a/url_parser.go b/url_parser.go new file mode 100644 index 0000000..07ec075 --- /dev/null +++ b/url_parser.go @@ -0,0 +1,153 @@ +package url + +import ( + "fmt" + "net/url" + "path" + "regexp" + "strconv" + "strings" +) + +// Parser encapsulates the logic for parsing URLs with additional domain-specific information. +// It enhances the standard URL parsing with the extraction of subdomain, root domain, and TLD. +// It also handles the addition of a default scheme if one is not present in the input URL. +type Parser struct { + scheme string // DefaultScheme is the default URL scheme to use if not specified in the URL. + + dp *DomainParser // DomainParser used for parsing the domain-specific details. +} + +// WithDefaultScheme allows setting a default scheme for the Parser. +// This default scheme is used if the input URL doesn't specify a scheme. +func (up *Parser) WithDefaultScheme(scheme string) { + ParserWithDefaultScheme(scheme)(up) +} + +// DefaultScheme returns the currently set default scheme of the Parser. +func (up *Parser) DefaultScheme() (scheme string) { + return up.scheme +} + +// Parse takes a raw URL string and parses it into a URL struct. +// It adds domain-specific details like subdomain, root domain, and TLD to the parsed URL. +// The method also ensures a default scheme is set if the URL does not specify one. +func (up *Parser) Parse(rawURL string) (parsedURL *URL, err error) { + parsedURL = &URL{} + + // Add default scheme if necessary + if up.scheme != "" { + rawURL = addScheme(rawURL, up.scheme) + } + + // Standard URL parsing + parsedURL.URL, err = url.Parse(rawURL) + if err != nil { + err = fmt.Errorf("error parsing URL: %w", err) + + return + } + + // Split host and port, and handle errors + parsedURL.Host, parsedURL.Port, err = splitHostPort(parsedURL.Host) + if err != nil { + err = fmt.Errorf("error splitting host and port: %w", err) + + return + } + + domainRegex := regexp.MustCompile(`(?i)(?:[a-z0-9](?:[a-z0-9-]{0,61}[a-z0-9])?\.)+[a-z]{2,}`) + + if domainRegex.MatchString(parsedURL.Host) { + parsedURL.Domain = up.dp.Parse(parsedURL.Host) + } + + // Extract file extension from the path + parsedURL.Extension = path.Ext(parsedURL.Path) + + return +} + +// ParserOptionsFunc defines a function type for configuring a Parser. +type ParserOptionsFunc func(*Parser) + +// ParserInterface defines the interface for URL parsing functionality. +type ParserInterface interface { + WithDefaultScheme(scheme string) + + DefaultScheme() (scheme string) + + Parse(rawURL string) (parsedURL *URL, err error) +} + +var _ ParserInterface = &Parser{} + +// NewParser creates a new Parser with the given options. +// It initializes a DomainParser for parsing domain details and applies any additional configuration options. +func NewParser(opts ...ParserOptionsFunc) (up *Parser) { + up = &Parser{} + + // Initialize the DomainParser + dp := NewDomainParser() + up.dp = dp + + // Apply additional options + for _, opt := range opts { + opt(up) + } + + return +} + +// ParserWithDefaultScheme returns a ParserOptionsFunc to set a default scheme. +// This is useful when parsing URLs that may not have a scheme included. +func ParserWithDefaultScheme(scheme string) ParserOptionsFunc { + return func(up *Parser) { + up.scheme = scheme + } +} + +// addScheme is a helper function that adds a scheme to the URL if it's missing. +// This ensures that the URL is parsed correctly as a network address rather than a relative path. +// This makes net/url.Parse() not put both host and path into the (relative) path. +func addScheme(inURL, scheme string) (outURL string) { + switch { + case strings.HasPrefix(inURL, "//"): + outURL = scheme + ":" + inURL + case strings.HasPrefix(inURL, "://"): + outURL = scheme + inURL + case !strings.Contains(inURL, "//"): + outURL = scheme + "://" + inURL + default: + outURL = inURL + } + + return +} + +// splitHostPort separates the host and port in a network address. +// It is designed to handle both IPv4 and IPv6 addresses and gracefully manages URLs without a port. +// Unlike net.SplitHostPort(), it doesn't remove brackets from [IPv6] hosts. +func splitHostPort(address string) (host string, port int, err error) { + host = address + + // Check for the last colon, which should separate host and port + i := strings.LastIndex(address, ":") + if i == -1 { + return + } + + // Handle IPv6 addresses enclosed in brackets + if strings.HasPrefix(address, "[") && strings.Contains(address[i:], "]") { + return + } + + // Split the host and port + host = address[:i] + + if port, err = strconv.Atoi(address[i+1:]); err != nil { + return + } + + return +} diff --git a/url_parser_test.go b/url_parser_test.go new file mode 100644 index 0000000..277767c --- /dev/null +++ b/url_parser_test.go @@ -0,0 +1,118 @@ +package url_test + +import ( + "fmt" + "net/url" + "reflect" + "testing" + + hqgourl "github.com/hueristiq/hq-go-url" +) + +func TestNewParser(t *testing.T) { + t.Parallel() + + up := hqgourl.NewParser() + + if up == nil { + t.Error("NewParser() = nil; want non-nil") + } + + scheme := "https" + + parserWithDefaultScheme := hqgourl.NewParser(hqgourl.ParserWithDefaultScheme(scheme)) + + if up == nil { + t.Errorf("NewParser(ParserWithDefaultScheme(%s)) = nil; want non-nil", scheme) + } + + expectedDefaultScheme := parserWithDefaultScheme.DefaultScheme() + + if expectedDefaultScheme != scheme { + t.Errorf("NewParser(ParserWithDefaultScheme(%s)).DefaultScheme() = '%s', want '%s'", scheme, expectedDefaultScheme, scheme) + } +} + +func TestParser_Parse(t *testing.T) { + t.Parallel() + + cases := []struct { + rawURL string + defaultScheme string + expectedParsedURL *hqgourl.URL + expectParseErr bool + }{ + { + "http://example.com", + "http", + &hqgourl.URL{ + URL: &url.URL{ + Scheme: "http", + Host: "example.com", + }, + Domain: &hqgourl.Domain{ + Sub: "", + Root: "example", + TopLevel: "com", + }, + }, + false, + }, + { + "example.com", + "http", + &hqgourl.URL{ + URL: &url.URL{ + Scheme: "http", + Host: "example.com", + }, + Domain: &hqgourl.Domain{ + Sub: "", + Root: "example", + TopLevel: "com", + }, + }, + false, + }, + { + "http://example.com/path/file.html", + "http", + &hqgourl.URL{ + URL: &url.URL{ + Scheme: "http", + Host: "example.com", + Path: "/path/file.html", + }, + Domain: &hqgourl.Domain{ + Sub: "", + Root: "example", + TopLevel: "com", + }, + Extension: ".html", + }, + false, + }, + } + + for _, c := range cases { + t.Run(fmt.Sprintf("Parse(%q)", c.rawURL), func(t *testing.T) { + t.Parallel() + + up := hqgourl.NewParser( + hqgourl.ParserWithDefaultScheme(c.defaultScheme), + ) + + parsedURL, err := up.Parse(c.rawURL) + + if (err != nil) != c.expectParseErr { + t.Errorf("Parse(%q) error = %v, expectParseErr %v", c.rawURL, err, c.expectParseErr) + + return + } + + if !reflect.DeepEqual(parsedURL, c.expectedParsedURL) { + t.Errorf("Parse(%q) = %+v, want %+v", c.rawURL, parsedURL, c.expectedParsedURL) + } + }) + } +}