// Code generated by go-swagger; DO NOT EDIT.

// Copyright 2018 The go-netbox Authors.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
//   http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.

package secrets

// This file was generated by the swagger tool.
// Editing this file might prove futile when you re-run the swagger generate command

import (
	"net/http"
	"time"

	"golang.org/x/net/context"

	"github.com/go-openapi/errors"
	"github.com/go-openapi/runtime"
	cr "github.com/go-openapi/runtime/client"
	"github.com/go-openapi/swag"

	strfmt "github.com/go-openapi/strfmt"
)

// NewSecretsSecretRolesListParams creates a new SecretsSecretRolesListParams object
// with the default values initialized.
func NewSecretsSecretRolesListParams() *SecretsSecretRolesListParams {
	var ()
	return &SecretsSecretRolesListParams{

		timeout: cr.DefaultTimeout,
	}
}

// NewSecretsSecretRolesListParamsWithTimeout creates a new SecretsSecretRolesListParams object
// with the default values initialized, and the ability to set a timeout on a request
func NewSecretsSecretRolesListParamsWithTimeout(timeout time.Duration) *SecretsSecretRolesListParams {
	var ()
	return &SecretsSecretRolesListParams{

		timeout: timeout,
	}
}

// NewSecretsSecretRolesListParamsWithContext creates a new SecretsSecretRolesListParams object
// with the default values initialized, and the ability to set a context for a request
func NewSecretsSecretRolesListParamsWithContext(ctx context.Context) *SecretsSecretRolesListParams {
	var ()
	return &SecretsSecretRolesListParams{

		Context: ctx,
	}
}

// NewSecretsSecretRolesListParamsWithHTTPClient creates a new SecretsSecretRolesListParams object
// with the default values initialized, and the ability to set a custom HTTPClient for a request
func NewSecretsSecretRolesListParamsWithHTTPClient(client *http.Client) *SecretsSecretRolesListParams {
	var ()
	return &SecretsSecretRolesListParams{
		HTTPClient: client,
	}
}

/*SecretsSecretRolesListParams contains all the parameters to send to the API endpoint
for the secrets secret roles list operation typically these are written to a http.Request
*/
type SecretsSecretRolesListParams struct {

	/*Limit
	  Number of results to return per page.

	*/
	Limit *int64
	/*Name*/
	Name *string
	/*Offset
	  The initial index from which to return the results.

	*/
	Offset *int64
	/*Slug*/
	Slug *string

	timeout    time.Duration
	Context    context.Context
	HTTPClient *http.Client
}

// WithTimeout adds the timeout to the secrets secret roles list params
func (o *SecretsSecretRolesListParams) WithTimeout(timeout time.Duration) *SecretsSecretRolesListParams {
	o.SetTimeout(timeout)
	return o
}

// SetTimeout adds the timeout to the secrets secret roles list params
func (o *SecretsSecretRolesListParams) SetTimeout(timeout time.Duration) {
	o.timeout = timeout
}

// WithContext adds the context to the secrets secret roles list params
func (o *SecretsSecretRolesListParams) WithContext(ctx context.Context) *SecretsSecretRolesListParams {
	o.SetContext(ctx)
	return o
}

// SetContext adds the context to the secrets secret roles list params
func (o *SecretsSecretRolesListParams) SetContext(ctx context.Context) {
	o.Context = ctx
}

// WithHTTPClient adds the HTTPClient to the secrets secret roles list params
func (o *SecretsSecretRolesListParams) WithHTTPClient(client *http.Client) *SecretsSecretRolesListParams {
	o.SetHTTPClient(client)
	return o
}

// SetHTTPClient adds the HTTPClient to the secrets secret roles list params
func (o *SecretsSecretRolesListParams) SetHTTPClient(client *http.Client) {
	o.HTTPClient = client
}

// WithLimit adds the limit to the secrets secret roles list params
func (o *SecretsSecretRolesListParams) WithLimit(limit *int64) *SecretsSecretRolesListParams {
	o.SetLimit(limit)
	return o
}

// SetLimit adds the limit to the secrets secret roles list params
func (o *SecretsSecretRolesListParams) SetLimit(limit *int64) {
	o.Limit = limit
}

// WithName adds the name to the secrets secret roles list params
func (o *SecretsSecretRolesListParams) WithName(name *string) *SecretsSecretRolesListParams {
	o.SetName(name)
	return o
}

// SetName adds the name to the secrets secret roles list params
func (o *SecretsSecretRolesListParams) SetName(name *string) {
	o.Name = name
}

// WithOffset adds the offset to the secrets secret roles list params
func (o *SecretsSecretRolesListParams) WithOffset(offset *int64) *SecretsSecretRolesListParams {
	o.SetOffset(offset)
	return o
}

// SetOffset adds the offset to the secrets secret roles list params
func (o *SecretsSecretRolesListParams) SetOffset(offset *int64) {
	o.Offset = offset
}

// WithSlug adds the slug to the secrets secret roles list params
func (o *SecretsSecretRolesListParams) WithSlug(slug *string) *SecretsSecretRolesListParams {
	o.SetSlug(slug)
	return o
}

// SetSlug adds the slug to the secrets secret roles list params
func (o *SecretsSecretRolesListParams) SetSlug(slug *string) {
	o.Slug = slug
}

// WriteToRequest writes these params to a swagger request
func (o *SecretsSecretRolesListParams) WriteToRequest(r runtime.ClientRequest, reg strfmt.Registry) error {

	if err := r.SetTimeout(o.timeout); err != nil {
		return err
	}
	var res []error

	if o.Limit != nil {

		// query param limit
		var qrLimit int64
		if o.Limit != nil {
			qrLimit = *o.Limit
		}
		qLimit := swag.FormatInt64(qrLimit)
		if qLimit != "" {
			if err := r.SetQueryParam("limit", qLimit); err != nil {
				return err
			}
		}

	}

	if o.Name != nil {

		// query param name
		var qrName string
		if o.Name != nil {
			qrName = *o.Name
		}
		qName := qrName
		if qName != "" {
			if err := r.SetQueryParam("name", qName); err != nil {
				return err
			}
		}

	}

	if o.Offset != nil {

		// query param offset
		var qrOffset int64
		if o.Offset != nil {
			qrOffset = *o.Offset
		}
		qOffset := swag.FormatInt64(qrOffset)
		if qOffset != "" {
			if err := r.SetQueryParam("offset", qOffset); err != nil {
				return err
			}
		}

	}

	if o.Slug != nil {

		// query param slug
		var qrSlug string
		if o.Slug != nil {
			qrSlug = *o.Slug
		}
		qSlug := qrSlug
		if qSlug != "" {
			if err := r.SetQueryParam("slug", qSlug); err != nil {
				return err
			}
		}

	}

	if len(res) > 0 {
		return errors.CompositeValidationError(res...)
	}
	return nil
}
