Create a usecase from API to extract domain creation on provider

Fixes a security issue where a user can create a domain on a provider
while DisableProviders option was set
This commit is contained in:
nemunaire 2025-06-09 10:45:46 +02:00
parent b0e0ab6d9e
commit ce91b49a73
5 changed files with 84 additions and 33 deletions

View file

@ -246,23 +246,12 @@ func (pc *ProviderController) GetDomainsHostedByProvider(c *gin.Context) {
// @Failure 401 {object} happydns.ErrorResponse "Authentication failure"
// @Failure 404 {object} happydns.ErrorResponse "Provider not found"
// @Router /providers/{providerId}/domains/{fqdn} [get]
func (pc *ProviderController) CreateDomainsOnProvider(c *gin.Context) {
func (pc *ProviderController) CreateDomainOnProvider(c *gin.Context) {
provider := c.MustGet("provider").(*happydns.Provider)
p, err := provider.InstantiateProvider()
err := pc.providerService.CreateDomainOnProvider(provider, c.Param("fqdn"))
if err != nil {
c.AbortWithStatusJSON(http.StatusBadRequest, gin.H{"errmsg": fmt.Sprintf("Unable to instantiate the provider: %s", err.Error())})
return
}
if !p.CanCreateDomain() {
c.AbortWithStatusJSON(http.StatusBadRequest, gin.H{"errmsg": "Provider doesn't support domain creation."})
return
}
err = p.CreateDomain(c.Param("fqdn"))
if err != nil {
c.AbortWithStatusJSON(http.StatusBadRequest, gin.H{"errmsg": err.Error()})
middleware.ErrorResponse(c, http.StatusBadRequest, err)
return
}

View file

@ -47,5 +47,5 @@ func DeclareProviderRoutes(router *gin.RouterGroup, dependancies happydns.Usecas
apiProviderRoutes.PUT("", pc.UpdateProvider)
apiProviderRoutes.GET("/domains", pc.GetDomainsHostedByProvider)
apiProviderRoutes.POST("/domains/:fqdn", pc.CreateDomainsOnProvider)
apiProviderRoutes.POST("/domains/:fqdn", pc.CreateDomainOnProvider)
}

View file

@ -0,0 +1,47 @@
// This file is part of the happyDomain (R) project.
// Copyright (c) 2020-2025 happyDomain
// Authors: Pierre-Olivier Mercier, et al.
//
// This program is offered under a commercial and under the AGPL license.
// For commercial licensing, contact us at <contact@happydomain.org>.
//
// For AGPL licensing:
// This program is free software: you can redistribute it and/or modify
// it under the terms of the GNU Affero General Public License as published by
// the Free Software Foundation, either version 3 of the License, or
// (at your option) any later version.
//
// This program is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU Affero General Public License for more details.
//
// You should have received a copy of the GNU Affero General Public License
// along with this program. If not, see <https://www.gnu.org/licenses/>.
package provider
import (
"fmt"
"git.happydns.org/happyDomain/model"
)
type CreateDomainOnProviderUsecase struct{}
func NewCreateDomainOnProviderUsecase() *CreateDomainOnProviderUsecase {
return &CreateDomainOnProviderUsecase{}
}
func (uc *CreateDomainOnProviderUsecase) Create(provider *happydns.Provider, fqdn string) error {
p, err := provider.InstantiateProvider()
if err != nil {
return fmt.Errorf("unable to instantiate the provider: %w", err)
}
if !p.CanCreateDomain() {
return fmt.Errorf("the provider doesn't support domain creation")
}
return p.CreateDomain(fqdn)
}

View file

@ -26,15 +26,16 @@ import (
)
type Service struct {
CreateProviderUC *CreateProviderUsecase
DeleteProviderUC *DeleteProviderUsecase
UpdateProviderUC *UpdateProviderUsecase
ListHostedDomainsUC *ListHostedDomainsUsecase
GetProviderUC *GetProviderUsecase
ListProvidersUC *ListProvidersUsecase
RetrieveZoneUC *ZoneRetrieverUsecase
ZoneCorrectionsUC *ZoneCorrectorUsecase
DomainExistenceUC *DomainExistenceUsecase
CreateProviderUC *CreateProviderUsecase
CreateDomainOnProviderUC *CreateDomainOnProviderUsecase
DeleteProviderUC *DeleteProviderUsecase
UpdateProviderUC *UpdateProviderUsecase
ListHostedDomainsUC *ListHostedDomainsUsecase
GetProviderUC *GetProviderUsecase
ListProvidersUC *ListProvidersUsecase
RetrieveZoneUC *ZoneRetrieverUsecase
ZoneCorrectionsUC *ZoneCorrectorUsecase
DomainExistenceUC *DomainExistenceUsecase
}
func NewProviderUsecases(store ProviderStorage) *Service {
@ -42,15 +43,16 @@ func NewProviderUsecases(store ProviderStorage) *Service {
validator := &DefaultProviderValidator{}
return &Service{
CreateProviderUC: NewCreateProviderUsecase(store, validator),
DeleteProviderUC: NewDeleteProviderUsecase(store),
UpdateProviderUC: NewUpdateProviderUsecase(store, getProvider, validator),
ListHostedDomainsUC: NewListHostedDomainsUsecase(),
GetProviderUC: getProvider,
ListProvidersUC: NewListProvidersUsecase(store),
RetrieveZoneUC: NewZoneRetrieverUsecase(),
ZoneCorrectionsUC: NewZoneCorrectorUsecase(),
DomainExistenceUC: NewDomainExistenceUsecase(),
CreateProviderUC: NewCreateProviderUsecase(store, validator),
CreateDomainOnProviderUC: NewCreateDomainOnProviderUsecase(),
DeleteProviderUC: NewDeleteProviderUsecase(store),
UpdateProviderUC: NewUpdateProviderUsecase(store, getProvider, validator),
ListHostedDomainsUC: NewListHostedDomainsUsecase(),
GetProviderUC: getProvider,
ListProvidersUC: NewListProvidersUsecase(store),
RetrieveZoneUC: NewZoneRetrieverUsecase(),
ZoneCorrectionsUC: NewZoneCorrectorUsecase(),
DomainExistenceUC: NewDomainExistenceUsecase(),
}
}
@ -58,6 +60,10 @@ func (s *Service) CreateProvider(user *happydns.User, msg *happydns.ProviderMess
return s.CreateProviderUC.Create(user, msg)
}
func (s *Service) CreateDomainOnProvider(provider *happydns.Provider, fqdn string) error {
return s.CreateDomainOnProviderUC.Create(provider, fqdn)
}
func (s *Service) DeleteProvider(user *happydns.User, providerID happydns.Identifier) error {
return s.DeleteProviderUC.Delete(user, providerID)
}
@ -119,6 +125,14 @@ func (s *RestrictedService) CreateProvider(user *happydns.User, msg *happydns.Pr
return s.Service.CreateProvider(user, msg)
}
func (s *RestrictedService) CreateDomainOnProvider(provider *happydns.Provider, fqdn string) error {
if s.config.DisableProviders {
return happydns.ForbiddenError{Msg: "cannot create domain on provider as DisableProviders parameter is set."}
}
return s.Service.CreateDomainOnProvider(provider, fqdn)
}
func (s *RestrictedService) DeleteProvider(user *happydns.User, providerID happydns.Identifier) error {
if s.config.DisableProviders {
return happydns.ForbiddenError{Msg: "cannot delete provider as DisableProviders parameter is set."}

View file

@ -124,6 +124,7 @@ func (p *Provider) Meta() *ProviderMeta {
type ProviderUsecase interface {
CreateProvider(*User, *ProviderMessage) (*Provider, error)
CreateDomainOnProvider(*Provider, string) error
DeleteProvider(*User, Identifier) error
GetUserProvider(*User, Identifier) (*Provider, error)
GetUserProviderMeta(*User, Identifier) (*ProviderMeta, error)