package schema import ( "context" "fmt" "net/mail" "net/url" "github.com/answerdev/answer/internal/base/constant" "github.com/answerdev/answer/internal/base/handler" "github.com/answerdev/answer/internal/base/reason" "github.com/answerdev/answer/internal/base/translator" "github.com/answerdev/answer/internal/base/validator" "github.com/segmentfault/pacman/errors" ) // SiteGeneralReq site general request type SiteGeneralReq struct { Name string `validate:"required,sanitizer,gt=1,lte=128" form:"name" json:"name"` ShortDescription string `validate:"omitempty,sanitizer,gt=3,lte=255" form:"short_description" json:"short_description"` Description string `validate:"omitempty,sanitizer,gt=3,lte=2000" form:"description" json:"description"` SiteUrl string `validate:"required,sanitizer,gt=1,lte=512,url" form:"site_url" json:"site_url"` ContactEmail string `validate:"required,sanitizer,gt=1,lte=512,email" form:"contact_email" json:"contact_email"` } func (r *SiteGeneralReq) FormatSiteUrl() { parsedUrl, err := url.Parse(r.SiteUrl) if err != nil { return } r.SiteUrl = fmt.Sprintf("%s://%s", parsedUrl.Scheme, parsedUrl.Host) } // SiteInterfaceReq site interface request type SiteInterfaceReq struct { Language string `validate:"required,gt=1,lte=128" form:"language" json:"language"` TimeZone string `validate:"required,gt=1,lte=128" form:"time_zone" json:"time_zone"` } // SiteBrandingReq site branding request type SiteBrandingReq struct { Logo string `validate:"omitempty,gt=0,lte=512" form:"logo" json:"logo"` MobileLogo string `validate:"omitempty,gt=0,lte=512" form:"mobile_logo" json:"mobile_logo"` SquareIcon string `validate:"omitempty,gt=0,lte=512" form:"square_icon" json:"square_icon"` Favicon string `validate:"omitempty,gt=0,lte=512" form:"favicon" json:"favicon"` } // SiteWriteReq site write request type SiteWriteReq struct { RequiredTag bool `validate:"omitempty" form:"required_tag" json:"required_tag"` RecommendTags []string `validate:"omitempty" form:"recommend_tags" json:"recommend_tags"` ReservedTags []string `validate:"omitempty" form:"reserved_tags" json:"reserved_tags"` UserID string `json:"-"` } // SiteLegalReq site branding request type SiteLegalReq struct { TermsOfServiceOriginalText string `json:"terms_of_service_original_text"` TermsOfServiceParsedText string `json:"terms_of_service_parsed_text"` PrivacyPolicyOriginalText string `json:"privacy_policy_original_text"` PrivacyPolicyParsedText string `json:"privacy_policy_parsed_text"` } // GetSiteLegalInfoReq site site legal request type GetSiteLegalInfoReq struct { InfoType string `validate:"required,oneof=tos privacy" form:"info_type"` } func (r *GetSiteLegalInfoReq) IsTOS() bool { return r.InfoType == "tos" } func (r *GetSiteLegalInfoReq) IsPrivacy() bool { return r.InfoType == "privacy" } // GetSiteLegalInfoResp get site legal info response type GetSiteLegalInfoResp struct { TermsOfServiceOriginalText string `json:"terms_of_service_original_text,omitempty"` TermsOfServiceParsedText string `json:"terms_of_service_parsed_text,omitempty"` PrivacyPolicyOriginalText string `json:"privacy_policy_original_text,omitempty"` PrivacyPolicyParsedText string `json:"privacy_policy_parsed_text,omitempty"` } // SiteUsersReq site users config request type SiteUsersReq struct { DefaultAvatar string `validate:"required,oneof=system gravatar" json:"default_avatar"` GravatarBaseURL string `json:"gravatar_base_url"` AllowUpdateDisplayName bool `json:"allow_update_display_name"` AllowUpdateUsername bool `json:"allow_update_username"` AllowUpdateAvatar bool `json:"allow_update_avatar"` AllowUpdateBio bool `json:"allow_update_bio"` AllowUpdateWebsite bool `json:"allow_update_website"` AllowUpdateLocation bool `json:"allow_update_location"` } // SiteLoginReq site login request type SiteLoginReq struct { AllowNewRegistrations bool `json:"allow_new_registrations"` AllowEmailRegistrations bool `json:"allow_email_registrations"` LoginRequired bool `json:"login_required"` AllowEmailDomains []string `json:"allow_email_domains"` } // SiteCustomCssHTMLReq site custom css html type SiteCustomCssHTMLReq struct { CustomHead string `validate:"omitempty,gt=0,lte=65536" json:"custom_head"` CustomCss string `validate:"omitempty,gt=0,lte=65536" json:"custom_css"` CustomHeader string `validate:"omitempty,gt=0,lte=65536" json:"custom_header"` CustomFooter string `validate:"omitempty,gt=0,lte=65536" json:"custom_footer"` CustomSideBar string `validate:"omitempty,gt=0,lte=65536" json:"custom_sidebar"` } // SiteThemeReq site theme config type SiteThemeReq struct { Theme string `validate:"required,gt=0,lte=255" json:"theme"` ThemeConfig map[string]interface{} `validate:"omitempty" json:"theme_config"` } type SiteSeoReq struct { PermaLink int `validate:"required,lte=4,gte=0" form:"permalink" json:"permalink"` Robots string `validate:"required" form:"robots" json:"robots"` } func (s *SiteSeoResp) IsShortLink() bool { return s.PermaLink == constant.PermaLinkQuestionIDAndTitleByShortID || s.PermaLink == constant.PermaLinkQuestionIDByShortID } // SiteGeneralResp site general response type SiteGeneralResp SiteGeneralReq // SiteInterfaceResp site interface response type SiteInterfaceResp SiteInterfaceReq // SiteBrandingResp site branding response type SiteBrandingResp SiteBrandingReq // SiteLoginResp site login response type SiteLoginResp SiteLoginReq // SiteCustomCssHTMLResp site custom css html response type SiteCustomCssHTMLResp SiteCustomCssHTMLReq // SiteUsersResp site users response type SiteUsersResp SiteUsersReq // SiteThemeResp site theme response type SiteThemeResp struct { ThemeOptions []*ThemeOption `json:"theme_options"` Theme string `json:"theme"` ThemeConfig map[string]interface{} `json:"theme_config"` } func (s *SiteThemeResp) TrTheme(ctx context.Context) { la := handler.GetLangByCtx(ctx) for _, option := range s.ThemeOptions { tr := translator.Tr(la, option.Value) // if tr is equal the option value means not found translation, so use the original label if tr != option.Value { option.Label = tr } } } // ThemeOption get label option type ThemeOption struct { Label string `json:"label"` Value string `json:"value"` } // SiteWriteResp site write response type SiteWriteResp SiteWriteReq // SiteLegalResp site write response type SiteLegalResp SiteLegalReq // SiteSeoResp site write response type SiteSeoResp SiteSeoReq // SiteInfoResp get site info response type SiteInfoResp struct { General *SiteGeneralResp `json:"general"` Interface *SiteInterfaceResp `json:"interface"` Branding *SiteBrandingResp `json:"branding"` Login *SiteLoginResp `json:"login"` Theme *SiteThemeResp `json:"theme"` CustomCssHtml *SiteCustomCssHTMLResp `json:"custom_css_html"` SiteSeo *SiteSeoResp `json:"site_seo"` SiteUsers *SiteUsersResp `json:"site_users"` Version string `json:"version"` Revision string `json:"revision"` } type TemplateSiteInfoResp struct { General *SiteGeneralResp `json:"general"` Interface *SiteInterfaceResp `json:"interface"` Branding *SiteBrandingResp `json:"branding"` SiteSeo *SiteSeoResp `json:"site_seo"` CustomCssHtml *SiteCustomCssHTMLResp `json:"custom_css_html"` Title string Year string Canonical string JsonLD string Keywords string Description string } // UpdateSMTPConfigReq get smtp config request type UpdateSMTPConfigReq struct { FromEmail string `validate:"omitempty,gt=0,lte=256" json:"from_email"` FromName string `validate:"omitempty,gt=0,lte=256" json:"from_name"` SMTPHost string `validate:"omitempty,gt=0,lte=256" json:"smtp_host"` SMTPPort int `validate:"omitempty,min=1,max=65535" json:"smtp_port"` Encryption string `validate:"omitempty,oneof=SSL" json:"encryption"` // "" SSL SMTPUsername string `validate:"omitempty,gt=0,lte=256" json:"smtp_username"` SMTPPassword string `validate:"omitempty,gt=0,lte=256" json:"smtp_password"` SMTPAuthentication bool `validate:"omitempty" json:"smtp_authentication"` TestEmailRecipient string `validate:"omitempty,email" json:"test_email_recipient"` } func (r *UpdateSMTPConfigReq) Check() (errField []*validator.FormErrorField, err error) { _, err = mail.ParseAddress(r.FromName) if err == nil { return append(errField, &validator.FormErrorField{ ErrorField: "from_name", ErrorMsg: reason.SMTPConfigFromNameCannotBeEmail, }), errors.BadRequest(reason.SMTPConfigFromNameCannotBeEmail) } return nil, nil } // GetSMTPConfigResp get smtp config response type GetSMTPConfigResp struct { FromEmail string `json:"from_email"` FromName string `json:"from_name"` SMTPHost string `json:"smtp_host"` SMTPPort int `json:"smtp_port"` Encryption string `json:"encryption"` // "" SSL SMTPUsername string `json:"smtp_username"` SMTPPassword string `json:"smtp_password"` SMTPAuthentication bool `json:"smtp_authentication"` } // GetManifestJsonResp get manifest json response type GetManifestJsonResp struct { ManifestVersion int `json:"manifest_version"` Version string `json:"version"` Revision string `json:"revision"` ShortName string `json:"short_name"` Name string `json:"name"` Icons map[string]string `json:"icons"` StartUrl string `json:"start_url"` Display string `json:"display"` ThemeColor string `json:"theme_color"` BackgroundColor string `json:"background_color"` } const ( // PrivilegeLevel1 low PrivilegeLevel1 PrivilegeLevel = 1 // PrivilegeLevel2 medium PrivilegeLevel2 PrivilegeLevel = 2 // PrivilegeLevel3 high PrivilegeLevel3 PrivilegeLevel = 3 ) type PrivilegeLevel int type PrivilegeOptions []*PrivilegeOption func (p PrivilegeOptions) Choose(level PrivilegeLevel) (option *PrivilegeOption) { for _, op := range p { if op.Level == level { return op } } return nil } // GetPrivilegesConfigResp get privileges config response type GetPrivilegesConfigResp struct { Options []*PrivilegeOption `json:"options"` SelectedLevel PrivilegeLevel `json:"selected_level"` } // PrivilegeOption privilege option type PrivilegeOption struct { Level PrivilegeLevel `json:"level"` LevelDesc string `json:"level_desc"` Privileges []*constant.Privilege `json:"privileges"` } // UpdatePrivilegesConfigReq update privileges config request type UpdatePrivilegesConfigReq struct { Level PrivilegeLevel `validate:"required,min=1,max=3" json:"level"` } var ( DefaultPrivilegeOptions PrivilegeOptions privilegeOptionsLevelMapping = map[string][]int{ constant.RankQuestionAddKey: {1, 1, 1}, constant.RankAnswerAddKey: {1, 1, 1}, constant.RankCommentAddKey: {1, 1, 1}, constant.RankReportAddKey: {1, 1, 1}, constant.RankCommentVoteUpKey: {1, 1, 1}, constant.RankLinkUrlLimitKey: {1, 10, 10}, constant.RankQuestionVoteUpKey: {1, 8, 15}, constant.RankAnswerVoteUpKey: {1, 8, 15}, constant.RankQuestionVoteDownKey: {125, 125, 125}, constant.RankAnswerVoteDownKey: {125, 125, 125}, constant.RankInviteSomeoneToAnswerKey: {1, 500, 1000}, constant.RankTagAddKey: {1, 750, 1500}, constant.RankTagEditKey: {1, 50, 100}, constant.RankQuestionEditKey: {1, 100, 200}, constant.RankAnswerEditKey: {1, 100, 200}, constant.RankQuestionEditWithoutReviewKey: {1, 1000, 2000}, constant.RankAnswerEditWithoutReviewKey: {1, 1000, 2000}, constant.RankQuestionAuditKey: {1, 1000, 2000}, constant.RankAnswerAuditKey: {1, 1000, 2000}, constant.RankTagAuditKey: {1, 2500, 5000}, constant.RankTagEditWithoutReviewKey: {1, 10000, 20000}, constant.RankTagSynonymKey: {1, 10000, 20000}, } ) func init() { DefaultPrivilegeOptions = append(DefaultPrivilegeOptions, &PrivilegeOption{ Level: PrivilegeLevel1, LevelDesc: reason.PrivilegeLevel1Desc, }, &PrivilegeOption{ Level: PrivilegeLevel2, LevelDesc: reason.PrivilegeLevel2Desc, }, &PrivilegeOption{ Level: PrivilegeLevel3, LevelDesc: reason.PrivilegeLevel3Desc, }) for _, option := range DefaultPrivilegeOptions { for _, privilege := range constant.RankAllPrivileges { if len(privilegeOptionsLevelMapping[privilege.Key]) == 0 { continue } option.Privileges = append(option.Privileges, &constant.Privilege{ Label: privilege.Label, Value: privilegeOptionsLevelMapping[privilege.Key][option.Level-1], Key: privilege.Key, }) } } }