From 34d829338d6eff98684949f3a49edfe5cb36e0ef Mon Sep 17 00:00:00 2001 From: WithoutPants <53250216+WithoutPants@users.noreply.github.com> Date: Wed, 11 Mar 2020 11:41:55 +1100 Subject: [PATCH] Add image scraping support (#370) * Add sub-scraper functionality * Add scraping of performer image * Add scene cover image scraping * Port UI changes to v2.5 * Fix v2.5 dialog suggest color * Don't convert eol of UI to support pretty --- .gitattributes | 3 +- go.mod | 1 + go.sum | 2 + graphql/documents/data/scrapers.graphql | 2 + .../schema/types/scraped-performer.graphql | 5 + graphql/schema/types/scraper.graphql | 3 + pkg/models/model_scraped_item.go | 34 ++++ pkg/scraper/image.go | 84 ++++++++ pkg/scraper/scrapers.go | 29 ++- pkg/scraper/stash.go | 39 +++- pkg/scraper/xpath.go | 50 +++++ .../PerformerDetailsPanel.tsx | 24 ++- .../Scenes/SceneDetails/SceneEditPanel.tsx | 6 + ui/v2.5/src/index.scss | 34 ++++ .../PerformerDetailsPanel.tsx | 23 ++- .../scenes/SceneDetails/SceneEditPanel.tsx | 6 + vendor/github.com/jinzhu/copier/Guardfile | 3 + vendor/github.com/jinzhu/copier/License | 20 ++ vendor/github.com/jinzhu/copier/README.md | 100 +++++++++ vendor/github.com/jinzhu/copier/copier.go | 189 ++++++++++++++++++ vendor/github.com/jinzhu/copier/wercker.yml | 23 +++ 21 files changed, 665 insertions(+), 15 deletions(-) create mode 100644 pkg/scraper/image.go create mode 100644 vendor/github.com/jinzhu/copier/Guardfile create mode 100644 vendor/github.com/jinzhu/copier/License create mode 100644 vendor/github.com/jinzhu/copier/README.md create mode 100644 vendor/github.com/jinzhu/copier/copier.go create mode 100644 vendor/github.com/jinzhu/copier/wercker.yml diff --git a/.gitattributes b/.gitattributes index 4fea60b95..201b307f0 100644 --- a/.gitattributes +++ b/.gitattributes @@ -1,2 +1,3 @@ go.mod text eol=lf -go.sum text eol=lf \ No newline at end of file +go.sum text eol=lf +ui/v2.5/** -text diff --git a/go.mod b/go.mod index ae4aff41b..9f6ad52d2 100644 --- a/go.mod +++ b/go.mod @@ -14,6 +14,7 @@ require ( github.com/h2non/filetype v1.0.8 // this is required for generate github.com/inconshreveable/mousetrap v1.0.0 // indirect + github.com/jinzhu/copier v0.0.0-20190924061706-b57f9002281a github.com/jmoiron/sqlx v1.2.0 github.com/mattn/go-sqlite3 v1.10.0 github.com/rs/cors v1.6.0 diff --git a/go.sum b/go.sum index bf4e10dbe..998a0b3c5 100644 --- a/go.sum +++ b/go.sum @@ -383,6 +383,8 @@ github.com/inconshreveable/mousetrap v1.0.0/go.mod h1:PxqpIevigyE2G7u3NXJIT2ANyt github.com/jackc/fake v0.0.0-20150926172116-812a484cc733/go.mod h1:WrMFNQdiFJ80sQsxDoMokWK1W5TQtxBFNpzWTD84ibQ= github.com/jackc/pgx v3.2.0+incompatible/go.mod h1:0ZGrqGqkRlliWnWB4zKnWtjbSWbGkVEFm4TeybAXq+I= github.com/jellevandenhooff/dkim v0.0.0-20150330215556-f50fe3d243e1/go.mod h1:E0B/fFc00Y+Rasa88328GlI/XbtyysCtTHZS8h7IrBU= +github.com/jinzhu/copier v0.0.0-20190924061706-b57f9002281a h1:zPPuIq2jAWWPTrGt70eK/BSch+gFAGrNzecsoENgu2o= +github.com/jinzhu/copier v0.0.0-20190924061706-b57f9002281a/go.mod h1:yL958EeXv8Ylng6IfnvG4oflryUi3vgA3xPs9hmII1s= github.com/jmespath/go-jmespath v0.0.0-20180206201540-c2b33e8439af/go.mod h1:Nht3zPeWKUH0NzdCt2Blrr5ys8VGpn0CEB0cQHVjt7k= github.com/jmoiron/sqlx v0.0.0-20180614180643-0dae4fefe7c0/go.mod h1:IiEW3SEiiErVyFdH8NTuWjSifiEQKUoyK3LNqr2kCHU= github.com/jmoiron/sqlx v1.2.0 h1:41Ip0zITnmWNR/vHV+S4m+VoUivnWY5E4OJfLZjCJMA= diff --git a/graphql/documents/data/scrapers.graphql b/graphql/documents/data/scrapers.graphql index b12ccbe24..2a46cc155 100644 --- a/graphql/documents/data/scrapers.graphql +++ b/graphql/documents/data/scrapers.graphql @@ -14,6 +14,7 @@ fragment ScrapedPerformerData on ScrapedPerformer { tattoos piercings aliases + image } fragment ScrapedScenePerformerData on ScrapedScenePerformer { @@ -75,6 +76,7 @@ fragment ScrapedSceneData on ScrapedScene { details url date + image file { size diff --git a/graphql/schema/types/scraped-performer.graphql b/graphql/schema/types/scraped-performer.graphql index a16f3df23..e57c7a2e3 100644 --- a/graphql/schema/types/scraped-performer.graphql +++ b/graphql/schema/types/scraped-performer.graphql @@ -15,6 +15,9 @@ type ScrapedPerformer { tattoos: String piercings: String aliases: String + + """This should be base64 encoded""" + image: String } input ScrapedPerformerInput { @@ -33,4 +36,6 @@ input ScrapedPerformerInput { tattoos: String piercings: String aliases: String + + # not including image for the input } \ No newline at end of file diff --git a/graphql/schema/types/scraper.graphql b/graphql/schema/types/scraper.graphql index 0b189e0eb..601621c61 100644 --- a/graphql/schema/types/scraper.graphql +++ b/graphql/schema/types/scraper.graphql @@ -76,6 +76,9 @@ type ScrapedScene { url: String date: String + """This should be base64 encoded""" + image: String + file: SceneFileType # Resolver studio: ScrapedSceneStudio diff --git a/pkg/models/model_scraped_item.go b/pkg/models/model_scraped_item.go index cf2585cd7..e09418e61 100644 --- a/pkg/models/model_scraped_item.go +++ b/pkg/models/model_scraped_item.go @@ -39,6 +39,26 @@ type ScrapedPerformer struct { Tattoos *string `graphql:"tattoos" json:"tattoos"` Piercings *string `graphql:"piercings" json:"piercings"` Aliases *string `graphql:"aliases" json:"aliases"` + Image *string `graphql:"image" json:"image"` +} + +// this type has no Image field +type ScrapedPerformerStash struct { + Name *string `graphql:"name" json:"name"` + URL *string `graphql:"url" json:"url"` + Twitter *string `graphql:"twitter" json:"twitter"` + Instagram *string `graphql:"instagram" json:"instagram"` + Birthdate *string `graphql:"birthdate" json:"birthdate"` + Ethnicity *string `graphql:"ethnicity" json:"ethnicity"` + Country *string `graphql:"country" json:"country"` + EyeColor *string `graphql:"eye_color" json:"eye_color"` + Height *string `graphql:"height" json:"height"` + Measurements *string `graphql:"measurements" json:"measurements"` + FakeTits *string `graphql:"fake_tits" json:"fake_tits"` + CareerLength *string `graphql:"career_length" json:"career_length"` + Tattoos *string `graphql:"tattoos" json:"tattoos"` + Piercings *string `graphql:"piercings" json:"piercings"` + Aliases *string `graphql:"aliases" json:"aliases"` } type ScrapedScene struct { @@ -46,6 +66,7 @@ type ScrapedScene struct { Details *string `graphql:"details" json:"details"` URL *string `graphql:"url" json:"url"` Date *string `graphql:"date" json:"date"` + Image *string `graphql:"image" json:"image"` File *SceneFileType `graphql:"file" json:"file"` Studio *ScrapedSceneStudio `graphql:"studio" json:"studio"` Movies []*ScrapedSceneMovie `graphql:"movies" json:"movies"` @@ -53,6 +74,19 @@ type ScrapedScene struct { Performers []*ScrapedScenePerformer `graphql:"performers" json:"performers"` } +// stash doesn't return image, and we need id +type ScrapedSceneStash struct { + ID string `graphql:"id" json:"id"` + Title *string `graphql:"title" json:"title"` + Details *string `graphql:"details" json:"details"` + URL *string `graphql:"url" json:"url"` + Date *string `graphql:"date" json:"date"` + File *SceneFileType `graphql:"file" json:"file"` + Studio *ScrapedSceneStudio `graphql:"studio" json:"studio"` + Tags []*ScrapedSceneTag `graphql:"tags" json:"tags"` + Performers []*ScrapedScenePerformer `graphql:"performers" json:"performers"` +} + type ScrapedScenePerformer struct { // Set if performer matched ID *string `graphql:"id" json:"id"` diff --git a/pkg/scraper/image.go b/pkg/scraper/image.go new file mode 100644 index 000000000..c44295100 --- /dev/null +++ b/pkg/scraper/image.go @@ -0,0 +1,84 @@ +package scraper + +import ( + "io/ioutil" + "net/http" + "strings" + "time" + + "github.com/stashapp/stash/pkg/models" + "github.com/stashapp/stash/pkg/utils" +) + +// Timeout to get the image. Includes transfer time. May want to make this +// configurable at some point. +const imageGetTimeout = time.Second * 30 + +func setPerformerImage(p *models.ScrapedPerformer) error { + if p == nil || p.Image == nil || !strings.HasPrefix(*p.Image, "http") { + // nothing to do + return nil + } + + img, err := getImage(*p.Image) + if err != nil { + return err + } + + p.Image = img + + return nil +} + +func setSceneImage(s *models.ScrapedScene) error { + // don't try to get the image if it doesn't appear to be a URL + if s == nil || s.Image == nil || !strings.HasPrefix(*s.Image, "http") { + // nothing to do + return nil + } + + img, err := getImage(*s.Image) + if err != nil { + return err + } + + s.Image = img + + return nil +} + +func getImage(url string) (*string, error) { + client := &http.Client{ + Timeout: imageGetTimeout, + } + + // assume is a URL for now + resp, err := client.Get(url) + if err != nil { + return nil, err + } + + defer resp.Body.Close() + + body, err := ioutil.ReadAll(resp.Body) + if err != nil { + return nil, err + } + + // determine the image type and set the base64 type + contentType := resp.Header.Get("Content-Type") + if contentType == "" { + contentType = http.DetectContentType(body) + } + + img := "data:" + contentType + ";base64," + utils.GetBase64StringFromData(body) + return &img, nil +} + +func getStashPerformerImage(stashURL string, performerID string) (*string, error) { + return getImage(stashURL + "/performer/" + performerID + "/image") +} + +func getStashSceneImage(stashURL string, sceneID string) (*string, error) { + return getImage(stashURL + "/scene/" + sceneID + "/screenshot") +} diff --git a/pkg/scraper/scrapers.go b/pkg/scraper/scrapers.go index f17e704f7..1eff1a5a8 100644 --- a/pkg/scraper/scrapers.go +++ b/pkg/scraper/scrapers.go @@ -108,7 +108,17 @@ func ScrapePerformer(scraperID string, scrapedPerformer models.ScrapedPerformerI // find scraper with the provided id s := findScraper(scraperID) if s != nil { - return s.ScrapePerformer(scrapedPerformer) + ret, err := s.ScrapePerformer(scrapedPerformer) + if err != nil { + return nil, err + } + + // post-process - set the image if applicable + if err := setPerformerImage(ret); err != nil { + logger.Warnf("Could not set image using URL %s: %s", *ret.Image, err.Error()) + } + + return ret, nil } return nil, errors.New("Scraper with ID " + scraperID + " not found") @@ -117,7 +127,17 @@ func ScrapePerformer(scraperID string, scrapedPerformer models.ScrapedPerformerI func ScrapePerformerURL(url string) (*models.ScrapedPerformer, error) { for _, s := range scrapers { if s.matchesPerformerURL(url) { - return s.ScrapePerformerURL(url) + ret, err := s.ScrapePerformerURL(url) + if err != nil { + return nil, err + } + + // post-process - set the image if applicable + if err := setPerformerImage(ret); err != nil { + logger.Warnf("Could not set image using URL %s: %s", *ret.Image, err.Error()) + } + + return ret, nil } } @@ -228,6 +248,11 @@ func postScrapeScene(ret *models.ScrapedScene) error { } } + // post-process - set the image if applicable + if err := setSceneImage(ret); err != nil { + logger.Warnf("Could not set image using URL %s: %s", *ret.Image, err.Error()) + } + return nil } diff --git a/pkg/scraper/stash.go b/pkg/scraper/stash.go index 3de82efae..69881ffd9 100644 --- a/pkg/scraper/stash.go +++ b/pkg/scraper/stash.go @@ -4,6 +4,7 @@ import ( "context" "strconv" + "github.com/jinzhu/copier" "github.com/shurcooL/graphql" "github.com/stashapp/stash/pkg/models" @@ -67,12 +68,14 @@ func scrapePerformerFragmentStash(c scraperTypeConfig, scrapedPerformer models.S client := getStashClient(c) var q struct { - FindPerformer *models.ScrapedPerformer `graphql:"findPerformer(id: $f)"` + FindPerformer *models.ScrapedPerformerStash `graphql:"findPerformer(id: $f)"` } + performerID := *scrapedPerformer.URL + // get the id from the URL field vars := map[string]interface{}{ - "f": *scrapedPerformer.URL, + "f": performerID, } err := client.Query(context.Background(), &q, vars) @@ -80,7 +83,20 @@ func scrapePerformerFragmentStash(c scraperTypeConfig, scrapedPerformer models.S return nil, err } - return q.FindPerformer, nil + // need to copy back to a scraped performer + ret := models.ScrapedPerformer{} + err = copier.Copy(&ret, q.FindPerformer) + if err != nil { + return nil, err + } + + // get the performer image directly + ret.Image, err = getStashPerformerImage(c.scraperConfig.StashServer.URL, performerID) + if err != nil { + return nil, err + } + + return &ret, nil } func scrapeSceneFragmentStash(c scraperTypeConfig, scene models.SceneUpdateInput) (*models.ScrapedScene, error) { @@ -99,7 +115,7 @@ func scrapeSceneFragmentStash(c scraperTypeConfig, scene models.SceneUpdateInput } var q struct { - FindScene *models.ScrapedScene `graphql:"findScene(checksum: $c)"` + FindScene *models.ScrapedSceneStash `graphql:"findScene(checksum: $c)"` } checksum := graphql.String(storedScene.Checksum) @@ -128,5 +144,18 @@ func scrapeSceneFragmentStash(c scraperTypeConfig, scene models.SceneUpdateInput } } - return q.FindScene, nil + // need to copy back to a scraped scene + ret := models.ScrapedScene{} + err = copier.Copy(&ret, q.FindScene) + if err != nil { + return nil, err + } + + // get the performer image directly + ret.Image, err = getStashSceneImage(c.scraperConfig.StashServer.URL, q.FindScene.ID) + if err != nil { + return nil, err + } + + return &ret, nil } diff --git a/pkg/scraper/xpath.go b/pkg/scraper/xpath.go index fad6755f7..9f063454c 100644 --- a/pkg/scraper/xpath.go +++ b/pkg/scraper/xpath.go @@ -135,6 +135,22 @@ func (c xpathScraperAttrConfig) getReplace() xpathRegexConfigs { return ret } +func (c xpathScraperAttrConfig) getSubScraper() xpathScraperAttrConfig { + const subScraperKey = "subScraper" + val, _ := c[subScraperKey] + + if val == nil { + return nil + } + + asMap, _ := val.(map[interface{}]interface{}) + if asMap != nil { + return xpathScraperAttrConfig(asMap) + } + + return nil +} + func (c xpathScraperAttrConfig) concatenateResults(nodes []*html.Node) string { separator := c.getConcat() result := []string{} @@ -174,10 +190,44 @@ func (c xpathScraperAttrConfig) replaceRegex(value string) string { return replace.apply(value) } +func (c xpathScraperAttrConfig) applySubScraper(value string) string { + subScraper := c.getSubScraper() + + if subScraper == nil { + return value + } + + doc, err := htmlquery.LoadURL(value) + + if err != nil { + logger.Warnf("Error getting URL '%s' for sub-scraper: %s", value, err.Error()) + return "" + } + + found := runXPathQuery(doc, subScraper.getSelector(), nil) + + if len(found) > 0 { + // check if we're concatenating the results into a single result + var result string + if subScraper.hasConcat() { + result = subScraper.concatenateResults(found) + } else { + result = htmlquery.InnerText(found[0]) + result = commonPostProcess(result) + } + + result = subScraper.postProcess(result) + return result + } + + return "" +} + func (c xpathScraperAttrConfig) postProcess(value string) string { // perform regex replacements first value = c.replaceRegex(value) value = c.parseDate(value) + value = c.applySubScraper(value) return value } diff --git a/ui/v2.5/src/components/Performers/PerformerDetails/PerformerDetailsPanel.tsx b/ui/v2.5/src/components/Performers/PerformerDetails/PerformerDetailsPanel.tsx index 906bdc9f7..5ca968d60 100644 --- a/ui/v2.5/src/components/Performers/PerformerDetails/PerformerDetailsPanel.tsx +++ b/ui/v2.5/src/components/Performers/PerformerDetails/PerformerDetailsPanel.tsx @@ -72,7 +72,7 @@ export const PerformerDetailsPanel: React.FC = ({ const [queryableScrapers, setQueryableScrapers] = useState([]); function updatePerformerEditState( - state: Partial + state: Partial ) { if ((state as GQL.PerformerDataFragment).favorite !== undefined) { setFavorite((state as GQL.PerformerDataFragment).favorite); @@ -94,6 +94,21 @@ export const PerformerDetailsPanel: React.FC = ({ setInstagram(state.instagram ?? undefined); } + function updatePerformerEditStateFromScraper( + state: Partial + ) { + updatePerformerEditState(state); + + // image is a base64 string + if ((state as GQL.ScrapedPerformerDataFragment).image !== undefined) { + let imageStr = (state as GQL.ScrapedPerformerDataFragment).image; + setImage(imageStr ?? undefined); + if (onImageChange) { + onImageChange(imageStr!); + } + } + } + useEffect(() => { setImage(undefined); updatePerformerEditState(performer); @@ -158,7 +173,8 @@ export const PerformerDetailsPanel: React.FC = ({ function getQueryScraperPerformerInput() { if (!scrapePerformerDetails) return {}; - const { __typename, ...ret } = scrapePerformerDetails; + // image is not supported + const { __typename, image, ...ret } = scrapePerformerDetails; return ret; } @@ -172,7 +188,7 @@ export const PerformerDetailsPanel: React.FC = ({ getQueryScraperPerformerInput() ); if (!result?.data?.scrapePerformer) return; - updatePerformerEditState(result.data.scrapePerformer); + updatePerformerEditStateFromScraper(result.data.scrapePerformer); } catch (e) { Toast.error(e); } finally { @@ -193,7 +209,7 @@ export const PerformerDetailsPanel: React.FC = ({ if (!result.data.scrapePerformerURL.url) { result.data.scrapePerformerURL.url = url; } - updatePerformerEditState(result.data.scrapePerformerURL); + updatePerformerEditStateFromScraper(result.data.scrapePerformerURL); } catch (e) { Toast.error(e); } finally { diff --git a/ui/v2.5/src/components/Scenes/SceneDetails/SceneEditPanel.tsx b/ui/v2.5/src/components/Scenes/SceneDetails/SceneEditPanel.tsx index b65b3d30b..99b68fe14 100644 --- a/ui/v2.5/src/components/Scenes/SceneDetails/SceneEditPanel.tsx +++ b/ui/v2.5/src/components/Scenes/SceneDetails/SceneEditPanel.tsx @@ -257,6 +257,12 @@ export const SceneEditPanel: React.FC = (props: IProps) => { setTagIds(newIds as string[]); } } + + if (scene.image) { + // image is a base64 string + setCoverImage(scene.image); + setCoverImagePreview(scene.image); + } } async function onScrapeSceneURL() { diff --git a/ui/v2.5/src/index.scss b/ui/v2.5/src/index.scss index 194f0076f..cd50f437c 100755 --- a/ui/v2.5/src/index.scss +++ b/ui/v2.5/src/index.scss @@ -140,6 +140,7 @@ code, /* this is a bit of a hack, because we can't supply direct class names to the react-select controls */ /* stylelint-disable selector-class-pattern */ + div.react-select__control { background-color: $secondary; border-color: $secondary; @@ -170,6 +171,39 @@ div.react-select__menu { cursor: pointer; } } + +/* we don't want to override this for dialogs, which are light colored */ +.modal { + div.react-select__control { + background-color: #fff; + border-color: inherit; + color: $dark-text; + + .react-select__single-value, + .react-select__input { + color: $dark-text; + } + + .react-select__multi-value { + background-color: #fff; + color: $dark-text; + } + } + + div.react-select__menu { + background-color: #fff; + color: $text-color; + + .react-select__option { + color: $dark-text; + } + + .react-select__option--is-focused { + background-color: rgba(167,182,194,.3); + } + } +} + /* stylelint-enable selector-class-pattern */ .image-thumbnail { diff --git a/ui/v2/src/components/performers/PerformerDetails/PerformerDetailsPanel.tsx b/ui/v2/src/components/performers/PerformerDetails/PerformerDetailsPanel.tsx index 3100ba51b..e4e9b795c 100644 --- a/ui/v2/src/components/performers/PerformerDetails/PerformerDetailsPanel.tsx +++ b/ui/v2/src/components/performers/PerformerDetails/PerformerDetailsPanel.tsx @@ -61,7 +61,7 @@ export const PerformerDetailsPanel: FunctionComponent = const Scrapers = StashService.useListPerformerScrapers(); const [queryableScrapers, setQueryableScrapers] = useState([]); - function updatePerformerEditState(state: Partial) { + function updatePerformerEditState(state: Partial) { if ((state as GQL.PerformerDataFragment).favorite !== undefined) { setFavorite((state as GQL.PerformerDataFragment).favorite); } @@ -82,6 +82,19 @@ export const PerformerDetailsPanel: FunctionComponent = setInstagram(state.instagram); } + function updatePerformerEditStateFromScraper(state: Partial) { + updatePerformerEditState(state); + + // image is a base64 string + if ((state as GQL.ScrapedPerformerDataFragment).image !== undefined) { + let imageStr = (state as GQL.ScrapedPerformerDataFragment).image; + setImage(imageStr); + if (props.onImageChange) { + props.onImageChange(imageStr!); + } + } + } + useEffect(() => { setImage(undefined); updatePerformerEditState(props.performer); @@ -169,6 +182,10 @@ export const PerformerDetailsPanel: FunctionComponent = let ret = _.clone(scrapePerformerDetails); delete ret.__typename; + + // image is not supported + delete ret.image; + return ret as GQL.ScrapedPerformerInput; } @@ -179,7 +196,7 @@ export const PerformerDetailsPanel: FunctionComponent = setIsLoading(true); const result = await StashService.queryScrapePerformer(isDisplayingScraperDialog.id, getQueryScraperPerformerInput()); if (!result.data || !result.data.scrapePerformer) { return; } - updatePerformerEditState(result.data.scrapePerformer); + updatePerformerEditStateFromScraper(result.data.scrapePerformer); } catch (e) { ErrorUtils.handle(e); } finally { @@ -199,7 +216,7 @@ export const PerformerDetailsPanel: FunctionComponent = result.data.scrapePerformerURL.url = url; } - updatePerformerEditState(result.data.scrapePerformerURL); + updatePerformerEditStateFromScraper(result.data.scrapePerformerURL); } catch (e) { ErrorUtils.handle(e); } finally { diff --git a/ui/v2/src/components/scenes/SceneDetails/SceneEditPanel.tsx b/ui/v2/src/components/scenes/SceneDetails/SceneEditPanel.tsx index 7295196cd..d224b8ccb 100644 --- a/ui/v2/src/components/scenes/SceneDetails/SceneEditPanel.tsx +++ b/ui/v2/src/components/scenes/SceneDetails/SceneEditPanel.tsx @@ -360,6 +360,12 @@ export const SceneEditPanel: FunctionComponent = (props: IProps) => { setTagIds(newIds as string[]); } } + + if (scene.image) { + // image is a base64 string + setCoverImage(scene.image); + setCoverImagePreview(scene.image); + } } async function onScrapeSceneURL() { diff --git a/vendor/github.com/jinzhu/copier/Guardfile b/vendor/github.com/jinzhu/copier/Guardfile new file mode 100644 index 000000000..0b860b065 --- /dev/null +++ b/vendor/github.com/jinzhu/copier/Guardfile @@ -0,0 +1,3 @@ +guard 'gotest' do + watch(%r{\.go$}) +end diff --git a/vendor/github.com/jinzhu/copier/License b/vendor/github.com/jinzhu/copier/License new file mode 100644 index 000000000..e2dc5381e --- /dev/null +++ b/vendor/github.com/jinzhu/copier/License @@ -0,0 +1,20 @@ +The MIT License (MIT) + +Copyright (c) 2015 Jinzhu + +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/vendor/github.com/jinzhu/copier/README.md b/vendor/github.com/jinzhu/copier/README.md new file mode 100644 index 000000000..f929b4679 --- /dev/null +++ b/vendor/github.com/jinzhu/copier/README.md @@ -0,0 +1,100 @@ +# Copier + + I am a copier, I copy everything from one to another + +[![wercker status](https://app.wercker.com/status/9d44ad2d4e6253929c8fb71359effc0b/s/master "wercker status")](https://app.wercker.com/project/byKey/9d44ad2d4e6253929c8fb71359effc0b) + +## Features + +* Copy from field to field with same name +* Copy from method to field with same name +* Copy from field to method with same name +* Copy from slice to slice +* Copy from struct to slice + +## Usage + +```go +package main + +import ( + "fmt" + "github.com/jinzhu/copier" +) + +type User struct { + Name string + Role string + Age int32 +} + +func (user *User) DoubleAge() int32 { + return 2 * user.Age +} + +type Employee struct { + Name string + Age int32 + DoubleAge int32 + EmployeId int64 + SuperRule string +} + +func (employee *Employee) Role(role string) { + employee.SuperRule = "Super " + role +} + +func main() { + var ( + user = User{Name: "Jinzhu", Age: 18, Role: "Admin"} + users = []User{{Name: "Jinzhu", Age: 18, Role: "Admin"}, {Name: "jinzhu 2", Age: 30, Role: "Dev"}} + employee = Employee{} + employees = []Employee{} + ) + + copier.Copy(&employee, &user) + + fmt.Printf("%#v \n", employee) + // Employee{ + // Name: "Jinzhu", // Copy from field + // Age: 18, // Copy from field + // DoubleAge: 36, // Copy from method + // EmployeeId: 0, // Ignored + // SuperRule: "Super Admin", // Copy to method + // } + + // Copy struct to slice + copier.Copy(&employees, &user) + + fmt.Printf("%#v \n", employees) + // []Employee{ + // {Name: "Jinzhu", Age: 18, DoubleAge: 36, EmployeId: 0, SuperRule: "Super Admin"} + // } + + // Copy slice to slice + employees = []Employee{} + copier.Copy(&employees, &users) + + fmt.Printf("%#v \n", employees) + // []Employee{ + // {Name: "Jinzhu", Age: 18, DoubleAge: 36, EmployeId: 0, SuperRule: "Super Admin"}, + // {Name: "jinzhu 2", Age: 30, DoubleAge: 60, EmployeId: 0, SuperRule: "Super Dev"}, + // } +} +``` + +## Contributing + +You can help to make the project better, check out [http://gorm.io/contribute.html](http://gorm.io/contribute.html) for things you can do. + +# Author + +**jinzhu** + +* +* +* + +## License + +Released under the [MIT License](https://github.com/jinzhu/copier/blob/master/License). diff --git a/vendor/github.com/jinzhu/copier/copier.go b/vendor/github.com/jinzhu/copier/copier.go new file mode 100644 index 000000000..3f339becc --- /dev/null +++ b/vendor/github.com/jinzhu/copier/copier.go @@ -0,0 +1,189 @@ +package copier + +import ( + "database/sql" + "errors" + "reflect" +) + +// Copy copy things +func Copy(toValue interface{}, fromValue interface{}) (err error) { + var ( + isSlice bool + amount = 1 + from = indirect(reflect.ValueOf(fromValue)) + to = indirect(reflect.ValueOf(toValue)) + ) + + if !to.CanAddr() { + return errors.New("copy to value is unaddressable") + } + + // Return is from value is invalid + if !from.IsValid() { + return + } + + fromType := indirectType(from.Type()) + toType := indirectType(to.Type()) + + // Just set it if possible to assign + // And need to do copy anyway if the type is struct + if fromType.Kind() != reflect.Struct && from.Type().AssignableTo(to.Type()) { + to.Set(from) + return + } + + if fromType.Kind() != reflect.Struct || toType.Kind() != reflect.Struct { + return + } + + if to.Kind() == reflect.Slice { + isSlice = true + if from.Kind() == reflect.Slice { + amount = from.Len() + } + } + + for i := 0; i < amount; i++ { + var dest, source reflect.Value + + if isSlice { + // source + if from.Kind() == reflect.Slice { + source = indirect(from.Index(i)) + } else { + source = indirect(from) + } + // dest + dest = indirect(reflect.New(toType).Elem()) + } else { + source = indirect(from) + dest = indirect(to) + } + + // check source + if source.IsValid() { + fromTypeFields := deepFields(fromType) + //fmt.Printf("%#v", fromTypeFields) + // Copy from field to field or method + for _, field := range fromTypeFields { + name := field.Name + + if fromField := source.FieldByName(name); fromField.IsValid() { + // has field + if toField := dest.FieldByName(name); toField.IsValid() { + if toField.CanSet() { + if !set(toField, fromField) { + if err := Copy(toField.Addr().Interface(), fromField.Interface()); err != nil { + return err + } + } + } + } else { + // try to set to method + var toMethod reflect.Value + if dest.CanAddr() { + toMethod = dest.Addr().MethodByName(name) + } else { + toMethod = dest.MethodByName(name) + } + + if toMethod.IsValid() && toMethod.Type().NumIn() == 1 && fromField.Type().AssignableTo(toMethod.Type().In(0)) { + toMethod.Call([]reflect.Value{fromField}) + } + } + } + } + + // Copy from method to field + for _, field := range deepFields(toType) { + name := field.Name + + var fromMethod reflect.Value + if source.CanAddr() { + fromMethod = source.Addr().MethodByName(name) + } else { + fromMethod = source.MethodByName(name) + } + + if fromMethod.IsValid() && fromMethod.Type().NumIn() == 0 && fromMethod.Type().NumOut() == 1 { + if toField := dest.FieldByName(name); toField.IsValid() && toField.CanSet() { + values := fromMethod.Call([]reflect.Value{}) + if len(values) >= 1 { + set(toField, values[0]) + } + } + } + } + } + if isSlice { + if dest.Addr().Type().AssignableTo(to.Type().Elem()) { + to.Set(reflect.Append(to, dest.Addr())) + } else if dest.Type().AssignableTo(to.Type().Elem()) { + to.Set(reflect.Append(to, dest)) + } + } + } + return +} + +func deepFields(reflectType reflect.Type) []reflect.StructField { + var fields []reflect.StructField + + if reflectType = indirectType(reflectType); reflectType.Kind() == reflect.Struct { + for i := 0; i < reflectType.NumField(); i++ { + v := reflectType.Field(i) + if v.Anonymous { + fields = append(fields, deepFields(v.Type)...) + } else { + fields = append(fields, v) + } + } + } + + return fields +} + +func indirect(reflectValue reflect.Value) reflect.Value { + for reflectValue.Kind() == reflect.Ptr { + reflectValue = reflectValue.Elem() + } + return reflectValue +} + +func indirectType(reflectType reflect.Type) reflect.Type { + for reflectType.Kind() == reflect.Ptr || reflectType.Kind() == reflect.Slice { + reflectType = reflectType.Elem() + } + return reflectType +} + +func set(to, from reflect.Value) bool { + if from.IsValid() { + if to.Kind() == reflect.Ptr { + //set `to` to nil if from is nil + if from.Kind() == reflect.Ptr && from.IsNil() { + to.Set(reflect.Zero(to.Type())) + return true + } else if to.IsNil() { + to.Set(reflect.New(to.Type().Elem())) + } + to = to.Elem() + } + + if from.Type().ConvertibleTo(to.Type()) { + to.Set(from.Convert(to.Type())) + } else if scanner, ok := to.Addr().Interface().(sql.Scanner); ok { + err := scanner.Scan(from.Interface()) + if err != nil { + return false + } + } else if from.Kind() == reflect.Ptr { + return set(to, from.Elem()) + } else { + return false + } + } + return true +} diff --git a/vendor/github.com/jinzhu/copier/wercker.yml b/vendor/github.com/jinzhu/copier/wercker.yml new file mode 100644 index 000000000..5e6ce981d --- /dev/null +++ b/vendor/github.com/jinzhu/copier/wercker.yml @@ -0,0 +1,23 @@ +box: golang + +build: + steps: + - setup-go-workspace + + # Gets the dependencies + - script: + name: go get + code: | + go get + + # Build the project + - script: + name: go build + code: | + go build ./... + + # Test the project + - script: + name: go test + code: | + go test ./...